Compare commits

...

68 Commits

Author SHA1 Message Date
2a1a47ecc1 Remove VMWare static since not applicable 2024-10-30 23:18:27 +01:00
4808ce4401 Fix DISK removal at cleanup 2024-10-30 23:10:53 +01:00
db1fd13623 Fix variable hierarchy 2024-10-30 22:19:00 +01:00
e5660b0ba7 Fix ISO mounting for VMware Hypervisor 2024-10-30 20:25:41 +01:00
173ecd299b Different aproche for ISO mounting 2024-10-30 19:30:12 +01:00
4d242ad987 Adjust controllerID for RHEL ISO for correct mounting 2024-10-30 19:23:01 +01:00
f8ac22cfab Allow passwordless ssh for VMware Setup 2024-10-30 19:12:36 +01:00
12a7549aaa Speed up setup on VMware if ssh is available 2024-10-30 18:59:32 +01:00
6705411b2d Enable root ssh login 2024-10-30 18:54:15 +01:00
fe2b216fc7 set cis default value 2024-10-30 18:14:29 +01:00
26824ca6bb Improve Ip set on VMware hypervisors 2024-10-30 18:04:46 +01:00
c60fcca86d Fix VM Connection if hypervisor is VMware 2024-10-30 17:57:22 +01:00
cdd8062937 Fix recursion 2024-10-30 17:09:22 +01:00
ebedff1c4e fix jinja syntax 2024-10-30 17:05:50 +01:00
04d05a4e8b Move hypervisor and disk variable from main playbook 2024-10-30 16:58:22 +01:00
ee6e06a3fe lower connection timeout 2024-10-30 16:48:23 +01:00
527bc11d1d Change VMware boot order to boot correctly from ArchISO 2024-10-30 15:59:16 +01:00
d331e07536 Fix VMware Network if no VLAN specified 2024-10-30 15:48:22 +01:00
287036bcb4 use the correct NetworkMask variable name 2024-10-30 14:38:25 +01:00
ca5a3c8807 Add network mask variables for Hypervisor static IP assigments 2024-10-30 14:33:38 +01:00
c8dd89681b move vm_ip back since it is not a permanent/static variable 2024-10-30 14:10:37 +01:00
9d4af56976 Move some persstent Vars to main playbook 2024-10-30 14:01:07 +01:00
3c55eaf4a1 Recommend Ansible Vault for variables storing secrets 2024-10-30 13:45:19 +01:00
d905dce89e Add missing RHEL variable examples 2024-10-30 00:49:37 +01:00
76f1382e3e Assertion for minimum filesystem size 2024-10-30 00:44:19 +01:00
04c27cd7d0 remove deperacted parameter causing sshd startup fails 2024-10-30 00:32:08 +01:00
147430b36e Add RHEL8 and RHEL9 support 2024-10-30 00:29:46 +01:00
f8ba5c41db Update Ubuntu to Oracular Oriole and Ubuntu-LTS to Noble Numbat 2024-10-29 15:08:43 +01:00
7a4fc24f32 Remove SSH Config multiline since OpenSSH does not support it 2024-10-29 14:25:53 +01:00
7bf7c29291 Update Fedora to Version 41 2024-10-29 14:17:01 +01:00
ccfce65673 Disable Cloud-init updates on boot to prevent loopdevice out of storage 2024-10-29 12:59:50 +01:00
528f2fc775 Use command module instead of shell if possible 2024-10-28 21:15:10 +01:00
505110f580 Fix command module formating 2024-10-28 21:07:33 +01:00
1d1b2fff42 Fix connection DNS resolving inside chroot 2024-10-28 20:26:15 +01:00
4cf4816be0 ensure variable is not empty 2024-10-28 19:25:49 +01:00
e37b5a535b Specify changed_when for shell commands 2024-10-28 19:20:05 +01:00
5312ec8cc6 Replace ignore_errors with failed_when 2024-10-28 18:56:00 +01:00
a3b772c543 fix risky-shell-pipe 2024-10-28 18:47:31 +01:00
adde811f47 Fix risky-file-permissions because of unpecified mode 2024-10-28 18:37:44 +01:00
f788767839 Fix line-length 2024-10-28 18:26:54 +01:00
8b773d2304 Adjust literal-compare to use correct bool comparison 2024-10-28 17:17:24 +01:00
c988ab8f9a dont use ignore-errors 2024-07-11 22:31:13 +02:00
8864db253b add missing task name 2024-07-11 22:22:43 +02:00
06ca8d8787 ansible-lint fixes 2024-07-11 22:20:45 +02:00
374b5fc7ef use correct boolean values 2024-07-11 22:09:58 +02:00
6bfd530c90 fix jinja formating 2024-07-11 22:03:15 +02:00
b077e549db remove btrfs quota limits 2024-05-21 14:20:28 +02:00
43ce280d11 correct README 2024-04-17 14:38:47 +02:00
a6b51b4cb4 Add supported distro to the README 2024-04-17 14:37:47 +02:00
6dd31cc95f fix cis support for all distros 2024-04-17 14:09:32 +02:00
4b98ec1434 add ubuntu-lts support 2024-04-17 12:17:19 +02:00
2444c5d7af add ubuntu support 2024-04-17 10:53:09 +02:00
ec6ca49265 fix fedora boot issue 2024-04-17 06:02:32 +02:00
fe43bf6733 add essential almalinux packages 2024-04-17 05:06:45 +02:00
31c155ce92 install dnf if {{ os }} is fedora 2024-04-17 04:47:33 +02:00
0c75114b94 add rocky to README example 2024-04-17 04:39:29 +02:00
cd9ed65c91 Add essential rockylinux packages 2024-04-17 04:32:11 +02:00
9986d19ed6 Add en and de langauge support for rockylinux 2024-04-17 04:19:32 +02:00
d73e78c5f2 Add cloud-init support 2024-04-16 01:17:48 +02:00
b6f620fb70 Add RockyLinux support 2024-04-16 01:14:12 +02:00
cc40bae858 Add RockyLinux support 2024-04-16 01:14:05 +02:00
344753fa5b Add RockyLinux Repo file 2024-04-15 21:30:04 +02:00
6be464a0e2 move assertion list to main playbook 2024-04-15 21:23:32 +02:00
48b5f602fa Enable systemd-resolved and systemd-timesyncd services for ArchLinux 2024-03-28 03:50:04 +01:00
cc118274a3 Update gitignore 2024-03-22 12:48:49 +01:00
sandwich
d733513e29 Delete vars_libvirt.yml 2024-03-22 12:46:31 +01:00
sandwich
402f2b9bc0 Delete inventory_libvirt.yml 2024-03-22 12:46:22 +01:00
4ec5432989 Add inventory example in yaml 2024-03-22 12:43:13 +01:00
25 changed files with 949 additions and 420 deletions

3
.gitignore vendored
View File

@@ -1,5 +1,8 @@
inventory.yml inventory.yml
inventory.yaml inventory.yaml
inventory_libvirt.yml
vars.yml vars.yml
vars.yaml vars.yaml
vars_kvm.yml
vars_libvirt.yml

View File

@@ -6,8 +6,26 @@ An Ansible playbook for automating system bootstrap processes in an Infrastructu
Most of the roles are adaptable for use with systems beyond ArchLinux, requiring only that the target system can install a necessary package manager, such as `dnf` for RHEL-based systems. Additionally, a replacement for the `arch-chroot` command may be required for these systems. Most of the roles are adaptable for use with systems beyond ArchLinux, requiring only that the target system can install a necessary package manager, such as `dnf` for RHEL-based systems. Additionally, a replacement for the `arch-chroot` command may be required for these systems.
**NOTE**: **NOTE**:
- RHEL Systems are not currently supported due to restricted access to their repositories. - For RHEL 8 and RHEL 9, repository access requires the `rhel_iso` variable. This variable specifies a local ISO or proxy repository.
A workaround could involve using an ISO as a local repository or setting up a proxy repository to facilitate access. - RHEL systems do not support `btrfs`. Use `ext4` or `xfs` as alternatives.
- For RHEL 8, `xfs` may cause installation issues; `ext4` is recommended.
# Supported Distributions
This playbook supports multiple Linux distributions with specific versions tailored to each. Below is a list of supported distributions:
| `os` | Distribution |
|------------|------------------------------------|
| archlinux | ArchLinux (Latest rolling release) |
| almalinux | AlmaLinux 9.x |
| debian11 | Debian 11 (Bullseye) |
| debian12 | Debian 12 (Bookworm) |
| fedora | Fedora 41 |
| rhel8 | Red Hat Enterprise Linux 8 |
| rhel9 | Red Hat Enterprise Linux 9 |
| rocky | Rocky Linux 9.x |
| ubuntu | Ubuntu 24.10 (Oracular Oriole) |
| ubuntu-lts | Ubuntu 24.04 LTS (Noble Numbat) |
# Documentation # Documentation
@@ -32,6 +50,7 @@ Global variables apply across your Ansible project and are loaded from `vars.yml
| Variable | Description | Example Value | | Variable | Description | Example Value |
|-----------------------|--------------------------------------------------------------------|-----------------------------------------| |-----------------------|--------------------------------------------------------------------|-----------------------------------------|
| `boot_iso` | Path to the boot ISO image. | `local-btrfs:iso/archlinux-x86_64.iso` | | `boot_iso` | Path to the boot ISO image. | `local-btrfs:iso/archlinux-x86_64.iso` |
| `rhel_iso` | Path to the RHEL ISO file, required for RHEL 8 and RHEL 9. |`local-btrfs:iso/rhel-9.4-x86_64-dvd.iso`|
| `hypervisor` | Type of hypervisor. | `libvirt`, `proxmox`, `vmware`, `none` | | `hypervisor` | Type of hypervisor. | `libvirt`, `proxmox`, `vmware`, `none` |
| `hypervisor_cluster` | Name of the hypervisor cluster. | `default-cluster` | | `hypervisor_cluster` | Name of the hypervisor cluster. | `default-cluster` |
| `hypervisor_node` | Hypervisor node name. | `node01` | | `hypervisor_node` | Hypervisor node name. | `node01` |
@@ -43,6 +62,8 @@ Global variables apply across your Ansible project and are loaded from `vars.yml
| `install_type` | Type of installation. | `virtual`, `physical` | | `install_type` | Type of installation. | `virtual`, `physical` |
| `vlan_name` (optional)| VLAN for the VM's network interface. | `vlan100` | | `vlan_name` (optional)| VLAN for the VM's network interface. | `vlan100` |
To protect sensitive information, such as passwords, API keys, and other confidential variables (e.g., `hypervisor_password`), **it is recommended to use Ansible Vault**.
## 3. Inventory Variables ## 3. Inventory Variables
Inventory variables are defined for individual hosts or VMs in the inventory file, allowing customization of settings such as the operating system, filesystem, and compliance with CIS benchmarks. These variables can be set globally and overridden for specific hosts or VMs. Inventory variables are defined for individual hosts or VMs in the inventory file, allowing customization of settings such as the operating system, filesystem, and compliance with CIS benchmarks. These variables can be set globally and overridden for specific hosts or VMs.
@@ -52,7 +73,7 @@ Inventory variables are defined for individual hosts or VMs in the inventory fil
| `cis` (optional) | Adjusts the installation to be CIS level 3 conformant. | `true`, `false` | | `cis` (optional) | Adjusts the installation to be CIS level 3 conformant. | `true`, `false` |
| `filesystem` | Filesystem type for the VM's primary storage. | `btrfs`, `ext4`, `xfs` | | `filesystem` | Filesystem type for the VM's primary storage. | `btrfs`, `ext4`, `xfs` |
| `hostname` | The hostname assigned to the virtual machine or system. | `vm01` | | `hostname` | The hostname assigned to the virtual machine or system. | `vm01` |
| `os` | Operating system to be installed on the VM. | `archlinux`, `almalinux`, `debian11`, `debian12`, `fedora` | | `os` | Operating system to be installed on the VM. | `archlinux`, `almalinux`, `debian11`, `debian12`, `fedora`, `rhel8`, `rhel9`, `rocky`, `ubuntu`, `ubuntu-lts` |
| `root_password` | Root password for the VM or system, used for initial setup or secure access. | `SecurePass123` | | `root_password` | Root password for the VM or system, used for initial setup or secure access. | `SecurePass123` |
| `user_name` | Username for a user account within the VM, often used with cloud-init. | `adminuser` | | `user_name` | Username for a user account within the VM, often used with cloud-init. | `adminuser` |
| `user_password` | Password for the user account within the VM. | `UserPass123` | | `user_password` | Password for the user account within the VM. | `UserPass123` |
@@ -62,6 +83,8 @@ Inventory variables are defined for individual hosts or VMs in the inventory fil
| `vm_gw` | Default gateway IP address for the virtual machine's network configuration. | `192.168.0.1` | | `vm_gw` | Default gateway IP address for the virtual machine's network configuration. | `192.168.0.1` |
| `vm_id` | Unique identifier for the virtual machine. | `101` | | `vm_id` | Unique identifier for the virtual machine. | `101` |
| `vm_ip` | IP address assigned to the virtual machine. | `192.168.0.10` | | `vm_ip` | IP address assigned to the virtual machine. | `192.168.0.10` |
| `vm_nm` (optional) | IP address netmask assigned to the virtual machine. | `255.255.255.0` |
| `vm_nms` (optional) | IP address netmask assigned to the virtual machine. | `24` |
| `vm_memory` | Amount of memory (in MB) allocated to the virtual machine. | `2048` | | `vm_memory` | Amount of memory (in MB) allocated to the virtual machine. | `2048` |
| `vm_nif` | Network interface type or identifier for the VM's network connection. | `vmbr0` | | `vm_nif` | Network interface type or identifier for the VM's network connection. | `vmbr0` |
| `vm_path (optional)` | Path or folder where the VM configuration or related files will be stored. | `/var/lib/libvirt/images/` | | `vm_path (optional)` | Path or folder where the VM configuration or related files will be stored. | `/var/lib/libvirt/images/` |

