Update pylorax.api.projects for DNF usage

And adjust tests to match.
This commit is contained in:
Brian C. Lane 2018-05-02 10:00:58 -07:00
parent e86f4f9a36
commit 9b9b627fe5
2 changed files with 144 additions and 187 deletions

View File

@ -17,10 +17,9 @@
import logging import logging
log = logging.getLogger("lorax-composer") log = logging.getLogger("lorax-composer")
import dnf
import time import time
from yum.Errors import YumBaseError
TIME_FORMAT = "%Y-%m-%dT%H:%M:%S" TIME_FORMAT = "%Y-%m-%dT%H:%M:%S"
@ -56,80 +55,80 @@ def api_changelog(changelog):
return entry return entry
def yaps_to_project(yaps): def pkg_to_project(pkg):
"""Extract the details from a YumAvailablePackageSqlite object """Extract the details from a hawkey.Package object
:param yaps: Yum object with package details :param pkgs: hawkey.Package object with package details
:type yaps: YumAvailablePackageSqlite :type pkgs: hawkey.Package
:returns: A dict with the name, summary, description, and url. :returns: A dict with the name, summary, description, and url.
:rtype: dict :rtype: dict
upstream_vcs is hard-coded to UPSTREAM_VCS upstream_vcs is hard-coded to UPSTREAM_VCS
""" """
return {"name": yaps.name, return {"name": pkg.name,
"summary": yaps.summary, "summary": pkg.summary,
"description": yaps.description, "description": pkg.description,
"homepage": yaps.url, "homepage": pkg.url,
"upstream_vcs": "UPSTREAM_VCS"} "upstream_vcs": "UPSTREAM_VCS"}
def yaps_to_project_info(yaps): def pkg_to_project_info(pkg):
"""Extract the details from a YumAvailablePackageSqlite object """Extract the details from a hawkey.Package object
:param yaps: Yum object with package details :param pkg: hawkey.Package object with package details
:type yaps: YumAvailablePackageSqlite :type pkg: hawkey.Package
:returns: A dict with the project details, as well as epoch, release, arch, build_time, changelog, ... :returns: A dict with the project details, as well as epoch, release, arch, build_time, changelog, ...
:rtype: dict :rtype: dict
metadata entries are hard-coded to {} metadata entries are hard-coded to {}
""" """
build = {"epoch": int(yaps.epoch), build = {"epoch": pkg.epoch,
"release": yaps.release, "release": pkg.release,
"arch": yaps.arch, "arch": pkg.arch,
"build_time": api_time(yaps.buildtime), "build_time": api_time(pkg.buildtime),
"changelog": api_changelog(yaps.returnChangelog()), "changelog": "CHANGELOG_NEEDED", # XXX Not in hawkey.Package
"build_config_ref": "BUILD_CONFIG_REF", "build_config_ref": "BUILD_CONFIG_REF",
"build_env_ref": "BUILD_ENV_REF", "build_env_ref": "BUILD_ENV_REF",
"metadata": {}, "metadata": {},
"source": {"license": yaps.license, "source": {"license": pkg.license,
"version": yaps.version, "version": pkg.version,
"source_ref": "SOURCE_REF", "source_ref": "SOURCE_REF",
"metadata": {}}} "metadata": {}}}
return {"name": yaps.name, return {"name": pkg.name,
"summary": yaps.summary, "summary": pkg.summary,
"description": yaps.description, "description": pkg.description,
"homepage": yaps.url, "homepage": pkg.url,
"upstream_vcs": "UPSTREAM_VCS", "upstream_vcs": "UPSTREAM_VCS",
"builds": [build]} "builds": [build]}
def tm_to_dep(tm): def pkg_to_dep(pkg):
"""Extract the info from a TransactionMember object """Extract the info from a hawkey.Package object
:param tm: A Yum transaction object :param pkg: A hawkey.Package object
:type tm: TransactionMember :type pkg: hawkey.Package
:returns: A dict with name, epoch, version, release, arch :returns: A dict with name, epoch, version, release, arch
:rtype: dict :rtype: dict
""" """
return {"name": tm.name, return {"name": pkg.name,
"epoch": int(tm.epoch), "epoch": pkg.epoch,
"version": tm.version, "version": pkg.version,
"release": tm.release, "release": pkg.release,
"arch": tm.arch} "arch": pkg.arch}
def yaps_to_module(yaps): def proj_to_module(proj):
"""Extract the name from a YumAvailablePackageSqlite object """Extract the name from a project_info dict
:param yaps: Yum object with package details :param pkg: dict with package details
:type yaps: YumAvailablePackageSqlite :type pkg: dict
:returns: A dict with name, and group_type :returns: A dict with name, and group_type
:rtype: dict :rtype: dict
group_type is hard-coded to "rpm" group_type is hard-coded to "rpm"
""" """
return {"name": yaps.name, return {"name": proj["name"],
"group_type": "rpm"} "group_type": "rpm"}
@ -150,122 +149,126 @@ def dep_nevra(dep):
"""Return the name-epoch:version-release.arch""" """Return the name-epoch:version-release.arch"""
return dep["name"]+"-"+dep_evra(dep) return dep["name"]+"-"+dep_evra(dep)
def projects_list(yb):
def projects_list(dbo):
"""Return a list of projects """Return a list of projects
:param yb: yum base object :param dbo: dnf base object
:type yb: YumBase :type dbo: dnf.Base
:returns: List of project info dicts with name, summary, description, homepage, upstream_vcs :returns: List of project info dicts with name, summary, description, homepage, upstream_vcs
:rtype: list of dicts :rtype: list of dicts
""" """
try: return projects_info(dbo, None)
ybl = yb.doPackageLists(pkgnarrow="available", showdups=False)
except YumBaseError as e:
raise ProjectsError("There was a problem listing projects: %s" % str(e))
finally:
yb.closeRpmDB()
return sorted(map(yaps_to_project, ybl.available), key=lambda p: p["name"].lower())
def projects_info(yb, project_names): def projects_info(dbo, project_names):
"""Return details about specific projects """Return details about specific projects
:param yb: yum base object :param dbo: dnf base object
:type yb: YumBase :type dbo: dnf.Base
:param project_names: List of names of projects to get info about :param project_names: List of names of projects to get info about
:type project_names: str :type project_names: str
:returns: List of project info dicts with yaps_to_project as well as epoch, version, release, etc. :returns: List of project info dicts with pkg_to_project as well as epoch, version, release, etc.
:rtype: list of dicts :rtype: list of dicts
If project_names is None it will return the full list of available packages
""" """
try: if project_names:
ybl = yb.doPackageLists(pkgnarrow="available", patterns=project_names, showdups=False) pkgs = dbo.sack.query().available().filter(name__glob=project_names)
except YumBaseError as e: else:
raise ProjectsError("There was a problem with info for %s: %s" % (project_names, str(e))) pkgs = dbo.sack.query().available()
finally: return sorted(map(pkg_to_project_info, pkgs), key=lambda p: p["name"].lower())
yb.closeRpmDB()
return sorted(map(yaps_to_project_info, ybl.available), key=lambda p: p["name"].lower())
def projects_depsolve(yb, project_names): def projects_depsolve(dbo, project_names):
"""Return the dependencies for a list of projects """Return the dependencies for a list of projects
:param yb: yum base object :param dbo: dnf base object
:type yb: YumBase :type dbo: dnf.Base
:param project_names: The projects to find the dependencies for :param project_names: The projects to find the dependencies for
:type project_names: List of Strings :type project_names: List of Strings
:returns: NEVRA's of the project and its dependencies :returns: NEVRA's of the project and its dependencies
:rtype: list of dicts :rtype: list of dicts
""" """
try:
# This resets the transaction # This resets the transaction
yb.closeRpmDB() dbo.reset(goal=True)
for p in project_names: for p in project_names:
yb.install(pattern=p) try:
(rc, msg) = yb.buildTransaction() dbo.install(p)
if rc not in [0, 1, 2]: except dnf.exceptions.MarkingError:
raise ProjectsError("There was a problem depsolving %s: %s" % (project_names, msg)) raise ProjectsError("No match for %s" % p)
yb.tsInfo.makelists()
deps = sorted(map(tm_to_dep, yb.tsInfo.installed + yb.tsInfo.depinstalled), key=lambda p: p["name"].lower()) try:
except YumBaseError as e: dbo.resolve()
except dnf.exceptions.DepsolveError as e:
raise ProjectsError("There was a problem depsolving %s: %s" % (project_names, str(e))) raise ProjectsError("There was a problem depsolving %s: %s" % (project_names, str(e)))
finally:
yb.closeRpmDB() if len(dbo.transaction) == 0:
return deps raise ProjectsError("No packages installed for %s" % project_names)
return sorted(map(pkg_to_dep, dbo.transaction.install_set), key=lambda p: p["name"].lower())
def estimate_size(packages, block_size=4096): def estimate_size(packages, block_size=4096):
"""Estimate the installed size of a package list """Estimate the installed size of a package list
:param packages: The packages to be installed :param packages: The packages to be installed
:type packages: list of TransactionMember objects :type packages: list of hawkey.Package objects
:param block_size: The block size to use for rounding up file sizes. :param block_size: The block size to use for rounding up file sizes.
:type block_size: int :type block_size: int
:returns: The estimated size of installed packages :returns: The estimated size of installed packages
:rtype: int :rtype: int
Estimating actual requirements is difficult without the actual file sizes, which Estimating actual requirements is difficult without the actual file sizes, which
yum doesn't provide access to. So use the file count and block size to estimate dnf doesn't provide access to. So use the file count and block size to estimate
a minimum size for each package. a minimum size for each package.
""" """
installed_size = 0 installed_size = 0
for p in packages: for p in packages:
installed_size += len(p.po.filelist) * block_size installed_size += len(p.files) * block_size
installed_size += p.po.installedsize installed_size += p.installsize
return installed_size return installed_size
def projects_depsolve_with_size(yb, project_names, with_core=True):
def projects_depsolve_with_size(dbo, project_names, with_core=True):
"""Return the dependencies and installed size for a list of projects """Return the dependencies and installed size for a list of projects
:param yb: yum base object :param dbo: dnf base object
:type yb: YumBase :type dbo: dnf.Base
:param project_names: The projects to find the dependencies for :param project_names: The projects to find the dependencies for
:type project_names: List of Strings :type project_names: List of Strings
:returns: installed size and a list of NEVRA's of the project and its dependencies :returns: installed size and a list of NEVRA's of the project and its dependencies
:rtype: tuple of (int, list of dicts) :rtype: tuple of (int, list of dicts)
""" """
try:
# This resets the transaction # This resets the transaction
yb.closeRpmDB() dbo.reset(goal=True)
for p in project_names: for p in project_names:
yb.install(pattern=p) try:
dbo.install(p)
except dnf.exceptions.MarkingError:
raise ProjectsError("No match for %s" % p)
if with_core: if with_core:
yb.selectGroup("core", group_package_types=['mandatory', 'default', 'optional']) dbo.group_install("core", ['mandatory', 'default', 'optional'])
(rc, msg) = yb.buildTransaction()
if rc not in [0, 1, 2]: try:
raise ProjectsError("There was a problem depsolving %s: %s" % (project_names, msg)) dbo.resolve()
yb.tsInfo.makelists() except dnf.exceptions.DepsolveError as e:
installed_size = estimate_size(yb.tsInfo.installed + yb.tsInfo.depinstalled)
deps = sorted(map(tm_to_dep, yb.tsInfo.installed + yb.tsInfo.depinstalled), key=lambda p: p["name"].lower())
except YumBaseError as e:
raise ProjectsError("There was a problem depsolving %s: %s" % (project_names, str(e))) raise ProjectsError("There was a problem depsolving %s: %s" % (project_names, str(e)))
finally:
yb.closeRpmDB() if len(dbo.transaction) == 0:
raise ProjectsError("No packages installed for %s" % project_names)
installed_size = estimate_size(dbo.transaction.install_set)
deps = sorted(map(pkg_to_dep, dbo.transaction.install_set), key=lambda p: p["name"].lower())
return (installed_size, deps) return (installed_size, deps)
def modules_list(yb, module_names):
def modules_list(dbo, module_names):
"""Return a list of modules """Return a list of modules
:param yb: yum base object :param dbo: dnf base object
:type yb: YumBase :type dbo: dnf.Base
:param offset: Number of modules to skip :param offset: Number of modules to skip
:type limit: int :type limit: int
:param limit: Maximum number of modules to return :param limit: Maximum number of modules to return
@ -275,37 +278,27 @@ def modules_list(yb, module_names):
Modules don't exist in RHEL7 so this only returns projects Modules don't exist in RHEL7 so this only returns projects
and sets the type to "rpm" and sets the type to "rpm"
""" """
try: # TODO - Figure out what to do with this for Fedora 'modules'
ybl = yb.doPackageLists(pkgnarrow="available", patterns=module_names, showdups=False) projs = projects_info(dbo, module_names)
except YumBaseError as e: return sorted(map(proj_to_module, projs), key=lambda p: p["name"].lower())
raise ProjectsError("There was a problem listing modules: %s" % str(e))
finally:
yb.closeRpmDB()
return sorted(map(yaps_to_module, ybl.available), key=lambda p: p["name"].lower())
def modules_info(yb, module_names): def modules_info(dbo, module_names):
"""Return details about a module, including dependencies """Return details about a module, including dependencies
:param yb: yum base object :param dbo: dnf base object
:type yb: YumBase :type dbo: dnf.Base
:param module_names: Names of the modules to get info about :param module_names: Names of the modules to get info about
:type module_names: str :type module_names: str
:returns: List of dicts with module details and dependencies. :returns: List of dicts with module details and dependencies.
:rtype: list of dicts :rtype: list of dicts
""" """
try: modules = projects_info(dbo, module_names)
# Get the info about each module
ybl = yb.doPackageLists(pkgnarrow="available", patterns=module_names, showdups=False)
except YumBaseError as e:
raise ProjectsError("There was a problem with info for %s: %s" % (module_names, str(e)))
finally:
yb.closeRpmDB()
modules = sorted(map(yaps_to_project, ybl.available), key=lambda p: p["name"].lower())
# Add the dependency info to each one # Add the dependency info to each one
for module in modules: for module in modules:
module["dependencies"] = projects_depsolve(yb, [module["name"]]) module["dependencies"] = projects_depsolve(dbo, [module["name"]])
return modules return modules

