Move metalsmith_instances from tripleo-ansible
Change-Id: I02407547154d8d6084fa30c5fe164c5b6a060043
This commit is contained in:
		| @@ -1,3 +1,4 @@ | ||||
| ansible==2.8.12 | ||||
| appdirs==1.4.3 | ||||
| certifi==2020.4.5.1 | ||||
| cffi==1.14.0 | ||||
|   | ||||
							
								
								
									
										252
									
								
								metalsmith/test/test_metalsmith_instances.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										252
									
								
								metalsmith/test/test_metalsmith_instances.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,252 @@ | ||||
| # Copyright 2019 Red Hat, Inc. | ||||
| # All Rights Reserved. | ||||
| # | ||||
| #    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 unittest | ||||
| from unittest import mock | ||||
|  | ||||
| from metalsmith_ansible.ansible_plugins.modules \ | ||||
|     import metalsmith_instances as mi | ||||
|  | ||||
| from metalsmith import exceptions as exc | ||||
|  | ||||
|  | ||||
| class TestMetalsmithInstances(unittest.TestCase): | ||||
|  | ||||
|     @mock.patch('metalsmith.sources.detect', autospec=True) | ||||
|     def test_get_source(self, mock_detect): | ||||
|         mi._get_source({ | ||||
|             'image': {'href': 'overcloud-full'} | ||||
|         }) | ||||
|         mi._get_source({ | ||||
|             'image': { | ||||
|                 'href': 'file://overcloud-full.qcow2', | ||||
|                 'checksum': 'asdf', | ||||
|                 'kernel': 'file://overcloud-full.vmlinuz', | ||||
|                 'ramdisk': 'file://overcloud-full.initrd' | ||||
|             } | ||||
|         }) | ||||
|         mock_detect.assert_has_calls([ | ||||
|             mock.call( | ||||
|                 image='overcloud-full', | ||||
|                 checksum=None, | ||||
|                 kernel=None, | ||||
|                 ramdisk=None | ||||
|             ), | ||||
|             mock.call( | ||||
|                 image='file://overcloud-full.qcow2', | ||||
|                 checksum='asdf', | ||||
|                 kernel='file://overcloud-full.vmlinuz', | ||||
|                 ramdisk='file://overcloud-full.initrd' | ||||
|             ) | ||||
|         ]) | ||||
|  | ||||
|     def test_reserve(self): | ||||
|         provisioner = mock.Mock() | ||||
|         instances = [{ | ||||
|             'name': 'node', | ||||
|             'resource_class': 'boxen', | ||||
|             'capabilities': {'foo': 'bar'}, | ||||
|             'traits': ['this', 'that'], | ||||
|             'conductor_group': 'group' | ||||
|         }, {}] | ||||
|         reserved = [ | ||||
|             mock.Mock(id=1), | ||||
|             mock.Mock(id=2), | ||||
|         ] | ||||
|  | ||||
|         # test reserve success | ||||
|         provisioner.reserve_node.side_effect = reserved | ||||
|  | ||||
|         result = mi.reserve(provisioner, instances, True) | ||||
|         provisioner.reserve_node.assert_has_calls([ | ||||
|             mock.call( | ||||
|                 candidates=['node'], | ||||
|                 capabilities={'foo': 'bar'}, | ||||
|                 conductor_group='group', | ||||
|                 resource_class='boxen', | ||||
|                 traits=['this', 'that'] | ||||
|             ), | ||||
|             mock.call( | ||||
|                 candidates=None, | ||||
|                 capabilities=None, | ||||
|                 conductor_group=None, | ||||
|                 resource_class='baremetal', | ||||
|                 traits=None | ||||
|             ) | ||||
|         ]) | ||||
|         self.assertTrue(result[0]) | ||||
|         self.assertEqual(reserved, result[1]) | ||||
|  | ||||
|         # test reserve failure with cleanup | ||||
|         instances = [{}, {}, {}] | ||||
|         reserved = [ | ||||
|             mock.Mock(id=1), | ||||
|             mock.Mock(id=2), | ||||
|             exc.ReservationFailed('ouch') | ||||
|         ] | ||||
|         provisioner.reserve_node.side_effect = reserved | ||||
|         self.assertRaises(exc.ReservationFailed, mi.reserve, | ||||
|                           provisioner, instances, True) | ||||
|         provisioner.unprovision_node.assert_has_calls([ | ||||
|             mock.call(1), | ||||
|             mock.call(2) | ||||
|         ]) | ||||
|  | ||||
|     @mock.patch('metalsmith.sources.detect', autospec=True) | ||||
|     @mock.patch('metalsmith.instance_config.CloudInitConfig', autospec=True) | ||||
|     def test_provision(self, mock_config, mock_detect): | ||||
|         config = mock_config.return_value | ||||
|         image = mock_detect.return_value | ||||
|  | ||||
|         provisioner = mock.Mock() | ||||
|         instances = [{ | ||||
|             'name': 'node-1', | ||||
|             'hostname': 'overcloud-controller-1', | ||||
|             'image': {'href': 'overcloud-full'} | ||||
|         }, { | ||||
|             'name': 'node-2', | ||||
|             'hostname': 'overcloud-controller-2', | ||||
|             'image': {'href': 'overcloud-full'}, | ||||
|             'nics': {'network': 'ctlplane'}, | ||||
|             'root_size_gb': 200, | ||||
|             'swap_size_mb': 16, | ||||
|             'netboot': True, | ||||
|             'ssh_public_keys': 'abcd', | ||||
|             'user_name': 'centos', | ||||
|             'passwordless_sudo': False | ||||
|         }, { | ||||
|             'name': 'node-3', | ||||
|             'hostname': 'overcloud-controller-3', | ||||
|             'image': {'href': 'overcloud-full'} | ||||
|         }, { | ||||
|             'name': 'node-4', | ||||
|             'hostname': 'overcloud-compute-0', | ||||
|             'image': {'href': 'overcloud-full'} | ||||
|         }] | ||||
|         provisioned = [ | ||||
|             mock.Mock(uuid=1), | ||||
|             mock.Mock(uuid=2), | ||||
|             mock.Mock(uuid=3), | ||||
|             mock.Mock(uuid=4), | ||||
|         ] | ||||
|  | ||||
|         # test provision success | ||||
|         provisioner.provision_node.side_effect = provisioned | ||||
|  | ||||
|         # provision 4 nodes with concurrency of 2 | ||||
|         result = mi.provision(provisioner, instances, 3600, 2, True, True) | ||||
|         provisioner.provision_node.assert_has_calls([ | ||||
|             mock.call( | ||||
|                 'node-1', | ||||
|                 config=config, | ||||
|                 hostname='overcloud-controller-1', | ||||
|                 image=image, | ||||
|                 netboot=False, | ||||
|                 nics=None, | ||||
|                 root_size_gb=None, | ||||
|                 swap_size_mb=None | ||||
|             ), | ||||
|             mock.call( | ||||
|                 'node-2', | ||||
|                 config=config, | ||||
|                 hostname='overcloud-controller-2', | ||||
|                 image=image, | ||||
|                 netboot=True, | ||||
|                 nics={'network': 'ctlplane'}, | ||||
|                 root_size_gb=200, | ||||
|                 swap_size_mb=16 | ||||
|             ), | ||||
|             mock.call( | ||||
|                 'node-3', | ||||
|                 config=config, | ||||
|                 hostname='overcloud-controller-3', | ||||
|                 image=image, | ||||
|                 netboot=False, | ||||
|                 nics=None, | ||||
|                 root_size_gb=None, | ||||
|                 swap_size_mb=None | ||||
|             ), | ||||
|             mock.call( | ||||
|                 'node-4', | ||||
|                 config=config, | ||||
|                 hostname='overcloud-compute-0', | ||||
|                 image=image, | ||||
|                 netboot=False, | ||||
|                 nics=None, | ||||
|                 root_size_gb=None, | ||||
|                 swap_size_mb=None | ||||
|             ), | ||||
|         ]) | ||||
|         mock_config.assert_has_calls([ | ||||
|             mock.call(ssh_keys=None), | ||||
|             mock.call(ssh_keys='abcd'), | ||||
|         ]) | ||||
|         config.add_user.assert_called_once_with( | ||||
|             'centos', admin=True, sudo=False) | ||||
|         mock_detect.assert_has_calls([ | ||||
|             mock.call( | ||||
|                 image='overcloud-full', | ||||
|                 checksum=None, | ||||
|                 kernel=None, | ||||
|                 ramdisk=None | ||||
|             ), | ||||
|             mock.call( | ||||
|                 image='overcloud-full', | ||||
|                 checksum=None, | ||||
|                 kernel=None, | ||||
|                 ramdisk=None | ||||
|             ), | ||||
|             mock.call( | ||||
|                 image='overcloud-full', | ||||
|                 checksum=None, | ||||
|                 kernel=None, | ||||
|                 ramdisk=None | ||||
|             ), | ||||
|             mock.call( | ||||
|                 image='overcloud-full', | ||||
|                 checksum=None, | ||||
|                 kernel=None, | ||||
|                 ramdisk=None | ||||
|             ), | ||||
|         ]) | ||||
|         self.assertTrue(result[0]) | ||||
|         self.assertEqual(provisioned, result[1]) | ||||
|  | ||||
|         # test provision failure with cleanup | ||||
|         instances = [{ | ||||
|             'name': 'node-1', | ||||
|             'hostname': 'overcloud-controller-1', | ||||
|             'image': {'href': 'overcloud-full'} | ||||
|         }, { | ||||
|             'name': 'node-2', | ||||
|             'hostname': 'overcloud-controller-2', | ||||
|             'image': {'href': 'overcloud-full'}, | ||||
|         }, { | ||||
|             'name': 'node-3', | ||||
|             'hostname': 'overcloud-controller-3', | ||||
|             'image': {'href': 'overcloud-full'}, | ||||
|         }] | ||||
|         provisioned = [ | ||||
|             mock.Mock(uuid=1), | ||||
|             mock.Mock(uuid=2), | ||||
|             exc.Error('ouch') | ||||
|         ] | ||||
|         provisioner.provision_node.side_effect = provisioned | ||||
|         self.assertRaises(exc.Error, mi.provision, | ||||
|                           provisioner, instances, 3600, 20, True, True) | ||||
|         provisioner.unprovision_node.assert_has_calls([ | ||||
|             mock.call(1), | ||||
|             mock.call(2) | ||||
|         ]) | ||||
| @@ -0,0 +1,368 @@ | ||||
| #!/usr/bin/python | ||||
| # Copyright 2020 Red Hat, Inc. | ||||
| # All Rights Reserved. | ||||
| # | ||||
| # 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. | ||||
|  | ||||
| from __future__ import absolute_import | ||||
| __metaclass__ = type | ||||
|  | ||||
| from concurrent import futures | ||||
|  | ||||
| from ansible.module_utils.basic import AnsibleModule | ||||
| from ansible.module_utils.openstack import openstack_cloud_from_module | ||||
| from ansible.module_utils.openstack import openstack_full_argument_spec | ||||
| from ansible.module_utils.openstack import openstack_module_kwargs | ||||
|  | ||||
| import metalsmith | ||||
| from metalsmith import instance_config | ||||
| from metalsmith import sources | ||||
|  | ||||
| import yaml | ||||
|  | ||||
|  | ||||
| ANSIBLE_METADATA = { | ||||
|     'metadata_version': '1.1', | ||||
|     'status': ['preview'], | ||||
|     'supported_by': 'community' | ||||
| } | ||||
|  | ||||
|  | ||||
| DOCUMENTATION = ''' | ||||
| --- | ||||
| module: metalsmith_instances | ||||
| short_description: Manage baremetal instances with metalsmith | ||||
| version_added: "2.9" | ||||
| author: "Steve Baker (@stevebaker)" | ||||
| description: | ||||
|   - Provision and unprovision ironic baremetal instances using metalsmith, | ||||
|     which is a a simple tool to provision bare metal machines using | ||||
|     OpenStack Bare Metal Service (ironic) and, optionally, OpenStack | ||||
|     Image Service (glance) and OpenStack Networking Service (neutron). | ||||
| options: | ||||
|   instances: | ||||
|     description: | ||||
|       - List of node description dicts to perform operations on | ||||
|     type: list | ||||
|     default: [] | ||||
|     elements: dict | ||||
|     suboptions: | ||||
|       hostname: | ||||
|         description: | ||||
|           - Host name to use, defaults to Node's name or UUID | ||||
|         type: str | ||||
|       name: | ||||
|         description: | ||||
|           - The name of an existing node to provision | ||||
|         type: str | ||||
|       image: | ||||
|         description: | ||||
|           - Details of the image you want to provision onto the node | ||||
|         type: dict | ||||
|         required: True | ||||
|         suboptions: | ||||
|           href: | ||||
|             description: | ||||
|               - Image to use (name, UUID or URL) | ||||
|             type: str | ||||
|             required: True | ||||
|           checksum : | ||||
|             description: | ||||
|               - Image MD5 checksum or URL with checksums | ||||
|             type: str | ||||
|           kernel: | ||||
|             description: | ||||
|               - URL of the image's kernel | ||||
|             type: str | ||||
|           ramdisk: | ||||
|             description: | ||||
|               - URL of the image's ramdisk | ||||
|             type: str | ||||
|       nics: | ||||
|         description: | ||||
|           - List of requested NICs | ||||
|         type: list | ||||
|         elements: dict | ||||
|         suboptions: | ||||
|           network: | ||||
|             description: | ||||
|               - Network to create a port on (name or UUID) | ||||
|           subnet: | ||||
|             description: | ||||
|               - Subnet to create a port on (name or UUID) | ||||
|           port: | ||||
|             description: | ||||
|               - Port to attach (name or UUID) | ||||
|           fixed_ip: | ||||
|             description: | ||||
|               - Attach IP from the network | ||||
|  | ||||
|       netboot: | ||||
|         description: | ||||
|           - Boot from network instead of local disk | ||||
|         default: no | ||||
|         type: bool | ||||
|       root_size_gb: | ||||
|         description: | ||||
|           - Root partition size (in GiB), defaults to (local_gb - 1) | ||||
|         type: int | ||||
|       swap_size_mb: | ||||
|         description: | ||||
|           - Swap partition size (in MiB), defaults to no swap | ||||
|         type: int | ||||
|       capabilities: | ||||
|         description: | ||||
|           - Selection criteria to match the node capabilities | ||||
|         type: dict | ||||
|       traits: | ||||
|         description: | ||||
|           - Traits the node should have | ||||
|         type: list | ||||
|         elements: str | ||||
|       ssh_public_keys: | ||||
|         description: | ||||
|           - SSH public keys to load | ||||
|         type: str | ||||
|       resource_class: | ||||
|         description: | ||||
|           - Node resource class to provision | ||||
|         type: str | ||||
|         default: baremetal | ||||
|       conductor_group: | ||||
|         description: | ||||
|           - Conductor group to pick the node from | ||||
|         type: str | ||||
|       user_name: | ||||
|         description: | ||||
|           - Name of the admin user to create | ||||
|         type: str | ||||
|       passwordless_sudo: | ||||
|         description: | ||||
|           - Allow password-less sudo for the user | ||||
|         default: yes | ||||
|         type: bool | ||||
|   clean_up: | ||||
|     description: | ||||
|       - Clean up resources on failure | ||||
|     default: yes | ||||
|     type: bool | ||||
|   state: | ||||
|     description: | ||||
|       - Desired provision state, "present" to provision, | ||||
|         "absent" to unprovision, "reserved" to create an allocation | ||||
|         record without changing the node state | ||||
|     default: present | ||||
|     choices: | ||||
|     - present | ||||
|     - absent | ||||
|     - reserved | ||||
|   wait: | ||||
|     description: | ||||
|       - A boolean value instructing the module to wait for node provision | ||||
|         to complete before returning.  A 'yes' is implied if the number of | ||||
|         instances is more than the concurrency. | ||||
|     type: bool | ||||
|     default: no | ||||
|   timeout: | ||||
|     description: | ||||
|       - An integer value representing the number of seconds to wait for the | ||||
|         node provision to complete. | ||||
|     type: int | ||||
|     default: 3660 | ||||
|   concurrency: | ||||
|     description: | ||||
|       - Maximum number of instances to provision at once. Set to 0 to have no | ||||
|         concurrency limit | ||||
|     type: int | ||||
| ''' | ||||
|  | ||||
|  | ||||
| def _get_source(instance): | ||||
|     image = instance.get('image') | ||||
|     return sources.detect(image=image.get('href'), | ||||
|                           kernel=image.get('kernel'), | ||||
|                           ramdisk=image.get('ramdisk'), | ||||
|                           checksum=image.get('checksum')) | ||||
|  | ||||
|  | ||||
| def reserve(provisioner, instances, clean_up): | ||||
|     nodes = [] | ||||
|     for instance in instances: | ||||
|         if instance.get('name') is not None: | ||||
|             # NOTE(dtantsur): metalsmith accepts list of instances to pick | ||||
|             # from. We implement a simplest case when a user can pick a | ||||
|             # node by its name (actually, UUID will also work). | ||||
|             candidates = [instance['name']] | ||||
|         else: | ||||
|             candidates = None | ||||
|         try: | ||||
|             node = provisioner.reserve_node( | ||||
|                 resource_class=instance.get('resource_class', 'baremetal'), | ||||
|                 capabilities=instance.get('capabilities'), | ||||
|                 candidates=candidates, | ||||
|                 traits=instance.get('traits'), | ||||
|                 conductor_group=instance.get('conductor_group')), | ||||
|             if isinstance(node, tuple): | ||||
|                 node = node[0] | ||||
|             nodes.append(node) | ||||
|             # side-effect of populating the instance name, which is passed to | ||||
|             # a later provision step | ||||
|             instance['name'] = node.id | ||||
|         except Exception as exc: | ||||
|             if clean_up: | ||||
|                 # Remove all reservations on failure | ||||
|                 _release_nodes(provisioner, [i.id for i in nodes]) | ||||
|             raise exc | ||||
|     return len(nodes) > 0, nodes | ||||
|  | ||||
|  | ||||
| def _release_nodes(provisioner, node_ids): | ||||
|     for node in node_ids: | ||||
|         try: | ||||
|             provisioner.unprovision_node(node) | ||||
|         except Exception: | ||||
|             pass | ||||
|  | ||||
|  | ||||
| def provision(provisioner, instances, timeout, concurrency, clean_up, wait): | ||||
|     if not instances: | ||||
|         return False, [] | ||||
|  | ||||
|     # first, ensure all instances are reserved | ||||
|     reserve(provisioner, [i for i in instances if not i.get('name')], clean_up) | ||||
|  | ||||
|     nodes = [] | ||||
|  | ||||
|     # no limit on concurrency, create a worker for every instance | ||||
|     if concurrency < 1: | ||||
|         concurrency = len(instances) | ||||
|  | ||||
|     # if concurrency is less than instances, need to wait for | ||||
|     # instance completion | ||||
|     if concurrency < len(instances): | ||||
|         wait = True | ||||
|  | ||||
|     provision_jobs = [] | ||||
|     exceptions = [] | ||||
|     with futures.ThreadPoolExecutor(max_workers=concurrency) as p: | ||||
|         for i in instances: | ||||
|             provision_jobs.append(p.submit( | ||||
|                 _provision_instance, provisioner, i, nodes, timeout, wait | ||||
|             )) | ||||
|     for job in futures.as_completed(provision_jobs): | ||||
|         e = job.exception() | ||||
|         if e: | ||||
|             exceptions.append(e) | ||||
|  | ||||
|             if clean_up: | ||||
|                 # first, cancel all jobs | ||||
|                 for job in provision_jobs: | ||||
|                     job.cancel() | ||||
|                 # Unprovision all provisioned so far. | ||||
|                 # This is best-effort as some provision calls may have | ||||
|                 # started but not yet appended to nodes. | ||||
|                 _release_nodes(provisioner, [i.uuid for i in nodes]) | ||||
|                 nodes = [] | ||||
|     if exceptions: | ||||
|         # TODO(sbaker) future enhancement to tolerate a proportion of failures | ||||
|         # so that provisioning and deployment can continue | ||||
|         raise exceptions[0] | ||||
|  | ||||
|     return len(nodes) > 0, nodes | ||||
|  | ||||
|  | ||||
| def _provision_instance(provisioner, instance, nodes, timeout, wait): | ||||
|     name = instance.get('name') | ||||
|  | ||||
|     image = _get_source(instance) | ||||
|     ssh_keys = instance.get('ssh_public_keys') | ||||
|     config = instance_config.CloudInitConfig(ssh_keys=ssh_keys) | ||||
|     if instance.get('user_name'): | ||||
|         config.add_user(instance.get('user_name'), admin=True, | ||||
|                         sudo=instance.get('passwordless_sudo', True)) | ||||
|     node = provisioner.provision_node( | ||||
|         name, | ||||
|         config=config, | ||||
|         hostname=instance.get('hostname'), | ||||
|         image=image, | ||||
|         nics=instance.get('nics'), | ||||
|         root_size_gb=instance.get('root_size_gb'), | ||||
|         swap_size_mb=instance.get('swap_size_mb'), | ||||
|         netboot=instance.get('netboot', False) | ||||
|     ) | ||||
|     nodes.append(node) | ||||
|     if wait: | ||||
|         provisioner.wait_for_provisioning( | ||||
|             [node.uuid], timeout=timeout) | ||||
|  | ||||
|  | ||||
| def unprovision(provisioner, instances): | ||||
|     for instance in instances: | ||||
|         provisioner.unprovision_node(instance.get('name')) | ||||
|     return True | ||||
|  | ||||
|  | ||||
| def main(): | ||||
|     argument_spec = openstack_full_argument_spec( | ||||
|         **yaml.safe_load(DOCUMENTATION)['options'] | ||||
|     ) | ||||
|     module_kwargs = openstack_module_kwargs() | ||||
|     module = AnsibleModule( | ||||
|         argument_spec=argument_spec, | ||||
|         supports_check_mode=False, | ||||
|         **module_kwargs | ||||
|     ) | ||||
|  | ||||
|     sdk, cloud = openstack_cloud_from_module(module) | ||||
|     provisioner = metalsmith.Provisioner(cloud_region=cloud.config) | ||||
|     instances = module.params['instances'] | ||||
|     state = module.params['state'] | ||||
|     concurrency = module.params['concurrency'] | ||||
|     timeout = module.params['timeout'] | ||||
|     wait = module.params['wait'] | ||||
|     clean_up = module.params['clean_up'] | ||||
|  | ||||
|     if state == 'present': | ||||
|         changed, nodes = provision(provisioner, instances, | ||||
|                                    timeout, concurrency, clean_up, | ||||
|                                    wait) | ||||
|         instances = [{ | ||||
|             'name': i.node.name or i.uuid, | ||||
|             'hostname': i.hostname, | ||||
|             'id': i.uuid, | ||||
|         } for i in nodes] | ||||
|         module.exit_json( | ||||
|             changed=changed, | ||||
|             msg="{} instances provisioned".format(len(nodes)), | ||||
|             instances=instances, | ||||
|         ) | ||||
|  | ||||
|     if state == 'reserved': | ||||
|         changed, nodes = reserve(provisioner, instances, clean_up) | ||||
|         module.exit_json( | ||||
|             changed=changed, | ||||
|             msg="{} instances reserved".format(len(nodes)), | ||||
|             ids=[node.id for node in nodes], | ||||
|             instances=instances | ||||
|         ) | ||||
|  | ||||
|     if state == 'absent': | ||||
|         changed = unprovision(provisioner, instances) | ||||
|         module.exit_json( | ||||
|             changed=changed, | ||||
|             msg="{} nodes unprovisioned".format(len(instances)) | ||||
|         ) | ||||
|  | ||||
|  | ||||
| if __name__ == '__main__': | ||||
|     main() | ||||
| @@ -27,6 +27,7 @@ packages = | ||||
|  | ||||
| data_files = | ||||
|     share/ansible/roles/ = metalsmith_ansible/roles/* | ||||
|     share/ansible/plugins/ = metalsmith_ansible/ansible_plugins/* | ||||
|  | ||||
| [entry_points] | ||||
| console_scripts = | ||||
|   | ||||
| @@ -7,3 +7,4 @@ flake8-import-order>=0.17.1 # LGPLv3 | ||||
| hacking>=3.0.0,<3.1.0 # Apache-2.0 | ||||
| stestr>=1.0.0 # Apache-2.0 | ||||
| Pygments>=2.2.0 # BSD | ||||
| ansible>=2.8 | ||||
		Reference in New Issue
	
	Block a user
	 Steve Baker
					Steve Baker