Backport 794 PR and escape macros

Signed-off-by: Patrick Uiterwijk <patrick@puiterwijk.org>
epel9
Patrick Uiterwijk 7 years ago
parent 1352cef7c3
commit cf88a30bc0

@ -0,0 +1,354 @@
From b975594ce4ec1df300eccc261bedbd607e2c2aa0 Mon Sep 17 00:00:00 2001
From: Mike McLean <mikem@redhat.com>
Date: Feb 01 2018 15:12:52 +0000
Subject: [PATCH 1/4] recheck for duplicate external rpm on insertion errors
fixes: https://pagure.io/koji/issue/788
---
diff --git a/hub/kojihub.py b/hub/kojihub.py
index 7e664af..4500acd 100644
--- a/hub/kojihub.py
+++ b/hub/kojihub.py
@@ -5730,7 +5730,7 @@ def add_external_rpm(rpminfo, external_repo, strict=True):
# [!] Calling function should perform access checks
- #sanity check rpminfo
+ # sanity check rpminfo
dtypes = (
('name', basestring),
('version', basestring),
@@ -5746,37 +5746,47 @@ def add_external_rpm(rpminfo, external_repo, strict=True):
if not isinstance(rpminfo[field], allowed):
#this will catch unwanted NULLs
raise koji.GenericError("Invalid value for %s: %r" % (field, rpminfo[field]))
- #TODO: more sanity checks for payloadhash
+ # strip extra fields
+ rpminfo = dslice(rpminfo, [x[0] for x in dtypes])
+ # TODO: more sanity checks for payloadhash
- #Check to see if we have it
- data = rpminfo.copy()
- data['location'] = external_repo
- previous = get_rpm(data, strict=False)
+ def check_dup():
+ # Check to see if we have it
+ data = rpminfo.copy()
+ data['location'] = external_repo
+ previous = get_rpm(data, strict=False)
+ if previous:
+ disp = "%(name)s-%(version)s-%(release)s.%(arch)s@%(external_repo_name)s" % previous
+ if strict:
+ raise koji.GenericError("external rpm already exists: %s" % disp)
+ elif data['payloadhash'] != previous['payloadhash']:
+ raise koji.GenericError("hash changed for external rpm: %s (%s -> %s)" \
+ % (disp, previous['payloadhash'], data['payloadhash']))
+ else:
+ return previous
+
+ previous = check_dup()
if previous:
- disp = "%(name)s-%(version)s-%(release)s.%(arch)s@%(external_repo_name)s" % previous
- if strict:
- raise koji.GenericError("external rpm already exists: %s" % disp)
- elif data['payloadhash'] != previous['payloadhash']:
- raise koji.GenericError("hash changed for external rpm: %s (%s -> %s)" \
- % (disp, previous['payloadhash'], data['payloadhash']))
- else:
- return previous
+ return previous
- #add rpminfo entry
- rpminfo['external_repo_id'] = get_external_repo_id(external_repo, strict=True)
- rpminfo['id'] = _singleValue("""SELECT nextval('rpminfo_id_seq')""")
- q = """INSERT INTO rpminfo (id, build_id, buildroot_id,
- name, version, release, epoch, arch,
- external_repo_id,
- payloadhash, size, buildtime)
- VALUES (%(id)i, NULL, NULL,
- %(name)s, %(version)s, %(release)s, %(epoch)s, %(arch)s,
- %(external_repo_id)i,
- %(payloadhash)s, %(size)i, %(buildtime)i)
- """
- _dml(q, rpminfo)
+ # add rpminfo entry
+ data = rpminfo.copy()
+ data['external_repo_id'] = get_external_repo_id(external_repo, strict=True)
+ data['id'] = nextval('rpminfo_id_seq')
+ data['build_id'] = None
+ data['buildroot_id'] = None
+ insert = InsertProcessor('rpminfo', data=data)
+ try:
+ insert.execute()
+ except Exception:
+ # check for dup again to work around a race
+ # see: https://pagure.io/koji/issue/788
+ previous = check_dup()
+ if previous:
+ return previous
+ raise
- return get_rpm(rpminfo['id'])
+ return get_rpm(data['id'])
def import_build_log(fn, buildinfo, subdir=None):
"""Move a logfile related to a build to the right place"""
From 6dd0cb86ce5aa07d7fb199ea55e35cdec9d3d49d Mon Sep 17 00:00:00 2001
From: Mike McLean <mikem@redhat.com>
Date: Feb 04 2018 13:01:27 +0000
Subject: [PATCH 2/4] use a savepoint
---
diff --git a/hub/kojihub.py b/hub/kojihub.py
index 4500acd..3da3def 100644
--- a/hub/kojihub.py
+++ b/hub/kojihub.py
@@ -5776,11 +5776,13 @@ def add_external_rpm(rpminfo, external_repo, strict=True):
data['build_id'] = None
data['buildroot_id'] = None
insert = InsertProcessor('rpminfo', data=data)
+ savepoint = Savepoint('pre_insert')
try:
insert.execute()
except Exception:
- # check for dup again to work around a race
+ # if this failed, it likely duplicates one just inserted
# see: https://pagure.io/koji/issue/788
+ savepoint.rollback()
previous = check_dup()
if previous:
return previous
@@ -7395,6 +7397,16 @@ def nextval(sequence):
return _singleValue("SELECT nextval(%(sequence)s)", data, strict=True)
+class Savepoint(object):
+
+ def __init__(self, name):
+ self.name = name
+ _dml("SAVEPOINT %s" % name, {})
+
+ def rollback(self):
+ _dml("ROLLBACK TO SAVEPOINT %s" % self.name, {})
+
+
def parse_json(value, desc=None, errstr=None):
if value is None:
return value
From 3c5c9f74d446ee6bca15cadceaf3c21b2f26be7c Mon Sep 17 00:00:00 2001
From: Mike McLean <mikem@redhat.com>
Date: Feb 04 2018 15:08:00 +0000
Subject: [PATCH 3/4] unit test for add_external_rpm
---
diff --git a/tests/test_hub/test_add_external_rpm.py b/tests/test_hub/test_add_external_rpm.py
new file mode 100644
index 0000000..7aa928b
--- /dev/null
+++ b/tests/test_hub/test_add_external_rpm.py
@@ -0,0 +1,158 @@
+import mock
+import unittest
+
+import koji
+import kojihub
+
+
+IP = kojihub.InsertProcessor
+
+
+class FakeException(Exception):
+ pass
+
+
+class TestAddExternalRPM(unittest.TestCase):
+
+ def setUp(self):
+ self.get_rpm = mock.patch('kojihub.get_rpm').start()
+ self.get_external_repo_id = mock.patch('kojihub.get_external_repo_id').start()
+ self.nextval = mock.patch('kojihub.nextval').start()
+ self.Savepoint = mock.patch('kojihub.Savepoint').start()
+ self.InsertProcessor = mock.patch('kojihub.InsertProcessor',
+ side_effect=self.getInsert).start()
+ self.inserts = []
+ self.insert_execute = mock.MagicMock()
+
+ self.rpminfo = {
+ 'name': 'NAME',
+ 'version': 'VERSION',
+ 'release': 'RELEASE',
+ 'epoch': None,
+ 'arch': 'noarch',
+ 'payloadhash': 'fakehash',
+ 'size': 42,
+ 'buildtime': 0,
+ }
+ self.repo = 'myrepo'
+
+ def tearDown(self):
+ mock.patch.stopall()
+
+ def getInsert(self, *args, **kwargs):
+ insert = IP(*args, **kwargs)
+ insert.execute = self.insert_execute
+ self.inserts.append(insert)
+ return insert
+
+ def test_add_ext_rpm(self):
+ self.get_rpm.return_value = None
+ self.get_external_repo_id.return_value = mock.sentinel.repo_id
+ self.nextval.return_value = mock.sentinel.rpm_id
+
+ # call it
+ kojihub.add_external_rpm(self.rpminfo, self.repo)
+
+ self.assertEqual(len(self.inserts), 1)
+ insert = self.inserts[0]
+ self.assertEqual(insert.data['external_repo_id'], mock.sentinel.repo_id)
+ self.assertEqual(insert.data['id'], mock.sentinel.rpm_id)
+ self.assertEqual(insert.table, 'rpminfo')
+
+ def test_add_ext_rpm_bad_data(self):
+ rpminfo = self.rpminfo.copy()
+ del rpminfo['size']
+
+ with self.assertRaises(koji.GenericError):
+ kojihub.add_external_rpm(rpminfo, self.repo)
+
+ self.get_rpm.assert_not_called()
+ self.nextval.assert_not_called()
+ self.assertEqual(len(self.inserts), 0)
+
+ rpminfo = self.rpminfo.copy()
+ rpminfo['size'] = ['invalid type']
+
+ with self.assertRaises(koji.GenericError):
+ kojihub.add_external_rpm(rpminfo, self.repo)
+
+ self.get_rpm.assert_not_called()
+ self.nextval.assert_not_called()
+ self.assertEqual(len(self.inserts), 0)
+
+ def test_add_ext_rpm_dup(self):
+ prev = self.rpminfo.copy()
+ prev['external_repo_id'] = mock.sentinel.repo_id
+ prev['external_repo_name'] = self.repo
+ self.get_rpm.return_value = prev
+ self.get_external_repo_id.return_value = mock.sentinel.repo_id
+
+ # call it (default is strict)
+ with self.assertRaises(koji.GenericError):
+ kojihub.add_external_rpm(self.rpminfo, self.repo)
+
+ self.assertEqual(len(self.inserts), 0)
+ self.nextval.assert_not_called()
+
+ # call it without strict
+ ret = kojihub.add_external_rpm(self.rpminfo, self.repo, strict=False)
+
+ self.assertEqual(ret, self.get_rpm.return_value)
+ self.assertEqual(len(self.inserts), 0)
+ self.nextval.assert_not_called()
+
+ # different hash
+ prev['payloadhash'] = 'different hash'
+ with self.assertRaises(koji.GenericError):
+ kojihub.add_external_rpm(self.rpminfo, self.repo, strict=False)
+
+ self.assertEqual(len(self.inserts), 0)
+ self.nextval.assert_not_called()
+
+ def test_add_ext_rpm_dup_late(self):
+ prev = self.rpminfo.copy()
+ prev['external_repo_id'] = mock.sentinel.repo_id
+ prev['external_repo_name'] = self.repo
+ self.get_rpm.side_effect = [None, prev]
+ self.get_external_repo_id.return_value = mock.sentinel.repo_id
+ self.insert_execute.side_effect = FakeException('insert failed')
+
+ # call it (default is strict)
+ with self.assertRaises(koji.GenericError):
+ kojihub.add_external_rpm(self.rpminfo, self.repo)
+
+ self.assertEqual(len(self.inserts), 1)
+ self.nextval.assert_called_once()
+
+ # call it without strict
+ self.inserts[:] = []
+ self.nextval.reset_mock()
+ self.get_rpm.side_effect = [None, prev]
+ ret = kojihub.add_external_rpm(self.rpminfo, self.repo, strict=False)
+
+ self.assertEqual(ret, prev)
+ self.assertEqual(len(self.inserts), 1)
+ self.nextval.assert_called_once()
+
+ # different hash
+ self.inserts[:] = []
+ self.nextval.reset_mock()
+ self.get_rpm.side_effect = [None, prev]
+ prev['payloadhash'] = 'different hash'
+ with self.assertRaises(koji.GenericError):
+ kojihub.add_external_rpm(self.rpminfo, self.repo, strict=False)
+
+ self.assertEqual(len(self.inserts), 1)
+ self.nextval.assert_called_once()
+
+ # no dup after failed insert
+ self.inserts[:] = []
+ self.nextval.reset_mock()
+ self.get_rpm.side_effect = [None, None]
+ with self.assertRaises(FakeException):
+ kojihub.add_external_rpm(self.rpminfo, self.repo, strict=False)
+
+ self.assertEqual(len(self.inserts), 1)
+ self.nextval.assert_called_once()
+
+
From a7e866cf8a39f56e694372cfd9d6099f7bc5be0d Mon Sep 17 00:00:00 2001
From: Mike McLean <mikem@redhat.com>
Date: Feb 04 2018 15:15:40 +0000
Subject: [PATCH 4/4] unit test for Savepoint
---
diff --git a/tests/test_hub/test_savepoint.py b/tests/test_hub/test_savepoint.py
new file mode 100644
index 0000000..279729e
--- /dev/null
+++ b/tests/test_hub/test_savepoint.py
@@ -0,0 +1,22 @@
+import mock
+import unittest
+
+import kojihub
+
+
+class TestSavepoint(unittest.TestCase):
+
+ def setUp(self):
+ self.dml = mock.patch('kojihub._dml').start()
+
+ def tearDown(self):
+ mock.patch.stopall()
+
+ def test_savepoint(self):
+ sp = kojihub.Savepoint('some_name')
+ self.assertEqual(sp.name, 'some_name')
+ self.dml.assert_called_once_with('SAVEPOINT some_name', {})
+
+ self.dml.reset_mock()
+ sp.rollback()
+ self.dml.assert_called_once_with('ROLLBACK TO SAVEPOINT some_name', {})

@ -26,7 +26,7 @@
Name: koji Name: koji
Version: 1.15.0 Version: 1.15.0
Release: 3%{?dist} Release: 4%{?dist}
# koji.ssl libs (from plague) are GPLv2+ # koji.ssl libs (from plague) are GPLv2+
License: LGPLv2 and GPLv2+ License: LGPLv2 and GPLv2+
Summary: Build system tools Summary: Build system tools
@ -36,6 +36,7 @@ Source0: https://releases.pagure.org/koji/koji-%{version}.tar.bz2
# Backported patches # Backported patches
Patch0: https://pagure.io/koji/pull-request/735.patch Patch0: https://pagure.io/koji/pull-request/735.patch
Patch1: https://pagure.io/koji/pull-request/794.patch
# Not upstreamable # Not upstreamable
Patch100: fedora-config.patch Patch100: fedora-config.patch
@ -242,6 +243,7 @@ koji-web is a web UI to the Koji system.
%prep %prep
%setup -q %setup -q
%patch0 -p1 %patch0 -p1
%patch1 -p1
%patch100 -p1 -b .fedoraconfig %patch100 -p1 -b .fedoraconfig
%build %build
@ -282,16 +284,16 @@ sed -i 's/\#\!\/usr\/bin\/python/\#\!\/usr\/bin\/python3/' $RPM_BUILD_ROOT/usr/b
%defattr(-,root,root) %defattr(-,root,root)
%{python2_sitelib}/koji_cli_plugins %{python2_sitelib}/koji_cli_plugins
# we don't have config files for default plugins yet # we don't have config files for default plugins yet
#%%dir %{_sysconfdir}/koji/plugins #%%dir %%{_sysconfdir}/koji/plugins
#%%config(noreplace) %{_sysconfdir}/koji/plugins/*.conf #%%config(noreplace) %%{_sysconfdir}/koji/plugins/*.conf
%if 0%{with python3} %if 0%{with python3}
%files -n python%{python3_pkgversion}-%{name}-cli-plugins %files -n python%{python3_pkgversion}-%{name}-cli-plugins
%defattr(-,root,root) %defattr(-,root,root)
%{python3_sitelib}/koji_cli_plugins %{python3_sitelib}/koji_cli_plugins
# we don't have config files for default plugins yet # we don't have config files for default plugins yet
#%%dir %{_sysconfdir}/koji/plugins #%%dir %%{_sysconfdir}/koji/plugins
#%%config(noreplace) %{_sysconfdir}/koji/plugins/*.conf #%%config(noreplace) %%{_sysconfdir}/koji/plugins/*.conf
%endif %endif
%files hub %files hub
@ -387,7 +389,7 @@ fi
%files vm %files vm
%defattr(-,root,root) %defattr(-,root,root)
%{_sbindir}/kojivmd %{_sbindir}/kojivmd
#dir %{_datadir}/kojivmd #dir %%{_datadir}/kojivmd
%{_datadir}/kojivmd/kojikamid %{_datadir}/kojivmd/kojikamid
%if %{use_systemd} %if %{use_systemd}
%{_unitdir}/kojivmd.service %{_unitdir}/kojivmd.service
@ -444,6 +446,10 @@ fi
%endif %endif
%changelog %changelog
* Fri Feb 16 2018 Patrick Uiterwijk <patrick@puiterwijk.org> - 1.15.0-4
- Backport patch from PR#794
- Fix macro escaping in comments
* Mon Feb 12 2018 Owen Taylor <otaylor@redhat.com> - 1.15.0-3 * Mon Feb 12 2018 Owen Taylor <otaylor@redhat.com> - 1.15.0-3
- Make hub, builder, etc, require python2-koji not koji - Make hub, builder, etc, require python2-koji not koji

Loading…
Cancel
Save