View File

@ -15,23 +15,19 @@
# along with this program. If not, see <http://www.gnu.org/licenses/>. # along with this program. If not, see <http://www.gnu.org/licenses/>.
# #
import os import os
import mock
import time
import shutil import shutil
import tempfile import tempfile
import time
import unittest import unittest
from yum.Errors import YumBaseError from pylorax.api.config import configure, make_dnf_dirs
from pylorax.api.projects import api_time, api_changelog, pkg_to_project, pkg_to_project_info, pkg_to_dep
from pylorax.api.config import configure, make_yum_dirs from pylorax.api.projects import proj_to_module, projects_list, projects_info, projects_depsolve
from pylorax.api.projects import api_time, api_changelog, yaps_to_project, yaps_to_project_info
from pylorax.api.projects import tm_to_dep, yaps_to_module, projects_list, projects_info, projects_depsolve
from pylorax.api.projects import modules_list, modules_info, ProjectsError, dep_evra, dep_nevra from pylorax.api.projects import modules_list, modules_info, ProjectsError, dep_evra, dep_nevra
from pylorax.api.yumbase import get_base_object from pylorax.api.dnfbase import get_base_object
class Package(object):
class Yaps(object): """Test class for hawkey.Package tests"""
"""Test class for yaps tests"""
name = "name" name = "name"
summary = "summary" summary = "summary"
description = "description" description = "description"
@ -43,26 +39,13 @@ class Yaps(object):
license = "license" license = "license"
version = "version" version = "version"
def returnChangelog(self):
return [[0, 1, "Heavy!"]]
class TM(object):
"""Test class for tm test"""
name = "name"
epoch = 1
version = "version"
release = "release"
arch = "arch"
class ProjectsTest(unittest.TestCase): class ProjectsTest(unittest.TestCase):
@classmethod @classmethod
def setUpClass(self): def setUpClass(self):
self.tmp_dir = tempfile.mkdtemp(prefix="lorax.test.repo.") self.tmp_dir = tempfile.mkdtemp(prefix="lorax.test.repo.")
self.config = configure(root_dir=self.tmp_dir, test_config=True) self.config = configure(root_dir=self.tmp_dir, test_config=True)
make_yum_dirs(self.config) make_dnf_dirs(self.config)
self.yb = get_base_object(self.config) self.dbo = get_base_object(self.config)
os.environ["TZ"] = "UTC" os.environ["TZ"] = "UTC"
time.tzset() time.tzset()
@ -82,22 +65,22 @@ class ProjectsTest(unittest.TestCase):
def test_api_changelog_missing_text_entry(self): def test_api_changelog_missing_text_entry(self):
self.assertEqual(api_changelog([('now', 'atodorov')]), '') self.assertEqual(api_changelog([('now', 'atodorov')]), '')
def test_yaps_to_project(self): def test_pkg_to_project(self):
result = {"name":"name", result = {"name":"name",
"summary":"summary", "summary":"summary",
"description":"description", "description":"description",
"homepage":"url", "homepage":"url",
"upstream_vcs":"UPSTREAM_VCS"} "upstream_vcs":"UPSTREAM_VCS"}
y = Yaps() pkg = Package()
self.assertEqual(yaps_to_project(y), result) self.assertEqual(pkg_to_project(pkg), result)
def test_yaps_to_project_info(self): def test_pkg_to_project_info(self):
build = {"epoch":1, build = {"epoch":1,
"release":"release", "release":"release",
"arch":"arch", "arch":"arch",
"build_time":"1985-10-27T01:00:00", "build_time":"1985-10-27T01:00:00",
"changelog":"Heavy!", "changelog":"CHANGELOG_NEEDED",
"build_config_ref": "BUILD_CONFIG_REF", "build_config_ref": "BUILD_CONFIG_REF",
"build_env_ref": "BUILD_ENV_REF", "build_env_ref": "BUILD_ENV_REF",
"metadata": {}, "metadata": {},
@ -113,25 +96,25 @@ class ProjectsTest(unittest.TestCase):
"upstream_vcs":"UPSTREAM_VCS", "upstream_vcs":"UPSTREAM_VCS",
"builds": [build]} "builds": [build]}
y = Yaps() pkg = Package()
self.assertEqual(yaps_to_project_info(y), result) self.assertEqual(pkg_to_project_info(pkg), result)
def test_tm_to_dep(self): def test_pkg_to_dep(self):
result = {"name":"name", result = {"name":"name",
"epoch":1, "epoch":1,
"version":"version", "version":"version",
"release":"release", "release":"release",
"arch":"arch"} "arch":"arch"}
tm = TM() pkg = Package()
self.assertEqual(tm_to_dep(tm), result) self.assertEqual(pkg_to_dep(pkg), result)
def test_yaps_to_module(self): def test_proj_to_module(self):
result = {"name":"name", result = {"name":"name",
"group_type":"rpm"} "group_type":"rpm"}
y = Yaps() proj = pkg_to_project(Package())
self.assertEqual(yaps_to_module(y), result) self.assertEqual(proj_to_module(proj), result)
def test_dep_evra(self): def test_dep_evra(self):
dep = {"arch": "noarch", dep = {"arch": "noarch",
@ -158,60 +141,41 @@ class ProjectsTest(unittest.TestCase):
self.assertEqual(dep_nevra(dep), "basesystem-10.0-7.el7.noarch") self.assertEqual(dep_nevra(dep), "basesystem-10.0-7.el7.noarch")
def test_projects_list(self): def test_projects_list(self):
projects = projects_list(self.yb) projects = projects_list(self.dbo)
self.assertEqual(len(projects) > 10, True) self.assertEqual(len(projects) > 10, True)
def test_projects_list_yum_raises_exception(self):
with self.assertRaises(ProjectsError):
with mock.patch.object(self.yb, 'doPackageLists', side_effect=YumBaseError('TESTING')):
projects_list(self.yb)
def test_projects_info(self): def test_projects_info(self):
projects = projects_info(self.yb, ["bash"]) projects = projects_info(self.dbo, ["bash"])
self.assertEqual(projects[0]["name"], "bash") self.assertEqual(projects[0]["name"], "bash")
self.assertEqual(projects[0]["builds"][0]["source"]["license"], "GPLv3+") self.assertEqual(projects[0]["builds"][0]["source"]["license"], "GPLv3+")
def test_projects_info_yum_raises_exception(self):
with self.assertRaises(ProjectsError):
with mock.patch.object(self.yb, 'doPackageLists', side_effect=YumBaseError('TESTING')):
projects_info(self.yb, ["bash"])
def test_projects_depsolve(self): def test_projects_depsolve(self):
deps = projects_depsolve(self.yb, ["bash"]) deps = projects_depsolve(self.dbo, ["bash"])
self.assertEqual(deps[0]["name"], "basesystem") self.assertEqual(deps[0]["name"], "basesystem")
def test_projects_depsolve_fail(self): def test_projects_depsolve_fail(self):
with self.assertRaises(ProjectsError): with self.assertRaises(ProjectsError):
projects_depsolve(self.yb, ["nada-package"]) projects_depsolve(self.dbo, ["nada-package"])
def test_modules_list(self): def test_modules_list_all(self):
modules = modules_list(self.yb, None) modules = modules_list(self.dbo, None)
self.assertEqual(len(modules) > 10, True) self.assertEqual(len(modules) > 10, True)
self.assertEqual(modules[0]["group_type"], "rpm") self.assertEqual(modules[0]["group_type"], "rpm")
modules = modules_list(self.yb, ["g*"]) def test_modules_list_glob(self):
modules = modules_list(self.dbo, ["g*"])
self.assertEqual(modules[0]["name"].startswith("g"), True) self.assertEqual(modules[0]["name"].startswith("g"), True)
def test_modules_list_yum_raises_exception(self):
with self.assertRaises(ProjectsError):
with mock.patch.object(self.yb, 'doPackageLists', side_effect=YumBaseError('TESTING')):
modules_list(self.yb, None)
def test_modules_info(self): def test_modules_info(self):
modules = modules_info(self.yb, ["bash"]) modules = modules_info(self.dbo, ["bash"])
print(modules) print(modules)
self.assertEqual(modules[0]["name"], "bash") self.assertEqual(modules[0]["name"], "bash")
self.assertEqual(modules[0]["dependencies"][0]["name"], "basesystem") self.assertEqual(modules[0]["dependencies"][0]["name"], "basesystem")
def test_modules_info_yum_raises_exception(self):
with self.assertRaises(ProjectsError):
with mock.patch.object(self.yb, 'doPackageLists', side_effect=YumBaseError('TESTING')):
modules_info(self.yb, ["bash"])
class ConfigureTest(unittest.TestCase): class ConfigureTest(unittest.TestCase):
@classmethod @classmethod