2019-07-09 19:47:25 +00:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
# Copyright: (c) 2019, Ansible Project
|
|
|
|
# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
|
|
|
|
|
|
|
|
# Make coding more python3-ish
|
|
|
|
from __future__ import (absolute_import, division, print_function)
|
|
|
|
__metaclass__ = type
|
|
|
|
|
2019-08-20 21:49:05 +00:00
|
|
|
import copy
|
2019-07-09 19:47:25 +00:00
|
|
|
import json
|
|
|
|
import os
|
|
|
|
import pytest
|
|
|
|
import re
|
|
|
|
import shutil
|
|
|
|
import tarfile
|
|
|
|
import yaml
|
|
|
|
|
|
|
|
from io import BytesIO, StringIO
|
|
|
|
from units.compat.mock import MagicMock
|
|
|
|
|
|
|
|
import ansible.module_utils.six.moves.urllib.error as urllib_error
|
|
|
|
|
2019-08-20 21:49:05 +00:00
|
|
|
from ansible import context
|
2019-07-09 19:47:25 +00:00
|
|
|
from ansible.cli.galaxy import GalaxyCLI
|
|
|
|
from ansible.errors import AnsibleError
|
2019-08-28 20:59:34 +00:00
|
|
|
from ansible.galaxy import collection, api
|
2019-07-09 19:47:25 +00:00
|
|
|
from ansible.module_utils._text import to_bytes, to_native, to_text
|
|
|
|
from ansible.utils import context_objects as co
|
|
|
|
from ansible.utils.display import Display
|
|
|
|
|
|
|
|
|
|
|
|
def call_galaxy_cli(args):
|
|
|
|
orig = co.GlobalCLIArgs._Singleton__instance
|
|
|
|
co.GlobalCLIArgs._Singleton__instance = None
|
|
|
|
try:
|
|
|
|
GalaxyCLI(args=['ansible-galaxy', 'collection'] + args).run()
|
|
|
|
finally:
|
|
|
|
co.GlobalCLIArgs._Singleton__instance = orig
|
|
|
|
|
|
|
|
|
|
|
|
def artifact_json(namespace, name, version, dependencies, server):
|
|
|
|
json_str = json.dumps({
|
|
|
|
'artifact': {
|
|
|
|
'filename': '%s-%s-%s.tar.gz' % (namespace, name, version),
|
|
|
|
'sha256': '2d76f3b8c4bab1072848107fb3914c345f71a12a1722f25c08f5d3f51f4ab5fd',
|
|
|
|
'size': 1234,
|
|
|
|
},
|
|
|
|
'download_url': '%s/download/%s-%s-%s.tar.gz' % (server, namespace, name, version),
|
|
|
|
'metadata': {
|
|
|
|
'namespace': namespace,
|
|
|
|
'name': name,
|
|
|
|
'dependencies': dependencies,
|
|
|
|
},
|
|
|
|
'version': version
|
|
|
|
})
|
|
|
|
return to_text(json_str)
|
|
|
|
|
|
|
|
|
2019-08-28 20:59:34 +00:00
|
|
|
def artifact_versions_json(namespace, name, versions, galaxy_api, available_api_versions=None):
|
2019-07-09 19:47:25 +00:00
|
|
|
results = []
|
2019-08-28 20:59:34 +00:00
|
|
|
available_api_versions = available_api_versions or {}
|
|
|
|
api_version = 'v2'
|
|
|
|
if 'v3' in available_api_versions:
|
|
|
|
api_version = 'v3'
|
2019-07-09 19:47:25 +00:00
|
|
|
for version in versions:
|
|
|
|
results.append({
|
2019-08-28 20:59:34 +00:00
|
|
|
'href': '%s/api/%s/%s/%s/versions/%s/' % (galaxy_api.api_server, api_version, namespace, name, version),
|
2019-07-09 19:47:25 +00:00
|
|
|
'version': version,
|
|
|
|
})
|
|
|
|
|
2019-08-28 20:59:34 +00:00
|
|
|
if api_version == 'v2':
|
|
|
|
json_str = json.dumps({
|
|
|
|
'count': len(versions),
|
|
|
|
'next': None,
|
|
|
|
'previous': None,
|
|
|
|
'results': results
|
|
|
|
})
|
|
|
|
|
|
|
|
if api_version == 'v3':
|
|
|
|
response = {'meta': {'count': len(versions)},
|
|
|
|
'data': results,
|
|
|
|
'links': {'first': None,
|
|
|
|
'last': None,
|
|
|
|
'next': None,
|
|
|
|
'previous': None},
|
|
|
|
}
|
|
|
|
json_str = json.dumps(response)
|
|
|
|
return to_text(json_str)
|
|
|
|
|
|
|
|
|
|
|
|
def error_json(galaxy_api, errors_to_return=None, available_api_versions=None):
|
|
|
|
errors_to_return = errors_to_return or []
|
|
|
|
available_api_versions = available_api_versions or {}
|
|
|
|
|
|
|
|
response = {}
|
|
|
|
|
|
|
|
api_version = 'v2'
|
|
|
|
if 'v3' in available_api_versions:
|
|
|
|
api_version = 'v3'
|
|
|
|
|
|
|
|
if api_version == 'v2':
|
|
|
|
assert len(errors_to_return) <= 1
|
|
|
|
if errors_to_return:
|
|
|
|
response = errors_to_return[0]
|
|
|
|
|
|
|
|
if api_version == 'v3':
|
|
|
|
response['errors'] = errors_to_return
|
|
|
|
|
|
|
|
json_str = json.dumps(response)
|
2019-07-09 19:47:25 +00:00
|
|
|
return to_text(json_str)
|
|
|
|
|
|
|
|
|
2019-08-20 21:49:05 +00:00
|
|
|
@pytest.fixture(autouse='function')
|
|
|
|
def reset_cli_args():
|
|
|
|
co.GlobalCLIArgs._Singleton__instance = None
|
|
|
|
yield
|
|
|
|
co.GlobalCLIArgs._Singleton__instance = None
|
|
|
|
|
|
|
|
|
2019-07-09 19:47:25 +00:00
|
|
|
@pytest.fixture()
|
|
|
|
def collection_artifact(request, tmp_path_factory):
|
|
|
|
test_dir = to_text(tmp_path_factory.mktemp('test-ÅÑŚÌβŁÈ Collections Input'))
|
|
|
|
namespace = 'ansible_namespace'
|
|
|
|
collection = 'collection'
|
|
|
|
|
|
|
|
skeleton_path = os.path.join(os.path.dirname(os.path.split(__file__)[0]), 'cli', 'test_data', 'collection_skeleton')
|
|
|
|
collection_path = os.path.join(test_dir, namespace, collection)
|
|
|
|
|
|
|
|
call_galaxy_cli(['init', '%s.%s' % (namespace, collection), '-c', '--init-path', test_dir,
|
|
|
|
'--collection-skeleton', skeleton_path])
|
|
|
|
dependencies = getattr(request, 'param', None)
|
|
|
|
if dependencies:
|
|
|
|
galaxy_yml = os.path.join(collection_path, 'galaxy.yml')
|
|
|
|
with open(galaxy_yml, 'rb+') as galaxy_obj:
|
|
|
|
existing_yaml = yaml.safe_load(galaxy_obj)
|
|
|
|
existing_yaml['dependencies'] = dependencies
|
|
|
|
|
|
|
|
galaxy_obj.seek(0)
|
|
|
|
galaxy_obj.write(to_bytes(yaml.safe_dump(existing_yaml)))
|
|
|
|
galaxy_obj.truncate()
|
|
|
|
|
|
|
|
call_galaxy_cli(['build', collection_path, '--output-path', test_dir])
|
|
|
|
|
|
|
|
collection_tar = os.path.join(test_dir, '%s-%s-0.1.0.tar.gz' % (namespace, collection))
|
|
|
|
return to_bytes(collection_path), to_bytes(collection_tar)
|
|
|
|
|
|
|
|
|
2019-08-20 21:49:05 +00:00
|
|
|
@pytest.fixture()
|
|
|
|
def galaxy_server():
|
|
|
|
context.CLIARGS._store = {'ignore_certs': False}
|
|
|
|
galaxy_api = api.GalaxyAPI(None, 'test_server', 'https://galaxy.ansible.com')
|
|
|
|
return galaxy_api
|
|
|
|
|
|
|
|
|
2019-07-09 19:47:25 +00:00
|
|
|
def test_build_requirement_from_path(collection_artifact):
|
2019-08-20 21:49:05 +00:00
|
|
|
actual = collection.CollectionRequirement.from_path(collection_artifact[0], True)
|
2019-07-09 19:47:25 +00:00
|
|
|
|
|
|
|
assert actual.namespace == u'ansible_namespace'
|
|
|
|
assert actual.name == u'collection'
|
|
|
|
assert actual.b_path == collection_artifact[0]
|
2019-08-20 21:49:05 +00:00
|
|
|
assert actual.api is None
|
2019-07-09 19:47:25 +00:00
|
|
|
assert actual.skip is True
|
|
|
|
assert actual.versions == set([u'*'])
|
|
|
|
assert actual.latest_version == u'*'
|
|
|
|
assert actual.dependencies == {}
|
|
|
|
|
|
|
|
|
|
|
|
def test_build_requirement_from_path_with_manifest(collection_artifact):
|
|
|
|
manifest_path = os.path.join(collection_artifact[0], b'MANIFEST.json')
|
|
|
|
manifest_value = json.dumps({
|
|
|
|
'collection_info': {
|
|
|
|
'namespace': 'namespace',
|
|
|
|
'name': 'name',
|
|
|
|
'version': '1.1.1',
|
|
|
|
'dependencies': {
|
|
|
|
'ansible_namespace.collection': '*'
|
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
with open(manifest_path, 'wb') as manifest_obj:
|
|
|
|
manifest_obj.write(to_bytes(manifest_value))
|
|
|
|
|
2019-08-20 21:49:05 +00:00
|
|
|
actual = collection.CollectionRequirement.from_path(collection_artifact[0], True)
|
2019-07-09 19:47:25 +00:00
|
|
|
|
|
|
|
# While the folder name suggests a different collection, we treat MANIFEST.json as the source of truth.
|
|
|
|
assert actual.namespace == u'namespace'
|
|
|
|
assert actual.name == u'name'
|
|
|
|
assert actual.b_path == collection_artifact[0]
|
2019-08-20 21:49:05 +00:00
|
|
|
assert actual.api is None
|
2019-07-09 19:47:25 +00:00
|
|
|
assert actual.skip is True
|
|
|
|
assert actual.versions == set([u'1.1.1'])
|
|
|
|
assert actual.latest_version == u'1.1.1'
|
|
|
|
assert actual.dependencies == {'ansible_namespace.collection': '*'}
|
|
|
|
|
|
|
|
|
|
|
|
def test_build_requirement_from_path_invalid_manifest(collection_artifact):
|
|
|
|
manifest_path = os.path.join(collection_artifact[0], b'MANIFEST.json')
|
|
|
|
with open(manifest_path, 'wb') as manifest_obj:
|
|
|
|
manifest_obj.write(b"not json")
|
|
|
|
|
|
|
|
expected = "Collection file at '%s' does not contain a valid json string." % to_native(manifest_path)
|
|
|
|
with pytest.raises(AnsibleError, match=expected):
|
2019-08-20 21:49:05 +00:00
|
|
|
collection.CollectionRequirement.from_path(collection_artifact[0], True)
|
2019-07-09 19:47:25 +00:00
|
|
|
|
|
|
|
|
|
|
|
def test_build_requirement_from_tar(collection_artifact):
|
|
|
|
actual = collection.CollectionRequirement.from_tar(collection_artifact[1], True, True)
|
|
|
|
|
|
|
|
assert actual.namespace == u'ansible_namespace'
|
|
|
|
assert actual.name == u'collection'
|
|
|
|
assert actual.b_path == collection_artifact[1]
|
2019-08-20 21:49:05 +00:00
|
|
|
assert actual.api is None
|
2019-07-09 19:47:25 +00:00
|
|
|
assert actual.skip is False
|
|
|
|
assert actual.versions == set([u'0.1.0'])
|
|
|
|
assert actual.latest_version == u'0.1.0'
|
|
|
|
assert actual.dependencies == {}
|
|
|
|
|
|
|
|
|
|
|
|
def test_build_requirement_from_tar_fail_not_tar(tmp_path_factory):
|
|
|
|
test_dir = to_bytes(tmp_path_factory.mktemp('test-ÅÑŚÌβŁÈ Collections Input'))
|
|
|
|
test_file = os.path.join(test_dir, b'fake.tar.gz')
|
|
|
|
with open(test_file, 'wb') as test_obj:
|
|
|
|
test_obj.write(b"\x00\x01\x02\x03")
|
|
|
|
|
|
|
|
expected = "Collection artifact at '%s' is not a valid tar file." % to_native(test_file)
|
|
|
|
with pytest.raises(AnsibleError, match=expected):
|
|
|
|
collection.CollectionRequirement.from_tar(test_file, True, True)
|
|
|
|
|
|
|
|
|
|
|
|
def test_build_requirement_from_tar_no_manifest(tmp_path_factory):
|
|
|
|
test_dir = to_bytes(tmp_path_factory.mktemp('test-ÅÑŚÌβŁÈ Collections Input'))
|
|
|
|
|
|
|
|
json_data = to_bytes(json.dumps(
|
|
|
|
{
|
|
|
|
'files': [],
|
|
|
|
'format': 1,
|
|
|
|
}
|
|
|
|
))
|
|
|
|
|
|
|
|
tar_path = os.path.join(test_dir, b'ansible-collections.tar.gz')
|
|
|
|
with tarfile.open(tar_path, 'w:gz') as tfile:
|
|
|
|
b_io = BytesIO(json_data)
|
|
|
|
tar_info = tarfile.TarInfo('FILES.json')
|
|
|
|
tar_info.size = len(json_data)
|
|
|
|
tar_info.mode = 0o0644
|
|
|
|
tfile.addfile(tarinfo=tar_info, fileobj=b_io)
|
|
|
|
|
|
|
|
expected = "Collection at '%s' does not contain the required file MANIFEST.json." % to_native(tar_path)
|
|
|
|
with pytest.raises(AnsibleError, match=expected):
|
|
|
|
collection.CollectionRequirement.from_tar(tar_path, True, True)
|
|
|
|
|
|
|
|
|
|
|
|
def test_build_requirement_from_tar_no_files(tmp_path_factory):
|
|
|
|
test_dir = to_bytes(tmp_path_factory.mktemp('test-ÅÑŚÌβŁÈ Collections Input'))
|
|
|
|
|
|
|
|
json_data = to_bytes(json.dumps(
|
|
|
|
{
|
|
|
|
'collection_info': {},
|
|
|
|
}
|
|
|
|
))
|
|
|
|
|
|
|
|
tar_path = os.path.join(test_dir, b'ansible-collections.tar.gz')
|
|
|
|
with tarfile.open(tar_path, 'w:gz') as tfile:
|
|
|
|
b_io = BytesIO(json_data)
|
|
|
|
tar_info = tarfile.TarInfo('MANIFEST.json')
|
|
|
|
tar_info.size = len(json_data)
|
|
|
|
tar_info.mode = 0o0644
|
|
|
|
tfile.addfile(tarinfo=tar_info, fileobj=b_io)
|
|
|
|
|
|
|
|
expected = "Collection at '%s' does not contain the required file FILES.json." % to_native(tar_path)
|
|
|
|
with pytest.raises(AnsibleError, match=expected):
|
|
|
|
collection.CollectionRequirement.from_tar(tar_path, True, True)
|
|
|
|
|
|
|
|
|
|
|
|
def test_build_requirement_from_tar_invalid_manifest(tmp_path_factory):
|
|
|
|
test_dir = to_bytes(tmp_path_factory.mktemp('test-ÅÑŚÌβŁÈ Collections Input'))
|
|
|
|
|
|
|
|
json_data = b"not a json"
|
|
|
|
|
|
|
|
tar_path = os.path.join(test_dir, b'ansible-collections.tar.gz')
|
|
|
|
with tarfile.open(tar_path, 'w:gz') as tfile:
|
|
|
|
b_io = BytesIO(json_data)
|
|
|
|
tar_info = tarfile.TarInfo('MANIFEST.json')
|
|
|
|
tar_info.size = len(json_data)
|
|
|
|
tar_info.mode = 0o0644
|
|
|
|
tfile.addfile(tarinfo=tar_info, fileobj=b_io)
|
|
|
|
|
|
|
|
expected = "Collection tar file member MANIFEST.json does not contain a valid json string."
|
|
|
|
with pytest.raises(AnsibleError, match=expected):
|
|
|
|
collection.CollectionRequirement.from_tar(tar_path, True, True)
|
|
|
|
|
|
|
|
|
2019-08-30 01:55:19 +00:00
|
|
|
def test_build_requirement_from_name(galaxy_server, monkeypatch):
|
|
|
|
mock_get_versions = MagicMock()
|
|
|
|
mock_get_versions.return_value = ['2.1.9', '2.1.10']
|
|
|
|
monkeypatch.setattr(galaxy_server, 'get_collection_versions', mock_get_versions)
|
2019-07-09 19:47:25 +00:00
|
|
|
|
|
|
|
actual = collection.CollectionRequirement.from_name('namespace.collection', [galaxy_server], '*', True, True)
|
|
|
|
|
|
|
|
assert actual.namespace == u'namespace'
|
|
|
|
assert actual.name == u'collection'
|
|
|
|
assert actual.b_path is None
|
2019-08-20 21:49:05 +00:00
|
|
|
assert actual.api == galaxy_server
|
2019-07-09 19:47:25 +00:00
|
|
|
assert actual.skip is False
|
|
|
|
assert actual.versions == set([u'2.1.9', u'2.1.10'])
|
|
|
|
assert actual.latest_version == u'2.1.10'
|
|
|
|
assert actual.dependencies is None
|
|
|
|
|
2019-08-30 01:55:19 +00:00
|
|
|
assert mock_get_versions.call_count == 1
|
|
|
|
assert mock_get_versions.mock_calls[0][1] == ('namespace', 'collection')
|
2019-07-09 19:47:25 +00:00
|
|
|
|
|
|
|
|
2019-08-30 01:55:19 +00:00
|
|
|
def test_build_requirement_from_name_with_prerelease(galaxy_server, monkeypatch):
|
|
|
|
mock_get_versions = MagicMock()
|
|
|
|
mock_get_versions.return_value = ['1.0.1', '2.0.1-beta.1', '2.0.1']
|
|
|
|
monkeypatch.setattr(galaxy_server, 'get_collection_versions', mock_get_versions)
|
2019-07-09 19:47:25 +00:00
|
|
|
|
|
|
|
actual = collection.CollectionRequirement.from_name('namespace.collection', [galaxy_server], '*', True, True)
|
|
|
|
|
|
|
|
assert actual.namespace == u'namespace'
|
|
|
|
assert actual.name == u'collection'
|
|
|
|
assert actual.b_path is None
|
2019-08-20 21:49:05 +00:00
|
|
|
assert actual.api == galaxy_server
|
2019-07-09 19:47:25 +00:00
|
|
|
assert actual.skip is False
|
|
|
|
assert actual.versions == set([u'1.0.1', u'2.0.1'])
|
|
|
|
assert actual.latest_version == u'2.0.1'
|
|
|
|
assert actual.dependencies is None
|
|
|
|
|
2019-08-30 01:55:19 +00:00
|
|
|
assert mock_get_versions.call_count == 1
|
|
|
|
assert mock_get_versions.mock_calls[0][1] == ('namespace', 'collection')
|
2019-08-28 20:59:34 +00:00
|
|
|
|
2019-07-09 19:47:25 +00:00
|
|
|
|
2019-08-30 01:55:19 +00:00
|
|
|
def test_build_requirment_from_name_with_prerelease_explicit(galaxy_server, monkeypatch):
|
|
|
|
mock_get_info = MagicMock()
|
|
|
|
mock_get_info.return_value = api.CollectionVersionMetadata('namespace', 'collection', '2.0.1-beta.1', None, None,
|
|
|
|
{})
|
|
|
|
monkeypatch.setattr(galaxy_server, 'get_collection_version_metadata', mock_get_info)
|
2019-07-09 19:47:25 +00:00
|
|
|
|
|
|
|
actual = collection.CollectionRequirement.from_name('namespace.collection', [galaxy_server], '2.0.1-beta.1', True,
|
|
|
|
True)
|
|
|
|
|
|
|
|
assert actual.namespace == u'namespace'
|
|
|
|
assert actual.name == u'collection'
|
|
|
|
assert actual.b_path is None
|
2019-08-20 21:49:05 +00:00
|
|
|
assert actual.api == galaxy_server
|
2019-07-09 19:47:25 +00:00
|
|
|
assert actual.skip is False
|
|
|
|
assert actual.versions == set([u'2.0.1-beta.1'])
|
|
|
|
assert actual.latest_version == u'2.0.1-beta.1'
|
|
|
|
assert actual.dependencies == {}
|
|
|
|
|
2019-08-30 01:55:19 +00:00
|
|
|
assert mock_get_info.call_count == 1
|
|
|
|
assert mock_get_info.mock_calls[0][1] == ('namespace', 'collection', '2.0.1-beta.1')
|
2019-07-09 19:47:25 +00:00
|
|
|
|
|
|
|
|
2019-08-30 01:55:19 +00:00
|
|
|
def test_build_requirement_from_name_second_server(galaxy_server, monkeypatch):
|
|
|
|
mock_get_versions = MagicMock()
|
|
|
|
mock_get_versions.return_value = ['1.0.1', '1.0.2', '1.0.3']
|
|
|
|
monkeypatch.setattr(galaxy_server, 'get_collection_versions', mock_get_versions)
|
2019-07-09 19:47:25 +00:00
|
|
|
|
2019-08-20 21:49:05 +00:00
|
|
|
broken_server = copy.copy(galaxy_server)
|
|
|
|
broken_server.api_server = 'https://broken.com/'
|
2019-08-30 01:55:19 +00:00
|
|
|
mock_404 = MagicMock()
|
|
|
|
mock_404.side_effect = api.GalaxyError(urllib_error.HTTPError('https://galaxy.server.com', 404, 'msg', {},
|
|
|
|
StringIO()), "custom msg")
|
|
|
|
monkeypatch.setattr(broken_server, 'get_collection_versions', mock_404)
|
|
|
|
|
2019-08-20 21:49:05 +00:00
|
|
|
actual = collection.CollectionRequirement.from_name('namespace.collection', [broken_server, galaxy_server],
|
2019-07-09 19:47:25 +00:00
|
|
|
'>1.0.1', False, True)
|
|
|
|
|
|
|
|
assert actual.namespace == u'namespace'
|
|
|
|
assert actual.name == u'collection'
|
|
|
|
assert actual.b_path is None
|
2019-08-28 20:59:34 +00:00
|
|
|
# assert actual.api == galaxy_server
|
2019-07-09 19:47:25 +00:00
|
|
|
assert actual.skip is False
|
|
|
|
assert actual.versions == set([u'1.0.2', u'1.0.3'])
|
|
|
|
assert actual.latest_version == u'1.0.3'
|
|
|
|
assert actual.dependencies is None
|
|
|
|
|
2019-08-30 01:55:19 +00:00
|
|
|
assert mock_404.call_count == 1
|
|
|
|
assert mock_404.mock_calls[0][1] == ('namespace', 'collection')
|
|
|
|
|
|
|
|
assert mock_get_versions.call_count == 1
|
|
|
|
assert mock_get_versions.mock_calls[0][1] == ('namespace', 'collection')
|
2019-07-09 19:47:25 +00:00
|
|
|
|
|
|
|
|
2019-08-20 21:49:05 +00:00
|
|
|
def test_build_requirement_from_name_missing(galaxy_server, monkeypatch):
|
2019-07-09 19:47:25 +00:00
|
|
|
mock_open = MagicMock()
|
2019-08-30 01:55:19 +00:00
|
|
|
mock_open.side_effect = api.GalaxyError(urllib_error.HTTPError('https://galaxy.server.com', 404, 'msg', {},
|
|
|
|
StringIO()), "")
|
2019-07-09 19:47:25 +00:00
|
|
|
|
2019-08-30 01:55:19 +00:00
|
|
|
monkeypatch.setattr(galaxy_server, 'get_collection_versions', mock_open)
|
2019-08-28 20:59:34 +00:00
|
|
|
|
2019-07-09 19:47:25 +00:00
|
|
|
expected = "Failed to find collection namespace.collection:*"
|
|
|
|
with pytest.raises(AnsibleError, match=expected):
|
2019-08-30 01:55:19 +00:00
|
|
|
collection.CollectionRequirement.from_name('namespace.collection', [galaxy_server, galaxy_server], '*', False,
|
|
|
|
True)
|
2019-08-28 20:59:34 +00:00
|
|
|
|
|
|
|
|
2019-08-30 01:55:19 +00:00
|
|
|
def test_build_requirement_from_name_401_unauthorized(galaxy_server, monkeypatch):
|
2019-08-28 20:59:34 +00:00
|
|
|
mock_open = MagicMock()
|
2019-08-30 01:55:19 +00:00
|
|
|
mock_open.side_effect = api.GalaxyError(urllib_error.HTTPError('https://galaxy.server.com', 401, 'msg', {},
|
|
|
|
StringIO()), "error")
|
2019-08-28 20:59:34 +00:00
|
|
|
|
2019-08-30 01:55:19 +00:00
|
|
|
monkeypatch.setattr(galaxy_server, 'get_collection_versions', mock_open)
|
2019-08-28 20:59:34 +00:00
|
|
|
|
2019-08-30 01:55:19 +00:00
|
|
|
expected = "error (HTTP Code: 401, Message: Unknown error returned by Galaxy server.)"
|
|
|
|
with pytest.raises(api.GalaxyError, match=re.escape(expected)):
|
|
|
|
collection.CollectionRequirement.from_name('namespace.collection', [galaxy_server, galaxy_server], '*', False)
|
2019-08-28 20:59:34 +00:00
|
|
|
|
|
|
|
|
2019-08-20 21:49:05 +00:00
|
|
|
def test_build_requirement_from_name_single_version(galaxy_server, monkeypatch):
|
2019-08-30 01:55:19 +00:00
|
|
|
mock_get_info = MagicMock()
|
|
|
|
mock_get_info.return_value = api.CollectionVersionMetadata('namespace', 'collection', '2.0.0', None, None,
|
|
|
|
{})
|
|
|
|
monkeypatch.setattr(galaxy_server, 'get_collection_version_metadata', mock_get_info)
|
2019-08-28 20:59:34 +00:00
|
|
|
|
2019-08-30 01:55:19 +00:00
|
|
|
actual = collection.CollectionRequirement.from_name('namespace.collection', [galaxy_server], '2.0.0', True,
|
|
|
|
True)
|
2019-07-09 19:47:25 +00:00
|
|
|
|
|
|
|
assert actual.namespace == u'namespace'
|
|
|
|
assert actual.name == u'collection'
|
|
|
|
assert actual.b_path is None
|
2019-08-20 21:49:05 +00:00
|
|
|
assert actual.api == galaxy_server
|
2019-07-09 19:47:25 +00:00
|
|
|
assert actual.skip is False
|
|
|
|
assert actual.versions == set([u'2.0.0'])
|
|
|
|
assert actual.latest_version == u'2.0.0'
|
|
|
|
assert actual.dependencies == {}
|
|
|
|
|
2019-08-30 01:55:19 +00:00
|
|
|
assert mock_get_info.call_count == 1
|
|
|
|
assert mock_get_info.mock_calls[0][1] == ('namespace', 'collection', '2.0.0')
|
2019-07-09 19:47:25 +00:00
|
|
|
|
|
|
|
|
2019-08-20 21:49:05 +00:00
|
|
|
def test_build_requirement_from_name_multiple_versions_one_match(galaxy_server, monkeypatch):
|
2019-08-30 01:55:19 +00:00
|
|
|
mock_get_versions = MagicMock()
|
|
|
|
mock_get_versions.return_value = ['2.0.0', '2.0.1', '2.0.2']
|
|
|
|
monkeypatch.setattr(galaxy_server, 'get_collection_versions', mock_get_versions)
|
2019-07-09 19:47:25 +00:00
|
|
|
|
2019-08-30 01:55:19 +00:00
|
|
|
mock_get_info = MagicMock()
|
|
|
|
mock_get_info.return_value = api.CollectionVersionMetadata('namespace', 'collection', '2.0.1', None, None,
|
|
|
|
{})
|
|
|
|
monkeypatch.setattr(galaxy_server, 'get_collection_version_metadata', mock_get_info)
|
2019-08-28 20:59:34 +00:00
|
|
|
|
2019-07-09 19:47:25 +00:00
|
|
|
actual = collection.CollectionRequirement.from_name('namespace.collection', [galaxy_server], '>=2.0.1,<2.0.2',
|
|
|
|
True, True)
|
|
|
|
|
|
|
|
assert actual.namespace == u'namespace'
|
|
|
|
assert actual.name == u'collection'
|
|
|
|
assert actual.b_path is None
|
2019-08-20 21:49:05 +00:00
|
|
|
assert actual.api == galaxy_server
|
2019-07-09 19:47:25 +00:00
|
|
|
assert actual.skip is False
|
|
|
|
assert actual.versions == set([u'2.0.1'])
|
|
|
|
assert actual.latest_version == u'2.0.1'
|
|
|
|
assert actual.dependencies == {}
|
|
|
|
|
2019-08-30 01:55:19 +00:00
|
|
|
assert mock_get_versions.call_count == 1
|
|
|
|
assert mock_get_versions.mock_calls[0][1] == ('namespace', 'collection')
|
2019-07-09 19:47:25 +00:00
|
|
|
|
2019-08-30 01:55:19 +00:00
|
|
|
assert mock_get_info.call_count == 1
|
|
|
|
assert mock_get_info.mock_calls[0][1] == ('namespace', 'collection', '2.0.1')
|
2019-07-09 19:47:25 +00:00
|
|
|
|
|
|
|
|
2019-08-30 01:55:19 +00:00
|
|
|
def test_build_requirement_from_name_multiple_version_results(galaxy_server, monkeypatch):
|
|
|
|
mock_get_versions = MagicMock()
|
|
|
|
mock_get_versions.return_value = ['2.0.0', '2.0.1', '2.0.2', '2.0.3', '2.0.4', '2.0.5']
|
|
|
|
monkeypatch.setattr(galaxy_server, 'get_collection_versions', mock_get_versions)
|
2019-08-28 20:59:34 +00:00
|
|
|
|
2019-07-09 19:47:25 +00:00
|
|
|
actual = collection.CollectionRequirement.from_name('namespace.collection', [galaxy_server], '!=2.0.2',
|
|
|
|
True, True)
|
|
|
|
|
|
|
|
assert actual.namespace == u'namespace'
|
|
|
|
assert actual.name == u'collection'
|
|
|
|
assert actual.b_path is None
|
2019-08-20 21:49:05 +00:00
|
|
|
assert actual.api == galaxy_server
|
2019-07-09 19:47:25 +00:00
|
|
|
assert actual.skip is False
|
|
|
|
assert actual.versions == set([u'2.0.0', u'2.0.1', u'2.0.3', u'2.0.4', u'2.0.5'])
|
|
|
|
assert actual.latest_version == u'2.0.5'
|
|
|
|
assert actual.dependencies is None
|
|
|
|
|
2019-08-30 01:55:19 +00:00
|
|
|
assert mock_get_versions.call_count == 1
|
|
|
|
assert mock_get_versions.mock_calls[0][1] == ('namespace', 'collection')
|
2019-07-09 19:47:25 +00:00
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('versions, requirement, expected_filter, expected_latest', [
|
|
|
|
[['1.0.0', '1.0.1'], '*', ['1.0.0', '1.0.1'], '1.0.1'],
|
|
|
|
[['1.0.0', '1.0.5', '1.1.0'], '>1.0.0,<1.1.0', ['1.0.5'], '1.0.5'],
|
|
|
|
[['1.0.0', '1.0.5', '1.1.0'], '>1.0.0,<=1.0.5', ['1.0.5'], '1.0.5'],
|
|
|
|
[['1.0.0', '1.0.5', '1.1.0'], '>=1.1.0', ['1.1.0'], '1.1.0'],
|
|
|
|
[['1.0.0', '1.0.5', '1.1.0'], '!=1.1.0', ['1.0.0', '1.0.5'], '1.0.5'],
|
|
|
|
[['1.0.0', '1.0.5', '1.1.0'], '==1.0.5', ['1.0.5'], '1.0.5'],
|
|
|
|
[['1.0.0', '1.0.5', '1.1.0'], '1.0.5', ['1.0.5'], '1.0.5'],
|
|
|
|
[['1.0.0', '2.0.0', '3.0.0'], '>=2', ['2.0.0', '3.0.0'], '3.0.0'],
|
|
|
|
])
|
|
|
|
def test_add_collection_requirements(versions, requirement, expected_filter, expected_latest):
|
|
|
|
req = collection.CollectionRequirement('namespace', 'name', None, 'https://galaxy.com', versions, requirement,
|
|
|
|
False)
|
|
|
|
assert req.versions == set(expected_filter)
|
|
|
|
assert req.latest_version == expected_latest
|
|
|
|
|
|
|
|
|
|
|
|
def test_add_collection_requirement_to_unknown_installed_version():
|
|
|
|
req = collection.CollectionRequirement('namespace', 'name', None, 'https://galaxy.com', ['*'], '*', False,
|
|
|
|
skip=True)
|
|
|
|
|
|
|
|
expected = "Cannot meet requirement namespace.name:1.0.0 as it is already installed at version 'unknown'."
|
|
|
|
with pytest.raises(AnsibleError, match=expected):
|
|
|
|
req.add_requirement(str(req), '1.0.0')
|
|
|
|
|
|
|
|
|
|
|
|
def test_add_collection_wildcard_requirement_to_unknown_installed_version():
|
|
|
|
req = collection.CollectionRequirement('namespace', 'name', None, 'https://galaxy.com', ['*'], '*', False,
|
|
|
|
skip=True)
|
|
|
|
req.add_requirement(str(req), '*')
|
|
|
|
|
|
|
|
assert req.versions == set('*')
|
|
|
|
assert req.latest_version == '*'
|
|
|
|
|
|
|
|
|
2019-08-20 21:49:05 +00:00
|
|
|
def test_add_collection_requirement_with_conflict(galaxy_server):
|
2019-07-09 19:47:25 +00:00
|
|
|
expected = "Cannot meet requirement ==1.0.2 for dependency namespace.name from source '%s'. Available versions " \
|
|
|
|
"before last requirement added: 1.0.0, 1.0.1\n" \
|
|
|
|
"Requirements from:\n" \
|
2019-08-20 21:49:05 +00:00
|
|
|
"\tbase - 'namespace.name:==1.0.2'" % galaxy_server.api_server
|
2019-07-09 19:47:25 +00:00
|
|
|
with pytest.raises(AnsibleError, match=expected):
|
2019-08-20 21:49:05 +00:00
|
|
|
collection.CollectionRequirement('namespace', 'name', None, galaxy_server, ['1.0.0', '1.0.1'], '==1.0.2',
|
|
|
|
False)
|
2019-07-09 19:47:25 +00:00
|
|
|
|
|
|
|
|
2019-08-20 21:49:05 +00:00
|
|
|
def test_add_requirement_to_existing_collection_with_conflict(galaxy_server):
|
|
|
|
req = collection.CollectionRequirement('namespace', 'name', None, galaxy_server, ['1.0.0', '1.0.1'], '*', False)
|
2019-07-09 19:47:25 +00:00
|
|
|
|
|
|
|
expected = "Cannot meet dependency requirement 'namespace.name:1.0.2' for collection namespace.collection2 from " \
|
|
|
|
"source '%s'. Available versions before last requirement added: 1.0.0, 1.0.1\n" \
|
|
|
|
"Requirements from:\n" \
|
|
|
|
"\tbase - 'namespace.name:*'\n" \
|
2019-08-20 21:49:05 +00:00
|
|
|
"\tnamespace.collection2 - 'namespace.name:1.0.2'" % galaxy_server.api_server
|
2019-07-09 19:47:25 +00:00
|
|
|
with pytest.raises(AnsibleError, match=re.escape(expected)):
|
|
|
|
req.add_requirement('namespace.collection2', '1.0.2')
|
|
|
|
|
|
|
|
|
|
|
|
def test_add_requirement_to_installed_collection_with_conflict():
|
|
|
|
source = 'https://galaxy.ansible.com'
|
|
|
|
req = collection.CollectionRequirement('namespace', 'name', None, source, ['1.0.0', '1.0.1'], '*', False,
|
|
|
|
skip=True)
|
|
|
|
|
|
|
|
expected = "Cannot meet requirement namespace.name:1.0.2 as it is already installed at version '1.0.1'. " \
|
|
|
|
"Use --force to overwrite"
|
|
|
|
with pytest.raises(AnsibleError, match=re.escape(expected)):
|
|
|
|
req.add_requirement(None, '1.0.2')
|
|
|
|
|
|
|
|
|
|
|
|
def test_add_requirement_to_installed_collection_with_conflict_as_dep():
|
|
|
|
source = 'https://galaxy.ansible.com'
|
|
|
|
req = collection.CollectionRequirement('namespace', 'name', None, source, ['1.0.0', '1.0.1'], '*', False,
|
|
|
|
skip=True)
|
|
|
|
|
|
|
|
expected = "Cannot meet requirement namespace.name:1.0.2 as it is already installed at version '1.0.1'. " \
|
|
|
|
"Use --force-with-deps to overwrite"
|
|
|
|
with pytest.raises(AnsibleError, match=re.escape(expected)):
|
|
|
|
req.add_requirement('namespace.collection2', '1.0.2')
|
|
|
|
|
|
|
|
|
|
|
|
def test_install_skipped_collection(monkeypatch):
|
|
|
|
mock_display = MagicMock()
|
|
|
|
monkeypatch.setattr(Display, 'display', mock_display)
|
|
|
|
|
|
|
|
req = collection.CollectionRequirement('namespace', 'name', None, 'source', ['1.0.0'], '*', False, skip=True)
|
|
|
|
req.install(None, None)
|
|
|
|
|
|
|
|
assert mock_display.call_count == 1
|
|
|
|
assert mock_display.mock_calls[0][1][0] == "Skipping 'namespace.name' as it is already installed"
|
|
|
|
|
|
|
|
|
|
|
|
def test_install_collection(collection_artifact, monkeypatch):
|
|
|
|
mock_display = MagicMock()
|
|
|
|
monkeypatch.setattr(Display, 'display', mock_display)
|
|
|
|
|
|
|
|
collection_tar = collection_artifact[1]
|
|
|
|
output_path = os.path.join(os.path.split(collection_tar)[0], b'output')
|
|
|
|
collection_path = os.path.join(output_path, b'ansible_namespace', b'collection')
|
|
|
|
os.makedirs(os.path.join(collection_path, b'delete_me')) # Create a folder to verify the install cleans out the dir
|
|
|
|
|
|
|
|
temp_path = os.path.join(os.path.split(collection_tar)[0], b'temp')
|
|
|
|
os.makedirs(temp_path)
|
|
|
|
|
|
|
|
req = collection.CollectionRequirement.from_tar(collection_tar, True, True)
|
|
|
|
req.install(to_text(output_path), temp_path)
|
|
|
|
|
|
|
|
# Ensure the temp directory is empty, nothing is left behind
|
|
|
|
assert os.listdir(temp_path) == []
|
|
|
|
|
|
|
|
actual_files = os.listdir(collection_path)
|
|
|
|
actual_files.sort()
|
|
|
|
assert actual_files == [b'FILES.json', b'MANIFEST.json', b'README.md', b'docs', b'playbooks', b'plugins', b'roles']
|
|
|
|
|
|
|
|
assert mock_display.call_count == 1
|
|
|
|
assert mock_display.mock_calls[0][1][0] == "Installing 'ansible_namespace.collection:0.1.0' to '%s'" \
|
|
|
|
% to_text(collection_path)
|
|
|
|
|
|
|
|
|
2019-08-20 21:49:05 +00:00
|
|
|
def test_install_collection_with_download(galaxy_server, collection_artifact, monkeypatch):
|
2019-07-09 19:47:25 +00:00
|
|
|
collection_tar = collection_artifact[1]
|
|
|
|
output_path = os.path.join(os.path.split(collection_tar)[0], b'output')
|
|
|
|
collection_path = os.path.join(output_path, b'ansible_namespace', b'collection')
|
|
|
|
|
|
|
|
mock_display = MagicMock()
|
|
|
|
monkeypatch.setattr(Display, 'display', mock_display)
|
|
|
|
|
|
|
|
mock_download = MagicMock()
|
|
|
|
mock_download.return_value = collection_tar
|
|
|
|
monkeypatch.setattr(collection, '_download_file', mock_download)
|
|
|
|
|
|
|
|
temp_path = os.path.join(os.path.split(collection_tar)[0], b'temp')
|
|
|
|
os.makedirs(temp_path)
|
|
|
|
|
2019-08-30 01:55:19 +00:00
|
|
|
meta = api.CollectionVersionMetadata('ansible_namespace', 'collection', '0.1.0', 'https://downloadme.com',
|
|
|
|
'myhash', {})
|
2019-08-20 21:49:05 +00:00
|
|
|
req = collection.CollectionRequirement('ansible_namespace', 'collection', None, galaxy_server,
|
2019-08-30 01:55:19 +00:00
|
|
|
['0.1.0'], '*', False, metadata=meta)
|
2019-07-09 19:47:25 +00:00
|
|
|
req.install(to_text(output_path), temp_path)
|
|
|
|
|
|
|
|
# Ensure the temp directory is empty, nothing is left behind
|
|
|
|
assert os.listdir(temp_path) == []
|
|
|
|
|
|
|
|
actual_files = os.listdir(collection_path)
|
|
|
|
actual_files.sort()
|
|
|
|
assert actual_files == [b'FILES.json', b'MANIFEST.json', b'README.md', b'docs', b'playbooks', b'plugins', b'roles']
|
|
|
|
|
|
|
|
assert mock_display.call_count == 1
|
|
|
|
assert mock_display.mock_calls[0][1][0] == "Installing 'ansible_namespace.collection:0.1.0' to '%s'" \
|
|
|
|
% to_text(collection_path)
|
|
|
|
|
|
|
|
assert mock_download.call_count == 1
|
|
|
|
assert mock_download.mock_calls[0][1][0] == 'https://downloadme.com'
|
|
|
|
assert mock_download.mock_calls[0][1][1] == temp_path
|
|
|
|
assert mock_download.mock_calls[0][1][2] == 'myhash'
|
|
|
|
assert mock_download.mock_calls[0][1][3] is True
|
|
|
|
|
|
|
|
|
|
|
|
def test_install_collections_from_tar(collection_artifact, monkeypatch):
|
|
|
|
collection_path, collection_tar = collection_artifact
|
|
|
|
temp_path = os.path.split(collection_tar)[0]
|
|
|
|
shutil.rmtree(collection_path)
|
|
|
|
|
|
|
|
mock_display = MagicMock()
|
|
|
|
monkeypatch.setattr(Display, 'display', mock_display)
|
|
|
|
|
|
|
|
collection.install_collections([(to_text(collection_tar), '*', None,)], to_text(temp_path),
|
|
|
|
[u'https://galaxy.ansible.com'], True, False, False, False, False)
|
|
|
|
|
|
|
|
assert os.path.isdir(collection_path)
|
|
|
|
|
|
|
|
actual_files = os.listdir(collection_path)
|
|
|
|
actual_files.sort()
|
|
|
|
assert actual_files == [b'FILES.json', b'MANIFEST.json', b'README.md', b'docs', b'playbooks', b'plugins', b'roles']
|
|
|
|
|
|
|
|
with open(os.path.join(collection_path, b'MANIFEST.json'), 'rb') as manifest_obj:
|
|
|
|
actual_manifest = json.loads(to_text(manifest_obj.read()))
|
|
|
|
|
|
|
|
assert actual_manifest['collection_info']['namespace'] == 'ansible_namespace'
|
|
|
|
assert actual_manifest['collection_info']['name'] == 'collection'
|
|
|
|
assert actual_manifest['collection_info']['version'] == '0.1.0'
|
|
|
|
|
2019-08-22 20:27:28 +00:00
|
|
|
# Filter out the progress cursor display calls.
|
2019-09-13 01:06:18 +00:00
|
|
|
display_msgs = [m[1][0] for m in mock_display.mock_calls if 'newline' not in m[2] and len(m[1]) == 1]
|
2019-08-22 20:27:28 +00:00
|
|
|
assert len(display_msgs) == 3
|
|
|
|
assert display_msgs[0] == "Process install dependency map"
|
|
|
|
assert display_msgs[1] == "Starting collection install process"
|
|
|
|
assert display_msgs[2] == "Installing 'ansible_namespace.collection:0.1.0' to '%s'" % to_text(collection_path)
|
2019-07-09 19:47:25 +00:00
|
|
|
|
|
|
|
|
|
|
|
def test_install_collections_existing_without_force(collection_artifact, monkeypatch):
|
|
|
|
collection_path, collection_tar = collection_artifact
|
|
|
|
temp_path = os.path.split(collection_tar)[0]
|
|
|
|
|
|
|
|
mock_display = MagicMock()
|
|
|
|
monkeypatch.setattr(Display, 'display', mock_display)
|
|
|
|
|
|
|
|
# If we don't delete collection_path it will think the original build skeleton is installed so we expect a skip
|
|
|
|
collection.install_collections([(to_text(collection_tar), '*', None,)], to_text(temp_path),
|
|
|
|
[u'https://galaxy.ansible.com'], True, False, False, False, False)
|
|
|
|
|
|
|
|
assert os.path.isdir(collection_path)
|
|
|
|
|
|
|
|
actual_files = os.listdir(collection_path)
|
|
|
|
actual_files.sort()
|
|
|
|
assert actual_files == [b'README.md', b'docs', b'galaxy.yml', b'playbooks', b'plugins', b'roles']
|
|
|
|
|
2019-08-22 20:27:28 +00:00
|
|
|
# Filter out the progress cursor display calls.
|
2019-09-13 01:06:18 +00:00
|
|
|
display_msgs = [m[1][0] for m in mock_display.mock_calls if 'newline' not in m[2] and len(m[1]) == 1]
|
2019-08-22 20:27:28 +00:00
|
|
|
assert len(display_msgs) == 4
|
2019-07-09 19:47:25 +00:00
|
|
|
# Msg1 is the warning about not MANIFEST.json, cannot really check message as it has line breaks which varies based
|
|
|
|
# on the path size
|
2019-08-22 20:27:28 +00:00
|
|
|
assert display_msgs[1] == "Process install dependency map"
|
|
|
|
assert display_msgs[2] == "Starting collection install process"
|
|
|
|
assert display_msgs[3] == "Skipping 'ansible_namespace.collection' as it is already installed"
|
2019-07-09 19:47:25 +00:00
|
|
|
|
|
|
|
|
|
|
|
# Makes sure we don't get stuck in some recursive loop
|
|
|
|
@pytest.mark.parametrize('collection_artifact', [
|
|
|
|
{'ansible_namespace.collection': '>=0.0.1'},
|
|
|
|
], indirect=True)
|
|
|
|
def test_install_collection_with_circular_dependency(collection_artifact, monkeypatch):
|
|
|
|
collection_path, collection_tar = collection_artifact
|
|
|
|
temp_path = os.path.split(collection_tar)[0]
|
|
|
|
shutil.rmtree(collection_path)
|
|
|
|
|
|
|
|
mock_display = MagicMock()
|
|
|
|
monkeypatch.setattr(Display, 'display', mock_display)
|
|
|
|
|
|
|
|
collection.install_collections([(to_text(collection_tar), '*', None,)], to_text(temp_path),
|
|
|
|
[u'https://galaxy.ansible.com'], True, False, False, False, False)
|
|
|
|
|
|
|
|
assert os.path.isdir(collection_path)
|
|
|
|
|
|
|
|
actual_files = os.listdir(collection_path)
|
|
|
|
actual_files.sort()
|
|
|
|
assert actual_files == [b'FILES.json', b'MANIFEST.json', b'README.md', b'docs', b'playbooks', b'plugins', b'roles']
|
|
|
|
|
|
|
|
with open(os.path.join(collection_path, b'MANIFEST.json'), 'rb') as manifest_obj:
|
|
|
|
actual_manifest = json.loads(to_text(manifest_obj.read()))
|
|
|
|
|
|
|
|
assert actual_manifest['collection_info']['namespace'] == 'ansible_namespace'
|
|
|
|
assert actual_manifest['collection_info']['name'] == 'collection'
|
|
|
|
assert actual_manifest['collection_info']['version'] == '0.1.0'
|
|
|
|
|
2019-08-22 20:27:28 +00:00
|
|
|
# Filter out the progress cursor display calls.
|
2019-09-13 01:06:18 +00:00
|
|
|
display_msgs = [m[1][0] for m in mock_display.mock_calls if 'newline' not in m[2] and len(m[1]) == 1]
|
2019-08-22 20:27:28 +00:00
|
|
|
assert len(display_msgs) == 3
|
|
|
|
assert display_msgs[0] == "Process install dependency map"
|
|
|
|
assert display_msgs[1] == "Starting collection install process"
|
|
|
|
assert display_msgs[2] == "Installing 'ansible_namespace.collection:0.1.0' to '%s'" % to_text(collection_path)
|