Depuis quelque temps, je travaille avec la plateforme de cloud Outscale qui
propose une API proche de celle d’AWS. Cela permet entre autre d’utiliser la cli
d’AWS. Mais voilà, je dois développer des rôles ansible sur cette plateforme et
le driver EC2 ne fonctionne pas avec Outscale. La raison les clés des réponses
ne sont pas identiques à celle d’AWS. Heureusement Outscale fournit un provider
Terraform. Oui, mais comment l’utiliser avec molecule ?
Driver delegated molecule
Malheureusement, la documentation du driver delegated de molecule ne contient
que de quelques lignes. D’où l’importance de bien documenter ses productions.
Je me suis retroussé les manches, fait des recherches sur github, sans succès.
Aucun des codes trouvés ne gère le process complet de molecule : du create au
destroy …
J’ai décidé de lancer la commande pour voir ce qu’y est créé par défaut avec le
driver delegated :
molecule init role --driver-name delegated --verifier-name ansible steph.test_role
INFO Initializing new role test_role...
No config file found ; using defaults
- Role test_role was created successfully
[WARNING]: No inventory was parsed, only implicit localhost is available
localhost | CHANGED = > { " backup " : "" , " changed " : true , " msg " : " line added " }
Regardons ce qui est produit :
cd test_role/molecule/default
-rw-rw-r-- 1 vagrant vagrant 171 mars 6 13:29 converge.yml
-rw-rw-r-- 1 vagrant vagrant 1114 mars 6 13:29 create.yml
-rw-rw-r-- 1 vagrant vagrant 599 mars 6 13:29 destroy.yml
-rw-rw-r-- 1 vagrant vagrant 336 mars 6 13:29 INSTALL.rst
-rw-rw-r-- 1 vagrant vagrant 142 mars 6 13:29 molecule.yml
-rw-rw-r-- 1 vagrant vagrant 193 mars 6 13:29 verify.yml
C’est en ouvrant le fichier create.yml
que le déclic est venu.
no_log : " {{ molecule_no_log }} "
# TODO: Developer must implement and populate 'server' variable
- when : server.changed | default(false) | bool
- name : Populate instance config dict
ansible.builtin.set_fact :
' identity_file ' : " {{ }} " , }
with_items : " {{ server.results }} "
register : instance_config_dict
- name : Convert instance config dict to a list
ansible.builtin.set_fact :
instance_conf : " {{ instance_config_dict.results | map(attribute='ansible_facts.instance_conf_dict') | list }} "
- name : Dump instance config
{{ instance_conf | to_json | from_json | to_yaml }}
dest : " {{ molecule_instance_config }} "
En lançant la commande molecule --debug create
j’ai compris que le fichier
instance_config.yml
stockait les informations des instances. Mais quoi mettre
dedans ? En fait le minimum de données est indiqué dans la structure ci-dessous,
qui se trouve dans le fichier create.yml
:
'identity_file' : " {{ }} " ,
Si on regarde ces champs, on voit qu’il y a toutes les informations permettant
de lancer un playbook ansible.
Donc parti de ce constat, j’ai construit un template ansible
terraform
qui en
sortie retournerait ces informations (peut-être trop).
Pour le moment, je fais au plus simple et j’ajouterai par la suite des options comme :
ajouter de règles de groupes de sécurité
ajouter d’autres tags que name
limiter les connexions à mon ip seule
…
Le code :
source = " outscale/outscale "
version = " {{ platform.provider_version }} "
region = " {{ platform.region }} "
access_key_id = " {{ platform.accessid }} "
secret_key_id = " {{ platform.secretid }} "
default = " {{ platform.instance_type }} "
default = " {{ platform.region }} "
default = " {{ platform.image_name }} "
variable "instance_name" {
default = " {{ platform.name }} "
default = " {{ platform.ssh_user }} "
default = " {{ platform.ssh_port }} "
variable "authorized_range_ip" {
default = {{ platform.authorized_range_ip | to_json }}
variable "private_key_path" {
default = " {{ platform.private_key_path}} "
variable "public_key_path" {
default = " {{ platform.public_key_path}} "
data "outscale_image" "image" {
values = [ var . image_name ]
resource "outscale_keypair" "ssh-keypair" {
keypair_name = format ( " molecule-%s-%s " ,var . instance_name ,var . run_name )
public_key = file (var . public_key_path )
resource "outscale_security_group" "security_group_molecule" {
security_group_name = format ( " molecule-%s-%s " ,var . instance_name ,var . run_name )
resource "outscale_security_group_rule" "security_group_molecule_ssh" {
security_group_id = outscale_security_group . security_group_molecule . security_group_id
from_port_range = var . ssh_port
to_port_range = var . ssh_port
ip_ranges = var . authorized_range_ip
resource "outscale_vm" "my_vm" {
image_id = data . outscale_image . image . id
keypair_name = outscale_keypair . ssh-keypair . keypair_name
security_group_ids = [ outscale_security_group . security_group_molecule . security_group_id ]
value = format ( " molecule-%s-%s " ,var . instance_name ,var . run_name )
resource "outscale_public_ip" "my_public_ip" {
value = format ( " molecule-%s-%s " ,var . instance_name ,var . run_name )
resource "outscale_public_ip_link" "my_public_ip_link" {
vm_id = outscale_vm . my_vm . vm_id
public_ip = outscale_public_ip . my_public_ip . public_ip
value = outscale_public_ip . my_public_ip . public_ip
value = outscale_vm . my_vm . tags
value = var . instance_name
value = var . private_key_path
Ce fichier template, je l’ai déposé dans le répertoire
molecule/default/templates/
sous le nom main.tf.j2
.
Pour rappel, il est possible de créer plusieurs plateformes dans le fichier de
config molecule :
image_name : " Ubuntu-22.04-2022.12.06-0 "
image_name : " RockyLinux-9-2022.12.06-0 "
Donc j’ai créé modifié le fichier create.yml pour qu’il puisse créer plusieurs
instances. Ce qui donne :
no_log : " {{ molecule_no_log }} "
default_assign_public_ip : true
default_accessid : " {{ lookup('env', 'OUTSCALE_ACCESSKEYID') }} "
default_secretid : " {{ lookup('env', 'OUTSCALE_SECRETKEYID') }} "
default_instance_type : t2.small
default_private_key_path : " {{ lookup('env', 'MOLECULE_EPHEMERAL_DIRECTORY') }}/id_ed25519 "
default_public_key_path : " {{ default_private_key_path }}.pub "
default_ssh_user : outscale
default_region : " eu-west-2 "
default_provider_version : " >= 0.8.2 "
default_authorized_range_ip : [ " 0.0.0.0/0 " ]
authorized_range_ip : " {{ default_authorized_range_ip }} "
provider_version : " {{ default_provider_version }} "
assign_public_ip : " {{ default_assign_public_ip }} "
accessid : " {{ default_accessid }} "
secretid : " {{ default_secretid }} "
instance_type : " {{ default_instance_type }} "
private_key_path : " {{ default_private_key_path }} "
public_key_path : " {{ default_public_key_path }} "
ssh_user : " {{ default_ssh_user }} "
ssh_port : " {{ default_ssh_port }} "
region : " {{ default_region }} "
# Merging defaults into a list of dicts is, it turns out, not straightforward
{{ [platform_defaults | dict2items]
| product(molecule_yml.platforms | map('dict2items') | list)
| map('flatten', levels=1)
ansible.builtin.include_tasks : terraform-create.yml
- name : Test provisionned vm connection
- name : Test SSH Port available
ansible.builtin.wait_for :
with_items : " {{ groups['molecule_hosts'] }} "
Dans la première partie, je définis des valeurs par défaut qui sont surchargés
par la suite avec celles définies dans le fichier molecule.yml
. Le premier play
de ce fichier ne fait que boucler sur les plateformes et lance un sous-playbook
terraform-create.yml
dont voici le contenu :
ansible.builtin.set_fact :
run_id : " {{ lookup('password', '/dev/null chars=ascii_lowercase length=5') }} "
- name : Generate local key pairs
community.crypto.openssh_keypair :
path : " {{ platform.private_key_path }} "
- name : Create Terraform directory
path : " {{ lookup('env', 'MOLECULE_EPHEMERAL_DIRECTORY') }}/terraform/{{ run_id }} "
- name : Create Terraform file from template
ansible.builtin.template :
src : templates/main.tf.j2
dest : " {{ lookup('env', 'MOLECULE_EPHEMERAL_DIRECTORY') }}/terraform/{{ run_id }}/main.tf "
- name : Provision VM {{ platform.name }}
community.general.terraform :
project_path : " {{ lookup('env', 'MOLECULE_EPHEMERAL_DIRECTORY') }}/terraform/{{ run_id }} "
- name : Create instance config file
path : " {{ lookup('env', 'MOLECULE_EPHEMERAL_DIRECTORY') }}/instance_config.yml "
- name : Register instance config for VM {{ platform.name }}
ansible.builtin.blockinfile :
path : " {{ lookup('env', 'MOLECULE_EPHEMERAL_DIRECTORY') }}/instance_config.yml "
block : " - { address: {{ state.outputs.my_public_ip.value }} , identity_file: {{ platform.private_key_path }}, instance: {{ platform.name }}, port: {{ platform.ssh_port }}, user: {{ platform.ssh_user }}, workspace: {{ run_id }} } "
marker : ' # {mark} Instance : {{ platform.name }} '
- name : Add to group molecule_hosts {{ platform.name }}
ansible.builtin.add_host :
name : " {{ state.outputs.my_public_ip.value }} "
Je crée un répertoire avec un nom unique, qui stockera chaque fichier d’instance
terraform. Je dépose le rendu du template dans ce répertoire et je lance le
module terraform . En sortie, je récupère les infos nécessaires pour le fichier
instance_config.yml
. J’utilise blockinfile
pour créer chacun des blocks. Et
pour finir, j’ajoute la VM à un groupe ansible dynamique
.
En sortie de boucle un second play permettant d’attendre que le port 22 des VM
soit bien démarré.
Tests des commandes molecule login et converge
Allez, on provisionne deux machines (il faut être patient 2 minutes) :
INFO Running default > list
INFO Running docker > list
Instance Name │ Driver Name │ Provisioner Name │ Scenario Name │ Created │ Converged
╶───────────────┼─────────────┼──────────────────┼───────────────┼─────────┼───────────╴
Ubuntu │ delegated │ ansible │ default │ false │ false
Rocky │ delegated │ ansible │ default │ false │ false
TASK [Create] ******************************************************************
included: /home/vagrant/Projets/personal/ansible/roles/pyenv/molecule/default/terraform-create.yml for localhost = > (item=(censored due to no_log ))
included: /home/vagrant/Projets/personal/ansible/roles/pyenv/molecule/default/terraform-create.yml for localhost = > (item=(censored due to no_log ))
TASK [Set run_id] **************************************************************
TASK [Generate local key pairs] ************************************************
TASK [Create Terraform directory] **********************************************
TASK [Create Terraform file from template] *************************************
TASK [Provision VM Ubuntu] *****************************************************
TASK [Create instance config file] *********************************************
TASK [Register instance config for VM Ubuntu] **********************************
TASK [Add to group molecule_hosts Ubuntu] **************************************
TASK [Set run_id] **************************************************************
TASK [Generate local key pairs] ************************************************
TASK [Create Terraform directory] **********************************************
TASK [Create Terraform file from template] *************************************
TASK [Provision VM Rocky] ******************************************************
TASK [Create instance config file] *********************************************
TASK [Register instance config for VM Rocky] ***********************************
TASK [Add to group molecule_hosts Rocky] ***************************************
PLAY [Test provisionned vm connection] *****************************************
TASK [Test SSH Port available] *************************************************
ok: [localhost] = > ( item = 142.44.35.234 )
ok: [localhost] = > ( item = 142.44.41.97 )
PLAY RECAP *********************************************************************
localhost : ok= 19 changed= 13 unreachable= 0 failed= 0 skipped= 0 rescued= 0 ignored= 0
INFO Running default > prepare
WARNING Skipping, prepare playbook not configured.
Petit tour dans le fichier /home/vagrant/.cache/molecule/pyenv/default/instance_config.yml
.
# BEGIN Instance : Ubuntu
- { address : 142.44.35.234 , identity_file : /home/vagrant/.cache/molecule/pyenv/default/id_ed25519 , instance : Ubuntu , port : 22 , user : outscale , workspace : osuyx }
- { address : 142.44.41.97 , identity_file : /home/vagrant/.cache/molecule/pyenv/default/id_ed25519 , instance : Rocky , port : 22 , user : outscale , workspace : chzsi }
Vous remarquez juste que j’ai juste ajouté une clé indiquant ou se trouve le
fichier state de terraform. Cela va nous servir pour la destruction.
Test de connexion sur la machine Ubuntu :
molecule login --host Ubuntu
INFO Running default > login
Welcome to Ubuntu 22.04.1 LTS (GNU/Linux 5.15.0-56-generic x86_64 )
* Documentation: https://help.ubuntu.com
* Management: https://landscape.canonical.com
* Support: https://ubuntu.com/advantage
System information as of Mon Mar 6 13:10:54 UTC 2023
System load: 0.587890625 Processes: 103
Usage of /: 16.7% of 9.51GB Users logged in: 0
Memory usage: 9% IPv4 address for eth0: 10.9.8.135
0 updates can be applied immediately.
The list of available updates is more than a week old.
To check for new updates run: sudo apt update
The programs included with the Ubuntu system are free software ;
the exact distribution terms for each program are described in the
individual files in /usr/share/doc/ * /copyright.
Ubuntu comes with ABSOLUTELY NO WARRANTY, to the extent permitted by
-bash: warning: setlocale: LC_ALL: cannot change locale (fr_FR.UTF-8)
To run a command as administrator (user " root " ), use " sudo <command> " .
See " man sudo_root " for details.
outscale@ip-10-9-8-135:~$
Un petit dans le cockpit outscale :
Cela fonctionne. Pour tester converge, j’ai ajouté trois taches au rôle
installant seulement un package :
- name : Update cache on Ubuntu
allow_unauthenticated : false
when : ansible_os_family == "Debian"
- name : Install a package for test
On lance le converge :
INFO default scenario test matrix: dependency, create, prepare, converge
INFO Performing prerun with role_name_check=0...
INFO Set ANSIBLE_LIBRARY=/home/vagrant/.cache/ansible-compat/4489a2/modules:/home/vagrant/.ansible/plugins/modules:/usr/share/ansible/plugins/modules
INFO Set ANSIBLE_COLLECTIONS_PATH=/home/vagrant/.cache/ansible-compat/4489a2/collections:/home/vagrant/.ansible/collections:/usr/share/ansible/collections
INFO Set ANSIBLE_ROLES_PATH=/home/vagrant/.cache/ansible-compat/4489a2/roles:/home/vagrant/.ansible/roles:/usr/share/ansible/roles:/etc/ansible/roles
INFO Using /home/vagrant/.cache/ansible-compat/4489a2/roles/steph.pyenv symlink to current repository in order to enable Ansible to find the role using its expected full name.
INFO Running default > dependency
WARNING Skipping, missing the requirements file.
WARNING Skipping, missing the requirements file.
INFO Running default > create
WARNING Skipping, instances already created.
INFO Running default > prepare
WARNING Skipping, prepare playbook not configured.
INFO Running default > converge
PLAY [Converge] ****************************************************************
TASK [Include pyenv] ***********************************************************
TASK [pyenv : Get Facts] *******************************************************
TASK [pyenv : Update cache on Ubuntu] ******************************************
TASK [pyenv : Install a package for test] **************************************
PLAY RECAP *********************************************************************
Rocky : ok= 2 changed= 1 unreachable= 0 failed= 0 skipped= 1 rescued= 0 ignored= 0
Ubuntu : ok= 3 changed= 2 unreachable= 0 failed= 0 skipped= 0 rescued= 0 ignored= 0
Et la destruction comment cela se passe t’il ?
Destruction des VM
Pour la destruction le principe est le même sauf que j’ai plus besoin du
template. En fait dans le fichier instance_config.yml
j’ai stocké le dossier
où se trouve le state de terraform. Le code du fichier destroy.yml
:
no_log : " {{ molecule_no_log }} "
# Developer must implement.
# Mandatory configuration for Molecule to function.
- name : Check instance config file exist
path : " {{ molecule_instance_config }} "
when : file_exist.stat.exists
- name : Load Instance config File
ansible.builtin.set_fact :
instance_conf : " {{ lookup('file', molecule_instance_config) | from_yaml }} "
ansible.builtin.include_tasks : terraform-destroy.yml
loop : ' {{ instance_conf }} '
On reprend le même principe, on boucle sur les platforms pour lancer un
sous-playbook terraform-destroy.yml
dont voici le contenu :
community.general.terraform :
project_path : " {{ lookup('env', 'MOLECULE_EPHEMERAL_DIRECTORY') }}/terraform/{{ platform.workspace }} "
- name : Destroy Directory
path : " {{ lookup('env', 'MOLECULE_EPHEMERAL_DIRECTORY') }}/terraform/{{ platform.workspace }} "
Rien de bien compliqué non ?
On lance la destruction :
INFO default scenario test matrix: dependency, cleanup, destroy
INFO Performing prerun with role_name_check=0...
INFO Set ANSIBLE_LIBRARY=/home/vagrant/.cache/ansible-compat/4489a2/modules:/home/vagrant/.ansible/plugins/modules:/usr/share/ansible/plugins/modules
INFO Set ANSIBLE_COLLECTIONS_PATH=/home/vagrant/.cache/ansible-compat/4489a2/collections:/home/vagrant/.ansible/collections:/usr/share/ansible/collections
INFO Set ANSIBLE_ROLES_PATH=/home/vagrant/.cache/ansible-compat/4489a2/roles:/home/vagrant/.ansible/roles:/usr/share/ansible/roles:/etc/ansible/roles
INFO Using /home/vagrant/.cache/ansible-compat/4489a2/roles/steph.pyenv symlink to current repository in order to enable Ansible to find the role using its expected full name.
INFO Running default > dependency
WARNING Skipping, missing the requirements file.
WARNING Skipping, missing the requirements file.
INFO Running default > cleanup
WARNING Skipping, cleanup playbook not configured.
INFO Running default > destroy
PLAY [Destroy] *****************************************************************
TASK [Check instance config file exist] ****************************************
TASK [Load Instance config File] ***********************************************
TASK [Destroy] *****************************************************************
included: /home/vagrant/Projets/personal/ansible/roles/pyenv/molecule/default/terraform-destroy.yml for localhost = > (item=(censored due to no_log ))
included: /home/vagrant/Projets/personal/ansible/roles/pyenv/molecule/default/terraform-destroy.yml for localhost = > (item=(censored due to no_log ))
TASK [Destroy VM] **************************************************************
TASK [Destroy Directory] *******************************************************
TASK [Destroy VM] **************************************************************
TASK [Destroy Directory] *******************************************************
PLAY RECAP *********************************************************************
localhost : ok= 8 changed= 4 unreachable= 0 failed= 0 skipped= 0 rescued= 0 ignored= 0
INFO Pruning extra files from scenario ephemeral directory
Petit tour dans le cockpit. Tout est rincé.
Plus loin
On voit que finalement le driver delegated
de molecule ne demande que la
création d’un seul fichier instance_config.yml
contenant les informations de
connexion ssh. Cela veut dire que l’on utilise toutes sortes de produits pour
provisionner ces machines, il faut juste en sortie récupérer les informations
nécessaires. Je lis pas mal d’articles qui mettent en opposition terraform
et ansible
, mais moi, j’aime à dire qu’ils sont complémentaires.
Vous pouvez modifier le code pour vos besoins en changeant de provider terraform
: gcp, ec2, …
Je vais publier le code d’ici aux prochains jours et vous pourrez m’aider à le
compléter avec d’autres templates par exemple.