Compare commits
	
		
			29 Commits
		
	
	
		
			1.0.0
			...
			22467fd1f4
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 22467fd1f4 | |||
| b0b5c04fbb | |||
| ef8c7e3381 | |||
| 8bf5ca3ed2 | |||
| 
						
						
							
						
						6427696ce4
	
				 | 
					
					
						|||
| 
						
						
							
						
						f9f989cf58
	
				 | 
					
					
						|||
| 
						
						
							
						
						f43a697629
	
				 | 
					
					
						|||
| 
						
						
							
						
						e479213a6e
	
				 | 
					
					
						|||
| 
						
						
							
						
						f7f843539d
	
				 | 
					
					
						|||
| 
						
						
							
						
						5180ae639d
	
				 | 
					
					
						|||
| 
						
						
							
						
						9831d31ed8
	
				 | 
					
					
						|||
| 
						
						
							
						
						060d2ad829
	
				 | 
					
					
						|||
| 
						
						
							
						
						3b75e39241
	
				 | 
					
					
						|||
| 79567304dd | |||
| a1669c0b04 | |||
| cfdb264fe5 | |||
| 
						
						
							
						
						534b1427f2
	
				 | 
					
					
						|||
| 
						
						
							
						
						2e9f4b8f06
	
				 | 
					
					
						|||
| 
						
						
							
						
						87c3b71df5
	
				 | 
					
					
						|||
| 
						
						
							
						
						6c4c20baa6
	
				 | 
					
					
						|||
| 3617624623 | |||
| c6ef65d93b | |||
| 
						
						
							
						
						0cbe1e327a
	
				 | 
					
					
						|||
| 
						
						
							
						
						d075adc50d
	
				 | 
					
					
						|||
| 2e3d6bf892 | |||
| d8125f4777 | |||
| 7eb47daace | |||
| 5e74393578 | |||
| ec238fe51b | 
@ -81,7 +81,7 @@ steps:
 | 
			
		||||
        from_secret: HCLOUD_TOKEN
 | 
			
		||||
    commands:
 | 
			
		||||
      - pip install -e .
 | 
			
		||||
      - pip install "ansible>=2.10, <2.11"
 | 
			
		||||
      - pip install "ansible>=2.10, <2.11" netaddr
 | 
			
		||||
      - export INSTANCE_UUID=$(openssl rand -hex 5)
 | 
			
		||||
      - cd integration && molecule test
 | 
			
		||||
depends_on:
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										20
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										20
									
								
								CHANGELOG.md
									
									
									
									
									
								
							@ -7,6 +7,24 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
 | 
			
		||||
 | 
			
		||||
## [Unreleased]
 | 
			
		||||
 | 
			
		||||
## [1.2.1] - 2021-06-02
 | 
			
		||||
 | 
			
		||||
### Fixed
 | 
			
		||||
 | 
			
		||||