33
inventory_example.yml Normal file
View File

@@ -0,0 +1,33 @@
all:
vars:
hypervisor: 'proxmox'
install_drive: '/dev/sda'
cis: true
children:
promox-kvm:
hosts:
192.168.122.10:
hostname: proxy
vm_id: 100
os: archlinux
filesystem: btrfs
vm_memory: "2048"
vm_ballo: "1024"
vm_cpus: "2"
vm_size: "5"
vm_nif: vmbr1
vm_gw: 192.168.122.1
vm_dns: 1.1.1.1
192.168.122.11:
hostname: database
vm_id: 101
os: rhel9
filesystem: xfs
vm_memory: "6144"
vm_ballo: "3072"
vm_cpus: "4"
vm_size: "40"
vm_nif: vmbr1
vm_gw: 192.168.122.1
vm_dns: 1.1.1.1
rhel_iso: "local-btrfs:iso/rhel-9.4-x86_64-dvd.iso"

114
main.yml
View File

@@ -5,87 +5,83 @@
gather_facts: false gather_facts: false
become: true become: true
vars_prompt: vars_prompt:
- name: user_name - name: user_name
prompt: | prompt: |
What is your username? What is your username?
private: false private: false
- name: user_password - name: user_password
prompt: | prompt: |
What is your password? What is your password?
confirm: true confirm: true
- name: root_password - name: root_password
prompt: | prompt: |
What is your root password? What is your root password?
confirm: true confirm: true
- name: hypervisor
prompt: |
Select an Hypervisor:
- libvirt
- proxmox
- vmware
private: false
default: "proxmox"
- name: install_drive
prompt: |
"Enter the drive to install the system (default: /dev/sda)"
confirm: true
private: false
default: "/dev/sda"
vars_files: vars.yml vars_files: vars.yml
pre_tasks: pre_tasks:
- name: Set ansible_python_interpreter - name: Set ansible_python_interpreter
when: os | lower in ["almalinux", "rhel9", "rhel8"] when: os | lower in ["almalinux", "rhel9", "rhel8", "rocky"]
set_fact: ansible.builtin.set_fact:
ansible_python_interpreter: /usr/bin/python3 ansible_python_interpreter: /usr/bin/python3
- name: Set SSH Access
when: hypervisor != "vmware"
ansible.builtin.set_fact:
ansible_user: "{{ user_name }}"
ansible_password: "{{ user_password }}"
ansible_become_password: "{{ user_password }}"
ansible_ssh_extra_args: '-o UserKnownHostsFile=/dev/null -o StrictHostKeyChecking=no'
- name: Validate variables - name: Validate variables
assert: ansible.builtin.assert:
that: that:
- hypervisor in hypervisor_list - hypervisor in ["libvirt", "proxmox", "vmware", "none"]
- filesystem in filesystem_list - filesystem in ["btrfs", "ext4", "xfs"]
- os in os_list - install_drive is defined
fail_msg: "Invalid input specified, please try again" - os in ["archlinux", "almalinux", "debian11", "debian12", "fedora", "rhel8", "rhel9", "rocky", "ubuntu", "ubuntu-lts"]
- os not in ["rhel8", "rhel9"] or rhel_iso is defined
- (filesystem == "btrfs" and (vm_size | int) >= 10) or (filesystem != "btrfs" and (vm_size | int) >= 20)
fail_msg: Invalid input specified, please try again.
- name: Set connection - name: Set connection
when: hypervisor == "vmware" when: hypervisor == "vmware"
set_fact: ansible.builtin.set_fact:
ansible_connection: vmware_tools ansible_connection: vmware_tools
roles: roles:
- role: virtualization
when: install_type == "virtual"
become: false
vars:
ansible_connection: local
- role: virtualization - role: environment
when: install_type == "virtual" vars:
become: false ansible_connection: "{{ 'vmware_tools' if hypervisor == 'vmware' else 'ssh' }}"
vars:
ansible_connection: local
- role: environment - role: partitioning
vars: vars:
ansible_connection: "{{ 'vmware_tools' if hypervisor == 'vmware' else 'ssh' }}" boot_partition_suffix: 1
main_partition_suffix: 2
- role: partitioning - role: bootstrap
vars:
boot_partition_suffix: 1
main_partition_suffix: 2
- role: bootstrap - role: configuration
- role: configuration - role: cis
when: cis | bool
- role: cis - role: cleanup
when: cis == true when: install_type == "virtual"
vars:
- role: cleanup ansible_connection: local
when: install_type == "virtual"
vars:
ansible_connection: local
tasks: tasks:
- name: Reboot system - name: Reboot system
when: hypervisor != "libvirt" when: hypervisor != "libvirt"
command: reboot ansible.builtin.command: reboot
ignore_errors: true failed_when: false
changed_when: result.rc == 0
register: result

View File

@@ -1,6 +1,6 @@
--- ---
- name: Include Packages - name: Include Packages
include_vars: ansible.builtin.include_vars:
file: packages.yml file: packages.yml
name: role_packages name: role_packages
@@ -8,36 +8,99 @@
block: block:
- name: Bootstrap ArchLinux - name: Bootstrap ArchLinux
when: os | lower == 'archlinux' when: os | lower == 'archlinux'
command: pacstrap /mnt {{ role_packages.archlinux | join(' ') }} --asexplicit ansible.builtin.command: pacstrap /mnt {{ role_packages.archlinux | join(' ') }} --asexplicit
changed_when: result.rc == 0
register: result
- name: Bootstrap Debian System - name: Bootstrap Debian System
when: os | lower in ['debian11', 'debian12'] when: os | lower in ['debian11', 'debian12']
shell: "{{ item }}" ansible.builtin.command: "{{ item }}"
changed_when: result.rc == 0
register: result
with_items: with_items:
- debootstrap --include={{ role_packages[os].base | join(',') }} {{ 'bullseye' if os == 'debian11' else 'bookworm' }} /mnt http://deb.debian.org/debian/ - debootstrap --include={{ role_packages[os].base | join(',') }} {{ 'bullseye' if os == 'debian11' else 'bookworm' }}
/mnt http://deb.debian.org/debian/
- arch-chroot /mnt apt install -y {{ role_packages[os].extra | join(' ') }} - arch-chroot /mnt apt install -y {{ role_packages[os].extra | join(' ') }}
- arch-chroot /mnt apt remove -y libcups2 libavahi-common3 libavahi-common-data - arch-chroot /mnt apt remove -y libcups2 libavahi-common3 libavahi-common-data
- name: Bootstrap Ubuntu System
when: os | lower in ['ubuntu', 'ubuntu-lts']
ansible.builtin.command: "{{ item }}"
changed_when: result.rc == 0
register: result
with_items:
- debootstrap --include={{ role_packages[os].base | join(',') }} {{ 'oracular' if os == 'ubuntu' else 'noble' }}
/mnt http://archive.ubuntu.com/ubuntu/
- ln -sf /run/systemd/resolve/resolv.conf /mnt/etc/resolv.conf
- arch-chroot /mnt sed -i '1s|$| universe|' /etc/apt/sources.list
- arch-chroot /mnt apt update -y
- arch-chroot /mnt apt install -y {{ role_packages[os].extra | join(' ') }}
- name: Bootstrap AlmaLinux 9 - name: Bootstrap AlmaLinux 9
when: os | lower == 'almalinux' when: os | lower == 'almalinux'
shell: "{{ item }}" ansible.builtin.command: "{{ item }}"
changed_when: result.rc == 0
register: result
with_items: with_items:
- dnf --releasever=9 --best --repo=alma-baseos --installroot=/mnt --setopt=install_weak_deps=False groupinstall -y base core - dnf --releasever=9 --best --repo=alma-baseos --installroot=/mnt --setopt=install_weak_deps=False groupinstall -y base core
- echo "nameserver 1.0.0.1" > /mnt/etc/resolv.conf - ln -sf /run/systemd/resolve/resolv.conf /mnt/etc/resolv.conf
- arch-chroot /mnt dnf --releasever=9 --setopt=install_weak_deps=False install -y {{ role_packages.almalinux | join(' ') }} - arch-chroot /mnt dnf --releasever=9 --setopt=install_weak_deps=False install -y {{ role_packages.almalinux | join(' ') }}
- name: Bootstrap Fedora 39 - name: Bootstrap Fedora 41
when: os | lower == 'fedora' when: os | lower == 'fedora'
shell: "{{ item }}" ansible.builtin.command: "{{ item }}"
changed_when: result.rc == 0
register: result
with_items: with_items:
- dnf --releasever=39 --best --repo=fedora --repo=fedora-updates --installroot=/mnt --setopt=install_weak_deps=False groupinstall -y critical-path-base core - dnf --releasever=41 --best --repo=fedora --repo=fedora-updates
- arch-chroot /mnt dnf --releasever=39 --setopt=install_weak_deps=False install -y {{ role_packages.fedora | join(' ') }} --installroot=/mnt --setopt=install_weak_deps=False groupinstall -y critical-path-base core
- arch-chroot /mnt dnf reinstall -y grub2-efi-x64 kernel - ln -sf /run/systemd/resolve/resolv.conf /mnt/etc/resolv.conf
- arch-chroot /mnt dnf --releasever=41 --setopt=install_weak_deps=False install -y {{ role_packages.fedora | join(' ') }}
- arch-chroot /mnt dnf reinstall -y kernel-core
- name: Bootstrap RockyLinux 9
when: os | lower == 'rocky'
ansible.builtin.command: "{{ item }}"
changed_when: result.rc == 0
register: result
with_items:
- dnf --releasever=9 --best --repo=rocky-baseos --installroot=/mnt
--setopt=install_weak_deps=False --setopt=optional_metadata_types=filelists
groupinstall -y base core
- ln -sf /run/systemd/resolve/resolv.conf /mnt/etc/resolv.conf
- arch-chroot /mnt dnf --releasever=9 --setopt=install_weak_deps=False install -y {{ role_packages.rocky | join(' ') }}
- name: Bootstrap RHEL System - name: Bootstrap RHEL System
when: os | lower in ['rhel8', 'rhel9'] when: os | lower in ['rhel8', 'rhel9']
shell: "{{ item }}" block:
with_items: - name: Install base packages in chroot environment
- "dnf --releasever={{ '8' if os == 'rhel8' else '9' }} --installroot=/mnt --setopt=install_weak_deps=False groupinstall -y base core" ansible.builtin.command: >-
- "echo 'nameserver 1.0.0.1' > /mnt/etc/resolv.conf" dnf --releasever={{ '8' if os == 'rhel8' else '9' }} --repo={{ os | lower }}-baseos
- "arch-chroot /mnt dnf --releasever={{ '8' if os == 'rhel8' else '9' }} --setopt=install_weak_deps=False install -y {{ role_packages[os] | join(' ') }}" --installroot=/mnt
--setopt=install_weak_deps=False --setopt=optional_metadata_types=filelists
groupinstall -y base core
changed_when: result.rc == 0
register: result
- name: Prepare chroot environment
ansible.builtin.shell: |
ln -sf /run/systemd/resolve/resolv.conf /mnt/etc/resolv.conf
mkdir -p /mnt/usr/local/install/redhat/dvd
mount --bind /usr/local/install/redhat/dvd /mnt/usr/local/install/redhat/dvd
arch-chroot /mnt rpm --rebuilddb
changed_when: result.rc == 0
register: result
- name: Copy RHEL repo file into chroot environment
ansible.builtin.copy:
src: /etc/yum.repos.d/{{ os | lower }}.repo
dest: /mnt/etc/yum.repos.d/{{ os | lower }}.repo
mode: '0644'
remote_src: true
- name: Install additional packages in chroot
ansible.builtin.command: >-
arch-chroot /mnt dnf --releasever={{ '8' if os == 'rhel8' else '9' }}
--setopt=install_weak_deps=False install -y {{ role_packages[os] | join(' ') }}
changed_when: result.rc == 0
register: result

