2017-11-07 20:00:35 +00:00
|
|
|
#
|
|
|
|
# Copyright (C) 2017 Red Hat, Inc.
|
|
|
|
#
|
|
|
|
# This program is free software; you can redistribute it and/or modify
|
|
|
|
# it under the terms of the GNU General Public License as published by
|
|
|
|
# the Free Software Foundation; either version 2 of the License, or
|
|
|
|
# (at your option) any later version.
|
|
|
|
#
|
|
|
|
# This program is distributed in the hope that it will be useful,
|
|
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
# GNU General Public License for more details.
|
|
|
|
#
|
|
|
|
# You should have received a copy of the GNU General Public License
|
|
|
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
#
|
2018-02-12 14:56:24 +00:00
|
|
|
import os
|
2018-02-10 01:39:42 +00:00
|
|
|
from glob import glob
|
2017-11-13 18:32:52 +00:00
|
|
|
import shutil
|
2017-11-08 01:09:56 +00:00
|
|
|
import tempfile
|
2018-02-10 01:39:42 +00:00
|
|
|
import time
|
2017-11-08 01:09:56 +00:00
|
|
|
from threading import Lock
|
2017-11-07 20:00:35 +00:00
|
|
|
import unittest
|
|
|
|
|
|
|
|
from flask import json
|
2017-11-10 16:25:57 +00:00
|
|
|
import pytoml as toml
|
2018-02-10 01:39:42 +00:00
|
|
|
from pylorax.api.config import configure, make_yum_dirs, make_queue_dirs
|
|
|
|
from pylorax.api.queue import start_queue_monitor
|
2017-11-08 01:09:56 +00:00
|
|
|
from pylorax.api.recipes import open_or_create_repo, commit_recipe_directory
|
2017-11-22 18:29:33 +00:00
|
|
|
from pylorax.api.server import server, GitLock, YumLock
|
|
|
|
from pylorax.api.yumbase import get_base_object
|
2017-11-10 16:25:57 +00:00
|
|
|
from pylorax.sysutils import joinpaths
|
2017-11-07 20:00:35 +00:00
|
|
|
|
|
|
|
class ServerTestCase(unittest.TestCase):
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def setUpClass(self):
|
2017-11-08 01:09:56 +00:00
|
|
|
repo_dir = tempfile.mkdtemp(prefix="lorax.test.repo.")
|
|
|
|
server.config["REPO_DIR"] = repo_dir
|
|
|
|
repo = open_or_create_repo(server.config["REPO_DIR"])
|
|
|
|
server.config["GITLOCK"] = GitLock(repo=repo, lock=Lock(), dir=repo_dir)
|
|
|
|
|
2017-11-22 18:29:33 +00:00
|
|
|
server.config["COMPOSER_CFG"] = configure(root_dir=repo_dir, test_config=True)
|
2018-02-10 01:39:42 +00:00
|
|
|
os.makedirs(joinpaths(server.config["COMPOSER_CFG"].get("composer", "share_dir"), "composer"))
|
|
|
|
errors = make_queue_dirs(server.config["COMPOSER_CFG"], 0)
|
|
|
|
if errors:
|
|
|
|
raise RuntimeError("\n".join(errors))
|
|
|
|
|
2018-02-05 18:40:39 +00:00
|
|
|
make_yum_dirs(server.config["COMPOSER_CFG"])
|
2017-11-22 18:29:33 +00:00
|
|
|
yb = get_base_object(server.config["COMPOSER_CFG"])
|
|
|
|
server.config["YUMLOCK"] = YumLock(yb=yb, lock=Lock())
|
|
|
|
|
2017-11-07 20:00:35 +00:00
|
|
|
server.config['TESTING'] = True
|
|
|
|
self.server = server.test_client()
|
|
|
|
|
2017-11-10 16:25:57 +00:00
|
|
|
self.examples_path = "./tests/pylorax/recipes/"
|
|
|
|
|
2018-02-10 01:39:42 +00:00
|
|
|
# Copy the shared files over to the directory tree we are using
|
|
|
|
share_path = "./share/composer/"
|
|
|
|
for f in glob(joinpaths(share_path, "*")):
|
|
|
|
shutil.copy(f, joinpaths(server.config["COMPOSER_CFG"].get("composer", "share_dir"), "composer"))
|
|
|
|
|
2017-11-08 01:09:56 +00:00
|
|
|
# Import the example recipes
|
2017-11-10 16:25:57 +00:00
|
|
|
commit_recipe_directory(server.config["GITLOCK"].repo, "master", self.examples_path)
|
2017-11-08 01:09:56 +00:00
|
|
|
|
2018-02-10 01:39:42 +00:00
|
|
|
start_queue_monitor(server.config["COMPOSER_CFG"], 0, 0)
|
|
|
|
|
2017-11-07 20:00:35 +00:00
|
|
|
@classmethod
|
|
|
|
def tearDownClass(self):
|
2017-11-13 18:32:52 +00:00
|
|
|
shutil.rmtree(server.config["REPO_DIR"])
|
2017-11-07 20:00:35 +00:00
|
|
|
|
2018-02-12 14:56:24 +00:00
|
|
|
def test_00_hello(self):
|
|
|
|
"""Test /"""
|
|
|
|
resp = self.server.get("/")
|
|
|
|
self.assertEqual(resp.data, 'Hello, World!')
|
|
|
|
|
2017-11-13 20:42:59 +00:00
|
|
|
def test_01_status(self):
|
2017-11-07 20:00:35 +00:00
|
|
|
"""Test the /api/v0/status route"""
|
|
|
|
status_dict = {"build":"devel", "api":"0", "db_version":"0", "schema_version":"0", "db_supported":False}
|
|
|
|
resp = self.server.get("/api/v0/status")
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertEqual(data, status_dict)
|
2017-11-08 01:09:56 +00:00
|
|
|
|
2017-11-13 20:42:59 +00:00
|
|
|
def test_02_recipes_list(self):
|
2017-11-08 01:09:56 +00:00
|
|
|
"""Test the /api/v0/recipes/list route"""
|
|
|
|
list_dict = {"recipes":["atlas", "development", "glusterfs", "http-server", "jboss", "kubernetes"],
|
|
|
|
"limit":20, "offset":0, "total":6}
|
|
|
|
resp = self.server.get("/api/v0/recipes/list")
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertEqual(data, list_dict)
|
2017-11-09 01:38:12 +00:00
|
|
|
|
2017-11-13 20:42:59 +00:00
|
|
|
def test_03_recipes_info(self):
|
2017-11-09 01:38:12 +00:00
|
|
|
"""Test the /api/v0/recipes/info route"""
|
|
|
|
info_dict_1 = {"changes":[{"changed":False, "name":"http-server"}],
|
|
|
|
"errors":[],
|
|
|
|
"recipes":[{"description":"An example http server with PHP and MySQL support.",
|
|
|
|
"modules":[{"name":"httpd", "version":"2.4.*"},
|
|
|
|
{"name":"mod_auth_kerb", "version":"5.4"},
|
|
|
|
{"name":"mod_ssl", "version":"2.4.*"},
|
|
|
|
{"name":"php", "version":"5.4.*"},
|
|
|
|
{"name": "php-mysql", "version":"5.4.*"}],
|
|
|
|
"name":"http-server",
|
|
|
|
"packages": [{"name":"openssh-server", "version": "6.6.*"},
|
|
|
|
{"name": "rsync", "version": "3.0.*"},
|
|
|
|
{"name": "tmux", "version": "2.2"}],
|
2017-11-10 22:45:13 +00:00
|
|
|
"version": "0.0.1"}]}
|
2017-11-09 01:38:12 +00:00
|
|
|
resp = self.server.get("/api/v0/recipes/info/http-server")
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertEqual(data, info_dict_1)
|
|
|
|
|
|
|
|
info_dict_2 = {"changes":[{"changed":False, "name":"glusterfs"},
|
|
|
|
{"changed":False, "name":"http-server"}],
|
|
|
|
"errors":[],
|
|
|
|
"recipes":[{"description": "An example GlusterFS server with samba",
|
|
|
|
"modules":[{"name":"glusterfs", "version":"3.7.*"},
|
|
|
|
{"name":"glusterfs-cli", "version":"3.7.*"}],
|
|
|
|
"name":"glusterfs",
|
|
|
|
"packages":[{"name":"samba", "version":"4.2.*"}],
|
|
|
|
"version": "0.0.1"},
|
|
|
|
{"description":"An example http server with PHP and MySQL support.",
|
|
|
|
"modules":[{"name":"httpd", "version":"2.4.*"},
|
|
|
|
{"name":"mod_auth_kerb", "version":"5.4"},
|
|
|
|
{"name":"mod_ssl", "version":"2.4.*"},
|
|
|
|
{"name":"php", "version":"5.4.*"},
|
|
|
|
{"name": "php-mysql", "version":"5.4.*"}],
|
|
|
|
"name":"http-server",
|
|
|
|
"packages": [{"name":"openssh-server", "version": "6.6.*"},
|
|
|
|
{"name": "rsync", "version": "3.0.*"},
|
|
|
|
{"name": "tmux", "version": "2.2"}],
|
2017-11-10 22:45:13 +00:00
|
|
|
"version": "0.0.1"},
|
2017-11-09 01:38:12 +00:00
|
|
|
]}
|
|
|
|
resp = self.server.get("/api/v0/recipes/info/http-server,glusterfs")
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertEqual(data, info_dict_2)
|
|
|
|
|
|
|
|
info_dict_3 = {"changes":[],
|
2018-02-05 20:20:12 +00:00
|
|
|
"errors":[{"recipe":"missing-recipe", "msg":"No commits for missing-recipe.toml on the master branch."}],
|
2017-11-09 01:38:12 +00:00
|
|
|
"recipes":[]
|
|
|
|
}
|
|
|
|
resp = self.server.get("/api/v0/recipes/info/missing-recipe")
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertEqual(data, info_dict_3)
|
|
|
|
|
2017-11-13 20:42:59 +00:00
|
|
|
def test_04_recipes_changes(self):
|
2017-11-10 02:43:42 +00:00
|
|
|
"""Test the /api/v0/recipes/changes route"""
|
|
|
|
resp = self.server.get("/api/v0/recipes/changes/http-server")
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
|
|
|
|
# Can't compare a whole dict since commit hash and timestamps will change.
|
|
|
|
# Should have 1 commit (for now), with a matching message.
|
|
|
|
self.assertEqual(data["limit"], 20)
|
|
|
|
self.assertEqual(data["offset"], 0)
|
|
|
|
self.assertEqual(len(data["errors"]), 0)
|
|
|
|
self.assertEqual(len(data["recipes"]), 1)
|
|
|
|
self.assertEqual(data["recipes"][0]["name"], "http-server")
|
|
|
|
self.assertEqual(len(data["recipes"][0]["changes"]), 1)
|
2017-11-10 16:25:57 +00:00
|
|
|
|
2017-11-13 20:42:59 +00:00
|
|
|
def test_05_recipes_new_json(self):
|
2017-11-10 16:25:57 +00:00
|
|
|
"""Test the /api/v0/recipes/new route with json recipe"""
|
|
|
|
test_recipe = {"description": "An example GlusterFS server with samba",
|
|
|
|
"name":"glusterfs",
|
|
|
|
"version": "0.2.0",
|
|
|
|
"modules":[{"name":"glusterfs", "version":"3.7.*"},
|
|
|
|
{"name":"glusterfs-cli", "version":"3.7.*"}],
|
|
|
|
"packages":[{"name":"samba", "version":"4.2.*"},
|
|
|
|
{"name":"tmux", "version":"2.2"}]}
|
|
|
|
|
|
|
|
resp = self.server.post("/api/v0/recipes/new",
|
|
|
|
data=json.dumps(test_recipe),
|
|
|
|
content_type="application/json")
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertEqual(data, {"status":True})
|
|
|
|
|
|
|
|
resp = self.server.get("/api/v0/recipes/info/glusterfs")
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertNotEqual(data, None)
|
|
|
|
recipes = data.get("recipes")
|
|
|
|
self.assertEqual(len(recipes), 1)
|
|
|
|
self.assertEqual(recipes[0], test_recipe)
|
|
|
|
|
2017-11-13 20:42:59 +00:00
|
|
|
def test_06_recipes_new_toml(self):
|
2017-11-10 16:25:57 +00:00
|
|
|
"""Test the /api/v0/recipes/new route with toml recipe"""
|
|
|
|
test_recipe = open(joinpaths(self.examples_path, "glusterfs.toml"), "rb").read()
|
|
|
|
resp = self.server.post("/api/v0/recipes/new",
|
|
|
|
data=test_recipe,
|
|
|
|
content_type="text/x-toml")
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertEqual(data, {"status":True})
|
|
|
|
|
|
|
|
resp = self.server.get("/api/v0/recipes/info/glusterfs")
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertNotEqual(data, None)
|
|
|
|
recipes = data.get("recipes")
|
|
|
|
self.assertEqual(len(recipes), 1)
|
|
|
|
|
|
|
|
# Returned recipe has had its version bumped to 0.2.1
|
|
|
|
test_recipe = toml.loads(test_recipe)
|
|
|
|
test_recipe["version"] = "0.2.1"
|
|
|
|
|
|
|
|
self.assertEqual(recipes[0], test_recipe)
|
2017-11-13 18:32:52 +00:00
|
|
|
|
2017-11-13 20:42:59 +00:00
|
|
|
def test_07_recipes_ws_json(self):
|
2017-11-13 18:32:52 +00:00
|
|
|
"""Test the /api/v0/recipes/workspace route with json recipe"""
|
|
|
|
test_recipe = {"description": "An example GlusterFS server with samba, ws version",
|
|
|
|
"name":"glusterfs",
|
|
|
|
"version": "0.3.0",
|
|
|
|
"modules":[{"name":"glusterfs", "version":"3.7.*"},
|
|
|
|
{"name":"glusterfs-cli", "version":"3.7.*"}],
|
|
|
|
"packages":[{"name":"samba", "version":"4.2.*"},
|
|
|
|
{"name":"tmux", "version":"2.2"}]}
|
|
|
|
|
|
|
|
resp = self.server.post("/api/v0/recipes/workspace",
|
|
|
|
data=json.dumps(test_recipe),
|
|
|
|
content_type="application/json")
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertEqual(data, {"status":True})
|
|
|
|
|
|
|
|
resp = self.server.get("/api/v0/recipes/info/glusterfs")
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertNotEqual(data, None)
|
|
|
|
recipes = data.get("recipes")
|
|
|
|
self.assertEqual(len(recipes), 1)
|
|
|
|
self.assertEqual(recipes[0], test_recipe)
|
|
|
|
changes = data.get("changes")
|
|
|
|
self.assertEqual(len(changes), 1)
|
|
|
|
self.assertEqual(changes[0], {"name":"glusterfs", "changed":True})
|
|
|
|
|
2017-11-13 20:42:59 +00:00
|
|
|
def test_08_recipes_ws_toml(self):
|
2017-11-13 18:32:52 +00:00
|
|
|
"""Test the /api/v0/recipes/workspace route with toml recipe"""
|
|
|
|
test_recipe = {"description": "An example GlusterFS server with samba, ws version",
|
|
|
|
"name":"glusterfs",
|
|
|
|
"version": "0.4.0",
|
|
|
|
"modules":[{"name":"glusterfs", "version":"3.7.*"},
|
|
|
|
{"name":"glusterfs-cli", "version":"3.7.*"}],
|
|
|
|
"packages":[{"name":"samba", "version":"4.2.*"},
|
|
|
|
{"name":"tmux", "version":"2.2"}]}
|
|
|
|
|
|
|
|
resp = self.server.post("/api/v0/recipes/workspace",
|
|
|
|
data=json.dumps(test_recipe),
|
|
|
|
content_type="application/json")
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertEqual(data, {"status":True})
|
|
|
|
|
|
|
|
resp = self.server.get("/api/v0/recipes/info/glusterfs")
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertNotEqual(data, None)
|
|
|
|
recipes = data.get("recipes")
|
|
|
|
self.assertEqual(len(recipes), 1)
|
|
|
|
self.assertEqual(recipes[0], test_recipe)
|
|
|
|
changes = data.get("changes")
|
|
|
|
self.assertEqual(len(changes), 1)
|
|
|
|
self.assertEqual(changes[0], {"name":"glusterfs", "changed":True})
|
2017-11-13 19:21:14 +00:00
|
|
|
|
2017-11-13 20:42:59 +00:00
|
|
|
def test_09_recipes_ws_delete(self):
|
2017-11-13 19:21:14 +00:00
|
|
|
"""Test DELETE /api/v0/recipes/workspace/<recipe_name>"""
|
|
|
|
# Write to the workspace first, just use the test_recipes_ws_json test for this
|
2017-11-13 20:42:59 +00:00
|
|
|
self.test_07_recipes_ws_json()
|
2017-11-13 19:21:14 +00:00
|
|
|
|
|
|
|
# Delete it
|
|
|
|
resp = self.server.delete("/api/v0/recipes/workspace/glusterfs")
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertEqual(data, {"status":True})
|
|
|
|
|
|
|
|
# Make sure it isn't the workspace copy and that changed is False
|
|
|
|
resp = self.server.get("/api/v0/recipes/info/glusterfs")
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertNotEqual(data, None)
|
|
|
|
recipes = data.get("recipes")
|
|
|
|
self.assertEqual(len(recipes), 1)
|
|
|
|
self.assertEqual(recipes[0]["version"], "0.2.1")
|
|
|
|
changes = data.get("changes")
|
|
|
|
self.assertEqual(len(changes), 1)
|
|
|
|
self.assertEqual(changes[0], {"name":"glusterfs", "changed":False})
|
2017-11-13 20:42:59 +00:00
|
|
|
|
|
|
|
def test_10_recipes_delete(self):
|
|
|
|
"""Test DELETE /api/v0/recipes/delete/<recipe_name>"""
|
|
|
|
resp = self.server.delete("/api/v0/recipes/delete/glusterfs")
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertEqual(data, {"status":True})
|
|
|
|
|
|
|
|
# Make sure glusterfs is no longer in the list of recipes
|
|
|
|
resp = self.server.get("/api/v0/recipes/list")
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertNotEqual(data, None)
|
|
|
|
recipes = data.get("recipes")
|
|
|
|
self.assertEqual("glusterfs" in recipes, False)
|
2017-11-13 23:44:46 +00:00
|
|
|
|
|
|
|
def test_11_recipes_undo(self):
|
|
|
|
"""Test POST /api/v0/recipes/undo/<recipe_name/<commit>"""
|
|
|
|
resp = self.server.get("/api/v0/recipes/changes/glusterfs")
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertNotEqual(data, None)
|
|
|
|
|
|
|
|
# Revert it to the first commit
|
|
|
|
recipes = data.get("recipes")
|
|
|
|
self.assertNotEqual(recipes, None)
|
|
|
|
changes = recipes[0].get("changes")
|
|
|
|
self.assertEqual(len(changes) > 1, True)
|
|
|
|
|
|
|
|
# Revert it to the first commit
|
|
|
|
commit = changes[-1]["commit"]
|
|
|
|
resp = self.server.post("/api/v0/recipes/undo/glusterfs/%s" % commit)
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertEqual(data, {"status":True})
|
|
|
|
|
|
|
|
resp = self.server.get("/api/v0/recipes/changes/glusterfs")
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertNotEqual(data, None)
|
|
|
|
|
|
|
|
recipes = data.get("recipes")
|
|
|
|
self.assertNotEqual(recipes, None)
|
|
|
|
changes = recipes[0].get("changes")
|
|
|
|
self.assertEqual(len(changes) > 1, True)
|
|
|
|
|
|
|
|
expected_msg = "Recipe glusterfs.toml reverted to commit %s" % commit
|
|
|
|
self.assertEqual(changes[0]["message"], expected_msg)
|
2017-11-14 00:34:23 +00:00
|
|
|
|
|
|
|
def test_12_recipes_tag(self):
|
|
|
|
"""Test POST /api/v0/recipes/tag/<recipe_name>"""
|
|
|
|
resp = self.server.post("/api/v0/recipes/tag/glusterfs")
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertEqual(data, {"status":True})
|
|
|
|
|
|
|
|
resp = self.server.get("/api/v0/recipes/changes/glusterfs")
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertNotEqual(data, None)
|
|
|
|
|
|
|
|
# Revert it to the first commit
|
|
|
|
recipes = data.get("recipes")
|
|
|
|
self.assertNotEqual(recipes, None)
|
|
|
|
changes = recipes[0].get("changes")
|
|
|
|
self.assertEqual(len(changes) > 1, True)
|
|
|
|
self.assertEqual(changes[0]["revision"], 1)
|
2017-11-15 01:01:35 +00:00
|
|
|
|
|
|
|
def test_13_recipes_diff(self):
|
|
|
|
"""Test /api/v0/recipes/diff/<recipe_name>/<from_commit>/<to_commit>"""
|
|
|
|
resp = self.server.get("/api/v0/recipes/changes/glusterfs")
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertNotEqual(data, None)
|
|
|
|
recipes = data.get("recipes")
|
|
|
|
self.assertNotEqual(recipes, None)
|
|
|
|
changes = recipes[0].get("changes")
|
|
|
|
self.assertEqual(len(changes) >= 2, True)
|
|
|
|
|
|
|
|
from_commit = changes[1].get("commit")
|
|
|
|
self.assertNotEqual(from_commit, None)
|
|
|
|
to_commit = changes[0].get("commit")
|
|
|
|
self.assertNotEqual(to_commit, None)
|
|
|
|
|
|
|
|
# Get the differences between the two commits
|
|
|
|
resp = self.server.get("/api/v0/recipes/diff/glusterfs/%s/%s" % (from_commit, to_commit))
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertNotEqual(data, None)
|
|
|
|
self.assertEqual(data, {"diff": [{"new": {"Version": "0.0.1"}, "old": {"Version": "0.2.1"}}]})
|
|
|
|
|
|
|
|
# Write to the workspace and check the diff
|
|
|
|
test_recipe = {"description": "An example GlusterFS server with samba, ws version",
|
|
|
|
"name":"glusterfs",
|
|
|
|
"version": "0.3.0",
|
|
|
|
"modules":[{"name":"glusterfs", "version":"3.7.*"},
|
|
|
|
{"name":"glusterfs-cli", "version":"3.7.*"}],
|
|
|
|
"packages":[{"name":"samba", "version":"4.2.*"},
|
|
|
|
{"name":"tmux", "version":"2.2"}]}
|
|
|
|
|
|
|
|
resp = self.server.post("/api/v0/recipes/workspace",
|
|
|
|
data=json.dumps(test_recipe),
|
|
|
|
content_type="application/json")
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertEqual(data, {"status":True})
|
|
|
|
|
|
|
|
# Get the differences between the newest commit and the workspace
|
|
|
|
resp = self.server.get("/api/v0/recipes/diff/glusterfs/NEWEST/WORKSPACE")
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertNotEqual(data, None)
|
|
|
|
result = {"diff": [{"new": {"Description": "An example GlusterFS server with samba, ws version"},
|
|
|
|
"old": {"Description": "An example GlusterFS server with samba"}},
|
|
|
|
{"new": {"Version": "0.3.0"},
|
|
|
|
"old": {"Version": "0.0.1"}},
|
|
|
|
{"new": {"Package": {"version": "2.2", "name": "tmux"}},
|
|
|
|
"old": None}]}
|
|
|
|
self.assertEqual(data, result)
|
2017-11-22 18:29:33 +00:00
|
|
|
|
2017-11-22 21:50:02 +00:00
|
|
|
def test_14_recipes_depsolve(self):
|
|
|
|
"""Test /api/v0/recipes/depsolve/<recipe_names>"""
|
|
|
|
resp = self.server.get("/api/v0/recipes/depsolve/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")
|
|
|
|
self.assertEqual(len(recipes[0]["dependencies"]) > 10, True)
|
2017-12-12 10:38:56 +00:00
|
|
|
self.assertFalse(data.get("errors"))
|
|
|
|
|
|
|
|
def test_14_recipes_depsolve_empty(self):
|
|
|
|
"""Test /api/v0/recipes/depsolve/<recipe_names> on empty recipe"""
|
|
|
|
test_recipe = {"description": "An empty recipe",
|
|
|
|
"name":"void",
|
|
|
|
"version": "0.1.0"}
|
|
|
|
resp = self.server.post("/api/v0/recipes/new",
|
|
|
|
data=json.dumps(test_recipe),
|
|
|
|
content_type="application/json")
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertEqual(data, {"status":True})
|
|
|
|
|
|
|
|
resp = self.server.get("/api/v0/recipes/depsolve/void")
|
|
|
|
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"], "void")
|
|
|
|
self.assertEqual(recipes[0]["recipe"]["packages"], [])
|
|
|
|
self.assertEqual(recipes[0]["recipe"]["modules"], [])
|
|
|
|
self.assertEqual(recipes[0]["dependencies"], [])
|
|
|
|
self.assertFalse(data.get("errors"))
|
2017-11-22 21:50:02 +00:00
|
|
|
|
2017-11-22 22:46:43 +00:00
|
|
|
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)
|
|
|
|
|
2017-11-22 18:29:33 +00:00
|
|
|
def test_projects_list(self):
|
|
|
|
"""Test /api/v0/projects/list"""
|
|
|
|
resp = self.server.get("/api/v0/projects/list")
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertNotEqual(data, None)
|
|
|
|
projects = data.get("projects")
|
|
|
|
self.assertEqual(len(projects) > 10, True)
|
|
|
|
|
|
|
|
def test_projects_info(self):
|
|
|
|
"""Test /api/v0/projects/info/<project_names>"""
|
|
|
|
resp = self.server.get("/api/v0/projects/info/bash")
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertNotEqual(data, None)
|
|
|
|
projects = data.get("projects")
|
|
|
|
self.assertEqual(len(projects), 1)
|
|
|
|
self.assertEqual(projects[0]["name"], "bash")
|
|
|
|
self.assertEqual(projects[0]["builds"][0]["source"]["license"], "GPLv3+")
|
|
|
|
|
|
|
|
def test_projects_depsolve(self):
|
|
|
|
"""Test /api/v0/projects/depsolve/<project_names>"""
|
|
|
|
resp = self.server.get("/api/v0/projects/depsolve/bash")
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertNotEqual(data, None)
|
|
|
|
deps = data.get("projects")
|
|
|
|
self.assertEqual(len(deps) > 10, True)
|
|
|
|
self.assertEqual(deps[0]["name"], "basesystem")
|
|
|
|
|
|
|
|
def test_modules_list(self):
|
|
|
|
"""Test /api/v0/modules/list"""
|
|
|
|
resp = self.server.get("/api/v0/modules/list")
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertNotEqual(data, None)
|
|
|
|
modules = data.get("modules")
|
|
|
|
self.assertEqual(len(modules) > 10, True)
|
|
|
|
self.assertEqual(modules[0]["group_type"], "rpm")
|
|
|
|
|
2017-11-22 23:31:21 +00:00
|
|
|
resp = self.server.get("/api/v0/modules/list/d*")
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertNotEqual(data, None)
|
|
|
|
modules = data.get("modules")
|
|
|
|
self.assertEqual(len(modules) > 0, True)
|
|
|
|
self.assertEqual(modules[0]["name"].startswith("d"), True)
|
|
|
|
|
2017-11-22 18:29:33 +00:00
|
|
|
def test_modules_info(self):
|
|
|
|
"""Test /api/v0/modules/info"""
|
|
|
|
resp = self.server.get("/api/v0/modules/info/bash")
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertNotEqual(data, None)
|
|
|
|
modules = data.get("modules")
|
|
|
|
self.assertEqual(len(modules), 1)
|
|
|
|
self.assertEqual(modules[0]["name"], "bash")
|
|
|
|
self.assertEqual(modules[0]["dependencies"][0]["name"], "basesystem")
|
2017-12-21 17:44:59 +00:00
|
|
|
|
|
|
|
def test_recipe_new_branch(self):
|
|
|
|
"""Test the /api/v0/recipes/new route with a new branch"""
|
|
|
|
test_recipe = {"description": "An example GlusterFS server with samba",
|
|
|
|
"name":"glusterfs",
|
|
|
|
"version": "0.2.0",
|
|
|
|
"modules":[{"name":"glusterfs", "version":"3.7.*"},
|
|
|
|
{"name":"glusterfs-cli", "version":"3.7.*"}],
|
|
|
|
"packages":[{"name":"samba", "version":"4.2.*"},
|
|
|
|
{"name":"tmux", "version":"2.2"}]}
|
|
|
|
|
|
|
|
resp = self.server.post("/api/v0/recipes/new?branch=test",
|
|
|
|
data=json.dumps(test_recipe),
|
|
|
|
content_type="application/json")
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertEqual(data, {"status":True})
|
|
|
|
|
|
|
|
resp = self.server.get("/api/v0/recipes/info/glusterfs?branch=test")
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertNotEqual(data, None)
|
|
|
|
recipes = data.get("recipes")
|
|
|
|
self.assertEqual(len(recipes), 1)
|
|
|
|
self.assertEqual(recipes[0], test_recipe)
|
2018-02-12 14:56:24 +00:00
|
|
|
|
|
|
|
def test_api_docs(self):
|
|
|
|
"""Test the /api/docs/"""
|
|
|
|
resp = self.server.get("/api/docs/")
|
|
|
|
doc_str = open(os.path.abspath(joinpaths(os.path.dirname(__file__), "../../docs/html/index.html"))).read()
|
|
|
|
self.assertEqual(doc_str, resp.data)
|
|
|
|
|
|
|
|
def test_api_docs_with_existing_path(self):
|
|
|
|
"""Test the /api/docs/modules.html"""
|
|
|
|
resp = self.server.get("/api/docs/modules.html")
|
|
|
|
doc_str = open(os.path.abspath(joinpaths(os.path.dirname(__file__), "../../docs/html/modules.html"))).read()
|
|
|
|
self.assertEqual(doc_str, resp.data)
|
2018-02-10 01:39:42 +00:00
|
|
|
|
|
|
|
def wait_for_status(self, uuid, wait_status):
|
|
|
|
"""Helper function that waits for a status
|
|
|
|
|
|
|
|
:param uuid: UUID of the build to check
|
|
|
|
:type uuid: str
|
|
|
|
:param wait_status: List of statuses to exit on
|
|
|
|
:type wait_status: list of str
|
|
|
|
:returns: True if status was found, False if it timed out
|
|
|
|
:rtype: bool
|
|
|
|
|
|
|
|
This will time out after 60 seconds
|
|
|
|
"""
|
|
|
|
start = time.time()
|
|
|
|
while True:
|
|
|
|
resp = self.server.get("/api/v0/compose/info/%s" % uuid)
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertNotEqual(data, None)
|
|
|
|
queue_status = data.get("queue_status")
|
|
|
|
if queue_status in wait_status:
|
|
|
|
return True
|
|
|
|
if time.time() > start + 60:
|
|
|
|
return False
|
|
|
|
time.sleep(5)
|
|
|
|
|
|
|
|
def test_compose_01_types(self):
|
|
|
|
"""Test the /api/v0/compose/types route"""
|
|
|
|
resp = self.server.get("/api/v0/compose/types")
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertNotEqual(data, None)
|
|
|
|
self.assertEqual({"name": "tar", "enabled": True} in data["types"], True)
|
|
|
|
|
|
|
|
def test_compose_02_create_failed(self):
|
|
|
|
"""Test the /api/v0/compose routes with a failed test compose"""
|
|
|
|
test_compose = {"recipe_name": "glusterfs",
|
|
|
|
"compose_type": "tar",
|
|
|
|
"branch": "master"}
|
|
|
|
|
|
|
|
resp = self.server.post("/api/v0/compose?test=1",
|
|
|
|
data=json.dumps(test_compose),
|
|
|
|
content_type="application/json")
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertNotEqual(data, None)
|
|
|
|
self.assertEqual(data["status"], True, "Failed to start test compose: %s" % data)
|
|
|
|
|
|
|
|
build_id = data["build_id"]
|
|
|
|
|
|
|
|
# Is it in the queue list (either new or run is fine, based on timing)
|
|
|
|
resp = self.server.get("/api/v0/compose/queue")
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertNotEqual(data, None)
|
|
|
|
ids = [e["id"] for e in data["new"] + data["run"]]
|
|
|
|
self.assertEqual(build_id in ids, True, "Failed to add build to the queue")
|
|
|
|
|
|
|
|
# Wait for it to start
|
|
|
|
self.assertEqual(self.wait_for_status(build_id, ["RUNNING"]), True, "Failed to start test compose")
|
|
|
|
|
|
|
|
# Wait for it to finish
|
|
|
|
self.assertEqual(self.wait_for_status(build_id, ["FAILED"]), True, "Failed to finish test compose")
|
|
|
|
|
|
|
|
resp = self.server.get("/api/v0/compose/info/%s" % build_id)
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertNotEqual(data, None)
|
|
|
|
self.assertEqual(data["queue_status"], "FAILED", "Build not in FAILED state")
|
|
|
|
|
|
|
|
# Test the /api/v0/compose/failed route
|
|
|
|
resp = self.server.get("/api/v0/compose/failed")
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertNotEqual(data, None)
|
|
|
|
ids = [e["id"] for e in data["failed"]]
|
|
|
|
self.assertEqual(build_id in ids, True, "Failed build not listed by /compose/failed")
|
|
|
|
|
|
|
|
# Test the /api/v0/compose/finished route
|
|
|
|
resp = self.server.get("/api/v0/compose/finished")
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertNotEqual(data, None)
|
|
|
|
self.assertEqual(data["finished"], [], "Finished build not listed by /compose/finished")
|
|
|
|
|
|
|
|
# Test the /api/v0/compose/status/<uuid> route
|
|
|
|
resp = self.server.get("/api/v0/compose/status/%s" % build_id)
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertNotEqual(data, None)
|
|
|
|
ids = [(e["id"], e["queue_status"]) for e in data["uuids"]]
|
|
|
|
self.assertEqual((build_id, "FAILED") in ids, True, "Failed build not listed by /compose/status")
|
|
|
|
|
|
|
|
# Test the /api/v0/compose/cancel/<uuid> route
|
|
|
|
resp = self.server.post("/api/v0/compose?test=1",
|
|
|
|
data=json.dumps(test_compose),
|
|
|
|
content_type="application/json")
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertNotEqual(data, None)
|
|
|
|
self.assertEqual(data["status"], True, "Failed to start test compose: %s" % data)
|
|
|
|
|
|
|
|
cancel_id = data["build_id"]
|
|
|
|
|
|
|
|
# Wait for it to start
|
|
|
|
self.assertEqual(self.wait_for_status(cancel_id, ["RUNNING"]), True, "Failed to start test compose")
|
|
|
|
|
|
|
|
# Cancel the build
|
|
|
|
resp = self.server.delete("/api/v0/compose/cancel/%s" % cancel_id)
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertNotEqual(data, None)
|
|
|
|
self.assertEqual(data["status"], True, "Failed to cancel test compose: %s" % data)
|
|
|
|
|
|
|
|
# Delete the failed build
|
|
|
|
# Test the /api/v0/compose/delete/<uuid> route
|
|
|
|
resp = self.server.delete("/api/v0/compose/delete/%s" % build_id)
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertNotEqual(data, None)
|
|
|
|
ids = [(e["uuid"], e["status"]) for e in data["uuids"]]
|
|
|
|
self.assertEqual((build_id, True) in ids, True, "Failed to delete test compose: %s" % data)
|
|
|
|
|
|
|
|
# Make sure the failed list is empty
|
|
|
|
resp = self.server.get("/api/v0/compose/failed")
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertNotEqual(data, None)
|
|
|
|
self.assertEqual(data["failed"], [], "Failed to delete the failed build: %s" % data)
|
|
|
|
|
|
|
|
def test_compose_03_create_finished(self):
|
|
|
|
"""Test the /api/v0/compose routes with a finished test compose"""
|
|
|
|
test_compose = {"recipe_name": "glusterfs",
|
|
|
|
"compose_type": "tar",
|
|
|
|
"branch": "master"}
|
|
|
|
|
|
|
|
resp = self.server.post("/api/v0/compose?test=2",
|
|
|
|
data=json.dumps(test_compose),
|
|
|
|
content_type="application/json")
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertNotEqual(data, None)
|
|
|
|
self.assertEqual(data["status"], True, "Failed to start test compose: %s" % data)
|
|
|
|
|
|
|
|
build_id = data["build_id"]
|
|
|
|
|
|
|
|
# Is it in the queue list (either new or run is fine, based on timing)
|
|
|
|
resp = self.server.get("/api/v0/compose/queue")
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertNotEqual(data, None)
|
|
|
|
ids = [e["id"] for e in data["new"] + data["run"]]
|
|
|
|
self.assertEqual(build_id in ids, True, "Failed to add build to the queue")
|
|
|
|
|
|
|
|
# Wait for it to start
|
|
|
|
self.assertEqual(self.wait_for_status(build_id, ["RUNNING"]), True, "Failed to start test compose")
|
|
|
|
|
|
|
|
# Wait for it to finish
|
|
|
|
self.assertEqual(self.wait_for_status(build_id, ["FINISHED"]), True, "Failed to finish test compose")
|
|
|
|
|
|
|
|
resp = self.server.get("/api/v0/compose/info/%s" % build_id)
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertNotEqual(data, None)
|
|
|
|
self.assertEqual(data["queue_status"], "FINISHED", "Build not in FINISHED state")
|
|
|
|
|
|
|
|
# Test the /api/v0/compose/finished route
|
|
|
|
resp = self.server.get("/api/v0/compose/finished")
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertNotEqual(data, None)
|
|
|
|
ids = [e["id"] for e in data["finished"]]
|
|
|
|
self.assertEqual(build_id in ids, True, "Finished build not listed by /compose/finished")
|
|
|
|
|
|
|
|
# Test the /api/v0/compose/failed route
|
|
|
|
resp = self.server.get("/api/v0/compose/failed")
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertNotEqual(data, None)
|
|
|
|
self.assertEqual(data["failed"], [], "Failed build not listed by /compose/failed")
|
|
|
|
|
|
|
|
# Test the /api/v0/compose/status/<uuid> route
|
|
|
|
resp = self.server.get("/api/v0/compose/status/%s" % build_id)
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertNotEqual(data, None)
|
|
|
|
ids = [(e["id"], e["queue_status"]) for e in data["uuids"]]
|
|
|
|
self.assertEqual((build_id, "FINISHED") in ids, True, "Finished build not listed by /compose/status")
|
|
|
|
|
|
|
|
# Test the /api/v0/compose/metadata/<uuid> route
|
|
|
|
resp = self.server.get("/api/v0/compose/metadata/%s" % build_id)
|
|
|
|
self.assertEqual(resp.status_code, 200)
|
|
|
|
self.assertEqual(len(resp.data) > 1024, True)
|
|
|
|
|
|
|
|
# Test the /api/v0/compose/results/<uuid> route
|
|
|
|
resp = self.server.get("/api/v0/compose/results/%s" % build_id)
|
|
|
|
self.assertEqual(resp.status_code, 200)
|
|
|
|
self.assertEqual(len(resp.data) > 1024, True)
|
|
|
|
|
|
|
|
# Test the /api/v0/compose/image/<uuid> route
|
|
|
|
resp = self.server.get("/api/v0/compose/image/%s" % build_id)
|
|
|
|
self.assertEqual(resp.status_code, 200)
|
|
|
|
self.assertEqual(len(resp.data) > 0, True)
|
|
|
|
self.assertEqual(resp.data, "TEST IMAGE")
|
|
|
|
|
|
|
|
# Delete the finished build
|
|
|
|
# Test the /api/v0/compose/delete/<uuid> route
|
|
|
|
resp = self.server.delete("/api/v0/compose/delete/%s" % build_id)
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertNotEqual(data, None)
|
|
|
|
ids = [(e["uuid"], e["status"]) for e in data["uuids"]]
|
|
|
|
self.assertEqual((build_id, True) in ids, True, "Failed to delete test compose: %s" % data)
|
|
|
|
|
|
|
|
# Make sure the finished list is empty
|
|
|
|
resp = self.server.get("/api/v0/compose/finished")
|
|
|
|
data = json.loads(resp.data)
|
|
|
|
self.assertNotEqual(data, None)
|
|
|
|
self.assertEqual(data["finished"], [], "Failed to delete the failed build: %s" % data)
|