This implements the current openqa logic - while we have no official API this will do as long as we keep it uptodate
306 lines
11 KiB
Python
306 lines
11 KiB
Python
#!/usr/bin/env python
|
|
# -*- coding: utf-8 -*-
|
|
#
|
|
# (C) 2014 mhrusecky@suse.cz, openSUSE.org
|
|
# (C) 2014 tchvatal@suse.cz, openSUSE.org
|
|
# Distribute under GPLv2 or GPLv3
|
|
|
|
import logging
|
|
import os.path
|
|
import re
|
|
import sys
|
|
from xml.etree import cElementTree as ET
|
|
|
|
from osc import cmdln, oscerr
|
|
from osc.core import delete_package
|
|
from osc.core import delete_project
|
|
from osc.core import make_meta_url
|
|
from osc.core import makeurl
|
|
from osc.core import meta_get_packagelist
|
|
from osc.core import http_GET
|
|
from osc.core import http_POST
|
|
from osc.core import server_diff
|
|
|
|
# Expand sys.path to search modules inside the pluging directory
|
|
_plugin_dir = os.path.expanduser('~/.osc-plugins')
|
|
sys.path.append(_plugin_dir)
|
|
from osclib.stagingapi import StagingAPI
|
|
from osclib.request_finder import RequestFinder
|
|
|
|
OSC_STAGING_VERSION='0.0.1'
|
|
|
|
def _print_version(self):
|
|
""" Print version information about this extension. """
|
|
print('%s'%(self.OSC_STAGING_VERSION))
|
|
quit(0)
|
|
|
|
|
|
def _get_changed(opts, project, everything):
|
|
ret = []
|
|
# Check for local changes
|
|
for pkg in meta_get_packagelist(opts.apiurl, project):
|
|
if len(ret) != 0 and not everything:
|
|
break
|
|
f = http_GET(makeurl(opts.apiurl, ['source', project, pkg]))
|
|
linkinfo = ET.parse(f).getroot().find('linkinfo')
|
|
if linkinfo is None:
|
|
ret.append({'pkg': pkg, 'code': 'NOT_LINK', 'msg': 'Not a source link'})
|
|
continue
|
|
if linkinfo.get('error'):
|
|
ret.append({'pkg': pkg, 'code': 'BROKEN', 'msg': 'Broken source link'})
|
|
continue
|
|
t = linkinfo.get('project')
|
|
p = linkinfo.get('package')
|
|
r = linkinfo.get('revision')
|
|
if len(server_diff(opts.apiurl, t, p, r, project, pkg, None, True)) > 0:
|
|
ret.append({'pkg': pkg, 'code': 'MODIFIED', 'msg': 'Has local modifications', 'pprj': t, 'ppkg': p})
|
|
continue
|
|
return ret
|
|
|
|
|
|
def _staging_remove(self, project, opts):
|
|
"""
|
|
Remove staging project.
|
|
:param project: staging project to delete
|
|
:param opts: pointer to options
|
|
"""
|
|
chng = _get_changed(opts, project, True)
|
|
if len(chng) > 0:
|
|
print('Staging project "%s" is not clean:'%(project))
|
|
print('')
|
|
for pair in chng:
|
|
print(' * %s : %s'%(pair['pkg'],pair['msg']))
|
|
print('')
|
|
print('Really delete? (N/y)')
|
|
answer = sys.stdin.readline()
|
|
if not re.search("^\s*[Yy]", answer):
|
|
print('Aborting...')
|
|
exit(1)
|
|
delete_project(opts.apiurl, project, force=True, msg=None)
|
|
print("Deleted.")
|
|
return
|
|
|
|
|
|
def _staging_submit_devel(self, project, opts):
|
|
"""
|
|
Generate new review requests for devel-projects based on our staging changes.
|
|
:param project: staging project to submit into devel projects
|
|
"""
|
|
chng = _get_changed(opts, project, True)
|
|
msg = "Fixes from staging project %s" % project
|
|
if opts.message is not None:
|
|
msg = opts.message
|
|
if len(chng) > 0:
|
|
for pair in chng:
|
|
if pair['code'] != 'MODIFIED':
|
|
print('Error: Package "%s": %s'%(pair['pkg'],pair['msg']))
|
|
else:
|
|
print('Sending changes back %s/%s -> %s/%s'%(project,pair['pkg'],pair['pprj'],pair['ppkg']))
|
|
action_xml = '<request>';
|
|
action_xml += ' <action type="submit"> <source project="%s" package="%s" /> <target project="%s" package="%s" />' % (project, pair['pkg'], pair['pprj'], pair['ppkg'])
|
|
action_xml += ' </action>'
|
|
action_xml += ' <state name="new"/> <description>%s</description>' % msg
|
|
action_xml += '</request>'
|
|
|
|
u = makeurl(opts.apiurl, ['request'], query='cmd=create&addrevision=1')
|
|
f = http_POST(u, data=action_xml)
|
|
|
|
root = ET.parse(f).getroot()
|
|
print("Created request %s" % (root.get('id')))
|
|
else:
|
|
print("No changes to submit")
|
|
return
|
|
|
|
|
|
def _staging_change_review_state(self, opts, id, newstate, by_group='', by_user='', message='', supersed=None):
|
|
""" taken from osc/osc/core.py, improved:
|
|
- verbose option added,
|
|
- empty by_user=& removed.
|
|
- numeric id can be int().
|
|
"""
|
|
query = {'cmd': 'changereviewstate', 'newstate': newstate }
|
|
if by_group: query['by_group'] = by_group
|
|
if by_user: query['by_user'] = by_user
|
|
if supersed: query['superseded_by'] = supersed
|
|
# if message: query['comment'] = message
|
|
u = makeurl(opts.apiurl, ['request', str(id)], query=query)
|
|
f = http_POST(u, data=message)
|
|
root = ET.parse(f).getroot()
|
|
return root.attrib['code']
|
|
|
|
|
|
def _staging_get_rings(self, opts):
|
|
ret = dict()
|
|
for prj in ['openSUSE:Factory:Rings:0-Bootstrap', 'openSUSE:Factory:Rings:1-MinimalX']:
|
|
u = makeurl(opts.apiurl, ['source', prj])
|
|
f = http_GET(u)
|
|
for entry in ET.parse(f).getroot().findall('entry'):
|
|
ret[entry.attrib['name']] = prj
|
|
return ret
|
|
|
|
def _staging_one_request(self, rq, opts):
|
|
if (opts.verbose):
|
|
ET.dump(rq)
|
|
print(opts)
|
|
id = int(rq.get('id'))
|
|
act_id = 0
|
|
actions = rq.findall('action')
|
|
act = actions[0]
|
|
|
|
tprj = act.find('target').get('project')
|
|
tpkg = act.find('target').get('package')
|
|
|
|
e = []
|
|
if not tpkg:
|
|
e.append('no target/package in request %d, action %d; ' % (id, act_id))
|
|
if not tprj:
|
|
e.append('no target/project in request %d, action %d; ' % (id, act_id))
|
|
# it is no error, if the target package dies not exist
|
|
|
|
ring = self.rings.get(tpkg, None)
|
|
if ring is None:
|
|
msg = "ok"
|
|
else:
|
|
stage_info = self.packages_staged.get(tpkg, ('', 0))
|
|
if stage_info[0] == self.letter_to_accept and int(stage_info[1]) == id:
|
|
# TODO make api for that
|
|
stprj = 'openSUSE:Factory:Staging:%s' % self.letter_to_accept
|
|
msg = 'ok, tested in %s' % stprj
|
|
delete_package(opts.apiurl, stprj, tpkg, msg='done')
|
|
elif stage_info[1] != 0 and int(stage_info[1]) != id:
|
|
print(stage_info)
|
|
print("osc staging select %s %s" % (stage_info[0], id))
|
|
return
|
|
elif stage_info[1] != 0: # keep silent about those already asigned
|
|
return
|
|
else:
|
|
print("Request(%d): %s -> %s" % (id, tpkg, ring))
|
|
return
|
|
|
|
self._staging_change_review_state(opts, id, 'accepted', by_group='factory-staging', message=msg)
|
|
|
|
@cmdln.option('-e', '--everything', action='store_true',
|
|
help='during check do not stop on first first issue and show them all')
|
|
@cmdln.option('-p', '--parent', metavar='TARGETPROJECT',
|
|
help='manually specify different parent project during creation of staging')
|
|
@cmdln.option('-m', '--message', metavar='TEXT',
|
|
help='manually specify different parent project during creation of staging')
|
|
@cmdln.option('-v', '--version', action='store_true',
|
|
help='show version of the plugin')
|
|
def do_staging(self, subcmd, opts, *args):
|
|
"""${cmd_name}: Commands to work with staging projects
|
|
|
|
"check" will check if all packages are links without changes
|
|
|
|
"remove" (or "r") will delete the staging project into submit requests for openSUSE:Factory
|
|
|
|
"submit-devel" (or "s") will create review requests for changed packages in staging project
|
|
into their respective devel projects to obtain approval from maitnainers for pushing the
|
|
changes to openSUSE:Factory
|
|
|
|
"freeze" will freeze the sources of the project's links (not affecting the packages actually in)
|
|
|
|
"accept" will accept all requests openSUSE:Factory:Staging:<LETTER>
|
|
|
|
"list" will pick the requests not in rings
|
|
|
|
"select" will add requests to the project
|
|
|
|
Usage:
|
|
osc staging check [--everything] REPO
|
|
osc staging remove REPO
|
|
osc staging submit-devel [-m message] REPO
|
|
osc staging freeze PROJECT
|
|
osc staging list
|
|
osc staging select LETTER REQUEST...
|
|
osc staging accept LETTER
|
|
osc staging cleanup_rings
|
|
"""
|
|
if opts.version:
|
|
self._print_version()
|
|
|
|
# verify the argument counts match the commands
|
|
if len(args) == 0:
|
|
raise oscerr.WrongArgs('No command given, see "osc help staging"!')
|
|
cmd = args[0]
|
|
if cmd in ['submit-devel', 's', 'remove', 'r', 'accept', 'freeze']:
|
|
min_args, max_args = 1, 1
|
|
elif cmd in ['check']:
|
|
min_args, max_args = 0, 2
|
|
elif cmd in ['select']:
|
|
min_args, max_args = 2, None
|
|
elif cmd in ['move']:
|
|
min_args, max_args = 3, None
|
|
elif cmd in ['list', 'cleanup_rings']:
|
|
min_args, max_args = 0, 0
|
|
else:
|
|
raise oscerr.WrongArgs('Unknown command: %s'%(cmd))
|
|
if len(args) - 1 < min_args:
|
|
raise oscerr.WrongArgs('Too few arguments.')
|
|
if not max_args is None and len(args) - 1 > max_args:
|
|
raise oscerr.WrongArgs('Too many arguments.')
|
|
|
|
# init the obs access
|
|
opts.apiurl = self.get_api_url()
|
|
opts.verbose = False
|
|
|
|
self.rings = self._staging_get_rings(opts)
|
|
api = StagingAPI(opts.apiurl)
|
|
|
|
# call the respective command and parse args by need
|
|
if cmd in ['check']:
|
|
if len(args) > 1:
|
|
return api.check_project_status(api.prj_from_letter(args[1]))
|
|
for prj in api.get_staging_projects():
|
|
print("Checking {}".format(prj))
|
|
api.check_project_status(prj)
|
|
print("")
|
|
return True
|
|
elif cmd in ['remove', 'r']:
|
|
project = args[1]
|
|
self._staging_remove(project, opts)
|
|
elif cmd in ['submit-devel', 's']:
|
|
project = args[1]
|
|
self._staging_submit_devel(project, opts)
|
|
elif cmd in ['freeze']:
|
|
import osclib.freeze_command
|
|
osclib.freeze_command.FreezeCommand(opts.apiurl).perform(api.prj_from_letter(args[1]))
|
|
elif cmd in ['select']:
|
|
stprj = api.prj_from_letter(args[1])
|
|
for rq in RequestFinder.find_sr(args[2:], opts.apiurl):
|
|
api.rq_to_prj(rq, stprj)
|
|
elif cmd in ['move']:
|
|
sprj = api.prj_from_letter(args[1])
|
|
tprj = api.prj_from_letter(args[2])
|
|
for rq in RequestFinder.find_sr(args[3:], opts.apiurl):
|
|
api.move_between_project(sprj, rq, tprj)
|
|
elif cmd in ['cleanup_rings']:
|
|
import osclib.cleanup_rings
|
|
osclib.cleanup_rings.CleanupRings(opts.apiurl).perform()
|
|
elif cmd in ['accept', 'list']:
|
|
self.letter_to_accept = None
|
|
if cmd == 'accept':
|
|
self.letter_to_accept = args[1]
|
|
|
|
self.packages_staged = dict()
|
|
for prj in api.get_staging_projects():
|
|
meta = api.get_prj_pseudometa(prj)
|
|
for req in meta['requests']:
|
|
self.packages_staged[req['package']] = (prj[-1], req['id'])
|
|
|
|
# xpath query, using the -m, -r, -s options
|
|
where = "@by_group='factory-staging'+and+@state='new'"
|
|
|
|
url = makeurl(opts.apiurl, ['search','request'], "match=state/@name='review'+and+review["+where+"]")
|
|
f = http_GET(url)
|
|
root = ET.parse(f).getroot()
|
|
for rq in root.findall('request'):
|
|
tprj = rq.find('action/target').get('project')
|
|
self._staging_one_request(rq, opts)
|
|
|
|
if self.letter_to_accept:
|
|
url = makeurl(opts.apiurl, ['source', 'openSUSE:Factory:Staging:%s' % self.letter_to_accept])
|
|
f = http_GET(url)
|
|
root = ET.parse(f).getroot()
|
|
print(ET.tostring(root))
|