View File

@@ -1,7 +1,31 @@
--- ---
almalinux:
- bind-utils
- cloud-init
- dbus-daemon
- dhcp-client
- efibootmgr
- glibc-langpack-de
- glibc-langpack-en
- grub2
- grub2-efi
- lrzsz
- lvm2
- nc
- nfs-utils
- nfsv4-client-utils
- open-vm-tools
- ppp
- shim
- telnet
- vim
- wget
- zstd
archlinux: archlinux:
- base - base
- btrfs-progs - btrfs-progs
- cloud-init
- cronie - cronie
- dhcpcd - dhcpcd
- efibootmgr - efibootmgr
@@ -9,6 +33,7 @@ archlinux:
- fish - fish
- grub - grub
- htop - htop
- libpwquality
- linux - linux
- logrotate - logrotate
- lrzsz - lrzsz
@@ -20,6 +45,7 @@ archlinux:
- nfs-utils - nfs-utils
- openssh - openssh
- open-vm-tools - open-vm-tools
- ppp
- prometheus-node-exporter - prometheus-node-exporter
- python-psycopg2 - python-psycopg2
- qemu-guest-agent - qemu-guest-agent
@@ -34,13 +60,12 @@ debian11:
base: base:
- apparmor-utils - apparmor-utils
- btrfs-progs - btrfs-progs
- xfsprogs
- chrony - chrony
- cron - cron
- gnupg
- grub-efi - grub-efi
- grub-efi-amd64-signed - grub-efi-amd64-signed
- grub2-common - grub2-common
- gnupg
- linux-image-amd64 - linux-image-amd64
- locales - locales
- logrotate - logrotate
@@ -49,98 +74,107 @@ debian11:
- openssh-server - openssh-server
- python3 - python3
- sudo - sudo
- xfsprogs
extra: extra:
- cloud-init
- curl - curl
- firewalld - firewalld
- fish
- htop - htop
- network-manager - libpam-pwquality
- screen - lrzsz
- open-vm-tools
- python-is-python3
- ncdu - ncdu
- neofetch - neofetch
- lrzsz - network-manager
- libpam-pwquality - open-vm-tools
- python-is-python3
- rsync - rsync
- screen
- software-properties-common - software-properties-common
- syslog-ng - syslog-ng
- tcpd - tcpd
- fish
- vim - vim
- wget - wget
- zstd
debian12: debian12:
base: base:
- btrfs-progs - btrfs-progs
- xfsprogs
- cron - cron
- gnupg
- grub-efi - grub-efi
- grub-efi-amd64-signed - grub-efi-amd64-signed
- grub2-common - grub2-common
- gnupg
- linux-image-amd64 - linux-image-amd64
- locales - locales
- logrotate
- lvm2 - lvm2
- xfsprogs
extra: extra:
- apparmor-utils - apparmor-utils
- chrony - chrony
- cloud-init
- curl - curl
- firewalld - firewalld
- fish - fish
- htop - htop
- network-manager - libpam-pwquality
- screen
- open-vm-tools
- python-is-python3
- ncdu
- neofetch
- logrotate - logrotate
- lrzsz - lrzsz
- libpam-pwquality - ncdu
- neofetch
- net-tools
- network-manager
- open-vm-tools
- openssh-server
- python-is-python3
- python3
- rsync - rsync
- screen
- software-properties-common - software-properties-common
- sudo - sudo
- syslog-ng - syslog-ng
- tcpd - tcpd
- net-tools
- openssh-server
- python3
- vim - vim
- wget - wget
- zstd
fedora: fedora:
- bind-utils
- btrfs-progs
- cloud-init
- cronie
- dhcp-client - dhcp-client
- efibootmgr - efibootmgr
- glibc-langpack-de
- glibc-langpack-en
- grub2 - grub2
- grub2-efi-x64-modules - grub2-efi
- logrotate
- lrzsz - lrzsz
- lvm2
- nc
- nfs-utils - nfs-utils
- nfsv4-client-utils
- open-vm-tools - open-vm-tools
- polkit
- ppp
- shim - shim
- telnet - telnet
- vim-default-editor - vim-default-editor
- zstd - wget
almalinux:
- dhcp-client
- efibootmgr
- grub2
- grub2-efi-x64-modules
- lrzsz
- nfs-utils
- open-vm-tools
- shims
- telnet
- vim
- zstd - zstd
rhel8: rhel8:
- cloud-init
- dhcp-client - dhcp-client
- efibootmgr - efibootmgr
- grub2 - grub2
- grub2-efi-x64-modules - grub2-efi-x64
- lrzsz - lrzsz
- lvm2
- nfs-utils - nfs-utils
- open-vm-tools - open-vm-tools
- shim - shim
@@ -148,13 +182,127 @@ rhel8:
- zstd - zstd
rhel9: rhel9:
- cloud-init
- dhcp-client - dhcp-client
- efibootmgr - efibootmgr
- grub2 - grub2
- grub2-efi-x64-modules - grub2-efi
- lrzsz - lrzsz
- lvm2
- nfs-utils - nfs-utils
- open-vm-tools - open-vm-tools
- shim - shim
- telnet - telnet
- zstd - zstd
rocky:
- bind-utils
- cloud-init
- dbus-daemon
- dhcp-client
- efibootmgr
- glibc-langpack-de
- glibc-langpack-en
- grub2
- grub2-efi
- lrzsz
- lvm2
- nc
- nfs-utils
- nfsv4-client-utils
- open-vm-tools
- ppp
- shim
- telnet
- util-linux-core
- vim
- wget
- zstd
ubuntu:
base:
- btrfs-progs
- cron
- gnupg
- grub-efi
- grub-efi-amd64-signed
- grub2-common
- initramfs-tools
- linux-image-generic
- locales
- lvm2
- xfsprogs
extra:
- apparmor-utils
- bash-completion
- chrony
- cloud-init
- curl
- dnsutils
- firewalld
- fish
- htop
- libpam-pwquality
- logrotate
- lrzsz
- ncdu
- net-tools
- network-manager
- open-vm-tools
- openssh-server
- python-is-python3
- python3
- rsync
- screen
- software-properties-common
- sudo
- syslog-ng
- tcpd
- vim
- wget
- zstd
ubuntu-lts:
base:
- btrfs-progs
- cron
- gnupg
- grub-efi
- grub-efi-amd64-signed
- grub2-common
- initramfs-tools
- linux-image-generic
- locales
- lvm2
- xfsprogs
extra:
- apparmor-utils
- bash-completion
- chrony
- cloud-init
- curl
- dnsutils
- firewalld
- fish
- htop
- libpam-pwquality
- logrotate
- lrzsz
- ncdu
- net-tools
- network-manager
- open-vm-tools
- openssh-server
- python-is-python3
- python3
- rsync
- screen
- software-properties-common
- sudo
- syslog-ng
- tcpd
- vim
- wget
- zstd

View File

