diff --git a/ansible.cfg b/ansible.cfg new file mode 100644 index 0000000..560f2b6 --- /dev/null +++ b/ansible.cfg @@ -0,0 +1,11 @@ +[defaults] +stdout_callback = yaml +callbacks_enabled = timer, profile_tasks, profile_roles +host_key_checking = False +pipelining = True +forks = 30 +deprecation_warnings=False +roles_path = roles + +[ssh_connection] +ssh_args = -o ControlMaster=auto -o ControlPersist=60s \ No newline at end of file diff --git a/ansible/ansible.cfg b/ansible/ansible.cfg index 35367c1..39be919 100644 --- a/ansible/ansible.cfg +++ b/ansible/ansible.cfg @@ -5,6 +5,7 @@ host_key_checking = False pipelining = True forks = 30 deprecation_warnings=False +roles_path = ../roles [ssh_connection] ssh_args = -o ControlMaster=auto -o ControlPersist=60s \ No newline at end of file diff --git a/ansible/fix-homedir-ownership.yml b/ansible/fix-homedir-ownership.yml index 5b08dbf..fa48e94 100644 --- a/ansible/fix-homedir-ownership.yml +++ b/ansible/fix-homedir-ownership.yml @@ -1,6 +1,6 @@ --- - name: Fix Home Directory Ownership - hosts: all + hosts: all, !localhost gather_facts: false vars: # At the time of running this playbook the home directory is not owned by the user. @@ -16,7 +16,7 @@ gather_subset: - user_dir - - name: Ensure homedir is owned by {{ ansible_user }} + - name: Ensure homedir is owned by "{{ ansible_user }}" ansible.builtin.file: dest: "{{ ansible_env.HOME }}" state: directory diff --git a/ansible/vars/defaults.yml b/ansible/vars/defaults.yml index c61cb2d..4655c98 100644 --- a/ansible/vars/defaults.yml +++ b/ansible/vars/defaults.yml @@ -2,12 +2,12 @@ src_directory: "{{ ansible_env.HOME }}/src" kayobe_config_repo: https://github.com/stackhpc/stackhpc-kayobe-config.git -kayobe_config_version: stackhpc/yoga +kayobe_config_version: "{{ kayobe_config_branch | default('stackhpc/yoga')}}" kayobe_config_name: kayobe-config kayobe_config_environment: ci-multinode kayobe_repo: https://github.com/stackhpc/kayobe.git -kayobe_version: stackhpc/yoga +kayobe_version: "{{ kayobe_version_branch | default('stackhpc/yoga') }}" kayobe_name: kayobe openstack_config_repo: https://github.com/stackhpc/openstack-config-multinode @@ -16,9 +16,7 @@ openstack_config_name: openstack-config vault_password_path: "~/vault.password" -ssh_key_path: - -vxlan_vni: +ssh_key_path: "{{ cluster_ssh_private_key_file }}" root_domain: sms-lab.cloud diff --git a/authentication.tf b/authentication.tf index 266e107..0599337 100644 --- a/authentication.tf +++ b/authentication.tf @@ -1,4 +1,4 @@ resource "openstack_compute_keypair_v2" "keypair" { name = var.multinode_keypair - public_key = file(var.ssh_public_key) + public_key = var.ssh_public_key } \ No newline at end of file diff --git a/group_vars/openstack.yml b/group_vars/openstack.yml new file mode 100644 index 0000000..c1d37d6 --- /dev/null +++ b/group_vars/openstack.yml @@ -0,0 +1,26 @@ +# The default Terraform state key for backends that support it +terraform_state_key: "cluster/{{ cluster_id }}/tfstate" + +# Set up the terraform backend +# This setup allows us to use the Consul backend when enabled without any changes +#terraform_backend_type: 'local' +terraform_backend_type: "{{ 'consul' if 'CONSUL_HTTP_ADDR' in ansible_env else 'local' }}" +terraform_backend_config_defaults: + consul: + path: "{{ terraform_state_key }}" + gzip: "true" + local: {} +terraform_backend_config: "{{ terraform_backend_config_defaults[terraform_backend_type] }}" + +# These variables control the location of the Terraform binary +terraform_binary_directory: "{{ playbook_dir }}/bin" +terraform_binary_path: "{{ terraform_binary_directory }}/terraform" + +# This controls the location where the Terraform files are rendered +terraform_project_path: "{{ playbook_dir }}" + +# Indicates whether the Terraform operation is reconciling or removing resources +# Valid values are 'present' and 'absent' +terraform_state: "{{ cluster_state | default('present') }}" + +cluster_ssh_user: "{{ ssh_user }}" diff --git a/multinode-app.yml b/multinode-app.yml new file mode 100644 index 0000000..f231ef6 --- /dev/null +++ b/multinode-app.yml @@ -0,0 +1,66 @@ +--- + +- hosts: localhost + tasks: + - name: Show Playbook Directory + debug: + msg: "{{ playbook_dir }}" + + - name: Template Terraform files into project directory + template: + src: terraform.tfvars.j2 + dest: "{{ playbook_dir }}/terraform.tfvars" + + - name: Template Terraform userdata.cfg.tpl files into project template directory + template: + src: "{{ playbook_dir }}/templates/userdata.cfg.tpl.j2" + dest: "{{ playbook_dir }}/templates/userdata.cfg.tpl" + + +# Provision the infrastructure + +# The CaaS puts hosts for accessing the OpenStack +# API into the 'openstack' group +- hosts: openstack + roles: + - cluster_infra + +- hosts: localhost + tasks: + # Check whether an ans_vlt_pwd variable is defined and if so, save it into a + # file called '~/vault.password'. If it doesn't exist, create a the + # '~/vault.password' file with ans_vlt_pwd = "password_not_set" as the + # password. + - name: Create vault password file + vars: + ans_dflt: 'default_password' + ansible.builtin.copy: + content: "{{ ans_vlt_pwd | default( ans_dflt , true ) }}" + dest: "~/vault.password" + mode: 0600 + +# If openstack_deploy is true then continue if not end the playbook. + +# Import the playbook to start configuring the multi-node hosts. +- name: Configure hosts and deploy ansible + import_playbook: ansible/configure-hosts.yml + when: openstack_deploy == true + + +- hosts: ansible_control + vars: + ansible_pipelining: true + ansible_ssh_pipelining: true + tasks: + - name: Deploy OpenStack. + ansible.builtin.command: + cmd: "bash ~/deploy-openstack.sh" + when: openstack_deploy == true + +# This is to get the ip of the ansible-controller host. +- hosts: localhost + tasks: + - debug: var=outputs + vars: + outputs: + cluster_access_ip: "{{ hostvars[groups['openstack'][0]].cluster_gateway_ip }}" \ No newline at end of file diff --git a/outputs.tf b/outputs.tf index aaec76c..35a0531 100644 --- a/outputs.tf +++ b/outputs.tf @@ -2,6 +2,10 @@ output "ansible_control_access_ip_v4" { value = openstack_compute_instance_v2.ansible_control.access_ip_v4 } +output "cluster_gateway_ip" { + value = openstack_compute_instance_v2.ansible_control.access_ip_v4 +} + output "seed_access_ip_v4" { value = openstack_compute_instance_v2.seed.access_ip_v4 } @@ -75,38 +79,94 @@ resource "local_file" "deploy_openstack" { file_permission = "0755" } -resource "ansible_host" "control_host" { - name = openstack_compute_instance_v2.ansible_control.access_ip_v4 - groups = ["ansible_control"] +output "cluster_nodes" { + description = "A list of the cluster nodes and their IP addresses which will be used by the Ansible inventory" + value = concat( + [ + { + name = openstack_compute_instance_v2.ansible_control.name + ip = openstack_compute_instance_v2.ansible_control.access_ip_v4 + groups = ["ansible_control"] + variables = { + ansible_user = var.ssh_user + } + } + ], + flatten([ + for node in openstack_compute_instance_v2.compute: { + name = node.name + ip = node.access_ip_v4 + groups = ["compute"] + variables = { + ansible_user = var.ssh_user + } + } + ]), + flatten([ + for node in openstack_compute_instance_v2.controller: { + name = node.name + ip = node.access_ip_v4 + groups = ["controllers"] + variables = { + ansible_user = var.ssh_user + } + } + ]), + [{ + name = openstack_compute_instance_v2.seed.name + ip = openstack_compute_instance_v2.seed.access_ip_v4 + groups = ["seed"] + variables = { + ansible_user = var.ssh_user + } + }], + flatten([ + for node in openstack_compute_instance_v2.storage: { + name = node.name + ip = node.access_ip_v4 + groups = ["storage"] + variables = { + ansible_user = var.ssh_user + } + } + ]) + ) } -resource "ansible_host" "compute_host" { - for_each = { for host in openstack_compute_instance_v2.compute : host.name => host.access_ip_v4 } - name = each.value - groups = ["compute"] -} +# Template of all the hosts' configuration which can be used to generate Ansible varables. -resource "ansible_host" "controllers_hosts" { - for_each = { for host in openstack_compute_instance_v2.controller : host.name => host.access_ip_v4 } - name = each.value - groups = ["controllers"] -} +# resource "ansible_host" "control_host" { +# name = openstack_compute_instance_v2.ansible_control.access_ip_v4 +# groups = ["ansible_control"] +# } -resource "ansible_host" "seed_host" { - name = openstack_compute_instance_v2.seed.access_ip_v4 - groups = ["seed"] -} +# resource "ansible_host" "compute_host" { +# for_each = { for host in openstack_compute_instance_v2.compute : host.name => host.access_ip_v4 } +# name = each.value +# groups = ["compute"] +# } -resource "ansible_host" "storage" { - for_each = { for host in openstack_compute_instance_v2.storage : host.name => host.access_ip_v4 } - name = each.value - groups = ["storage"] -} +# resource "ansible_host" "controllers_hosts" { +# for_each = { for host in openstack_compute_instance_v2.controller : host.name => host.access_ip_v4 } +# name = each.value +# groups = ["controllers"] +# } -resource "ansible_group" "cluster_group" { - name = "cluster" - children = ["compute", "ansible_control", "controllers", "seed", "storage"] - variables = { - ansible_user = var.ssh_user - } -} +# resource "ansible_host" "seed_host" { +# name = openstack_compute_instance_v2.seed.access_ip_v4 +# groups = ["seed"] +# } + +# resource "ansible_host" "storage" { +# for_each = { for host in openstack_compute_instance_v2.storage : host.name => host.access_ip_v4 } +# name = each.value +# groups = ["storage"] +# } + +# resource "ansible_group" "cluster_group" { +# name = "cluster" +# children = ["compute", "ansible_control", "controllers", "seed", "storage"] +# variables = { +# ansible_user = var.ssh_user +# } +# } diff --git a/requirements.yml b/requirements.yml new file mode 100644 index 0000000..71a93fe --- /dev/null +++ b/requirements.yml @@ -0,0 +1,9 @@ +--- +collections: + - name: https://github.com/stackhpc/ansible-collection-terraform + type: git + version: 8c7acce4538aab8c0e928972155a2ccb5cb1b2a1 + - name: cloud.terraform + - name: ansible.posix +roles: + - src: mrlesmithjr.manage_lvm \ No newline at end of file diff --git a/roles/cluster_infra/tasks/main.yml b/roles/cluster_infra/tasks/main.yml new file mode 100644 index 0000000..2bcec93 --- /dev/null +++ b/roles/cluster_infra/tasks/main.yml @@ -0,0 +1,42 @@ +--- + +- name: Install Terraform binary + include_role: + name: stackhpc.terraform.install + +- name: Make Terraform project directory + file: + path: "{{ terraform_project_path }}" + state: directory + +- name: Write backend configuration + copy: + content: | + terraform { + backend "{{ terraform_backend_type }}" { } + } + dest: "{{ terraform_project_path }}/backend.tf" + +# Patching in this appliance is implemented as a switch to a new base image +# So unless explicitly patching, we want to use the same image as last time +# To do this, we query the previous Terraform state before updating +- block: + - name: Get previous Terraform state + stackhpc.terraform.terraform_output: + binary_path: "{{ terraform_binary_path }}" + project_path: "{{ terraform_project_path }}" + backend_config: "{{ terraform_backend_config }}" + register: cluster_infra_terraform_output + + - name: Extract image from Terraform state + set_fact: + cluster_previous_image: "{{ cluster_infra_terraform_output.outputs.cluster_image.value }}" + when: '"cluster_image" in cluster_infra_terraform_output.outputs' + when: + - terraform_state == "present" + - cluster_upgrade_system_packages is not defined or not cluster_upgrade_system_packages + + +- name: Provision infrastructure + include_role: + name: stackhpc.terraform.infra \ No newline at end of file diff --git a/roles/requirements.yml b/roles/requirements.yml new file mode 120000 index 0000000..6e76d52 --- /dev/null +++ b/roles/requirements.yml @@ -0,0 +1 @@ +../requirements.yml \ No newline at end of file diff --git a/templates/deploy-openstack.tpl b/templates/deploy-openstack.tpl index 65a2961..9b7a59a 100644 --- a/templates/deploy-openstack.tpl +++ b/templates/deploy-openstack.tpl @@ -131,7 +131,7 @@ fi if [[ "$(sudo docker image ls)" == *"kayobe"* ]]; then echo "Image already exists skipping docker build" else - sudo DOCKER_BUILDKIT=1 docker build --network host --build-arg BASE_IMAGE=$$BASE_IMAGE --file $${config_directories[kayobe]}/.automation/docker/kayobe/Dockerfile --tag kayobe:latest $${config_directories[kayobe]} + sudo DOCKER_BUILDKIT=1 docker build --network host --build-arg BASE_IMAGE=$BASE_IMAGE --file $${config_directories[kayobe]}/.automation/docker/kayobe/Dockerfile --tag kayobe:latest $${config_directories[kayobe]} fi set +x diff --git a/templates/userdata.cfg.tpl b/templates/userdata.cfg.tpl.j2 similarity index 58% rename from templates/userdata.cfg.tpl rename to templates/userdata.cfg.tpl.j2 index 63e65db..faf8726 100644 --- a/templates/userdata.cfg.tpl +++ b/templates/userdata.cfg.tpl.j2 @@ -7,3 +7,6 @@ packages: - git - vim - tmux +ssh_authorized_keys: + - "{{ cluster_deploy_ssh_public_key }}" + - "{{ cluster_user_ssh_public_key }}" \ No newline at end of file diff --git a/terraform.tfvars.j2 b/terraform.tfvars.j2 new file mode 100644 index 0000000..86b6476 --- /dev/null +++ b/terraform.tfvars.j2 @@ -0,0 +1,29 @@ +prefix = "{{ cluster_name }}" + +ansible_control_vm_flavor = "general.v1.small" +ansible_control_vm_name = "ansible-control" +ansible_control_disk_size = 100 + +seed_vm_flavor = "general.v1.small" +seed_disk_size = 100 + +multinode_flavor = "general.v1.medium" +multinode_image = "{{ multinode_image }}" +multinode_keypair = "MaxMNKP" +multinode_vm_network = "stackhpc-ipv4-geneve" +multinode_vm_subnet = "stackhpc-ipv4-geneve-subnet" +compute_count = "2" +controller_count = "3" +compute_disk_size = 100 +controller_disk_size = 100 + +ssh_public_key = "{{ cluster_user_ssh_public_key }}" +ssh_user = "{{ ssh_user }}" + +storage_count = "3" +storage_flavor = "general.v1.small" +storage_disk_size = 100 + +deploy_wazuh = false +infra_vm_flavor = "general.v1.small" +infra_vm_disk_size = 100 \ No newline at end of file diff --git a/ui-meta/multinode-appliance.yml b/ui-meta/multinode-appliance.yml new file mode 100644 index 0000000..50d97cc --- /dev/null +++ b/ui-meta/multinode-appliance.yml @@ -0,0 +1,183 @@ +# The machine-readable name of the appliance +name: "multinode-app" +# The human-readable label for the appliance +label: "Multinode Customisable Deployment" +# A short description of the appliance, rendered as-is +description: | + An appliance for deploying a multinode cluster on Azimuth with options to deploy it with a custom version of OpenStack or just the infrastructure. +# The URL of the logo for the appliance +# This should be an HTTPS URL, because plain HTTP URLs will cause content warnings when Azimuth is served over HTTPS +# Alternatively, as shown here, a data URL can be used to encode the image directly +logo:  +# A list of parameters to be gathered from the user +# parameters: + # The name of the Ansible variable that will be populated with the parameter value. + # This field is required. + # - name: vxlan_vni + # A human-readable name for the parameter. Defaults to the name if not given. + # label: Set the VXLAN VNI value. + # An optional short description for the parameter. + # description: Please provide a VXLAN VNI. A unique value from 1 to 100,000. + # The kind of the parameter. This field is required. + # Valid values are: + #  list + # A list of items. + # string + #  A string or something that can be coerced. + #  integer + #  An integer or a string that can be coerced. + #  number + #  Any number, i.e. an integer or a float, or a string that can be coerced. + #  boolean + # A boolean flag. + #  Valid values are: true/false, 1/0, "1"/"0", "true"/"false" and "yes"/"no". + #  choice + # A value from a set of choices. + #  cloud.size + # The ID of a size in the target tenancy. + #  cloud.machine + #  The ID of a machine in the target tenancy. + # cloud.ip + #  The ID of an external IP in the target tenancy. + #  cloud.volume + #  The ID of a volume in the target tenancy. + #  cloud.cluster + #  The ID of another cluster in the target tenancy. + # kind: "integer" + # Indicates whether the parameter is required. Defaults to true if not given. + # required: true + # The default value value for the parameter. Defaults to null if not given, indicating no default. + # default: null + # Indicates whether the parameter is immutable, i.e. cannot be changed after the initial creation. + # Defaults to false if not given. + # immutable: true + # Additional options for the parameter. The valid options depend on the kind of the parameter. + # list + # min_length: An optional minimum length for the list. + #  max_length: An optional maximum length for the list. + # item: An optional item validator, consisting of a kind and some options. + # string + # min_length: An optional minimum length for the string. + # max_length: An optional maximum length for the string. + #  pattern: An optional regular expression pattern to validate the string against. + #  integer + #  number + # options: + # min: 1 + # max: 100000 + #  boolean + # permanent: If true, indicates that the parameter cannot become false again once it has + #  been set to true. Defaults to false if not given. + #  choice + #  choices: The list of valid choices for the parameter. This field is required. + #  cloud.size + #  min_cpus: An optional minimum number of CPUs that the size must have. + #  min_ram: An optional minimum amount of RAM in MB that the size must have. + # min_disk: An optional minimum root disk size in GB that the size must have. + #  cloud.volume + #  min_size: An optional minimum size in GB for the volume. + #  cloud.cluster + # tag: An optional tag that the cluster must have. + # options: + # min_ram: 1024 + # min_disk: 10 + # secret: true + # confirm: true + +# A list of parameters to be gathered from the user +parameters: + - name: vxlan_vni + label: Set VXLAN VNI. + description: Please provide a VXLAN VNI. A unique value from 1 to 100,000. + kind: "integer" + required: true + default: null + immutable: true + options: + min: 1 + max: 100000 + + + - name: ans_vlt_pwd + # A human-readable name for the parameter. Defaults to the name if not given. + label: Ansible Vault password. + # An optional short description for the parameter. + description: Please provide the Ansible Vault password to enable the deployment of the cluster. + kind: "string" + required: false + #default: "password_not_set" + immutable: false + # Replace the characters of the parameter with asterisks in the UI. + options: + secret: true + confirm: true + #private: true + + - name: openstack_deploy + # A boolean flag which will be set to true if the user selects the checkbox, + # this will allow the user to decide whether they wish to have the + # infrastructure built with or without OpenStack. + label: Deploy Infrastructure with OpenStack services? #(True/False) + description: If you wish to deploy OpenStack services in the infrastructure select this option. + kind: "boolean" + default: false + required: false + immutable: true + + - name: kayobe_version_branch + label: Kayobe GitHub Branch. + description: | + Please provide the OpenStack GitHub Branch name you wish to deploy. + i.e. stackhpc/yoga (for https://github.com/stackhpc/kayobe/tree/stackhpc/yoga) + kind: "string" + required: false + default: "stackhpc/yoga" + immutable: true + + - name: kayobe_config_branch + label: Kayobe configuration GitHub Branch. + description: | + Please provide the GitHub Branch with the OpenStack configuration you wish to deploy. + i.e. stackhpc/yoga (for https://github.com/stackhpc/stackhpc-kayobe-config/tree/stackhpc/yoga) + kind: "string" + required: false + default: "stackhpc/yoga" + immutable: true + + - name: multinode_image + label: Operating system image. + description: | + Please provide the OS you wish to deploy on this multinode infrastructure. + The image must be available in the OpenStack tenancy, two supported images are: + - Ubuntu-22.04-lvm + - rocky9-lvm + kind: "string" + required: true + default: "rocky9-lvm" + immutable: true + + - name: ssh_user + label: Operating system SSH username. + description: | + Please select the OS username in order to connect to the infrastructure. + Examples are: + - ubuntu (Ubuntu-22.04-lvm) + - cloud_user (rocky9-lvm) + kind: "string" + required: true + default: "cloud_user" + immutable: false + + +# A template describing the usage of the appliance +# This is rendered in the "Cluster details" modal in the Azimuth UI +usage_template: |- + # Accessing the cluster + + To assess the clusters, use the following ssh commands: + + {% if cluster.outputs.cluster_access_ip %} + Ansible Control Host => READY -> ssh {{ ssh_user }}@{{ cluster.outputs.cluster_access_ip }} + {% else %} + Not Available + {% endif %} diff --git a/versions.tf b/versions.tf index a380cc4..8582a8c 100644 --- a/versions.tf +++ b/versions.tf @@ -1,15 +1,9 @@ terraform { required_version = ">= 0.14" - backend "local" { - } required_providers { openstack = { source = "terraform-provider-openstack/openstack" version = "1.49.0" } - ansible = { - source = "ansible/ansible" - version = "1.1.0" - } } }