
Openstack common has a wrapper for generating uuids. We should use that function to generate uuids for consistency. Change-Id: I726f0f2687945336c0e48c9f5a0897bd81a3b2b3
458 lines
18 KiB
Python
458 lines
18 KiB
Python
# Copyright 2013 - Noorul Islam K M
|
|
#
|
|
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
|
# not use this file except in compliance with the License. You may obtain
|
|
# a copy of the License at
|
|
#
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
#
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
|
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
|
# License for the specific language governing permissions and limitations
|
|
# under the License.
|
|
|
|
import collections
|
|
import re
|
|
import sys
|
|
|
|
import fixtures
|
|
import mock
|
|
|
|
from oslo_utils import uuidutils
|
|
|
|
import six
|
|
from stevedore import extension
|
|
import testtools
|
|
from testtools import matchers
|
|
|
|
from solumclient import client
|
|
from solumclient.common.apiclient import auth
|
|
from solumclient.common import yamlutils
|
|
from solumclient import solum
|
|
from solumclient.tests import base
|
|
from solumclient.v1 import component
|
|
from solumclient.v1 import languagepack
|
|
from solumclient.v1 import pipeline
|
|
from solumclient.v1 import plan
|
|
|
|
FAKE_ENV = {'OS_USERNAME': 'username',
|
|
'OS_PASSWORD': 'password',
|
|
'OS_TENANT_NAME': 'tenant_name',
|
|
'OS_AUTH_URL': 'http://no.where'}
|
|
|
|
languagepack_file_data = (
|
|
'{"language-pack-type":"Java", "language-pack-name":"Java version 1.4."}')
|
|
|
|
|
|
class MockEntrypoint(object):
|
|
def __init__(self, name, plugin):
|
|
self.name = name
|
|
self.plugin = plugin
|
|
|
|
|
|
class BaseFakePlugin(auth.BaseAuthPlugin):
|
|
def _do_authenticate(self, http_client):
|
|
pass
|
|
|
|
def token_and_endpoint(self, endpoint_type, service_type):
|
|
pass
|
|
|
|
|
|
class TestSolum(base.TestCase):
|
|
"""Test the Solum CLI."""
|
|
|
|
re_options = re.DOTALL | re.MULTILINE
|
|
|
|
# Patch os.environ to avoid required auth info.
|
|
def make_env(self, exclude=None):
|
|
env = dict((k, v) for k, v in FAKE_ENV.items() if k != exclude)
|
|
self.useFixture(fixtures.MonkeyPatch('os.environ', env))
|
|
|
|
@mock.patch.object(extension.ExtensionManager, "map")
|
|
def shell(self, argstr, mock_mgr_map, exit_code=0):
|
|
class FakePlugin(BaseFakePlugin):
|
|
def authenticate(self, cls):
|
|
cls.request(
|
|
"POST", "http://auth/tokens",
|
|
json={"fake": "me"}, allow_redirects=True)
|
|
|
|
mock_mgr_map.side_effect = (
|
|
lambda func: func(MockEntrypoint("fake", FakePlugin)))
|
|
|
|
orig = sys.stdout
|
|
try:
|
|
sys.stdout = six.StringIO()
|
|
argv = [__file__, ]
|
|
argv.extend(argstr.split())
|
|
self.useFixture(
|
|
fixtures.MonkeyPatch('sys.argv', argv))
|
|
|
|
solum.main()
|
|
except SystemExit:
|
|
exc_type, exc_value, exc_traceback = sys.exc_info()
|
|
self.assertEqual(exit_code, exc_value.code)
|
|
finally:
|
|
out = sys.stdout.getvalue()
|
|
sys.stdout.close()
|
|
sys.stdout = orig
|
|
|
|
return out
|
|
|
|
@testtools.skip("Skip it when found solution")
|
|
def test_get_client_debug(self):
|
|
env = dict((k, v) for k, v in FAKE_ENV.items() if k is not None)
|
|
test_client = client.get_client('1', **env)
|
|
self.assertFalse(test_client.http_client.debug)
|
|
test_client = client.get_client('1', debug=True)
|
|
self.assertTrue(test_client.http_client.debug)
|
|
|
|
@testtools.skip("Skip it when found solution")
|
|
def test_get_client_insecure(self):
|
|
env = dict((k, v) for k, v in FAKE_ENV.items() if k is not None)
|
|
test_client = client.get_client('1', **env)
|
|
self.assertTrue(test_client.http_client.verify)
|
|
test_client = client.get_client('1', verify=False)
|
|
self.assertFalse(test_client.http_client.verify)
|
|
|
|
def test_help(self):
|
|
required = [
|
|
'.*?^Available commands:'
|
|
|
|
]
|
|
help_text = self.shell('--help')
|
|
for r in required:
|
|
self.assertThat(help_text,
|
|
matchers.MatchesRegex(r,
|
|
self.re_options))
|
|
|
|
# Workflow Tests #
|
|
def test_workflow_error(self):
|
|
self.make_env()
|
|
out = self.shell("workflow unknown-command")
|
|
self.assertIn("Available commands", out)
|
|
|
|
# Pipeline Tests #
|
|
@mock.patch.object(pipeline.PipelineManager, "list")
|
|
def test_pipeline_list(self, mock_pipeline_list):
|
|
self.make_env()
|
|
self.shell("pipeline list")
|
|
mock_pipeline_list.assert_called_once_with()
|
|
|
|
@mock.patch.object(pipeline.PipelineManager, "create")
|
|
def test_pipeline_create(self, mock_pipeline_create):
|
|
self.make_env()
|
|
self.shell("pipeline create http://example.com/a.yaml workbook test")
|
|
mock_pipeline_create.assert_called_once_with(
|
|
name='test',
|
|
workbook_name='workbook',
|
|
plan_uri='http://example.com/a.yaml')
|
|
|
|
@mock.patch.object(pipeline.PipelineManager, "create")
|
|
def test_pipeline_create_without_name(self, mock_pipeline_create):
|
|
self.make_env()
|
|
self.shell("pipeline create http://example.com/a.yaml workbook",
|
|
exit_code=2)
|
|
|
|
@mock.patch.object(plan.PlanManager, "find")
|
|
@mock.patch.object(pipeline.PipelineManager, "create")
|
|
def test_pipeline_create_with_plan_name(self, mock_pipeline_create,
|
|
mock_plan_find):
|
|
class FakePlan(object):
|
|
uri = 'http://example.com/the-plan.yaml'
|
|
|
|
self.make_env()
|
|
mock_plan_find.return_value = FakePlan()
|
|
self.shell("pipeline create the-plan-name workbook test")
|
|
mock_plan_find.assert_called_once_with(name_or_id='the-plan-name')
|
|
mock_pipeline_create.assert_called_once_with(
|
|
name='test',
|
|
workbook_name='workbook',
|
|
plan_uri='http://example.com/the-plan.yaml')
|
|
|
|
@mock.patch.object(pipeline.PipelineManager, "delete")
|
|
@mock.patch.object(pipeline.PipelineManager, "find")
|
|
def test_pipeline_delete(self, mock_pipeline_find, mock_pipeline_delete):
|
|
self.make_env()
|
|
the_id = uuidutils.generate_uuid()
|
|
self.shell("pipeline delete %s" % the_id)
|
|
mock_pipeline_find.assert_called_once_with(
|
|
name_or_id=the_id)
|
|
self.assertEqual(1, mock_pipeline_delete.call_count)
|
|
|
|
@mock.patch.object(pipeline.PipelineManager, "find")
|
|
def test_pipeline_get(self, mock_pipeline_find):
|
|
self.make_env()
|
|
the_id = uuidutils.generate_uuid()
|
|
self.shell("pipeline show %s" % the_id)
|
|
mock_pipeline_find.assert_called_once_with(name_or_id=the_id)
|
|
|
|
@mock.patch.object(pipeline.PipelineManager, "find")
|
|
def test_pipeline_get_by_name(self, mock_pipeline_find):
|
|
self.make_env()
|
|
self.shell("pipeline show app2")
|
|
mock_pipeline_find.assert_called_once_with(name_or_id='app2')
|
|
|
|
# App Tests #
|
|
def test_app_create_with_missing_workflow_config(self):
|
|
raw_data = 'version: 1\nname: ex_app\nworkflow_config:\n'
|
|
mopen = mock.mock_open(read_data=raw_data)
|
|
|
|
with mock.patch('%s.open' % solum.__name__, mopen, create=True):
|
|
self.make_env()
|
|
out = self.shell("app create --app-file /dev/null")
|
|
self.assertEqual("ERROR: Workflow config cannot be empty\n", out)
|
|
|
|
def test_app_create_with_missing_trigger_actions(self):
|
|
raw_data = 'version: 1\nname: ex_app\ntrigger_actions:\n'
|
|
mopen = mock.mock_open(read_data=raw_data)
|
|
|
|
with mock.patch('%s.open' % solum.__name__, mopen, create=True):
|
|
self.make_env()
|
|
out = self.shell("app create --app-file /dev/null")
|
|
self.assertEqual("ERROR: Trigger actions cannot be empty\n", out)
|
|
|
|
def test_app_create_with_bad_name(self):
|
|
raw_data = '\n'.join([
|
|
'version: 1',
|
|
'name: ex=app1',
|
|
'description: python web app',
|
|
'workflow_config:',
|
|
' run_cmd: python app.py',
|
|
'ports: [5000]'])
|
|
|
|
mopen = mock.mock_open(read_data=raw_data)
|
|
|
|
with mock.patch('%s.open' % solum.__name__, mopen, create=True):
|
|
self.make_env()
|
|
out = self.shell("app create --app-file /dev/null")
|
|
|
|
self.assertEqual("ERROR: Application name must be 1-100 "
|
|
"characters long, only contain a-z,0-9,-,_ and "
|
|
"start with an alphabet character.\n", out)
|
|
|
|
# OldApp Tests #
|
|
def test_oldapp_create_with_missing_artifacts(self):
|
|
raw_data = 'version: 1\nname: ex_plan1\ndescription: dsc1.'
|
|
mopen = mock.mock_open(read_data=raw_data)
|
|
|
|
with mock.patch('%s.open' % solum.__name__, mopen, create=True):
|
|
self.make_env()
|
|
out = self.shell("oldapp create --plan-file /dev/null")
|
|
self.assertEqual("ERROR: Missing artifacts section\n", out)
|
|
|
|
def test_oldapp_create_with_bad_name(self):
|
|
raw_data = '\n'.join([
|
|
'version: 1',
|
|
'name: ex=plan1',
|
|
'description: python web app',
|
|
'artifacts:',
|
|
'- name: web',
|
|
' content:',
|
|
' href: https://github.com/user/repo.git',
|
|
' language_pack: auto',
|
|
' unittest_cmd: ./unit_tests.sh',
|
|
' run_cmd: python app.py',
|
|
' ports: 5000'])
|
|
|
|
mopen = mock.mock_open(read_data=raw_data)
|
|
|
|
with mock.patch('%s.open' % solum.__name__, mopen, create=True):
|
|
self.make_env()
|
|
out = self.shell("oldapp create --plan-file /dev/null")
|
|
|
|
self.assertEqual("ERROR: Application name must be 1-100 "
|
|
"characters long, only contain a-z,0-9,-,_ and "
|
|
"start with an alphabet character.\n", out)
|
|
|
|
@testtools.skip("Skip it when found solution")
|
|
def test_oldapp_create_with_bad_artifact_name(self):
|
|
raw_data = '\n'.join([
|
|
'version: 1',
|
|
'name: explan1',
|
|
'description: python web app',
|
|
'artifacts:',
|
|
'- name:',
|
|
' content:',
|
|
' href: https://github.com/user/repo.git',
|
|
' language_pack: auto',
|
|
' unittest_cmd: ./unit_tests.sh',
|
|
' run_cmd: python app.py',
|
|
' ports: 5000'])
|
|
|
|
mopen = mock.mock_open(read_data=raw_data)
|
|
|
|
with mock.patch('%s.open' % solum.__name__, mopen, create=True):
|
|
self.make_env()
|
|
out = self.shell("oldapp create --plan-file /dev/null")
|
|
|
|
# No part of the plan is in error; the next step in the test
|
|
# is authorization, which is deliberately mocked.
|
|
self.assertIn("ERROR: Authorization Failed:", out)
|
|
self.assertIn("http://no.where/tokens", out)
|
|
|
|
def test_oldapp_create_with_artifacts_empty(self):
|
|
raw_data = 'version: 1\nname: ex_plan1\ndescription: dsc1.\nartifacts:'
|
|
mopen = mock.mock_open(read_data=raw_data)
|
|
|
|
with mock.patch('%s.open' % solum.__name__, mopen, create=True):
|
|
self.make_env()
|
|
out = self.shell("oldapp create --plan-file /dev/null")
|
|
self.assertEqual("ERROR: Artifacts cannot be empty\n", out)
|
|
|
|
def test_oldapp_create_with_artifacts_no_content(self):
|
|
raw_data = 'version: 1\nname: ex_plan1\ndescription: d1.\nartifacts:\n'
|
|
raw_data += '- name: asdfds\n'
|
|
raw_data += ' artifact_type: heroku\n'
|
|
raw_data += ' language_pack: lp'
|
|
|
|
mopen = mock.mock_open(read_data=raw_data)
|
|
|
|
with mock.patch('%s.open' % solum.__name__, mopen, create=True):
|
|
self.make_env()
|
|
out = self.shell("oldapp create --plan-file /dev/null")
|
|
self.assertEqual("ERROR: Artifact content missing\n", out)
|
|
|
|
# Plan Tests #
|
|
@mock.patch.object(plan.PlanManager, "create")
|
|
def test_plan_create(self, mock_plan_create):
|
|
FakeResource = collections.namedtuple("FakeResource",
|
|
"uuid name description uri")
|
|
|
|
mock_plan_create.return_value = FakeResource('foo', 'foo', 'foo',
|
|
'foo')
|
|
raw_data = 'version: 1\nname: ex_plan1\ndescription: dsc1.'
|
|
plan_data = yamlutils.dump(yamlutils.load(raw_data))
|
|
mopen = mock.mock_open(read_data=raw_data)
|
|
with mock.patch('%s.open' % solum.__name__, mopen, create=True):
|
|
self.make_env()
|
|
self.shell("plan create /dev/null")
|
|
mock_plan_create.assert_called_once_with(plan_data)
|
|
|
|
@mock.patch.object(plan.PlanManager, "create")
|
|
def test_plan_create_with_private_github_repo(self, mock_plan_create):
|
|
FakeResource = collections.namedtuple(
|
|
"FakeResource", "uuid name description uri artifacts")
|
|
|
|
mock_plan_create.return_value = FakeResource('foo', 'foo', 'foo',
|
|
'foo', 'artifacts')
|
|
expected_printed_dict_args = mock_plan_create.return_value._asdict()
|
|
expected_printed_dict_args.pop('artifacts')
|
|
raw_data = 'version: 1\nname: ex_plan1\ndescription: dsc1.'
|
|
plan_data = yamlutils.dump(yamlutils.load(raw_data))
|
|
mopen = mock.mock_open(read_data=raw_data)
|
|
with mock.patch('%s.open' % solum.__name__, mopen, create=True):
|
|
self.make_env()
|
|
self.shell("plan create /dev/null")
|
|
mock_plan_create.assert_called_once_with(plan_data)
|
|
|
|
@mock.patch.object(plan.PlanManager, "list")
|
|
def test_plan_list(self, mock_plan_list):
|
|
self.make_env()
|
|
self.shell("plan list")
|
|
mock_plan_list.assert_called_once_with()
|
|
|
|
@mock.patch.object(plan.PlanManager, "delete")
|
|
@mock.patch.object(plan.PlanManager, "find")
|
|
def test_plan_delete(self, mock_plan_find, mock_plan_delete):
|
|
self.make_env()
|
|
the_id = uuidutils.generate_uuid()
|
|
self.shell("plan delete %s" % the_id)
|
|
mock_plan_find.assert_called_once_with(name_or_id=the_id)
|
|
self.assertEqual(1, mock_plan_delete.call_count)
|
|
|
|
@mock.patch.object(plan.PlanManager, "find")
|
|
def test_plan_get(self, mock_plan_find):
|
|
self.make_env()
|
|
the_id = uuidutils.generate_uuid()
|
|
self.shell("plan show %s" % the_id)
|
|
mock_plan_find.assert_called_once_with(name_or_id=the_id)
|
|
|
|
@mock.patch.object(plan.PlanManager, "find")
|
|
def test_plan_get_private_github_repo(self, mock_plan_find):
|
|
self.make_env()
|
|
the_id = uuidutils.generate_uuid()
|
|
FakeResource = collections.namedtuple(
|
|
"FakeResource", "uuid name description uri artifacts")
|
|
mock_plan_find.return_value = FakeResource('foo', 'foo', 'foo', 'foo',
|
|
'artifacts')
|
|
self.shell("plan show %s" % the_id)
|
|
mock_plan_find.assert_called_once_with(name_or_id=the_id)
|
|
|
|
# LanguagePack Tests #
|
|
@mock.patch.object(languagepack.LanguagePackManager, "list")
|
|
def test_languagepack_list(self, mock_lp_list):
|
|
self.make_env()
|
|
self.shell("languagepack list")
|
|
self.assertEqual(1, mock_lp_list.call_count)
|
|
|
|
@mock.patch.object(languagepack.LanguagePackManager, "create")
|
|
def test_languagepack_create(self, mock_lp_create):
|
|
FakeResource = collections.namedtuple("FakeResource",
|
|
"uuid name description "
|
|
"compiler_versions os_platform")
|
|
mock_lp_create.return_value = FakeResource(
|
|
'foo', 'foo', 'foo', 'foo', 'foo')
|
|
lp_metadata = '{"OS": "Ubuntu"}'
|
|
mopen = mock.mock_open(read_data=lp_metadata)
|
|
with mock.patch('%s.open' % solum.__name__, mopen, create=True):
|
|
self.make_env()
|
|
|
|
self.shell("languagepack create lp_name github.com/test "
|
|
"--lp_metadata=/dev/null")
|
|
mock_lp_create.assert_called_once_with(
|
|
name='lp_name',
|
|
source_uri='github.com/test',
|
|
lp_metadata=lp_metadata,
|
|
lp_params={})
|
|
|
|
@mock.patch.object(languagepack.LanguagePackManager, "delete")
|
|
def test_languagepack_delete(self, mock_lp_delete):
|
|
self.make_env()
|
|
self.shell("languagepack delete fake-lp-id")
|
|
mock_lp_delete.assert_called_once_with(lp_id='fake-lp-id')
|
|
|
|
@mock.patch.object(languagepack.LanguagePackManager, "find")
|
|
def test_languagepack_get(self, mock_lp_get):
|
|
self.make_env()
|
|
self.shell("languagepack show fake-lp-id1")
|
|
mock_lp_get.assert_called_once_with(name_or_id='fake-lp-id1')
|
|
|
|
# Component Tests #
|
|
@mock.patch.object(component.ComponentManager, "list")
|
|
def test_component_list(self, mock_component_list):
|
|
self.make_env()
|
|
self.shell("component list")
|
|
mock_component_list.assert_called_once_with()
|
|
|
|
@mock.patch.object(component.ComponentManager, "find")
|
|
def test_component_get(self, mock_component_find):
|
|
self.make_env()
|
|
the_id = uuidutils.generate_uuid()
|
|
self.shell("component show %s" % the_id)
|
|
mock_component_find.assert_called_once_with(name_or_id=the_id)
|
|
|
|
@mock.patch.object(component.ComponentManager, "find")
|
|
def test_component_get_by_name(self, mock_component_find):
|
|
self.make_env()
|
|
self.shell("component show comp1")
|
|
mock_component_find.assert_called_once_with(name_or_id='comp1')
|
|
|
|
def test_transform_git_url(self):
|
|
private_uri = 'git@github.com:solum/python.git'
|
|
public_uri = 'https://github.com/solum/python.git'
|
|
private = True
|
|
public = False
|
|
|
|
result = solum.transform_git_url(private_uri, private)
|
|
self.assertEqual(result, private_uri)
|
|
|
|
result = solum.transform_git_url(public_uri, private)
|
|
self.assertEqual(result, private_uri)
|
|
|
|
result = solum.transform_git_url(private_uri, public)
|
|
self.assertEqual(result, public_uri)
|
|
|
|
result = solum.transform_git_url(public_uri, public)
|
|
self.assertEqual(result, public_uri)
|