@@ -1,8 +1,10 @@
---
- name: Configurationg System for CIS conformity - name: Configurationg System for CIS conformity
block: block:
- name: Disable Kernel Modules - name: Disable Kernel Modules
copy: ansible.builtin.copy:
dest: /mnt/etc/modprobe.d/cis.conf dest: /mnt/etc/modprobe.d/cis.conf
mode: '0644'
content: | content: |
CIS LVL 3 Restrictions CIS LVL 3 Restrictions
install freevxfs /bin/true install freevxfs /bin/true
@@ -19,8 +21,9 @@
install tipc /bin/true install tipc /bin/true
- name: Create USB Rules - name: Create USB Rules
copy: ansible.builtin.copy:
dest: /mnt/etc/udev/rules.d/10-cis_usb_devices.sh dest: /mnt/etc/udev/rules.d/10-cis_usb_devices.sh
mode: '0644'
content: | content: |
By default, disable all. By default, disable all.
ACTION=="add", SUBSYSTEMS=="usb", TEST=="authorized_default", ATTR{authorized_default}="0" ACTION=="add", SUBSYSTEMS=="usb", TEST=="authorized_default", ATTR{authorized_default}="0"
@@ -35,8 +38,9 @@
ACTION=="add", ATTR{product}=="*Thinnet TM*", TEST=="authorized", ATTR{authorized}="1" ACTION=="add", ATTR{product}=="*Thinnet TM*", TEST=="authorized", ATTR{authorized}="1"
- name: Create a consolidated sysctl configuration file - name: Create a consolidated sysctl configuration file
copy: ansible.builtin.copy:
dest: /mnt/etc/sysctl.d/10-cis.conf dest: /mnt/etc/sysctl.d/10-cis.conf
mode: '0644'
content: | content: |
## CIS Sysctl configurations ## CIS Sysctl configurations
net.ipv4.conf.all.log_martians = 1 net.ipv4.conf.all.log_martians = 1
@@ -65,96 +69,100 @@
# - { regexp: '^PASS_MIN_DAYS.*', replace: 'PASS_MIN_DAYS 7' } # - { regexp: '^PASS_MIN_DAYS.*', replace: 'PASS_MIN_DAYS 7' }
# - { regexp: '^UMASK.*', replace: 'UMASK 027' } # - { regexp: '^UMASK.*', replace: 'UMASK 027' }
- name: Create allow files - name: Ensure files exist
file: ansible.builtin.file:
path: "{{ item }}" path: "{{ item }}"
state: touch state: touch
mode: '0600' mode: "0600"
loop: loop:
- /mnt/etc/at.allow - /mnt/etc/at.allow
- /mnt/etc/cron.allow - /mnt/etc/cron.allow
- /mnt/etc/hosts.allow
- /mnt/etc/hosts.deny
- name: Add Security related lines into config files - name: Add Security related lines into config files
lineinfile: ansible.builtin.lineinfile:
path: "{{ item.path }}" path: "{{ item.path }}"
line: "{{ item.content }}" line: "{{ item.content }}"
loop: loop:
- { path: '/mnt/etc/security/limits.conf', content: '* hard core 0' } - { path: /mnt/etc/security/limits.conf, content: "* hard core 0" }
- { path: '/mnt/etc/security/pwquality.conf', content: 'minlen = 14' } - { path: /mnt/etc/security/pwquality.conf, content: minlen = 14 }
- { path: '/mnt/etc/security/pwquality.conf', content: 'dcredit = -1' } - { path: /mnt/etc/security/pwquality.conf, content: dcredit = -1 }
- { path: '/mnt/etc/security/pwquality.conf', content: 'ucredit = -1' } - { path: /mnt/etc/security/pwquality.conf, content: ucredit = -1 }
- { path: '/mnt/etc/security/pwquality.conf', content: 'ocredit = -1' } - { path: /mnt/etc/security/pwquality.conf, content: ocredit = -1 }
- { path: '/mnt/etc/security/pwquality.conf', content: 'lcredit = -1' } - { path: /mnt/etc/security/pwquality.conf, content: lcredit = -1 }
- { path: '/mnt/etc/bash.bashrc', content: 'umask 077' } - { path: '/mnt/etc/{{ "bashrc" if os in ["almalinux", "fedora", "rhel8", "rhel9", "rocky"] else "bash.bashrc" }}', content: umask 077 }
- { path: '/mnt/etc/bash.bashrc', content: 'export TMOUT=3000' } - { path: '/mnt/etc/{{ "bashrc" if os in ["almalinux", "fedora", "rhel8", "rhel9", "rocky"] else "bash.bashrc" }}', content: export TMOUT=3000 }
- { path: '/mnt/etc/systemd/journald.conf', content: 'Storage=persistent' } - { path: '/mnt/{{ "usr/lib/systemd/journald.conf" if os == "fedora" else "etc/systemd/journald.conf" }}', content: Storage=persistent }
- { path: '/mnt/etc/sudoers', content: 'Defaults logfile="/var/log/sudo.log"' } - { path: /mnt/etc/sudoers, content: Defaults logfile="/var/log/sudo.log" }
- { path: '/mnt/etc/pam.d/su', content: 'auth required pam_wheel.so' } - { path: /mnt/etc/pam.d/su, content: auth required pam_wheel.so }
- { path: '/mnt/etc/pam.d/common-auth', content: 'auth required pam_faillock.so onerr=fail audit silent deny=5 unlock_time=900' } - { path: '/mnt/etc/{{ "pam.d/common-auth" if os in ["debian11", "debian12", "ubuntu", "ubuntu-lts"]
- { path: '/mnt/etc/pam.d/common-account', content: 'account required pam_faillock.so' } else "authselect/system-auth" if os == "fedora" else "pam.d/system-auth" }}',
- { path: '/mnt/etc/pam.d/common-password', content: 'password [success=1 default=ignore] pam_unix.so obscure sha512 remember=5' } content: auth required pam_faillock.so onerr=fail audit silent deny=5 unlock_time=900 }
- { path: '/mnt/etc/hosts.deny', content: 'ALL: ALL' } - { path: '/mnt/etc/{{ "pam.d/common-account" if os in ["debian11", "debian12", "ubuntu", "ubuntu-lts"] else "authselect/system-auth"
- { path: '/mnt/etc/hosts.allow', content: 'sshd: ALL' } if os == "fedora" else "pam.d/system-auth" }}', content: account required pam_faillock.so }
- { path: '/mnt/etc/pam.d/{{ "common-password" if os in ["debian11", "debian12", "ubuntu", "ubuntu-lts"] else "passwd" }}',
content: "password [success=1 default=ignore] pam_unix.so obscure sha512 remember=5" }
- { path: /mnt/etc/hosts.deny, content: "ALL: ALL" }
- { path: /mnt/etc/hosts.allow, content: "sshd: ALL" }
- name: Set permissions for various files and directories - name: Set permissions for various files and directories
file: ansible.builtin.file:
path: "{{ item.path }}" path: "{{ item.path }}"
owner: "{{ item.owner | default(omit) }}" owner: "{{ item.owner | default(omit) }}"
group: "{{ item.group | default(omit) }}" group: "{{ item.group | default(omit) }}"
mode: "{{ item.mode }}" mode: "{{ item.mode }}"
loop: loop: >
- { path: '/mnt/etc/ssh/sshd_config', mode: '0600' } {{ [
- { path: '/mnt/etc/cron.hourly', mode: '0700' } { "path": "/mnt/etc/ssh/sshd_config", "mode": "0600" },
- { path: '/mnt/etc/cron.daily', mode: '0700' } { "path": "/mnt/etc/cron.hourly", "mode": "0700" },
- { path: '/mnt/etc/cron.weekly', mode: '0700' } { "path": "/mnt/etc/cron.daily", "mode": "0700" },
- { path: '/mnt/etc/cron.monthly', mode: '0700' } { "path": "/mnt/etc/cron.weekly", "mode": "0700" },
- { path: '/mnt/etc/cron.d', mode: '0700' } { "path": "/mnt/etc/cron.monthly", "mode": "0700" },
- { path: '/mnt/etc/crontab', mode: '0600' } { "path": "/mnt/etc/cron.d", "mode": "0700" },
- { path: '/mnt/etc/logrotate.conf', mode: '0644' } { "path": "/mnt/etc/crontab", "mode": "0600" },
- { path: '/mnt/usr/sbin/pppd', mode: '754' } { "path": "/mnt/etc/logrotate.conf", "mode": "0644" },
- { path: '/mnt/usr/lib/dbus-1.0/dbus-daemon-launch-helper', mode: '754' } { "path": "/mnt/usr/sbin/pppd", "mode": "0754" } if os not in ["rhel8", "rhel9"] else None,
- { path: '/mnt/usr/libexec/polkit-agent-helper-1', mode: '755' } { "path": "/mnt/usr/bin/" + ("fusermount3" if os in ["almalinux", "archlinux", "debian12", "fedora", "rhel9", "rocky"]
- { path: '/mnt/usr/bin/{{ "fusermount" if os == "debian11" else "fusermount3" }}', mode: '755' } else "fusermount"), "mode": "755" },
- { path: '/mnt/usr/bin/{{ "write.ul" if os == "debian11" else "write" }}', mode: '755' } { "path": "/mnt/usr/bin/" + ("write.ul" if os == "debian11" else "write"), "mode": "755" }
- { path: '/mnt/usr/lib/x86_64-linux-gnu/utempter/utempter', mode: '755' } ] | reject("none") }}
- { path: '/mnt/home/svcansible', mode: '750' }
- name: Adjust SSHD config - name: Adjust SSHD config
lineinfile: ansible.builtin.lineinfile:
path: /mnt/etc/ssh/sshd_config path: /mnt/etc/ssh/sshd_config
regexp: '^\s*#?{{ item.option }}\s+.*$' regexp: ^\s*#?{{ item.option }}\s+.*$
line: '{{ item.option }} {{ item.value }}' line: "{{ item.option }} {{ item.value }}"
with_items: with_items:
- {option: 'LogLevel', value: 'VERBOSE'} - { option: LogLevel, value: VERBOSE }
- {option: 'LoginGraceTime', value: '60'} - { option: LoginGraceTime, value: "60" }
- {option: 'PermitRootLogin', value: 'no'} - { option: PermitRootLogin, value: "no" }
- {option: 'StrictModes', value: 'yes'} - { option: StrictModes, value: "yes" }
- {option: 'MaxAuthTries', value: '4'} - { option: MaxAuthTries, value: "4" }
- {option: 'MaxSessions', value: '10'} - { option: MaxSessions, value: "10" }
- {option: 'MaxStartups', value: '10:30:60'} - { option: MaxStartups, value: 10:30:60 }
- {option: 'PubkeyAuthentication', value: 'yes'} - { option: PubkeyAuthentication, value: "yes" }
- {option: 'HostbasedAuthentication', value: 'no'} - { option: HostbasedAuthentication, value: "no" }
- {option: 'IgnoreRhosts', value: 'yes'} - { option: IgnoreRhosts, value: "yes" }
- {option: 'PasswordAuthentication', value: 'no'} - { option: PasswordAuthentication, value: "no" }
- {option: 'PermitEmptyPasswords', value: 'no'} - { option: PermitEmptyPasswords, value: "no" }
- {option: 'KerberosAuthentication', value: 'no'} - { option: KerberosAuthentication, value: "no" }
- {option: 'GSSAPIAuthentication', value: 'no'} - { option: GSSAPIAuthentication, value: "no" }
- {option: 'GSSAPIKeyExchange', value: 'no'} - { option: AllowAgentForwarding, value: "no" }
- {option: 'AllowAgentForwarding', value: 'no'} - { option: AllowTcpForwarding, value: "no" }
- {option: 'AllowTcpForwarding', value: 'no'} - { option: ChallengeResponseAuthentication, value: "no" }
- {option: 'ChallengeResponseAuthentication', value: 'no'} - { option: GatewayPorts, value: "no" }
- {option: 'GatewayPorts', value: 'no'} - { option: X11Forwarding, value: "no" }
- {option: 'X11Forwarding', value: 'no'} - { option: PermitUserEnvironment, value: "no" }
- {option: 'PermitUserEnvironment', value: 'no'} - { option: ClientAliveInterval, value: "300" }
- {option: 'ClientAliveInterval', value: '300'} - { option: ClientAliveCountMax, value: "0" }
- {option: 'ClientAliveCountMax', value: '0'} - { option: PermitTunnel, value: "no" }
- {option: 'PermitTunnel', value: 'no'} - { option: Banner, value: /etc/issue.net }
- {option: 'Banner', value: '/etc/issue.net'}
- name: Append CIS Specific configurations to sshd_config - name: Append CIS Specific configurations to sshd_config
lineinfile: ansible.builtin.lineinfile:
path: /mnt/etc/ssh/sshd_config path: /mnt/etc/ssh/sshd_config
line: | line: |2-
## CIS Specific ## CIS Specific
Protocol 2 Protocol 2
@@ -169,7 +177,7 @@
AllowStreamLocalForwarding no AllowStreamLocalForwarding no
PermitUserRC no PermitUserRC no
AllowUsers svcansible AllowUsers *
AllowGroups * AllowGroups *
DenyUsers nobody DenyUsers nobody
DenyGroups nobody DenyGroups nobody

View File

@@ -1,3 +1,4 @@
---
- name: Setup Cleanup - name: Setup Cleanup
when: hypervisor == "proxmox" when: hypervisor == "proxmox"
delegate_to: localhost delegate_to: localhost
@@ -14,26 +15,33 @@
state: absent state: absent
loop: loop:
- ide0 - ide0
- ide1 - ide2
- name: Remove CD-ROM from VM in vCenter - name: Remove CD-ROM from VM in vCenter
when: hypervisor == "vmware" when: hypervisor == "vmware"
delegate_to: localhost delegate_to: localhost
ignore_errors: true become: false
vmware_guest: failed_when: false
community.vmware.vmware_guest:
hostname: "{{ hypervisor_url }}" hostname: "{{ hypervisor_url }}"
username: "{{ hypervisor_username }}" username: "{{ hypervisor_username }}"
password: "{{ hypervisor_password }}" password: "{{ hypervisor_password }}"
validate_certs: no validate_certs: false
datacenter: "{{ hypervisor_cluster }}" datacenter: "{{ hypervisor_cluster }}"
name: "{{ hostname }}" name: "{{ hostname }}"
cdrom: cdrom:
- controller_number: 0 - controller_number: 0
unit_number: 0 unit_number: 0
controller_type: "sata" controller_type: sata
type: iso type: iso
iso_path: "{{ boot_iso }}" iso_path: "{{ boot_iso }}"
state: absent state: absent
- controller_number: 0
unit_number: 1
controller_type: sata
type: iso
iso_path: "{{ rhel_iso | default(omit) }}"
state: absent
- name: Remove Archiso and cloud-init disks - name: Remove Archiso and cloud-init disks
when: hypervisor == "libvirt" when: hypervisor == "libvirt"
@@ -46,23 +54,25 @@
state: shutdown state: shutdown
- name: Remove cloud-init disk - name: Remove cloud-init disk
file: ansible.builtin.file:
path: "{{ vm_path | default('/var/lib/libvirt/images/') }}{{ hostname }}-cloudinit.iso" path: "{{ vm_path | default('/var/lib/libvirt/images/') }}{{ hostname }}-cloudinit.iso"
state: absent state: absent
- name: Get list of CD-ROM devices - name: Get list of CD-ROM devices
shell: virsh --connect qemu:///system domblklist {{ hostname }} --details | grep 'cdrom' | awk '{print $3}' ansible.builtin.shell: set -o pipefail && virsh --connect qemu:///system domblklist {{ hostname }} --details | grep 'cdrom' | awk '{print $3}'
changed_when: false changed_when: false
register: cdrom_devices register: cdrom_devices
- name: Wait for VM to spin down - name: Wait for VM to spin down
wait_for: ansible.builtin.wait_for:
timeout: 15 timeout: 15
- name: Remove CD-ROM devices - name: Remove CD-ROM devices
when: cdrom_devices.stdout_lines | length > 0 when: cdrom_devices.stdout_lines | length > 0
command: virsh --connect qemu:///system detach-disk {{ hostname }} {{ item }} --persistent ansible.builtin.command: virsh --connect qemu:///system detach-disk {{ hostname }} {{ item }} --persistent
with_items: "{{ cdrom_devices.stdout_lines }}" with_items: "{{ cdrom_devices.stdout_lines | select('ne', 'sdc') | list }}"
changed_when: result.rc == 0
register: result
- name: Start the VM - name: Start the VM
community.libvirt.virt: community.libvirt.virt:
@@ -71,5 +81,5 @@
- name: Wait for VM to boot up - name: Wait for VM to boot up
delegate_to: "{{ inventory_hostname }}" delegate_to: "{{ inventory_hostname }}"
wait_for_connection: ansible.builtin.wait_for_connection:
timeout: 300 timeout: 300

View File

