Add /recipes/freeze route and tests.
This commit is contained in:
		
							parent
							
								
									b8212358bc
								
							
						
					
					
						commit
						6b4bf92e7f
					
				| @ -133,6 +133,19 @@ def yaps_to_module(yaps): | |||||||
|             "group_type": "rpm"} |             "group_type": "rpm"} | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | def dep_evra(dep): | ||||||
|  |     """Return the epoch:version-release.arch for the dep | ||||||
|  | 
 | ||||||
|  |     :param dep: dependency dict | ||||||
|  |     :type dep: dict | ||||||
|  |     :returns: epoch:version-release.arch | ||||||
|  |     :rtype: str | ||||||
|  |     """ | ||||||
|  |     if dep["epoch"] == "0": | ||||||
|  |         return dep["version"]+"-"+dep["release"]+"."+dep["arch"] | ||||||
|  |     else: | ||||||
|  |         return dep["epoch"]+":"+dep["version"]+"-"+dep["release"]+"."+dep["arch"] | ||||||
|  | 
 | ||||||
| def projects_list(yb): | def projects_list(yb): | ||||||
|     """Return a list of projects |     """Return a list of projects | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -24,11 +24,11 @@ from pykickstart.parser import KickstartParser | |||||||
| from pykickstart.version import makeVersion, RHEL7 | from pykickstart.version import makeVersion, RHEL7 | ||||||
| 
 | 
 | ||||||
| from pylorax.api.crossdomain import crossdomain | from pylorax.api.crossdomain import crossdomain | ||||||
| from pylorax.api.projects import projects_list, projects_info, projects_depsolve | from pylorax.api.projects import projects_list, projects_info, projects_depsolve, dep_evra | ||||||
| from pylorax.api.projects import modules_list, modules_info, ProjectsError | from pylorax.api.projects import modules_list, modules_info, ProjectsError | ||||||
| from pylorax.api.recipes import list_branch_files, read_recipe_commit, recipe_filename, list_commits | from pylorax.api.recipes import list_branch_files, read_recipe_commit, recipe_filename, list_commits | ||||||
| from pylorax.api.recipes import recipe_from_dict, recipe_from_toml, commit_recipe, delete_recipe, revert_recipe | from pylorax.api.recipes import recipe_from_dict, recipe_from_toml, commit_recipe, delete_recipe, revert_recipe | ||||||
| from pylorax.api.recipes import tag_recipe_commit, recipe_diff | from pylorax.api.recipes import tag_recipe_commit, recipe_diff, Recipe, RecipePackage, RecipeModule | ||||||
| from pylorax.api.workspace import workspace_read, workspace_write, workspace_delete | from pylorax.api.workspace import workspace_read, workspace_write, workspace_delete | ||||||
| from pylorax.creator import DRACUT_DEFAULT, mount_boot_part_over_root | from pylorax.creator import DRACUT_DEFAULT, mount_boot_part_over_root | ||||||
| from pylorax.creator import make_appliance, make_image, make_livecd, make_live_images | from pylorax.creator import make_appliance, make_image, make_livecd, make_live_images | ||||||
| @ -282,6 +282,66 @@ def v0_api(api): | |||||||
|         diff = recipe_diff(old_recipe, new_recipe) |         diff = recipe_diff(old_recipe, new_recipe) | ||||||
|         return jsonify(diff=diff) |         return jsonify(diff=diff) | ||||||
| 
 | 
 | ||||||
|  |     @api.route("/api/v0/recipes/freeze/<recipe_names>") | ||||||
|  |     @crossdomain(origin="*") | ||||||
|  |     def v0_recipes_freeze(recipe_names): | ||||||
|  |         """Return the recipe with the exact modules and packages selected by depsolve""" | ||||||
|  |         recipes = [] | ||||||
|  |         errors = [] | ||||||
|  |         for recipe_name in [n.strip() for n in sorted(recipe_names.split(","), key=lambda n: n.lower())]: | ||||||
|  |             # get the recipe | ||||||
|  |             # Get the workspace version (if it exists) | ||||||
|  |             recipe = None | ||||||
|  |             try: | ||||||
|  |                 with api.config["GITLOCK"].lock: | ||||||
|  |                     recipe = workspace_read(api.config["GITLOCK"].repo, "master", recipe_name) | ||||||
|  |             except Exception: | ||||||
|  |                 pass | ||||||
|  | 
 | ||||||
|  |             if not recipe: | ||||||
|  |                 # No workspace version, get the git version (if it exists) | ||||||
|  |                 try: | ||||||
|  |                     with api.config["GITLOCK"].lock: | ||||||
|  |                         recipe = read_recipe_commit(api.config["GITLOCK"].repo, "master", recipe_name) | ||||||
|  |                 except Exception as e: | ||||||
|  |                     errors.append({"recipe":recipe_name, "msg":str(e)}) | ||||||
|  |                     log.error("(v0_recipes_depsolve) %s", str(e)) | ||||||
|  | 
 | ||||||
|  |             # No recipe found, skip it. | ||||||
|  |             if not recipe: | ||||||
|  |                 errors.append({"recipe":recipe_name, "msg":"Recipe not found"}) | ||||||
|  |                 continue | ||||||
|  | 
 | ||||||