- Remove async task handling for network deletion ([#30](https://github.com/ansible-community/molecule-hetznercloud/pull/30), credit @ggggut)
 | 
			
		||||
 | 
			
		||||
## [1.2.0] - 2021-06-02
 | 
			
		||||
 | 
			
		||||
### Added
 | 
			
		||||
 | 
			
		||||
- Allow to create networks during test runs ([#29](https://github.com/ansible-community/molecule-hetznercloud/pull/29), thanks @ggggut!)
 | 
			
		||||
 | 
			
		||||
## [1.1.0] - 2021-03-30
 | 
			
		||||
 | 
			
		||||
## Changed
 | 
			
		||||
 | 
			
		||||
- Relaxed bounds on Molecule to allow all versions less than `v4` ([#27](https://github.com/ansible-community/molecule-hetznercloud/pull/27))
 | 
			
		||||
 | 
			
		||||
## [1.0.0] - 2021-01-06
 | 
			
		||||
 | 
			
		||||
This is a major release with breaking changes for your schema and support for a
 | 
			
		||||
@ -23,7 +41,7 @@ tracker so that we can mention them here.
 | 
			
		||||
 | 
			
		||||
## [0.2.2] - 2020-06-15
 | 
			
		||||
 | 
			
		||||
## Fixed
 | 
			
		||||
### Fixed
 | 
			
		||||
 | 
			
		||||
- Point to an open issue tracker
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										95
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										95
									
								
								README.md
									
									
									
									
									
								
							@ -21,6 +21,22 @@ useful, please support my maintenance work financially through my
 | 
			
		||||
Sponsor profile](https://github.com/sponsors/decentral1se). I do not receive
 | 
			
		||||
any financial support from RedHat or Hetzner Cloud for this work.
 | 
			
		||||
 | 
			
		||||
## Install
 | 
			
		||||
 | 
			
		||||
```bash
 | 
			
		||||
$ pip install molecule-hetznercloud
 | 
			
		||||
```
 | 
			
		||||
 | 
			
		||||
If you're looking for a container approach, see [ansible-community/toolset](https://github.com/ansible-community/toolset).
 | 
			
		||||
 | 
			
		||||
## Upgrade
 | 
			
		||||
 | 
			
		||||
Please see the [CHANGELOG.md](./CHANGELOG.md) for migration guides.
 | 
			
		||||
 | 
			
		||||
```bash
 | 
			
		||||
$ pip install --upgrade molecule-hetznercloud
 | 
			
		||||
```
 | 
			
		||||
 | 
			
		||||
## Usage
 | 
			
		||||
 | 
			
		||||
You need to expose a `HCLOUD_TOKEN` environment variable in your environment.
 | 
			
		||||
@ -31,10 +47,9 @@ Find out more about how to get one of those [over here](https://docs.hetzner.clo
 | 
			
		||||
$ export HCLOUD_TOKEN=mycoolapitoken
 | 
			
		||||
```
 | 
			
		||||
 | 
			
		||||
Then install the required Python package.
 | 
			
		||||
Then create a role using the driver plugin.
 | 
			
		||||
 | 
			
		||||
```bash
 | 
			
		||||
$ pip install molecule-hetznercloud
 | 
			
		||||
$ molecule init role myrolename -d hetznercloud
 | 
			
		||||
```
 | 
			
		||||
 | 
			
		||||
@ -74,16 +89,22 @@ $ export MOLECULE_DEBUG=True  # very verbose, last ditch effort
 | 
			
		||||
 | 
			
		||||
## Volume Handling
 | 
			
		||||
 | 
			
		||||
It is possible to have the driver manage volumes during the test run.
 | 
			
		||||
> **WARNING**: this feature appears to be broke. See [#24](https://github.com/ansible-community/molecule-hetznercloud/issues/24) for more
 | 
			
		||||
 | 
			
		||||
You can add the following stanza to your Molecule configuration to have
 | 
			
		||||
Molecule create this volume for the managed VPS. This volume will be cleaned up
 | 
			
		||||
after use.
 | 
			
		||||
It is possible to have the driver manage volumes during the test run. You can
 | 
			
		||||
add the following stanza to your Molecule configuration to have Molecule create
 | 
			
		||||
this volume for the managed VPS. This volume will be cleaned up after use.
 | 
			
		||||
 | 
			
		||||
```yaml
 | 
			
		||||
volumes:
 | 
			
		||||
platforms:
 | 
			
		||||
  - name: instance
 | 
			
		||||
    server_type: cx11
 | 
			
		||||
    image: debian-10
 | 
			
		||||
    volumes:
 | 
			
		||||
      - name: "molecule-hetznercloud-volume-1-${INSTANCE_UUID}"
 | 
			
		||||
        location: /foo/bar
 | 
			
		||||
      - name: "molecule-hetznercloud-volume-2-${INSTANCE_UUID}"
 | 
			
		||||
        size: 20
 | 
			
		||||
```
 | 
			
		||||
 | 
			
		||||
Supported keys are:
 | 
			
		||||
@ -92,6 +113,48 @@ Supported keys are:
 | 
			
		||||
- **size** (optional, default: `10GB`): size of volume
 | 
			
		||||
- **location** (optional, default: `omitted`): path for volume
 | 
			
		||||
 | 
			
		||||
## Network Creation
 | 
			
		||||
 | 
			
		||||
This Driver is able to generate networks and subnetworks during the test run.
 | 
			
		||||
This can be useful for cluster tests. You can create networks with the
 | 
			
		||||
following snippet:
 | 
			
		||||
 | 
			
		||||
```yaml
 | 
			
		||||
platforms:
 | 
			
		||||
  - name: instance1
 | 
			
		||||
    server_type: cx11
 | 
			
		||||
    image: debian-10
 | 
			
		||||
    networks:
 | 
			
		||||
      test-network:
 | 
			
		||||
        ip_range: 10.10.0.0/16
 | 
			
		||||
        subnet:
 | 
			
		||||
          ip: 10.10.10.1/24
 | 
			
		||||
          type: cloud
 | 
			
		||||
          network_zone: eu-central
 | 
			
		||||
      test-network-2:
 | 
			
		||||
        ip_range: 10.20.0.0/16
 | 
			
		||||
        subnet:
 | 
			
		||||
          ip: 10.20.10.1/24
 | 
			
		||||
  - name: instance2
 | 
			
		||||
    server_type: cx11
 | 
			
		||||
    image: debian-10
 | 
			
		||||
    networks:
 | 
			
		||||
      test-network:
 | 
			
		||||
        subnet:
 | 
			
		||||
          ip: 10.10.10.2/24
 | 
			
		||||
```
 | 
			
		||||
 | 
			
		||||
The networks **ip_range** is only important for creating. If you have multiple
 | 
			
		||||
hosts, it is okay to only define **ip_range** once. The supported keys are:
 | 
			
		||||
 | 
			
		||||
- **networks**
 | 
			
		||||
    - **ip_range** (required): ip range of network (usually `/16`)
 | 
			
		||||
 | 
			
		||||
- **subnet**
 | 
			
		||||
    - **ip** (required): ip that should be assigned to host (also generates subnetwork) - prefix mandatory
 | 
			
		||||
    - **type** (optional, default: `cloud`): type of subnetwork
 | 
			
		||||
    - **network_zone** (optional, default: `eu-central`): network zone of subnetwork
 | 
			
		||||
 | 
			
		||||
## Only use `molecule.yml` for configuration
 | 
			
		||||
 | 
			
		||||
It is being worked on that it is possible to remove all the files except the
 | 
			
		||||
@ -120,7 +183,7 @@ See [CHANGELOG.md](./CHANGELOG.md).
 | 
			
		||||
 | 
			
		||||
## Contact
 | 
			
		||||
 | 
			
		||||
- Ping @decentral1se on the `#ansible-molecule` channel on [Freenode](https://webchat.freenode.net).
 | 
			
		||||
- Ping @decentral1se on the `#ansible-molecule` channel on [Libera](https://libera.chat/).
 | 
			
		||||
 | 
			
		||||
## License
 | 
			
		||||
 | 
			
		||||
@ -139,12 +202,12 @@ $ tox -v
 | 
			
		||||
 | 
			
		||||
### Integration
 | 
			
		||||
 | 
			
		||||
Only doable by [Autonomic Cooperative](https://autonomic.zone/) members.
 | 
			
		||||
 | 
			
		||||
```bash
 | 
			
		||||
$ sudo apt install -y direnv
 | 
			
		||||
$ cp .envrc.sample .envrc
 | 
			
		||||
$ direnv allow
 | 
			
		||||
$ pip install -e .
 | 
			
		||||
$ cd integration && molecule test
 | 
			
		||||
```
 | 
			
		||||
git clone https://github.com/ansible-community/molecule-hetznercloud.git
 | 
			
		||||
cd molecule-hetznercloud
 | 
			
		||||
python3 -m venv .venv && source .venv/bin/activate
 | 
			
		||||
pip install -e . "ansible<4" netaddr
 | 
			
		||||
export INSTANCE_UUID=$(openssl rand -hex 5)
 | 
			
		||||
export HCLOUD_TOKEN=YOURKEY
 | 
			
		||||
cd integration && molecule test
 | 
			
		||||
```
 | 
			
		||||
 | 
			
		||||
@ -7,9 +7,21 @@ platforms:
 | 
			
		||||
  - name: "molecule-hetznercloud-${INSTANCE_UUID}"
 | 
			
		||||
    server_type: cx11
 | 
			
		||||
    image: debian-10
 | 
			
		||||
    volumes:
 | 
			
		||||
      - name: "molecule-hetznercloud-volume-1-${INSTANCE_UUID}"
 | 
			
		||||
    # https://github.com/ansible-community/molecule-hetznercloud/issues/24
 | 
			
		||||
    # volumes:
 | 
			
		||||
    # - name: "molecule-hetznercloud-volume-1-${INSTANCE_UUID}"
 | 
			
		||||
    # - name: "molecule-hetznercloud-volume-2-${INSTANCE_UUID}"
 | 
			
		||||
    networks:
 | 
			
		||||
      molecule-hetznercloud-network-1:
 | 
			
		||||
        ip_range: 10.10.0.0/16
 | 
			
		||||
        subnet:
 | 
			
		||||
          ip: 10.10.10.1/24
 | 
			
		||||
          type: cloud
 | 
			
		||||
          network_zone: eu-central
 | 
			
		||||
      molecule-hetznercloud-network-2:
 | 
			
		||||
        ip_range: 10.20.0.0/16
 | 
			
		||||
        subnet:
 | 
			
		||||
          ip: 10.20.10.1/24
 | 
			
		||||
provisioner:
 | 
			
		||||
  name: ansible
 | 
			
		||||
verifier:
 | 
			
		||||
 | 
			
		||||
@ -1,115 +0,0 @@
 | 
			
		||||
---
 | 
			
		||||
{% raw -%}
 | 
			
		||||
- name: Create
 | 
			
		||||
  hosts: localhost
 | 
			
		||||
  connection: local
 | 
			
		||||
  gather_facts: false
 | 
			
		||||
  no_log: "{{ molecule_no_log }}"
 | 
			
		||||
  vars:
 | 
			
		||||
    ssh_port: 22
 | 
			
		||||
    ssh_user: root
 | 
			
		||||
    ssh_path: "{{ lookup('env', 'MOLECULE_EPHEMERAL_DIRECTORY') }}/ssh_key"
 | 
			
		||||
  tasks:
 | 
			
		||||
    - name: Create SSH key
 | 
			
		||||
      openssh_keypair:
 | 
			
		||||
        path: "{{ ssh_path }}"
 | 
			
		||||
        force: true
 | 
			
		||||
      register: generated_ssh_key
 | 
			
		||||
 | 
			
		||||
    - name: Register the SSH key name
 | 
			
		||||
      set_fact:
 | 
			
		||||
        ssh_key_name: "molecule-generated-{{ 12345 | random | to_uuid }}"
 | 
			
		||||
 | 
			
		||||
    - name: Register SSH key for test instance(s)
 | 
			
		||||
      hcloud_ssh_key:
 | 
			
		||||
        name: "{{ ssh_key_name }}"
 | 
			
		||||
        public_key: "{{ generated_ssh_key.public_key }}"
 | 
			
		||||
        api_token: "{{ lookup('env', 'HCLOUD_TOKEN') }}"
 | 
			
		||||
        state: present
 | 
			
		||||
 | 
			
		||||
    - name: Create molecule instance(s)
 | 
			
		||||
      hcloud_server:
 | 
			
		||||
        name: "{{ item.name }}"
 | 
			
		||||
        server_type: "{{ item.server_type }}"
 | 
			
		||||
        ssh_keys:
 | 
			
		||||
          - "{{ ssh_key_name }}"
 | 
			
		||||
        image: "{{ item.image }}"
 | 
			
		||||
        location: "{{ item.location | default(omit) }}"
 | 
			
		||||
        datacenter: "{{ item.datacenter | default(omit) }}"
 | 
			
		||||
        user_data: "{{ item.user_data | default(omit) }}"
 | 
			
		||||
        api_token: "{{ lookup('env', 'HCLOUD_TOKEN') }}"
 | 
			
		||||
        state: present
 | 
			
		||||
      register: server
 | 
			
		||||
      with_items: "{{ molecule_yml.platforms }}"
 | 
			
		||||
      async: 7200
 | 
			
		||||
      poll: 0
 | 
			
		||||
 | 
			
		||||
    - name: Wait for instance(s) creation to complete
 | 
			
		||||
      async_status:
 | 
			
		||||
        jid: "{{ item.ansible_job_id }}"
 | 
			
		||||
      register: hetzner_jobs
 | 
			
		||||
      until: hetzner_jobs.finished
 | 
			
		||||
      retries: 300
 | 
			
		||||
      with_items: "{{ server.results }}"
 | 
			
		||||
 | 
			
		||||
    - name: Create volume(s)
 | 
			
		||||
      hcloud_volume:
 | 
			
		||||
        name: "{{ item.volumes.name | default(item.name) }}"
 | 
			
		||||
        server: "{{ item.name }}"
 | 
			
		||||
        location: "{{ item.volumes.location | default(omit) }}"
 | 
			
		||||
        size: "{{ item.volumes.size | default(10) }}"
 | 
			
		||||
        api_token: "{{ lookup('env', 'HCLOUD_TOKEN') }}"
 | 
			
		||||
        state: "present"
 | 
			
		||||
      with_items: "{{ molecule_yml.platforms }}"
 | 
			
		||||
      when:
 | 
			
		||||
        - item.volumes is defined
 | 
			
		||||
        - item.volumes.create | default(False) | bool
 | 
			
		||||
      register: volumes
 | 
			
		||||
      async: 7200
 | 
			
		||||
      poll: 0
 | 
			
		||||
 | 
			
		||||
    - name: Wait for volume(s) creation to complete
 | 
			
		||||
      async_status:
 | 
			
		||||
        jid: "{{ item.ansible_job_id }}"
 | 
			
		||||
      register: hetzner_volumes
 | 
			
		||||
      until: hetzner_volumes.finished
 | 
			
		||||
      retries: 300
 | 
			
		||||
      when: volumes.changed
 | 
			
		||||
      with_items: "{{ volumes.results }}"
 | 
			
		||||
 | 
			
		||||
    - name: Populate instance config dict
 | 
			
		||||
      set_fact:
 | 
			
		||||
        instance_conf_dict: {
 | 
			
		||||
          'instance': "{{ item.hcloud_server.name }}",
 | 
			
		||||
          'ssh_key_name': "{{ ssh_key_name }}",
 | 
			
		||||
          'address': "{{ item.hcloud_server.ipv4_address }}",
 | 
			
		||||
          'user': "{{ ssh_user }}",
 | 
			
		||||
          'port': "{{ ssh_port }}",
 | 
			
		||||
          'identity_file': "{{ ssh_path }}",
 | 
			
		||||
          'volumes': "{{ item.item.item.volumes | default({}) }}", }
 | 
			
		||||
      with_items: "{{ hetzner_jobs.results }}"
 | 
			
		||||
      register: instance_config_dict
 | 
			
		||||
      when: server.changed | bool
 | 
			
		||||
 | 
			
		||||
    - name: Convert instance config dict to a list
 | 
			
		||||
      set_fact:
 | 
			
		||||
        instance_conf: "{{ instance_config_dict.results | map(attribute='ansible_facts.instance_conf_dict') | list }}"
 | 
			
		||||
      when: server.changed | bool
 | 
			
		||||
 | 
			
		||||
    - name: Dump instance config
 | 
			
		||||
      copy:
 | 
			
		||||
        content: |
 | 
			
		||||
          # Molecule managed
 | 
			
		||||
 | 
			
		||||
          {{ instance_conf | to_json | from_json | to_yaml }}
 | 
			
		||||
        dest: "{{ molecule_instance_config }}"
 | 
			
		||||
      when: server.changed | bool
 | 
			
		||||
 | 
			
		||||
    - name: Wait for SSH
 | 
			
		||||
      wait_for:
 | 
			
		||||
        port: "{{ ssh_port }}"
 | 
			
		||||
        host: "{{ item.address }}"
 | 
			
		||||
        search_regex: SSH
 | 
			
		||||
        delay: 10
 | 
			
		||||
      with_items: "{{ lookup('file', molecule_instance_config) | from_yaml }}"
 | 
			
		||||
{%- endraw %}
 | 
			
		||||
@ -1,84 +0,0 @@
 | 
			
		||||
---
 | 
			
		||||
{% raw -%}
 | 
			
		||||
- name: Destroy
 | 
			
		||||
  hosts: localhost
 | 
			
		||||
  connection: local
 | 
			
		||||
  gather_facts: false
 | 
			
		||||
  no_log: "{{ molecule_no_log }}"
 | 
			
		||||
  tasks:
 | 
			
		||||
    - name: Populate the instance config
 | 
			
		||||
      block:
 | 
			
		||||
        - name: Populate instance config from file
 | 
			
		||||
          set_fact:
 | 
			
		||||
            instance_conf: "{{ lookup('file', molecule_instance_config) | from_yaml }}"
 | 
			
		||||
            skip_instances: false
 | 
			
		||||
      rescue:
 | 
			
		||||
        - name: Populate instance config when file missing
 | 
			
		||||
          set_fact:
 | 
			
		||||
            instance_conf: {}
 | 
			
		||||
            skip_instances: true
 | 
			
		||||
 | 
			
		||||
    - name: Destroy molecule instance(s)
 | 
			
		||||
      hcloud_server:
 | 
			
		||||
        name: "{{ item.instance }}"
 | 
			
		||||
        api_token: "{{ lookup('env', 'HCLOUD_TOKEN') }}"
 | 
			
		||||
        state: absent
 | 
			
		||||
      register: server
 | 
			
		||||
      with_items: "{{ instance_conf }}"
 | 
			
		||||
      when: not skip_instances
 | 
			
		||||
      async: 7200
 | 
			
		||||
      poll: 0
 | 
			
		||||
 | 
			
		||||
    - name: Wait for instance(s) deletion to complete
 | 
			
		||||
      async_status:
 | 
			
		||||
        jid: "{{ item.ansible_job_id }}"
 | 
			
		||||
      register: hetzner_jobs
 | 
			
		||||
      until: hetzner_jobs.finished
 | 
			
		||||
      retries: 300
 | 
			
		||||
      with_items: "{{ server.results }}"
 | 
			
		||||
 | 
			
		||||
    - name: Destroy volume(s)
 | 
			
		||||
      hcloud_volume:
 | 
			
		||||
        name: "{{ item.volumes.name | default(item.instance) }}"
 | 
			
		||||
        server: "{{ item.instance }}"
 | 
			
		||||
        api_token: "{{ lookup('env', 'HCLOUD_TOKEN') }}"
 | 
			
		||||
        state: "absent"
 | 
			
		||||
      register: volumes
 | 
			
		||||
      with_items: "{{ instance_conf }}"
 | 
			
		||||
      when:
 | 
			
		||||
        - item.volumes is defined
 | 
			
		||||
        - item.volumes.create | default(False) | bool
 | 
			
		||||
      async: 7200
 | 
			
		||||
      poll: 0
 | 
			
		||||
 | 
			
		||||
    - name: Wait for volume(s) deletion to complete
 | 
			
		||||
      async_status:
 | 
			
		||||
        jid: "{{ item.ansible_job_id }}"
 | 
			
		||||
      register: hetzner_volumes
 | 
			
		||||
      until: hetzner_volumes.finished
 | 
			
		||||
      retries: 300
 | 
			
		||||
      when: volumes.changed
 | 
			
		||||
      with_items: "{{ volumes.results }}"
 | 
			
		||||
 | 
			
		||||
    - name: Remove registered SSH key
 | 
			
		||||
      hcloud_ssh_key:
 | 
			
		||||
        name: "{{ instance_conf[0].ssh_key_name }}"
 | 
			
		||||
        api_token: "{{ lookup('env', 'HCLOUD_TOKEN') }}"
 | 
			
		||||
        state: absent
 | 
			
		||||
      when:
 | 
			
		||||
        - not skip_instances
 | 
			
		||||
        - instance_conf | length  # must contain at least one instance
 | 
			
		||||
 | 
			
		||||
    - name: Populate instance config
 | 
			
		||||
      set_fact:
 | 
			
		||||
        instance_conf: {}
 | 
			
		||||
 | 
			
		||||
    - name: Dump instance config
 | 
			
		||||
      copy:
 | 
			
		||||
        content: |
 | 
			
		||||
          # Molecule managed
 | 
			
		||||
 | 
			
		||||
          {{ instance_conf | to_json | from_json | to_yaml }}
 | 
			
		||||
        dest: "{{ molecule_instance_config }}"
 | 
			
		||||
      when: server.changed | bool
 | 
			
		||||
{%- endraw %}
 | 
			
		||||
@ -7,6 +7,19 @@ platforms:
 | 
			
		||||
  - name: "{{ cookiecutter.role_name }}"
 | 
			
		||||
    server_type: cx11
 | 
			
		||||
    image: debian-10
 | 
			
		||||
    volumes:
 | 
			
		||||
      - name: "molecule-hetznercloud-volume-1"
 | 
			
		||||
    networks:
 | 
			
		||||
      molecule-hetznercloud-network-1:
 | 
			
		||||
        ip_range: 10.10.0.0/16
 | 
			
		||||
        subnet:
 | 
			
		||||
          ip: 10.10.10.1/24
 | 
			
		||||
          type: cloud
 | 
			
		||||
          network_zone: eu-central
 | 
			
		||||
      molecule-hetznercloud-network-2:
 | 
			
		||||
        ip_range: 10.20.0.0/16
 | 
			
		||||
        subnet:
 | 
			
		||||
          ip: 10.20.10.1/24
 | 
			
		||||
provisioner:
 | 
			
		||||
  name: ansible
 | 
			
		||||
lint: |
 | 
			
		||||
 | 
			
		||||
@ -75,6 +75,48 @@
 | 
			
		||||
        - volumes.changed
 | 
			
		||||
      with_items: "{{ volumes.results }}"
 | 
			
		||||
 | 
			
		||||
    - name: Create private network(s)
 | 
			
		||||
      hcloud_network:
 | 
			
		||||
        name: "{{ item.name }}"
 | 
			
		||||
        ip_range: "{{ item.ip_range | default(omit) }}"
 | 
			
		||||
        api_token: "{{ lookup('env', 'HCLOUD_TOKEN') }}"
 | 
			
		||||
        state: "present"
 | 
			
		||||
      loop: "{{ molecule_yml.platforms|molecule_get_hetznercloud_networks('networks') }}"
 | 
			
		||||
      register: networks
 | 
			
		||||
      async: 7200
 | 
			
		||||
      poll: 0
 | 
			
		||||
 | 
			
		||||
    - name: Wait for network(s) creation to complete
 | 
			
		||||
      async_status:
 | 
			
		||||
        jid: "{{ item.ansible_job_id }}"
 | 
			
		||||
      register: hetzner_networks
 | 
			
		||||
      until: hetzner_networks.finished
 | 
			
		||||
      retries: 300
 | 
			
		||||
      when:
 | 
			
		||||
        - networks is defined
 | 
			
		||||
        - networks.changed
 | 
			
		||||
      with_items: "{{ networks.results }}"
 | 
			
		||||
 | 
			
		||||
    - name: Create private subnetwork(s)
 | 
			
		||||
      hcloud_subnetwork:
 | 
			
		||||
        network: "{{ item.network_name }}"
 | 
			
		||||
        ip_range: "{{ item.ip|ipaddr('network/prefix') }}"
 | 
			
		||||
        network_zone: "{{ item.network_zone | default('eu-central') }}"
 | 
			
		||||
        type: "{{ item.type | default('cloud') }}"
 | 
			
		||||
        api_token: "{{ lookup('env', 'HCLOUD_TOKEN') }}"
 | 
			
		||||
        state: "present"
 | 
			
		||||
      loop: "{{ molecule_yml.platforms|molecule_get_hetznercloud_networks('subnetworks') }}"
 | 
			
		||||
      register: subnetworks
 | 
			
		||||
 | 
			
		||||
    - name: Attach Server to Subnetwork(s)
 | 
			
		||||
      hcloud_server_network:
 | 
			
		||||
        network: "{{ item.network_name }}"
 | 
			
		||||
        server: "{{ item.server_name }}"
 | 
			
		||||
        ip: "{{ item.ip|ipaddr('address') }}"
 | 
			
		||||
        api_token: "{{ lookup('env', 'HCLOUD_TOKEN') }}"
 | 
			
		||||
        state: "present"
 | 
			
		||||
      loop: "{{ molecule_yml.platforms|molecule_get_hetznercloud_networks('subnetworks') }}"
 | 
			
		||||
 | 
			
		||||
    - name: Populate instance config dict
 | 
			
		||||
      set_fact:
 | 
			
		||||
        instance_conf_dict:
 | 
			
		||||
@ -86,6 +128,7 @@
 | 
			
		||||
            "port": "{{ ssh_port }}",
 | 
			
		||||
            "identity_file": "{{ ssh_path }}",
 | 
			
		||||
            "volumes": "{{ item.item.item.volumes | default({}) }}",
 | 
			
		||||
            "networks": "{{ item.item.item.networks | default({}) | dict2items(key_name='name', value_name='data') }}",
 | 
			
		||||
          }
 | 
			
		||||
      with_items: "{{ hetzner_jobs.results }}"
 | 
			
		||||
      register: instance_config_dict
 | 
			
		||||
 | 
			
		||||
@ -6,16 +6,16 @@
 | 
			
		||||
  no_log: "{{ molecule_no_log }}"
 | 
			
		||||
  tasks:
 | 
			
		||||
    - name: Populate the instance config
 | 
			
		||||
      block:
 | 
			
		||||
        - name: Populate instance config from file
 | 
			
		||||
      set_fact:
 | 
			
		||||
            instance_conf: "{{ lookup('file', molecule_instance_config) | from_yaml }}"
 | 
			
		||||
        instance_conf: "{{ lookup('file', molecule_instance_config, errors='warn') | from_yaml }}"
 | 
			
		||||
        skip_instances: false
 | 
			
		||||
      rescue:
 | 
			
		||||
      register: instance_config_lookup
 | 
			
		||||
 | 
			
		||||
    - name: Populate instance config when file missing
 | 
			
		||||
      set_fact:
 | 
			
		||||
        instance_conf: {}
 | 
			
		||||
        skip_instances: true
 | 
			
		||||
      when: not instance_config_lookup.ansible_facts.instance_conf
 | 
			
		||||
 | 
			
		||||
    - name: Destroy molecule instance(s)
 | 
			
		||||
      hcloud_server:
 | 
			
		||||
@ -56,6 +56,14 @@
 | 
			
		||||
      when: volumes.changed
 | 
			
		||||
      with_items: "{{ volumes.results }}"
 | 
			
		||||
 | 
			
		||||
    - name: Destroy network(s)
 | 
			
		||||
      hcloud_network:
 | 
			
		||||
        name: "{{ item.1.name }}"
 | 
			
		||||
        api_token: "{{ lookup('env', 'HCLOUD_TOKEN') }}"
 | 
			
		||||
        state: absent
 | 
			
		||||
      register: networks
 | 
			
		||||
      loop: "{{ instance_conf|subelements('networks', skip_missing=True) }}"
 | 
			
		||||
 | 
			
		||||
    - name: Remove registered SSH key
 | 
			
		||||
      hcloud_ssh_key:
 | 
			
		||||
        name: "{{ instance_conf[0].ssh_key_name }}"
 | 
			
		||||
 | 
			
		||||
@ -0,0 +1,54 @@
 | 
			
		||||
#!/usr/bin/env python
 | 
			
		||||
"""Usage:"""
 | 
			
		||||
""" loop: "{{ molecule_yml.platforms|molecule_get_hetznercloud_networks('networks') }}" """  # noqa
 | 
			
		||||
""" loop: "{{ molecule_yml.platforms|molecule_get_hetznercloud_networks('subnetworks') }}" """  # noqa
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def merge_two_dicts(x, y):
 | 
			
		||||
    z = x.copy()
 | 
			
		||||
    z.update(y)
 | 
			
		||||
    return z
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def get_hetznercloud_networks(data, request):
 | 
			
		||||
    network_list = {}
 | 
			
		||||
    subnetwork_list = []
 | 
			
		||||
 | 
			
		||||
    if request == "networks":
 | 
			
		||||
 | 
			
		||||
        for platform in data:
 | 
			
		||||
            if "networks" in platform:
 | 
			
		||||
                for network_name, values in platform["networks"].items():
 | 
			
		||||
                    del values["subnet"]
 | 
			
		||||
                    values["name"] = network_name
 | 
			
		||||
                    if network_name in network_list:
 | 
			
		||||
                        network_list[network_name] = merge_two_dicts(
 | 
			
		||||
                            network_list[network_name], values
 | 
			
		||||
                        )
 | 
			
		||||
                    else:
 | 
			
		||||
                        network_list[network_name] = values
 | 
			
		||||
 | 
			
		||||
        return [x for x in network_list.values()]
 | 
			
		||||
 | 
			
		||||
    elif request == "subnetworks":
 | 
			
		||||
 | 
			
		||||
        for platform in data:
 | 
			
		||||
            name = platform["name"]
 | 
			
		||||
            if "networks" in platform:
 | 
			
		||||
                for network_name, values in platform["networks"].items():
 | 
			
		||||
                    values["name"] = network_name
 | 
			
		||||
                    if "subnet" in values:
 | 
			
		||||
                        values["subnet"]["server_name"] = name
 | 
			
		||||
                        values["subnet"]["network_name"] = network_name
 | 
			
		||||
                        subnetwork_list.append(values["subnet"])
 | 
			
		||||
 | 
			
		||||
        return subnetwork_list
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class FilterModule(object):
 | 
			
		||||
    """Core Molecule filter plugins."""
 | 
			
		||||
 | 
			
		||||
    def filters(self):
 | 
			
		||||
        return {
 | 
			
		||||
            "molecule_get_hetznercloud_networks": get_hetznercloud_networks,
 | 
			
		||||
        }
 | 
			
		||||
@ -76,6 +76,48 @@
 | 
			
		||||
      when: volumes.changed
 | 
			
		||||
      with_items: "{{ volumes.results }}"
 | 
			
		||||
 | 
			
		||||
    - name: Create private network(s)
 | 
			
		||||
      hcloud_network:
 | 
			
		||||
        name: "{{ item.name }}"
 | 
			
		||||
        ip_range: "{{ item.ip_range | default(omit) }}"
 | 
			
		||||
        api_token: "{{ lookup('env', 'HCLOUD_TOKEN') }}"
 | 
			
		||||
        state: "present"
 | 
			
		||||
      loop: "{{ molecule_yml.platforms|molecule_get_hetznercloud_networks('networks') }}"
 | 
			
		||||
      register: networks
 | 
			
		||||
      async: 7200
 | 
			
		||||
      poll: 0
 | 
			
		||||
 | 
			
		||||
    - name: Wait for network(s) creation to complete
 | 
			
		||||
      async_status:
 | 
			
		||||
        jid: "{{ item.ansible_job_id }}"
 | 
			
		||||
      register: hetzner_networks
 | 
			
		||||
      until: hetzner_networks.finished
 | 
			
		||||
      retries: 300
 | 
			
		||||
      when:
 | 
			
		||||
        - networks is defined
 | 
			
		||||
        - networks.changed
 | 
			
		||||
      with_items: "{{ networks.results }}"
 | 
			
		||||
 | 
			
		||||
    - name: Create private subnetwork(s)
 | 
			
		||||
      hcloud_subnetwork:
 | 
			
		||||
        network: "{{ item.network_name }}"
 | 
			
		||||
        ip_range: "{{ item.ip|ipaddr('network/prefix') }}"
 | 
			
		||||
        network_zone: "{{ item.network_zone | default('eu-central') }}"
 | 
			
		||||
        type: "{{ item.type | default('cloud') }}"
 | 
			
		||||
        api_token: "{{ lookup('env', 'HCLOUD_TOKEN') }}"
 | 
			
		||||
        state: "present"
 | 
			
		||||
      loop: "{{ molecule_yml.platforms|molecule_get_hetznercloud_networks('subnetworks') }}"
 | 
			
		||||
      register: subnetworks
 | 
			
		||||
 | 
			
		||||
    - name: Attach Server to Subnetwork(s)
 | 
			
		||||
      hcloud_server_network:
 | 
			
		||||
        network: "{{ item.network_name }}"
 | 
			
		||||
        server: "{{ item.server_name }}"
 | 
			
		||||
        ip: "{{ item.ip|ipaddr('address') }}"
 | 
			
		||||
        api_token: "{{ lookup('env', 'HCLOUD_TOKEN') }}"
 | 
			
		||||
        state: "present"
 | 
			
		||||
      loop: "{{ molecule_yml.platforms|molecule_get_hetznercloud_networks('subnetworks') }}"
 | 
			
		||||
 | 
			
		||||
    - name: Populate instance config dict
 | 
			
		||||
      set_fact:
 | 
			
		||||
        instance_conf_dict: {
 | 
			
		||||
@ -85,7 +127,8 @@
 | 
			
		||||
          'user': "{{ ssh_user }}",
 | 
			
		||||
          'port': "{{ ssh_port }}",
 | 
			
		||||
          'identity_file': "{{ ssh_path }}",
 | 
			
		||||
          'volumes': "{{ item.item.item.volumes | default({}) }}", }
 | 
			
		||||
          'volumes': "{{ item.item.item.volumes | default({}) }}",
 | 
			
		||||
          'networks': "{{ item.item.item.networks | default({}) | dict2items(key_name='name', value_name='data') }}", }
 | 
			
		||||
      with_items: "{{ hetzner_jobs.results }}"
 | 
			
		||||
      register: instance_config_dict
 | 
			
		||||
      when: server.changed | bool
 | 
			
		||||
 | 
			
		||||
@ -59,6 +59,14 @@
 | 
			
		||||
      when: volumes.changed
 | 
			
		||||
      with_items: "{{ volumes.results }}"
 | 
			
		||||
 | 
			
		||||
    - name: Destroy network(s)
 | 
			
		||||
      hcloud_network:
 | 
			
		||||
        name: "{{ item.1.name }}"
 | 
			
		||||
        api_token: "{{ lookup('env', 'HCLOUD_TOKEN') }}"
 | 
			
		||||
        state: absent
 | 
			
		||||
      register: networks
 | 
			
		||||
      loop: "{{ instance_conf|subelements('networks', skip_missing=True) }}"
 | 
			
		||||
 | 
			
		||||
    - name: Remove registered SSH key
 | 
			
		||||
      hcloud_ssh_key:
 | 
			
		||||
        name: "{{ instance_conf[0].ssh_key_name }}"
 | 
			
		||||
 | 
			
		||||
		Reference in New Issue
	
	Block a user