@@ -1,165 +1,262 @@
---
- name: Configuration - name: Configuration
block: block:
- name: Generate fstab - name: Generate fstab
shell: genfstab -LU /mnt > /mnt/etc/fstab ansible.builtin.shell: genfstab -LU /mnt > /mnt/etc/fstab
changed_when: result.rc == 0
register: result
- name: Remove depricated attr2 and disable large extent
when: os in ["almalinux", "rhel8", "rhel9", "rocky"] and filesystem == "xfs"
ansible.builtin.replace:
path: /mnt/etc/fstab
regexp: '(xfs.*?)(attr2)'
replace: '\1allocsize=64m'
- name: Replace ISO UUID entry with /dev/sr0 in fstab
when: os in ["rhel8", "rhel9"]
ansible.builtin.lineinfile:
path: /mnt/etc/fstab
regexp: '^.*\/dvd.*$'
line: "{{ '/usr/local/install/redhat/rhel.iso /usr/local/install/redhat/dvd iso9660 loop,nofail 0 0' if hypervisor == 'vmware'
else '/dev/sr0 /usr/local/install/redhat/dvd iso9660 ro,relatime,nojoliet,check=s,map=n,nofail 0 0' }}"
state: present
backrefs: true
- name: Write image from RHEL ISO to the target machine
when: os in ["rhel8", "rhel9"] and hypervisor == 'vmware'
ansible.builtin.command: dd if=/dev/sr1 of=/mnt/usr/local/install/redhat/rhel.iso bs=4M
changed_when: result.rc == 0
register: result
- name: Append TempFS to fstab - name: Append TempFS to fstab
lineinfile: ansible.builtin.lineinfile:
path: /mnt/etc/fstab path: /mnt/etc/fstab
line: "{{ item }}" line: "{{ item }}"
insertafter: EOF insertafter: EOF
with_items: with_items:
- "" - ""
- "# TempFS" - "# TempFS"
- "tmpfs /tmp tmpfs defaults,nosuid,nodev,noexec 0 0" - tmpfs /tmp tmpfs defaults,nosuid,nodev,noexec 0 0
- "tmpfs /var/tmp tmpfs defaults,nosuid,nodev,noexec 0 0" - tmpfs /var/tmp tmpfs defaults,nosuid,nodev,noexec 0 0
- "tmpfs /dev/shm tmpfs defaults,noexec 0 0" - tmpfs /dev/shm tmpfs defaults,noexec 0 0
- name: Set local timezone - name: Set local timezone
command: '{{ item }}' ansible.builtin.command: "{{ item }}"
changed_when: result.rc == 0
register: result
with_items: with_items:
- systemctl daemon-reload - systemctl daemon-reload
- arch-chroot /mnt ln -sf /usr/share/zoneinfo/Europe/Vienna /etc/localtime - arch-chroot /mnt ln -sf /usr/share/zoneinfo/Europe/Vienna /etc/localtime
- name: Generate adjtime file
command: arch-chroot /mnt /usr/sbin/hwclock --systohc
- name: Setup locales - name: Setup locales
block: block:
- name: Configure locale.gen - name: Configure locale.gen
lineinfile: when: os | lower not in ['almalinux', 'fedora', 'rhel8', 'rhel9', 'rocky']
ansible.builtin.lineinfile:
dest: /mnt/etc/locale.gen dest: /mnt/etc/locale.gen
regexp: '{{ item.regex }}' regexp: "{{ item.regex }}"
line: '{{ item.line }}' line: "{{ item.line }}"
loop: loop:
- {regex: en_US\.UTF-8 UTF-8, line: en_US.UTF-8 UTF-8} - { regex: en_US\.UTF-8 UTF-8, line: en_US.UTF-8 UTF-8 }
- name: Generate locales - name: Generate locales
command: arch-chroot /mnt /usr/sbin/locale-gen when: os | lower not in ['almalinux', 'fedora', 'rhel8', 'rhel9', 'rocky']
ansible.builtin.command: arch-chroot /mnt /usr/sbin/locale-gen
changed_when: result.rc == 0
register: result
- name: Set hostname - name: Set hostname
copy: ansible.builtin.copy:
content: "{{ hostname }}" content: "{{ hostname }}"
dest: /mnt/etc/hostname dest: /mnt/etc/hostname
mode: '0644'
- name: Add host entry to /etc/hosts - name: Add host entry to /etc/hosts
lineinfile: ansible.builtin.lineinfile:
path: /mnt/etc/hosts path: /mnt/etc/hosts
line: "{{ ansible_host }} {{ hostname }}" line: "{{ ansible_host }} {{ hostname }}"
state: present state: present
- name: Create vconsole.conf - name: Create vconsole.conf
copy: ansible.builtin.copy:
content: "KEYMAP=de-latin1-nodeadkeys" content: KEYMAP=us
dest: /mnt/etc/vconsole.conf dest: /mnt/etc/vconsole.conf
mode: '0644'
- name: Create locale.conf - name: Create locale.conf
copy: ansible.builtin.copy:
content: "LANG=en_US.UTF-8" content: LANG=en_US.UTF-8
dest: /mnt/etc/locale.conf dest: /mnt/etc/locale.conf
mode: '0644'
- name: SSH permit Password - name: SSH permit Password
replace: ansible.builtin.replace:
path: /mnt/etc/ssh/sshd_config path: /mnt/etc/ssh/sshd_config
regexp: '#PasswordAuthentication yes' regexp: "#PasswordAuthentication yes"
replace: 'PasswordAuthentication yes' replace: PasswordAuthentication yes
- name: SSH permit root login
ansible.builtin.replace:
path: /mnt/etc/ssh/sshd_config
regexp: "^#?PermitRootLogin.*"
replace: "PermitRootLogin yes"
- name: Enable Systemd Services - name: Enable Systemd Services
block: ansible.builtin.command: >
- name: Enable sshd arch-chroot /mnt systemctl enable NetworkManager
when: os | lower == "archlinux" {{
command: arch-chroot /mnt systemctl enable sshd NetworkManager logrotate ' ssh' if os | lower in ['ubuntu', 'ubuntu-lts'] else
(' sshd' if os | lower not in ['debian11', 'debian12'] else '')
}}
{{
'logrotate systemd-resolved systemd-timesyncd systemd-networkd'
if os | lower == 'archlinux' else ''
}}
changed_when: result.rc == 0
register: result
- name: Configure grub - name: Configure grub
when: os | lower != "fedora" and os | lower != "almalinux" and os | lower != "rhel8" and os | lower != "rhel9" when: os | lower not in ['almalinux', 'fedora', 'rhel8', 'rhel9', 'rocky']
block: block:
- name: Add commandline information to grub config - name: Add commandline information to grub config
lineinfile: ansible.builtin.lineinfile:
dest: /mnt/etc/default/grub dest: /mnt/etc/default/grub
regexp: ^GRUB_CMDLINE_LINUX_DEFAULT= regexp: ^GRUB_CMDLINE_LINUX_DEFAULT=
line: 'GRUB_CMDLINE_LINUX_DEFAULT="loglevel=3"' line: GRUB_CMDLINE_LINUX_DEFAULT="loglevel=3"
- name: Change Grub time - name: Change Grub time
lineinfile: ansible.builtin.lineinfile:
dest: /mnt/etc/default/grub dest: /mnt/etc/default/grub
regexp: ^GRUB_TIMEOUT= regexp: ^GRUB_TIMEOUT=
line: 'GRUB_TIMEOUT=0' line: GRUB_TIMEOUT=1
- name: Configure Bootloader - name: Configure Bootloader
block: block:
- name: Install Bootloader - name: Install Bootloader
command: arch-chroot /mnt {% if os | lower != "archlinux" and os | lower != "debian11" and os | lower != "debian12" %}/usr/sbin/efibootmgr -c -L '{{ os }}' -d "{{ install_drive }}" -wwp 1 -l '\efi\EFI\{{ os }}\shimx64.efi'{% else %}/usr/sbin/grub-install --target=x86_64-efi --efi-directory=/boot --bootloader-id={{ os }}{% endif %} ansible.builtin.command: arch-chroot /mnt
{% if os | lower not in ["archlinux", "debian11", "debian12", "ubuntu", "ubuntu-lts"] %} /usr/sbin/efibootmgr
-c -L '{{ os }}' -d "{{ install_drive }}" -p 1
-l '\efi\EFI\{% if os | lower in ["rhel8", "rhel9"] %}redhat{% else %}{{ os | lower }}{% endif %}\shimx64.efi'
{% else %}/usr/sbin/grub-install --target=x86_64-efi --efi-directory={{ "/boot/efi" if os | lower in ["ubuntu", "ubuntu-lts"] else "/boot" }}
--bootloader-id={{ "ubuntu" if os | lower in ["ubuntu", "ubuntu-lts"] else os }}
{% endif %}
changed_when: result.rc == 0
register: result
- name: Generate grub config - name: Generate grub config
command: arch-chroot /mnt {% if os | lower != "archlinux" and os | lower != "debian11" and os | lower != "debian12" %}/usr/sbin/grub2-mkconfig -o /boot/efi/EFI/{{ os }}/grub.cfg{% else %}/usr/sbin/grub-mkconfig -o /boot/grub/grub.cfg{% endif %} ansible.builtin.command: arch-chroot /mnt
{% if os | lower not in ["archlinux", "debian11", "debian12", "ubuntu", "ubuntu-lts"] %}
/usr/sbin/grub2-mkconfig -o /boot/efi/EFI/{% if os | lower in ["rhel8", "rhel9"] %}redhat{% else %}{{ os | lower }}{% endif %}/grub.cfg
{% else %}
/usr/sbin/grub-mkconfig -o {{ "/boot/efi/EFI/ubuntu/grub.cfg" if os | lower in ["ubuntu", "ubuntu-lts"] else "/boot/grub/grub.cfg" }}
{% endif %}
changed_when: result.rc == 0
register: result
- name: Regenerate initramfs
when: os | lower not in ["debian11", "debian12", "ubuntu", "ubuntu-lts"]
ansible.builtin.command: arch-chroot /mnt
{% if os | lower == "archlinux" %} /usr/sbin/mkinitcpio -P
{% elif os | lower not in ["debian11", "debian12", "ubuntu", "ubuntu-lts", "archlinux"] %} /usr/bin/dracut --regenerate-all --force
{% else %} echo "Skipping initramfs regeneration"
{% endif %}
changed_when: result.rc == 0
register: result
- name: Extra Configuration - name: Extra Configuration
when: os | lower != "archlinux"
block: block:
- name: Append lines to vimrc - name: Append lines to vimrc
lineinfile: failed_when: false
path: "{{ '/mnt/etc/vim/vimrc' if os|lower == 'debian11' or os|lower == 'debian12' else '/mnt/etc/vimrc' }}" ansible.builtin.lineinfile:
path: "{{ '/mnt/etc/vim/vimrc' if os | lower in ['debian11', 'debian12', 'ubuntu', 'ubuntu-lts'] else '/mnt/etc/vimrc' }}"
line: "{{ item }}" line: "{{ item }}"
insertafter: EOF insertafter: EOF
with_items: with_items:
- "set encoding=utf-8" - set encoding=utf-8
- "set number" - set number
- "set autoindent" - set autoindent
- "set smartindent" - set smartindent
- "set mouse=a" - set mouse=a
- name: Copy FirstRun Script - name: Copy FirstRun Script
template: when: os | lower != "archlinux"
ansible.builtin.template:
src: firstrun.sh.j2 src: firstrun.sh.j2
dest: /mnt/root/firstrun.sh dest: /mnt/root/firstrun.sh
mode: '0755' mode: "0755"
- name: Copy Custom Shell config - name: Copy Custom Shell config
template: ansible.builtin.template:
src: custom.sh.j2 src: custom.sh.j2
dest: /mnt/etc/profile.d/custom.sh dest: /mnt/etc/profile.d/custom.sh
mode: '0644'
- name: Setup Network - name: Setup Network
block: block:
- name: Generate UUID for Network Profile - name: Generate UUID for Network Profile
command: "uuidgen" ansible.builtin.command: uuidgen
register: net_uuid changed_when: net_uuid.rc == 0
register: net_uuid
- name: Retrieve Network Interface Name - name: Retrieve Network Interface Name
shell: "ip r | awk 'NR==1 {print $5}'" ansible.builtin.shell: set -o pipefail && ip r | awk 'NR==1 {print $5}'
register: net_inf changed_when: net_inf.rc == 0
register: net_inf
- name: Copy NetworkManager keyfile - name: Copy NetworkManager keyfile
template: ansible.builtin.template:
src: network.j2 src: network.j2
dest: /mnt/etc/NetworkManager/system-connections/LAN.nmconnection dest: /mnt/etc/NetworkManager/system-connections/LAN.nmconnection
mode: '0600' mode: "0600"
- name: Fix Ubuntu unmanaged devices
when: os | lower in ["ubuntu", "ubuntu-lts"]
ansible.builtin.file:
path: /mnt/etc/NetworkManager/conf.d/10-globally-managed-devices.conf
state: touch
mode: '0644'
- name: Setup user account - name: Setup user account
block: block:
- name: Create user account - name: Create user account
command: '{{ item }}' ansible.builtin.command: "{{ item }}"
with_items: with_items:
- arch-chroot /mnt /usr/sbin/useradd --create-home --user-group --groups {{ "sudo" if os|lower == "debian11" or os|lower == "debian12" else "wheel" }} {{ user_name }} --password {{ user_password | password_hash('sha512') }} --shell /bin/bash - arch-chroot /mnt /usr/sbin/useradd --create-home --user-group --groups
- arch-chroot /mnt /usr/sbin/usermod --password '{{ root_password | password_hash('sha512') }}' root --shell /bin/bash {{ "sudo" if os | lower in ["debian11", "debian12", "ubuntu", "ubuntu-lts"] else "wheel" }}
{{ user_name }} --password {{ user_password | password_hash('sha512') }} --shell /bin/bash
- arch-chroot /mnt /usr/sbin/usermod --password '{{ root_password | password_hash('sha512') }}' root --shell /bin/bash
changed_when: result.rc == 0
register: result
- name: Add SSH public key to authorized_keys - name: Add SSH public key to authorized_keys
when: user_public_key is defined when: user_public_key is defined
lineinfile: ansible.builtin.lineinfile:
path: "/mnt/home/{{ user_name }}/.ssh/authorized_keys" path: /mnt/home/{{ user_name }}/.ssh/authorized_keys
line: "{{ user_public_key }}" line: "{{ user_public_key }}"
owner: 1000 owner: 1000
group: 1000 group: 1000
mode: "0600" mode: "0600"
create: yes create: true
- name: Give sudo access to wheel group - name: Give sudo access to wheel group
copy: ansible.builtin.copy:
content: "{{ '%sudo ALL=(ALL) ALL' if os|lower == 'debian11' or os|lower == 'debian12' else '%wheel ALL=(ALL) ALL' }}" content: "{{ '%sudo ALL=(ALL) ALL' if os | lower in ['debian11', 'debian12', 'ubuntu', 'ubuntu-lts'] else '%wheel ALL=(ALL) ALL' }}"
dest: /mnt/etc/sudoers.d/01-wheel dest: /mnt/etc/sudoers.d/01-wheel
mode: 0440 mode: "0440"
validate: /usr/sbin/visudo --check --file=%s validate: /usr/sbin/visudo --check --file=%s
- name: Fix SELinux - name: Fix SELinux
when: (os | lower == "almalinux" or os | lower == "fedora" or os | lower == "rhel8" or os | lower == "rhel9") block:
command: touch /mnt/.autorelabel - name: Relabel the filesystem
when: os | lower in ['almalinux', 'rhel8', 'rhel9', 'rocky']
ansible.builtin.command: "arch-chroot /mnt /sbin/fixfiles onboot"
changed_when: result.rc == 0
register: result
- name: Disable SELinux
when: os | lower == "fedora"
ansible.builtin.lineinfile:
path: /mnt/etc/selinux/config
regexp: ^SELINUX=
line: SELINUX=permissive