|  |             # Combine modules and packages and depsolve the list | ||||||
|  |             # TODO include the version/glob in the depsolving | ||||||
|  |             module_names = map(lambda m: m["name"], recipe["modules"] or []) | ||||||
|  |             package_names = map(lambda p: p["name"], recipe["packages"] or []) | ||||||
|  |             projects = sorted(set(module_names+package_names), key=lambda n: n.lower()) | ||||||
|  |             deps = [] | ||||||
|  |             try: | ||||||
|  |                 with api.config["YUMLOCK"].lock: | ||||||
|  |                     deps = projects_depsolve(api.config["YUMLOCK"].yb, projects) | ||||||
|  |             except ProjectsError as e: | ||||||
|  |                 errors.append({"recipe":recipe_name, "msg":str(e)}) | ||||||
|  |                 log.error("(v0_recipes_depsolve) %s", str(e)) | ||||||
|  | 
 | ||||||
|  |             # Change the recipe's modules and packages to use the depsolved version | ||||||
|  |             new_modules = [] | ||||||
|  |             new_packages = [] | ||||||
|  |             for dep in deps: | ||||||
|  |                 if dep["name"] in package_names: | ||||||
|  |                     new_packages.append(RecipePackage(dep["name"], dep_evra(dep))) | ||||||
|  |                 elif dep["name"] in module_names: | ||||||
|  |                     new_modules.append(RecipeModule(dep["name"], dep_evra(dep))) | ||||||
|  | 
 | ||||||
|  |             recipes.append({"recipe":Recipe(recipe["name"], | ||||||
|  |                                             recipe["description"], | ||||||
|  |                                             recipe["version"], | ||||||
|  |                                             new_modules, | ||||||
|  |                                             new_packages)}) | ||||||
|  | 
 | ||||||
|  |         return jsonify(recipes=recipes, errors=errors) | ||||||
|  | 
 | ||||||
|     @api.route("/api/v0/recipes/depsolve/<recipe_names>") |     @api.route("/api/v0/recipes/depsolve/<recipe_names>") | ||||||
|     @crossdomain(origin="*") |     @crossdomain(origin="*") | ||||||
|     def v0_recipes_depsolve(recipe_names): |     def v0_recipes_depsolve(recipe_names): | ||||||
|  | |||||||
| @ -21,7 +21,7 @@ import unittest | |||||||
| from pylorax.api.config import configure | from pylorax.api.config import configure | ||||||
| from pylorax.api.projects import api_time, api_changelog, yaps_to_project, yaps_to_project_info | 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 tm_to_dep, yaps_to_module, projects_list, projects_info, projects_depsolve | ||||||
| from pylorax.api.projects import modules_list, modules_info, ProjectsError | from pylorax.api.projects import modules_list, modules_info, ProjectsError, dep_evra | ||||||
| from pylorax.api.yumbase import get_base_object | from pylorax.api.yumbase import get_base_object | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| @ -119,6 +119,14 @@ class ProjectsTest(unittest.TestCase): | |||||||
|         y = Yaps() |         y = Yaps() | ||||||
|         self.assertEqual(yaps_to_module(y), result) |         self.assertEqual(yaps_to_module(y), result) | ||||||
| 
 | 
 | ||||||
|  |     def test_dep_evra(self): | ||||||
|  |         dep = {"arch": "noarch", | ||||||
|  |                "epoch": "0", | ||||||
|  |                "name": "basesystem", | ||||||
|  |                "release": "7.el7", | ||||||
|  |                "version": "10.0"} | ||||||
|  |         self.assertEqual(dep_evra(dep), "10.0-7.el7.noarch") | ||||||
|  | 
 | ||||||
|     def test_projects_list(self): |     def test_projects_list(self): | ||||||
|         projects = projects_list(self.yb) |         projects = projects_list(self.yb) | ||||||
|         self.assertEqual(len(projects) > 10, True) |         self.assertEqual(len(projects) > 10, True) | ||||||
|  | |||||||
| @ -370,6 +370,18 @@ class ServerTestCase(unittest.TestCase): | |||||||
|         self.assertEqual(recipes[0]["recipe"]["name"], "glusterfs") |         self.assertEqual(recipes[0]["recipe"]["name"], "glusterfs") | ||||||
|         self.assertEqual(len(recipes[0]["dependencies"]) > 10, True) |         self.assertEqual(len(recipes[0]["dependencies"]) > 10, True) | ||||||
| 
 | 
 | ||||||
|  |     def test_15_recipes_freeze(self): | ||||||
|  |         """Test /api/v0/recipes/freeze/<recipe_names>""" | ||||||
|  |         resp = self.server.get("/api/v0/recipes/freeze/glusterfs") | ||||||
|  |         data = json.loads(resp.data) | ||||||
|  |         self.assertNotEqual(data, None) | ||||||
|  |         recipes = data.get("recipes") | ||||||
|  |         self.assertNotEqual(recipes, None) | ||||||
|  |         self.assertEqual(len(recipes), 1) | ||||||
|  |         self.assertEqual(recipes[0]["recipe"]["name"], "glusterfs") | ||||||
|  |         evra = recipes[0]["recipe"]["modules"][0]["version"] | ||||||
|  |         self.assertEqual(len(evra) > 10, True) | ||||||
|  | 
 | ||||||
|     def test_projects_list(self): |     def test_projects_list(self): | ||||||
|         """Test /api/v0/projects/list""" |         """Test /api/v0/projects/list""" | ||||||
|         resp = self.server.get("/api/v0/projects/list") |         resp = self.server.get("/api/v0/projects/list") | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user