213 lines
6.7 KiB
Python
Raw Permalink Normal View History

2019-05-16 06:59:25 +02:00
#!/usr/bin/python3
2018-11-17 09:43:04 +01:00
from lxml import etree as ET
2016-10-06 18:40:51 +02:00
import cmdln
2017-10-17 09:10:39 +02:00
import datetime
2016-10-06 18:40:51 +02:00
import logging
2017-10-17 09:10:39 +02:00
import signal
2016-10-06 18:40:51 +02:00
import sys
import time
2018-11-16 08:32:25 +01:00
from urllib.error import HTTPError, URLError
2019-05-16 06:59:25 +02:00
2016-10-06 18:40:51 +02:00
import osc.conf
import osc.core
from osclib.memoize import memoize
logger = logging.getLogger()
http_GET = osc.core.http_GET
http_DELETE = osc.core.http_DELETE
http_POST = osc.core.http_POST
def chunks(line, n):
2016-10-06 18:40:51 +02:00
""" Yield successive n-sized chunks from l.
"""
# http://stackoverflow.com/questions/312443/how-do-you-split-a-list-into-evenly-sized-chunks-in-python
for i in range(0, len(line), n):
yield line[i:i + n]
2016-10-06 18:40:51 +02:00
2016-10-06 18:40:51 +02:00
class ToolBase(object):
def __init__(self):
self.apiurl = osc.conf.config['apiurl']
self.debug = osc.conf.config['debug']
self.caching = False
self.dryrun = False
@memoize(add_invalidate=True)
def _cached_GET(self, url):
return self.retried_GET(url).read()
def cached_GET(self, url):
if self.caching:
return self._cached_GET(url)
return self.retried_GET(url).read()
def retried_GET(self, url):
try:
return http_GET(url)
2018-11-16 08:32:25 +01:00
except HTTPError as e:
2016-10-06 18:40:51 +02:00
if 500 <= e.code <= 599:
2024-05-07 17:55:17 +02:00
print(f'Retrying {url}')
2016-10-06 18:40:51 +02:00
time.sleep(1)
return self.retried_GET(url)
2017-06-02 13:46:24 +02:00
logging.error('%s: %s', e, url)
2016-10-06 18:40:51 +02:00
raise e
except URLError as e:
logging.error('%s: "%s - %s" %s', e, e.reason, type(e.reason), url)
# connection timeout
if isinstance(e.reason, TimeoutError):
2024-05-07 17:55:17 +02:00
print(f'Retrying {url}')
time.sleep(1)
return self.retried_GET(url)
raise e
2016-10-06 18:40:51 +02:00
def http_PUT(self, *args, **kwargs):
if self.dryrun:
logging.debug("dryrun PUT %s %s", args, str(kwargs)[:200])
else:
osc.core.http_PUT(*args, **kwargs)
def http_POST(self, *args, **kwargs):
if self.dryrun:
logging.debug("dryrun POST %s %s", args, str(kwargs)[:200])
else:
osc.core.http_POST(*args, **kwargs)
2016-10-06 18:40:51 +02:00
def get_project_meta(self, prj):
2017-05-05 10:57:11 +02:00
url = self.makeurl(['source', prj, '_meta'])
2016-10-06 18:40:51 +02:00
return self.cached_GET(url)
def _meta_get_packagelist(self, prj, deleted=None, expand=False):
query = {}
if deleted:
query['deleted'] = 1
if expand:
query['expand'] = 1
2017-05-05 10:57:11 +02:00
u = self.makeurl(['source', prj], query)
2016-10-06 18:40:51 +02:00
return self.cached_GET(u)
def meta_get_packagelist(self, prj, deleted=None, expand=False):
root = ET.fromstring(self._meta_get_packagelist(prj, deleted, expand))
res = list()
for node in root.findall('entry'):
name = node.get('name')
if not (name == '000product' or name.startswith('patchinfo.')):
res.push(name)
return res
2016-10-06 18:40:51 +02:00
def latest_packages(self, project):
2017-05-05 10:57:11 +02:00
data = self.cached_GET(self.makeurl(['project', 'latest_commits', project]))
2016-10-06 18:40:51 +02:00
lc = ET.fromstring(data)
packages = set()
for entry in lc.findall('{http://www.w3.org/2005/Atom}entry'):
title = entry.find('{http://www.w3.org/2005/Atom}title').text
if title.startswith('In '):
packages.add(title[3:].split(' ')[0])
return sorted(packages)
def makeurl(self, paths, query=None):
2017-05-05 10:57:11 +02:00
"""
Wrapper around osc's makeurl passing our apiurl
:return url made for l and query
"""
query = [] if not query else query
return osc.core.makeurl(self.apiurl, paths, query)
2017-05-05 10:57:11 +02:00
2016-10-06 18:40:51 +02:00
def process(self, packages):
""" reimplement this """
True
2016-10-06 18:40:51 +02:00
class CommandLineInterface(cmdln.Cmdln):
def __init__(self, *args, **kwargs):
cmdln.Cmdln.__init__(self, *args, **kwargs)
def get_optparser(self):
parser = cmdln.Cmdln.get_optparser(self)
parser.add_option("--apiurl", '-A', metavar="URL", help="api url")
parser.add_option("--dry", action="store_true", help="dry run")
parser.add_option("-d", "--debug", action="store_true", help="debug output")
2016-10-06 18:40:51 +02:00
parser.add_option("--osc-debug", action="store_true", help="osc debug output")
parser.add_option("--verbose", action="store_true", help="verbose")
parser.add_option("--http-debug", action="store_true", help="osc http debug output")
parser.add_option('--http-full-debug', action='store_true',
help='debug HTTP traffic (filters no headers)')
2016-10-06 18:40:51 +02:00
parser.add_option('--cache-requests', action='store_true', default=False,
help='cache GET requests. Not recommended for daily use.')
2016-10-06 18:40:51 +02:00
return parser
def postoptparse(self):
level = None
if (self.options.debug):
level = logging.DEBUG
elif (self.options.verbose):
level = logging.INFO
logging.basicConfig(level=level)
osc.conf.get_config(override_apiurl=self.options.apiurl,
override_debug=self.options.osc_debug,
override_http_debug=self.options.http_debug,
override_http_full_debug=self.options.http_full_debug)
2016-10-06 18:40:51 +02:00
self.tool = self.setup_tool()
self.tool.dryrun = self.options.dry
self.tool.caching = self.options.cache_requests
def setup_tool(self, toolclass=ToolBase):
2016-10-06 18:40:51 +02:00
""" reimplement this """
tool = toolclass()
return tool
# example
# @cmdln.option('-n', '--interval', metavar="minutes", type="int", help="periodic interval in minutes")
# def do_process(self, subcmd, opts, project):
# def work():
# self.tool.process()
#
# self.runner(work, opts.interval)
def runner(self, workfunc, interval):
""" runs the specified callback every <interval> minutes or
once if interval is None or 0
"""
class ExTimeout(Exception):
"""raised on timeout"""
if interval:
def alarm_called(nr, frame):
raise ExTimeout()
signal.signal(signal.SIGALRM, alarm_called)
while True:
try:
workfunc()
except Exception as e:
2016-10-06 18:40:51 +02:00
logger.exception(e)
if interval:
logger.info("sleeping %d minutes. Press enter to check now ..." % interval)
signal.alarm(interval * 60)
2016-10-06 18:40:51 +02:00
try:
input()
2016-10-06 18:40:51 +02:00
except ExTimeout:
pass
signal.alarm(0)
2024-05-07 17:55:17 +02:00
logger.info(f"recheck at {datetime.datetime.now().isoformat()}")
2016-10-06 18:40:51 +02:00
continue
break
2016-10-06 18:40:51 +02:00
if __name__ == "__main__":
app = CommandLineInterface()
sys.exit(app.main())