View File

@@ -1,74 +1,130 @@
---
- name: Configre work environment - name: Configre work environment
become: true become: true
block: block:
- name: Wait for connection - name: Wait for connection
wait_for_connection: ansible.builtin.wait_for_connection:
timeout: 300 timeout: 60
delay: 5 delay: 5
- name: Gather facts - name: Gather facts
setup: ansible.builtin.setup:
- name: Check if host is booted from the Arch install media - name: Check if host is booted from the Arch install media
stat: ansible.builtin.stat:
path: /run/archiso path: /run/archiso
register: archiso_stat register: archiso_stat
- name: Abort if the host is not booted from the Arch install media - name: Abort if the host is not booted from the Arch install media
fail: ansible.builtin.fail:
msg: "This host is not booted from the Arch install media!" msg: This host is not booted from the Arch install media!
when: not archiso_stat.stat.exists when: not archiso_stat.stat.exists
- name: Setect Interface - name: Setect Interface
when: hypervisor == "vmware" when: hypervisor == "vmware"
shell: "ip l | awk -F': ' '!/lo/{print $2; exit}'" ansible.builtin.shell: "set -o pipefail && ip l | awk -F': ' '!/lo/{print $2; exit}'"
changed_when: interface_name.rc == 0
register: interface_name register: interface_name
- name: Set IP-Address - name: Set IP-Address
when: hypervisor == "vmware" when: hypervisor == "vmware"
command: ip addr replace {{ ansible_host }}/24 dev {{ interface_name.stdout }} ansible.builtin.command: "ip addr replace {{ ansible_host }}/{{ vm_nms | default(24) }} dev {{ interface_name.stdout }}"
changed_when: result.rc == 0
register: result
- name: Set Default Gateway - name: Set Default Gateway
when: hypervisor == "vmware" when: hypervisor == "vmware"
command: ip route replace default via {{ vm_gw }} ansible.builtin.command: "ip route replace default via {{ vm_gw }}"
changed_when: result.rc == 0
register: result
- name: Synchronize clock via NTP - name: Synchronize clock via NTP
command: timedatectl set-ntp true ansible.builtin.command: timedatectl set-ntp true
changed_when: result.rc == 0
register: result
- name: Configure SSH for root login
when: hypervisor == "vmware" and vmware_ssh | bool
block:
- name: Allow empty passwords temporarily
ansible.builtin.replace:
path: /etc/ssh/sshd_config
regexp: "^#?PermitEmptyPasswords.*"
replace: "PermitEmptyPasswords yes"
- name: Allow root login
ansible.builtin.replace:
path: /etc/ssh/sshd_config
regexp: "^#?PermitRootLogin.*"
replace: "PermitRootLogin yes"
- name: Reload SSH service to apply changes
ansible.builtin.service:
name: sshd
state: reloaded
- name: Set connection back to SSH
ansible.builtin.set_fact:
ansible_connection: ssh
ansible_user: "root"
ansible_password: ""
ansible_become_password: ""
ansible_ssh_extra_args: '-o UserKnownHostsFile=/dev/null -o StrictHostKeyChecking=no'
- name: Speed-up Bootstrap process - name: Speed-up Bootstrap process
lineinfile: ansible.builtin.lineinfile:
path: /etc/pacman.conf path: /etc/pacman.conf
regexp: '^#ParallelDownloads =' regexp: ^#ParallelDownloads =
line: 'ParallelDownloads = 20' line: ParallelDownloads = 20
- name: Wait for Pacman - name: Wait for Pacman
wait_for: ansible.builtin.wait_for:
timeout: 15 timeout: 15
- name: Setup Pacman - name: Setup Pacman
pacman: community.general.pacman:
update_cache: true update_cache: true
force: true force: true
name: "{{ item.name }}" name: "{{ item.name }}"
state: latest state: latest
loop: loop:
- { name: 'glibc' } - { name: glibc }
- { name: 'dnf', os: ['almalinux', 'rhel9', 'rhel8'] } - { name: dnf, os: [almalinux, fedora, rhel9, rhel8, rocky] }
- { name: 'debootstrap', os: ['debian11', 'debian12'] } - { name: debootstrap, os: [debian11, debian12, ubuntu, ubuntu-lts] }
- { name: 'debian-archive-keyring', os: ['debian11', 'debian12'] } - { name: debian-archive-keyring, os: [debian11, debian12] }
- { name: ubuntu-keyring, os: [ubuntu, ubuntu-lts] }
when: "'os' not in item or os in item.os" when: "'os' not in item or os in item.os"
retries: 4 retries: 4
delay: 15 delay: 15
- name: Configure RHEL Repos for installation - name: Prepare /iso mount and repository for RHEL-based systems
when: os | lower == "almalinux" or os | lower == "fedora" when: os | lower in ["rhel8", "rhel9"]
block: block:
- name: Create directories for repository files and RPM GPG keys - name: Create /iso directory
file: ansible.builtin.file:
path: /etc/yum.repos.d path: /usr/local/install/redhat/dvd
state: directory state: directory
mode: '0755'
- name: Create RHEL repository file - name: Mount RHEL ISO
template: ansible.posix.mount:
src: '{{ os | lower }}.repo.j2' src: "{{ '/dev/sr1' if hypervisor == 'vmware' else '/dev/sr2' }}"
dest: '/etc/yum.repos.d/{{ os | lower }}.repo' path: /usr/local/install/redhat/dvd
fstype: iso9660
opts: "ro,loop"
state: mounted
- name: Configure RHEL Repos for installation
when: os | lower in ["almalinux", "fedora", "rhel8", "rhel9", "rocky"]
block:
- name: Create directories for repository files and RPM GPG keys
ansible.builtin.file:
path: /etc/yum.repos.d
state: directory
mode: '0755'
- name: Create RHEL repository file
ansible.builtin.template:
src: "{{ os | lower }}.repo.j2"
dest: /etc/yum.repos.d/{{ os | lower }}.repo
mode: '0644'

View File

@@ -2,25 +2,29 @@
- name: Setup BTRFS - name: Setup BTRFS
block: block:
- name: Create btrfs filesystem in main volume - name: Create btrfs filesystem in main volume
filesystem: community.general.filesystem:
dev: '{{ install_drive }}{{ main_partition_suffix }}' dev: "{{ install_drive }}{{ main_partition_suffix }}"
fstype: btrfs fstype: btrfs
force: yes force: true
- name: Prepare BTRFS Subvolume - name: Prepare BTRFS Subvolume
mount: ansible.posix.mount:
path: /mnt path: /mnt
src: '{{ install_drive }}{{ main_partition_suffix }}' src: "{{ install_drive }}{{ main_partition_suffix }}"
fstype: btrfs fstype: btrfs
opts: rw,relatime,compress=zstd:15,ssd,space_cache=v2,discard=async opts: rw,relatime,compress=zstd:15,ssd,space_cache=v2,discard=async
state: mounted state: mounted
- name: Enable quotas on Btrfs filesystem - name: Enable quotas on Btrfs filesystem
command: btrfs quota enable /mnt ansible.builtin.command: btrfs quota enable /mnt
changed_when: result.rc == 0
register: result
- name: Make root subvolumes - name: Make root subvolumes
when: cis == true or item.subvol not in ['var_log', 'var_log_audit'] when: cis | bool or item.subvol not in ['var_log', 'var_log_audit']
command: btrfs su cr /mnt/{{ '@' if item.subvol == 'root' else '@' + item.subvol }} ansible.builtin.command: btrfs su cr /mnt/{{ '@' if item.subvol == 'root' else '@' + item.subvol }}
changed_when: result.rc == 0
register: result
loop: loop:
- { subvol: root } - { subvol: root }
- { subvol: home } - { subvol: home }
@@ -29,18 +33,16 @@
- { subvol: var_log_audit } - { subvol: var_log_audit }
- name: Set quotas for subvolumes - name: Set quotas for subvolumes
when: cis == true or item.subvol not in ['var_log', 'var_log_audit'] when: cis | bool or item.subvol not in ['var_log', 'var_log_audit']
command: btrfs qgroup limit {{ item.quota }} /mnt/{{ '@' if item.subvol == 'root' else '@' + item.subvol }} ansible.builtin.command: btrfs qgroup limit {{ item.quota }} /mnt/{{ '@' if item.subvol == 'root' else '@' + item.subvol }}
changed_when: result.rc == 0
register: result
loop: loop:
- { subvol: root, quota: '12G' } - { subvol: home, quota: 2G }
- { subvol: home, quota: '2G' }
- { subvol: var, quota: '2G' }
- { subvol: var_log, quota: '2G' }
- { subvol: var_log_audit, quota: '1536M' }
- name: Unmount Partition - name: Unmount Partition
mount: ansible.posix.mount:
path: /mnt path: /mnt
src: '{{ install_drive }}{{ main_partition_suffix }}' src: "{{ install_drive }}{{ main_partition_suffix }}"
fstype: btrfs fstype: btrfs
state: unmounted state: unmounted

View File

@@ -1,10 +1,10 @@
--- ---
- name: Create and format ext4 logical volumes - name: Create and format ext4 logical volumes
when: cis == true or item.lv not in ['var_log', 'var_log_audit'] when: cis | bool or item.lv not in ['var_log', 'var_log_audit']
filesystem: community.general.filesystem:
dev: '/dev/sys/{{ item.lv }}' dev: /dev/sys/{{ item.lv }}
fstype: ext4 fstype: ext4
force: yes force: true
loop: loop:
- { lv: root } - { lv: root }
- { lv: home } - { lv: home }
@@ -13,8 +13,10 @@
- { lv: var_log_audit } - { lv: var_log_audit }
- name: Remove Unsupported features for older Systems - name: Remove Unsupported features for older Systems
when: (os | lower == 'debian11') and (cis == true or item.lv not in ['var_log', 'var_log_audit']) when: (os | lower in ['almalinux', 'debian11', 'rhel8', 'rhel9', 'rocky']) and (cis | bool or item.lv not in ['var_log', 'var_log_audit'])
command: tune2fs -O "^orphan_file,^metadata_csum_seed" "/dev/sys/{{ item.lv }}" ansible.builtin.command: tune2fs -O "^orphan_file,^metadata_csum_seed" "/dev/sys/{{ item.lv }}"
changed_when: result.rc == 0
register: result
loop: loop:
- { lv: root } - { lv: root }
- { lv: home } - { lv: home }

View File

@@ -2,17 +2,19 @@
- name: Partition install drive - name: Partition install drive
block: block:
- name: Prepare partitions - name: Prepare partitions
ignore_errors: true failed_when: false
command: "{{ item.cmd }}" ansible.builtin.command: "{{ item.cmd }}"
changed_when: result.rc == 0
register: result
loop: loop:
- { cmd: "umount -l /mnt" } - { cmd: umount -l /mnt }
- { cmd: "vgremove -f sys" } - { cmd: vgremove -f sys }
- { cmd: "find /dev -wholename \"{{ install_drive }}*\" -exec wipefs --force --all {} \\;" } - { cmd: 'find /dev -wholename "{{ install_drive }}*" -exec wipefs --force --all {} \;' }
loop_control: loop_control:
label: "{{ item.cmd }}" label: "{{ item.cmd }}"
- name: Define partitions - name: Define partitions
parted: community.general.parted:
device: "{{ install_drive }}" device: "{{ install_drive }}"
label: gpt label: gpt
number: "{{ item.number }}" number: "{{ item.number }}"
@@ -22,56 +24,56 @@
flags: "{{ item.flags | default(omit) }}" flags: "{{ item.flags | default(omit) }}"
state: present state: present
loop: loop:
- { number: 1, part_end: '500MiB', name: 'boot', flags: ['boot', 'esp'] } - { number: 1, part_end: 500MiB, name: boot, flags: [boot, esp] }
- { number: 2, part_start: '500MiB', name: 'root' } - { number: 2, part_start: 500MiB, name: root }
- name: Create LVM logical volumes - name: Create LVM logical volumes
when: filesystem != 'btrfs' when: filesystem != 'btrfs'
block: block:
- name: Create LVM volume group - name: Create LVM volume group
lvg: community.general.lvg:
vg: sys vg: sys
pvs: '{{ install_drive }}{{ main_partition_suffix }}' pvs: "{{ install_drive }}{{ main_partition_suffix }}"
- name: Create LVM logical volumes - name: Create LVM logical volumes
when: cis or (not cis and item.lv != 'var_log' and item.lv != 'var_log_audit') when: cis | bool or item.lv not in ['var_log', 'var_log_audit']
lvol: community.general.lvol:
vg: sys vg: sys
lv: "{{ item.lv }}" lv: "{{ item.lv }}"
size: "{{ item.size }}" size: "{{ item.size }}"
state: present state: present
loop: loop:
- { lv: 'root', size: '12G' } - { lv: root, size: 12G }
- { lv: 'home', size: '2G' } - { lv: home, size: 2G }
- { lv: 'var', size: '2G' } - { lv: var, size: 2G }
- { lv: 'var_log', size: '2G' } - { lv: var_log, size: 2G }
- { lv: 'var_log_audit', size: '1.5G' } - { lv: var_log_audit, size: 1.5G }
- name: Create filesystems - name: Create filesystems
block: block:
- name: Create FAT32 filesystem in boot partition - name: Create FAT32 filesystem in boot partition
filesystem: community.general.filesystem:
dev: '{{ install_drive }}{{ boot_partition_suffix }}' dev: "{{ install_drive }}{{ boot_partition_suffix }}"
fstype: vfat fstype: vfat
opts: -F32 opts: -F32 -n BOOT
force: yes force: true
- name: Create filesystem - name: Create filesystem
include_tasks: "{{ filesystem }}.yml" ansible.builtin.include_tasks: "{{ filesystem }}.yml"
- name: Get UUID for boot filesystem - name: Get UUID for boot filesystem
command: blkid -s UUID -o value '{{ install_drive }}{{ boot_partition_suffix }}' ansible.builtin.command: blkid -s UUID -o value '{{ install_drive }}{{ boot_partition_suffix }}'
changed_when: false changed_when: false
register: boot_uuid register: boot_uuid
- name: Get UUID for main filesystem - name: Get UUID for main filesystem
command: blkid -s UUID -o value '{{ install_drive }}{{ main_partition_suffix }}' ansible.builtin.command: blkid -s UUID -o value '{{ install_drive }}{{ main_partition_suffix }}'
changed_when: false changed_when: false
register: main_uuid register: main_uuid
- name: Get UUIDs for LVM filesystems - name: Get UUIDs for LVM filesystems
when: filesystem != 'btrfs' and (cis == true or item not in ['var_log', 'var_log_audit']) when: filesystem != 'btrfs' and (cis | bool or item not in ['var_log', 'var_log_audit'])
command: blkid -s UUID -o value /dev/sys/{{ item }} ansible.builtin.command: blkid -s UUID -o value /dev/sys/{{ item }}
changed_when: false changed_when: false
register: uuid_result register: uuid_result
loop: loop:
@@ -81,7 +83,8 @@
- var_log - var_log
- var_log_audit - var_log_audit
- set_fact: - name: Assign UUIDs to Variables
ansible.builtin.set_fact:
uuid_root: "{{ uuid_result.results[0].stdout_lines }}" uuid_root: "{{ uuid_result.results[0].stdout_lines }}"
uuid_home: "{{ uuid_result.results[1].stdout_lines }}" uuid_home: "{{ uuid_result.results[1].stdout_lines }}"
uuid_var: "{{ uuid_result.results[2].stdout_lines }}" uuid_var: "{{ uuid_result.results[2].stdout_lines }}"
@@ -92,34 +95,48 @@
- name: Mount filesystems - name: Mount filesystems
block: block:
- name: Mount filesystems and subvolumes - name: Mount filesystems and subvolumes
when: "cis or (not cis and item.path != '/var/log' and item.path != '/var/log/audit')" when: cis | bool or (not cis and item.path != '/var/log' and item.path != '/var/log/audit')
mount: ansible.posix.mount:
path: "/mnt{{ item.path }}" path: /mnt{{ item.path }}
src: "{{ 'UUID=' + (main_uuid.stdout if filesystem == 'btrfs' else item.uuid) }}" src: "{{ 'UUID=' + (main_uuid.stdout if filesystem == 'btrfs' else item.uuid) }}"
fstype: "{{ filesystem }}" fstype: "{{ filesystem }}"
opts: "{{ item.opts }}" opts: "{{ item.opts }}"
state: mounted state: mounted
loop: loop:
- { path: '', uuid: "{{ uuid_root[0] | default(omit) }}", opts: "{{ 'defaults' if filesystem != 'btrfs' else 'rw,relatime,compress=zstd:15,ssd,space_cache=v2,discard=async,subvol=@' }}" } - path: ""
- { path: '/home', uuid: "{{ uuid_home[0] | default(omit) }}", opts: "{{ 'defaults,nosuid,nodev' if filesystem != 'btrfs' else 'rw,nosuid,nodev,relatime,compress=zstd:15,ssd,space_cache=v2,discard=async,subvol=@home' }}" } uuid: "{{ uuid_root[0] | default(omit) }}"
- { path: '/var', uuid: "{{ uuid_var[0] | default(omit) }}", opts: "{{ 'defaults,nosuid,nodev' if filesystem != 'btrfs' else 'rw,nosuid,nodev,relatime,compress=zstd:15,ssd,space_cache=v2,discard=async,subvol=@var' }}" } opts: "{{ 'defaults' if filesystem != 'btrfs' else 'rw,relatime,compress=zstd:15,ssd,space_cache=v2,discard=async,subvol=@' }}"
- { path: '/var/log', uuid: "{{ uuid_var_log[0] | default(omit) }}", opts: "{{ 'defaults,nosuid,nodev,noexec' if filesystem != 'btrfs' else 'rw,nosuid,nodev,noexec,relatime,compress=zstd:15,ssd,space_cache=v2,discard=async,subvol=@var_log' }}" } - path: /home
- { path: '/var/log/audit', uuid: "{{ uuid_var_log_audit[0] | default(omit) }}", opts: "{{ 'defaults,nosuid,nodev,noexec' if filesystem != 'btrfs' else 'rw,nosuid,nodev,noexec,relatime,compress=zstd:15,ssd,space_cache=v2,discard=async,subvol=@var_log_audit' }}" } uuid: "{{ uuid_home[0] | default(omit) }}"
opts: "{{ 'defaults,nosuid,nodev' if filesystem != 'btrfs'
else 'rw,nosuid,nodev,relatime,compress=zstd:15,ssd,space_cache=v2,discard=async,subvol=@home' }}"
- path: /var
uuid: "{{ uuid_var[0] | default(omit) }}"
opts: "{{ 'defaults,nosuid,nodev' if filesystem != 'btrfs'
else 'rw,nosuid,nodev,relatime,compress=zstd:15,ssd,space_cache=v2,discard=async,subvol=@var' }}"
- path: /var/log
uuid: "{{ uuid_var_log[0] | default(omit) }}"
opts: "{{ 'defaults,nosuid,nodev,noexec' if filesystem != 'btrfs'
else 'rw,nosuid,nodev,noexec,relatime,compress=zstd:15,ssd,space_cache=v2,discard=async,subvol=@var_log' }}"
- path: /var/log/audit
uuid: "{{ uuid_var_log_audit[0] | default(omit) }}"
opts: "{{ 'defaults,nosuid,nodev,noexec' if filesystem != 'btrfs'
else 'rw,nosuid,nodev,noexec,relatime,compress=zstd:15,ssd,space_cache=v2,discard=async,subvol=@var_log_audit' }}"
- name: Mount tmp and var_tmp filesystems - name: Mount tmp and var_tmp filesystems
mount: ansible.posix.mount:
path: "/mnt{{ item.path }}" path: /mnt{{ item.path }}
src: tmpfs src: tmpfs
fstype: tmpfs fstype: tmpfs
opts: defaults,nosuid,nodev,noexec opts: defaults,nosuid,nodev,noexec
state: mounted state: mounted
loop: loop:
- { path: '/tmp' } - { path: /tmp }
- { path: '/var/tmp' } - { path: /var/tmp }
- name: Mount boot filesystem - name: Mount boot filesystem
mount: ansible.posix.mount:
path: /mnt/boot path: "{{ '/mnt/boot/efi' if os | lower in ['rhel8', 'ubuntu', 'ubuntu-lts'] else '/mnt/boot' }}"
src: UUID={{ boot_uuid.stdout }} src: UUID={{ boot_uuid.stdout }}
fstype: vfat fstype: vfat
state: mounted state: mounted

View File

@@ -1,10 +1,10 @@
--- ---
- name: Create and format XFS logical volumes - name: Create and format XFS logical volumes
when: cis == true or item.lv not in ['var_log', 'var_log_audit'] when: cis | bool or item.lv not in ['var_log', 'var_log_audit']
filesystem: community.general.filesystem:
dev: '/dev/sys/{{ item.lv }}' dev: /dev/sys/{{ item.lv }}
fstype: xfs fstype: xfs
force: yes force: true
loop: loop:
- { lv: root } - { lv: root }
- { lv: home } - { lv: home }

View File

@@ -1,32 +1,41 @@
---
- name: Check if VM disk exists - name: Check if VM disk exists
delegate_to: localhost delegate_to: localhost
stat: ansible.builtin.stat:
path: "{{ vm_path | default('/var/lib/libvirt/images/') }}{{ hostname }}.qcow2" path: "{{ vm_path | default('/var/lib/libvirt/images/') }}{{ hostname }}.qcow2"
register: vm_disk_stat register: vm_disk_stat
- name: Create VM disk - name: Create VM disk
when: not vm_disk_stat.stat.exists when: not vm_disk_stat.stat.exists
delegate_to: localhost delegate_to: localhost
command: "qemu-img create -f qcow2 {{ vm_path | default('/var/lib/libvirt/images/') }}{{ hostname }}.qcow2 {{ vm_size }}G" ansible.builtin.command: qemu-img create -f qcow2 {{ vm_path | default('/var/lib/libvirt/images/') }}{{ hostname }}.qcow2 {{ vm_size }}G
changed_when: result.rc == 0
register: result
- name: Generate Random MAC Address - name: Generate Random MAC Address
delegate_to: localhost delegate_to: localhost
shell: openssl rand -hex 5 | sed 's/\(..\)/\1:/g; s/.$//' | sed 's/^/02:/' ansible.builtin.shell: set -o pipefail && openssl rand -hex 5 | sed 's/\(..\)/\1:/g; s/.$//' | sed 's/^/02:/'
changed_when: false changed_when: false
register: mac_address_output register: mac_address_output
- name: Render cloud config templates - name: Render cloud config templates
delegate_to: localhost delegate_to: localhost
template: ansible.builtin.template:
src: "{{ item.src }}" src: "{{ item.src }}"
dest: "/tmp/{{ item.dest_prefix }}-{{ hostname }}.yml" dest: /tmp/{{ item.dest_prefix }}-{{ hostname }}.yml
mode: '0644'
loop: loop:
- { src: "cloud-user-data.yml.j2", dest_prefix: "cloud-user-data" } - { src: cloud-user-data.yml.j2, dest_prefix: cloud-user-data }
- { src: "cloud-network-config.yml.j2", dest_prefix: "cloud-network-config" } - { src: cloud-network-config.yml.j2, dest_prefix: cloud-network-config }
- name: Create cloud-init disk - name: Create cloud-init disk
delegate_to: localhost delegate_to: localhost
command: "cloud-localds {{ vm_path | default('/var/lib/libvirt/images/') }}{{ hostname }}-cloudinit.iso /tmp/cloud-user-data-{{ hostname }}.yml -N /tmp/cloud-network-config-{{ hostname }}.yml" ansible.builtin.command: >
cloud-localds {{ vm_path | default('/var/lib/libvirt/images/') }}/{{ hostname }}-cloudinit.iso
/tmp/cloud-user-data-{{ hostname }}.yml
-N /tmp/cloud-network-config-{{ hostname }}.yml
changed_when: result.rc == 0
register: result
- name: Create VM using libvirt - name: Create VM using libvirt
delegate_to: localhost delegate_to: localhost
@@ -34,7 +43,7 @@
command: define command: define
xml: "{{ lookup('template', 'vm.xml.j2') }}" xml: "{{ lookup('template', 'vm.xml.j2') }}"
- name: start vm - name: Start vm
delegate_to: localhost delegate_to: localhost
community.libvirt.virt: community.libvirt.virt:
name: "{{ hostname }}" name: "{{ hostname }}"

View File

@@ -1,2 +1,3 @@
---
- name: Create Virtual Machine - name: Create Virtual Machine
include_tasks: "{{ hypervisor }}.yml" ansible.builtin.include_tasks: "{{ hypervisor }}.yml"

View File

@@ -1,48 +1,51 @@
---
- name: Deploy VM on Proxmox - name: Deploy VM on Proxmox
delegate_to: localhost delegate_to: localhost
proxmox_kvm: community.general.proxmox_kvm:
api_host: "{{ hypervisor_url }}" api_host: "{{ hypervisor_url }}"
api_user: "{{ hypervisor_username }}" api_user: "{{ hypervisor_username }}"
api_password: "{{ hypervisor_password }}" api_password: "{{ hypervisor_password }}"
ciuser: "{{ user_name }}" ciuser: "{{ user_name }}"
cipassword: "{{ user_password }}" cipassword: "{{ user_password }}"
node: "{{ hypervisor_node }}" # Proxmox node name ciupgrade: false
vmid: "{{ vm_id }}" # Unique ID for the VM node: "{{ hypervisor_node }}"
name: "{{ hostname }}" # Name of the VM vmid: "{{ vm_id }}"
cpu: "host" name: "{{ hostname }}"
cores: "{{ vm_cpus }}" # Number of CPU cores cpu: host
memory: "{{ vm_memory }}" # Memory size in MB cores: "{{ vm_cpus }}"
balloon: "{{ vm_ballo | default(omit) }}" # Minimum Memory size in MB memory: "{{ vm_memory }}"
balloon: "{{ vm_ballo | default(omit) }}"
numa_enabled: true numa_enabled: true
hotplug: "network,disk" hotplug: network,disk
bios: ovmf bios: ovmf
boot: "ac" boot: ac
scsihw: "virtio-scsi-single" scsihw: virtio-scsi-single
scsi: scsi:
scsi0: "{{ hypervisor_storage }}:{{ vm_size }}" # Disk configuration scsi0: "{{ hypervisor_storage }}:{{ vm_size }}"
efidisk0: efidisk0:
efitype: "4m" efitype: 4m
format: "raw" format: raw
pre_enrolled_keys: false pre_enrolled_keys: false
storage: "{{ hypervisor_storage }}" storage: "{{ hypervisor_storage }}"
ide: ide:
ide0: "{{ boot_iso }},media=cdrom" ide0: "{{ boot_iso }},media=cdrom"
ide1: "{{ hypervisor_storage }}:cloudinit" ide1: "{{ rhel_iso | default(omit) }},media=cdrom"
ide2: "{{ hypervisor_storage }}:cloudinit"
net: net:
net0: "virtio,bridge={{ vm_nif }}{% if vlan_name is defined and vlan_name %},tag={{ vlan_name }}{% endif %}" net0: virtio,bridge={{ vm_nif }}{% if vlan_name is defined and vlan_name %},tag={{ vlan_name }}{% endif %}
ipconfig: ipconfig:
ipconfig0: "ip={{ vm_ip }},gw={{ vm_gw }}" ipconfig0: ip={{ vm_ip }},gw={{ vm_gw }}
nameservers: "{{ vm_dns }}" nameservers: "{{ vm_dns }}"
onboot: true # Start the VM on boot onboot: true
state: present # Ensure the VM is present state: present
- name: Start VM on Proxmox - name: Start VM on Proxmox
delegate_to: localhost delegate_to: localhost
proxmox_kvm: community.general.proxmox_kvm:
api_host: "{{ hypervisor_url }}" api_host: "{{ hypervisor_url }}"
api_user: "{{ hypervisor_username }}" api_user: "{{ hypervisor_username }}"
api_password: "{{ hypervisor_password }}" api_password: "{{ hypervisor_password }}"
node: "{{ hypervisor_node }}" node: "{{ hypervisor_node }}"
name: "{{ hostname }}" name: "{{ hostname }}"
vmid: "{{ vm_id }}" vmid: "{{ vm_id }}"
state: started # Ensure the VM is present state: started

View File

@@ -1,15 +1,15 @@
- name: Create VM in vCenter - name: Create VM in vCenter
delegate_to: localhost delegate_to: localhost
vmware_guest: community.vmware.vmware_guest:
hostname: "{{ hypervisor_url }}" hostname: "{{ hypervisor_url }}"
username: "{{ hypervisor_username }}" username: "{{ hypervisor_username }}"
password: "{{ hypervisor_password }}" password: "{{ hypervisor_password }}"
validate_certs: no validate_certs: false
datacenter: "{{ hypervisor_cluster }}" datacenter: "{{ hypervisor_cluster }}"
cluster: "{{ hypervisor_node }}" cluster: "{{ hypervisor_node }}"
folder: "{{ vm_path }}" folder: "{{ vm_path }}"
name: "{{ hostname }}" name: "{{ hostname }}"
guest_id: "otherGuest64" guest_id: otherGuest64
state: poweredon state: poweredon
disk: disk:
- size_gb: "{{ vm_size }}" - size_gb: "{{ vm_size }}"
@@ -18,16 +18,28 @@
hardware: hardware:
memory_mb: "{{ vm_memory }}" memory_mb: "{{ vm_memory }}"
num_cpus: "{{ vm_cpus }}" num_cpus: "{{ vm_cpus }}"
boot_firmware: "efi" boot_firmware: efi
secure_boot: false secure_boot: false
cdrom: cdrom:
- controller_number: 0 - controller_number: 0
unit_number: 0 unit_number: 0
controller_type: "sata" controller_type: sata
state: present state: present
type: iso type: iso
iso_path: "{{ boot_iso }}" iso_path: "{{ boot_iso }}"
- controller_number: 0
unit_number: 1
controller_type: sata
state: present
type: iso
iso_path: "{{ rhel_iso | default(omit) }}"
networks: networks:
- vlan: "{{ vlan_name }}" - name: "{{ vm_nif }}"
type: dhcp type: dhcp
ignore_errors: yes vlan: "{{ vlan_name | default(omit) }}"
register: vmware_guest_result
failed_when:
- vmware_guest_result.failed is defined and vmware_guest_result.failed
- "'error' in vmware_guest_result"
- "'failed' in vmware_guest_result"
- vmware_guest_result.rc is defined and vmware_guest_result.rc != 0

View File

@@ -37,6 +37,13 @@
<source file="{{ vm_path | default('/var/lib/libvirt/images/') }}{{ hostname }}-cloudinit.iso"/> <source file="{{ vm_path | default('/var/lib/libvirt/images/') }}{{ hostname }}-cloudinit.iso"/>
<target dev="sdb" bus="sata"/> <target dev="sdb" bus="sata"/>
</disk> </disk>
{% if rhel_iso is defined %}
<disk type="file" device="cdrom">
<driver name="qemu" type="raw"/>
<source file="{{ rhel_iso }}"/>
<target dev="sdc" bus="sata"/>
</disk>
{% endif %}
<interface type='network'> <interface type='network'>
<mac address="{{ mac_address_output.stdout }}"/> <mac address="{{ mac_address_output.stdout }}"/>
<source network='default'/> <source network='default'/>

13
templates/rhel8.repo.j2 Normal file
View File

@@ -0,0 +1,13 @@
[rhel8-baseos]
name=RHEL 8 BaseOS
baseurl=file:///usr/local/install/redhat/dvd/BaseOS
enabled=1
gpgcheck=0
gpgkey=file:///usr/local/install/redhat/dvd/RPM-GPG-KEY-redhat-release
[rhel8-appstream]
name=RHEL 8 AppStream
baseurl=file:///usr/local/install/redhat/dvd/AppStream
enabled=1
gpgcheck=0
gpgkey=file:///usr/local/install/redhat/dvd/RPM-GPG-KEY-redhat-release

13
templates/rhel9.repo.j2 Normal file
View File

@@ -0,0 +1,13 @@
[rhel9-baseos]
name=RHEL 9 BaseOS
baseurl=file:///usr/local/install/redhat/dvd/BaseOS
enabled=1
gpgcheck=0
gpgkey=file:///usr/local/install/redhat/dvd/RPM-GPG-KEY-redhat-release
[rhel9-appstream]
name=RHEL 9 AppStream
baseurl=file:///usr/local/install/redhat/dvd/AppStream
enabled=1
gpgcheck=0
gpgkey=file:///usr/local/install/redhat/dvd/RPM-GPG-KEY-redhat-release

10
templates/rocky.repo.j2 Normal file
View File

@@ -0,0 +1,10 @@
[rocky-baseos]
name=Rocky Linux $releasever - BaseOS
mirrorlist=https://mirrors.rockylinux.org/mirrorlist?arch=$basearch&repo=BaseOS-$releasever
#baseurl=http://dl.rockylinux.org/$contentdir/$releasever/BaseOS/$basearch/os/
gpgcheck=1
enabled=1
countme=1
gpgkey=https://dl.rockylinux.org/pub/rocky/RPM-GPG-KEY-Rocky-$releasever
metadata_expire=86400
enabled_metadata=1

View File

@@ -1,12 +1,4 @@
ansible_user: "{{ user_name }}" vm_ip: "{{ inventory_hostname }}/{{ vm_nms }}"
ansible_password: "{{ user_password }}"
ansible_become_password: "{{ user_password }}"
ansible_ssh_extra_args: '-o UserKnownHostsFile=/dev/null -o StrictHostKeyChecking=no'
vm_ip: "{{ inventory_hostname }}/24"
hypervisor_list: ["libvirt", "proxmox", "vmware", "none"]
filesystem_list: ["btrfs", "ext4", "xfs"]
os_list: ["archlinux", "almalinux", "debian11", "debian12", "fedora"]
install_type: "virtual" install_type: "virtual"
cis: false cis: false
@@ -16,3 +8,14 @@ hypervisor_password: "SomePassword"
hypervisor_node: "NodeName" hypervisor_node: "NodeName"
hypervisor_storage: "local-btrfs" hypervisor_storage: "local-btrfs"
boot_iso: "local-btrfs:iso/archlinux-x86_64.iso" boot_iso: "local-btrfs:iso/archlinux-x86_64.iso"
rhel_iso: "local-btrfs:rhel-9.4-x86_64-dvd.iso"
# For VMware-Tools
ansible_vmware_host: "{{ hypervisor_url }}"
ansible_vmware_user: "{{ hypervisor_username }}"
ansible_vmware_password: "{{ hypervisor_password }}"
ansible_vmware_guest_path: "/{{ hypervisor_cluster }}/vm{{ vm_path }}/{{ hostname }}"
ansible_vmware_validate_certs: no
ansible_vmware_tools_user: "root"
ansible_vmware_tools_password: ""
vmware_ssh: true