Operations 40 min read

Master Ansible Automation: From Beginner to Expert with a Complete Project Walkthrough

This comprehensive guide walks you through every aspect of Ansible configuration automation—from fundamental concepts and installation to advanced playbook techniques, inventory management, modules, roles, Vault encryption, CI/CD integration, and enterprise‑level deployment patterns—empowering you to automate modern IT infrastructure with confidence.

Open Source Linux
Open Source Linux
Open Source Linux
Master Ansible Automation: From Beginner to Expert with a Complete Project Walkthrough

From Zero to Expert: Ansible Configuration Automation Full Guide (with Complete Project Example)

1. Introduction

In modern IT operations, configuration management automation has become a key technology for improving efficiency, ensuring consistency, and reducing human error. Ansible, as an open‑source configuration management tool, is favored by many operations engineers because of its simplicity, agent‑less architecture, and ease of use. This article provides a complete overview of Ansible configuration automation, covering everything from basic concepts to advanced applications, helping engineers master this powerful automation tool.

2. Ansible Basic Concepts

2.1 What is Ansible

Ansible is an open‑source automation platform for configuration management, application deployment, task automation, and IT orchestration. It uses an agent‑less architecture, executes tasks over SSH, and writes playbooks in YAML, offering low learning cost and simple deployment.

2.2 Core Components

Control Node

Machine where Ansible is installed

Executes playbooks and ad‑hoc commands

Manages the inventory file

Managed Node

Target machines managed by Ansible

SSH service must be enabled

Usually requires a Python environment

Inventory

Defines the list of managed nodes

Supports static and dynamic inventories

Allows host and group variables

Playbook

YAML‑written configuration file

Describes the execution flow of automation tasks

Contains one or more plays

Module

Execution unit of Ansible

Performs specific task operations

Includes thousands of built‑in modules

Plugin

Extends Ansible functionality

Includes connection plugins, filter plugins, etc.

3. Installation and Configuration

3.1 Installing Ansible

CentOS/RHEL

# Use EPEL repository
yum install epel-release -y
yum install ansible -y

# Or install via pip
pip install ansible

Ubuntu/Debian

# Add PPA repository
sudo apt-add-repository ppa:ansible/ansible
sudo apt update
sudo apt install ansible -y

# Or install via pip
pip install ansible

Verify Installation

ansible --version

3.2 Configuration File Structure

Ansible searches for configuration files in the following order: $ANSIBLE_CONFIG environment variable ansible.cfg in the current directory .ansible.cfg in the user’s home directory /etc/ansible/ansible.cfg system configuration file

Main configuration items

[defaults]
# Default inventory file path
inventory = /etc/ansible/hosts
# Remote user
remote_user = root
# SSH private key file
private_key_file = ~/.ssh/id_rsa
# Disable host key checking
host_key_checking = False
# Number of parallel forks
forks = 20
# Timeout (seconds)
timeout = 30
# Log file
log_path = /var/log/ansible.log

[ssh_connection]
# SSH arguments
ssh_args = -o ControlMaster=auto -o ControlPersist=60s
# Enable pipelining
pipelining = True

4. Inventory Management

4.1 Static Inventory

Basic format

# Single host
web1.example.com

# Host group
[webservers]
web1.example.com
web2.example.com
web3.example.com

[databases]
db1.example.com
db2.example.com

# Host variables
[webservers]
web1.example.com http_port=80 maxRequestsPerChild=808
web2.example.com http_port=8080 maxRequestsPerChild=909

# Group variables
[webservers:vars]
ntp_server=ntp.example.com
proxy=proxy.example.com

Host ranges

# Numeric range
[webservers]
web[01:50].example.com

# Alphabetic range
[databases]
db[a:f].example.com

# Multiple ranges
[servers]
server[01:20].example[1:3]

4.2 Dynamic Inventory

AWS EC2 dynamic inventory (Python example)

#!/usr/bin/env python3
import boto3, json

def get_ec2_instances():
    ec2 = boto3.client('ec2')
    response = ec2.describe_instances()
    inventory = {'_meta': {'hostvars': {}}, 'all': {'hosts': []}, 'webservers': {'hosts': []}, 'databases': {'hosts': []}}
    for reservation in response['Reservations']:
        for instance in reservation['Instances']:
            if instance['State']['Name'] == 'running':
                public_ip = instance.get('PublicIpAddress')
                if public_ip:
                    inventory['all']['hosts'].append(public_ip)
                    for tag in instance.get('Tags', []):
                        if tag['Key'] == 'Role':
                            role = tag['Value']
                            if role not in inventory:
                                inventory[role] = {'hosts': []}
                            inventory[role]['hosts'].append(public_ip)
    return inventory

if __name__ == '__main__':
    print(json.dumps(get_ec2_instances(), indent=2))

4.3 Host and Group Variables

Directory structure:

inventory/
├── hosts
├── group_vars/
│   ├── all.yml
│   ├── webservers.yml
│   └── databases.yml
└── host_vars/
    ├── web1.example.com.yml
    └── db1.example.com.yml

group_vars/all.yml

---
timezone: Asia/Shanghai
dns_servers:
  - 8.8.8.8
  - 8.8.4.4

group_vars/webservers.yml

---
http_port: 80
max_clients: 200
document_root: /var/www/html

group_vars/databases.yml

---
mysql_user: myapp
mysql_password: "{{ vault_database_password }}"
mysql_db: myapp

5. Ad‑hoc Commands

5.1 Basic Syntax

ansible <pattern> -m <module> -a <arguments>

5.2 Common Ad‑hoc Commands

Execute shell command

# Run on all hosts
ansible all -m shell -a "uptime"

# Run on a specific group
ansible webservers -m shell -a "systemctl status nginx"

# Use sudo privileges
ansible all -m shell -a "systemctl restart nginx" --become

File operations

# Copy file
ansible all -m copy -a "src=/tmp/test.txt dest=/tmp/test.txt"

# Create directory
ansible all -m file -a "path=/tmp/testdir state=directory"

# Change file permissions
ansible all -m file -a "path=/tmp/test.txt mode=0644"

Package management

# Install package
ansible all -m yum -a "name=nginx state=present"

# Remove package
ansible all -m yum -a "name=nginx state=absent"

# Update all packages
ansible all -m yum -a "name=* state=latest"

Service management

# Start service
ansible all -m service -a "name=nginx state=started"

# Stop service
ansible all -m service -a "name=nginx state=stopped"

# Restart service
ansible all -m service -a "name=nginx state=restarted"

6. Writing Playbooks

6.1 Basic Structure

---
- name: Configure Web Server
  hosts: webservers
  become: yes
  vars:
    http_port: 80
    max_clients: 20
  tasks:
    - name: Install Nginx
      yum:
        name: nginx
        state: present
    - name: Start Nginx service
      service:
        name: nginx
        state: started
        enabled: yes

6.2 Using Variables

Define variables

---
- name: Variable example
  hosts: all
  vars:
    packages:
      - nginx
      - mysql
      - php
    user_info:
      name: webapp
      group: webapp
      shell: /bin/bash
  tasks:
    - name: Install packages
      yum:
        name: "{{ packages }}"
        state: present
    - name: Create user
      user:
        name: "{{ user_info.name }}"
        group: "{{ user_info.group }}"
        shell: "{{ user_info.shell }}"

6.3 Conditional Statements

---
- name: Conditional example
  hosts: all
  tasks:
    - name: Install Apache on RedHat
      yum:
        name: httpd
        state: present
      when: ansible_os_family == "RedHat"
    - name: Install Apache on Debian
      apt:
        name: apache2
        state: present
      when: ansible_os_family == "Debian"
    - name: Check disk space
      shell: df -h /
      register: disk_usage
    - name: Warn if disk usage > 90%
      debug:
        msg: "Disk space low, please clean up"
      when: "'90%' in disk_usage.stdout"

6.4 Loops

---
- name: Loop example
  hosts: all
  tasks:
    - name: Install multiple packages
      yum:
        name: "{{ item }}"
        state: present
      loop:
        - nginx
        - mysql
        - php
        - redis
    - name: Create multiple users
      user:
        name: "{{ item.name }}"
        group: "{{ item.group }}"
        state: present
      loop:
        - { name: 'user1', group: 'web' }
        - { name: 'user2', group: 'db' }
        - { name: 'user3', group: 'app' }
    - name: Process dictionary loop
      debug:
        msg: "{{ item.key }}: {{ item.value }}"
      loop: "{{ lookup('dict', {'a': 1, 'b': 2, 'c': 3}) }}"

6.5 Error Handling

---
- name: Error handling example
  hosts: all
  tasks:
    - name: Try to start service
      service:
        name: nginx
        state: started
      ignore_errors: yes
    - name: Capture error output
      shell: /bin/false
      register: result
      failed_when: result.rc != 0
      ignore_errors: yes
    - name: Act on failure
      debug:
        msg: "Command failed"
      when: result.failed
    - name: Rescue block
      block:
        - name: Potentially failing task
          shell: /bin/false
      rescue:
        - name: Rescue task
          debug:
            msg: "Running rescue actions"
      always:
        - name: Always run task
          debug:
            msg: "Runs regardless of success or failure"

7. Module Details

7.1 System Modules

user module

- name: Create user
  user:
    name: webapp
    group: webapp
    shell: /bin/bash
    home: /home/webapp
    createhome: yes
    password: "{{ 'password' | password_hash('sha512') }}"

group module

- name: Create group
  group:
    name: webapp
    gid: 1001
    state: present

cron module

- name: Add scheduled task
  cron:
    name: "backup database"
    minute: "0"
    hour: "2"
    job: "/usr/local/bin/backup.sh"
    user: root

7.2 File Modules

file module

- name: Create directory
  file:
    path: /var/www/html
    state: directory
    owner: nginx
    group: nginx
    mode: '0755'

- name: Create symbolic link
  file:
    src: /var/www/html
    dest: /var/www/site
    state: link

copy module

- name: Copy file
  copy:
    src: /tmp/source.txt
    dest: /tmp/dest.txt
    owner: root
    group: root
    mode: '0644'
    backup: yes

template module

- name: Generate configuration file
  template:
    src: nginx.conf.j2
    dest: /etc/nginx/nginx.conf
    owner: root
    group: root
    mode: '0644'
    notify: restart nginx

7.3 Package Modules

yum module

- name: Install packages
  yum:
    name:
      - nginx
      - mysql-server
      - php
    state: present
    update_cache: yes

- name: Install specific version
  yum:
    name: nginx-1.18.0
    state: present

apt module

- name: Update package cache
  apt:
    update_cache: yes
    cache_valid_time: 3600

- name: Install packages
  apt:
    name: "{{ packages }}"
    state: present
  vars:
    packages:
      - nginx
      - mysql-server
      - php

7.4 Service Modules

service module

- name: Start and enable service
  service:
    name: nginx
    state: started
    enabled: yes

- name: Restart service
  service:
    name: nginx
    state: restarted

systemd module

- name: Reload systemd
  systemd:
    daemon_reload: yes

- name: Start service via systemd
  systemd:
    name: nginx
    state: started
    enabled: yes

8. Templates and Filters

8.1 Jinja2 Templates

# nginx.conf.j2
user {{ nginx_user }};
worker_processes {{ ansible_processor_cores }};
error_log /var/log/nginx/error.log;
pid /var/run/nginx.pid;

events {
    worker_connections {{ nginx_worker_connections }};
}

http {
    include /etc/nginx/mime.types;
    default_type application/octet-stream;

    {% for upstream in nginx_upstreams %}
    upstream {{ upstream.name }} {
        {% for server in upstream.servers %}
        server {{ server }};
        {% endfor %}
    }
    {% endfor %}

    server {
        listen {{ http_port }};
        server_name {{ server_name }};

        location / {
            root {{ document_root }};
            index index.html index.htm;
        }
    }
}

8.2 Common Filters

---
- name: Filter example
  hosts: all
  vars:
    items: [1, 2, 3, 4, 5]
    text: "Hello World"
  tasks:
    - name: Max value filter
      debug:
        msg: "{{ items | max }}"  # maximum value
    - name: Lowercase filter
      debug:
        msg: "{{ text | lower }}"  # to lower case
    - name: Default value filter
      debug:
        msg: "{{ undefined_var | default('default_value') }}"
    - name: Password hash filter
      debug:
        msg: "{{ 'password' | password_hash('sha512') }}"
    - name: Time filter
      debug:
        msg: "{{ ansible_date_time.iso8601 | to_datetime }}"

9. Handlers and Notifications

9.1 Basic Handler Usage

---
- name: Web server configuration
  hosts: webservers
  tasks:
    - name: Install Nginx
      yum:
        name: nginx
        state: present
    - name: Configure Nginx
      template:
        src: nginx.conf.j2
        dest: /etc/nginx/nginx.conf
        notify:
          - restart nginx
          - reload nginx
    - name: Ensure Nginx is running
      service:
        name: nginx
        state: started
        enabled: yes
  handlers:
    - name: restart nginx
      service:
        name: nginx
        state: restarted
    - name: reload nginx
      service:
        name: nginx
        state: reloaded

9.2 Conditional Handler

---
- name: Conditional handler example
  hosts: all
  tasks:
    - name: Check configuration file
      stat:
        path: /etc/nginx/nginx.conf
      register: nginx_config
    - name: Update configuration
      template:
        src: nginx.conf.j2
        dest: /etc/nginx/nginx.conf
      when: nginx_config.stat.exists
      notify: restart nginx
  handlers:
    - name: restart nginx
      service:
        name: nginx
        state: restarted
      when: nginx_config.stat.exists

10. Roles

10.1 Role Structure

roles/
└── nginx/
    ├── tasks/
    │   └── main.yml
    ├── handlers/
    │   └── main.yml
    ├── templates/
    │   └── nginx.conf.j2
    ├── files/
    │   └── index.html
    ├── vars/
    │   └── main.yml
    ├── defaults/
    │   └── main.yml
    ├── meta/
    │   └── main.yml
    └── README.md

10.2 Creating a Role

tasks/main.yml

---
- name: Install Nginx
  yum:
    name: nginx
    state: present

- name: Create configuration file
  template:
    src: nginx.conf.j2
    dest: /etc/nginx/nginx.conf
    notify: restart nginx

- name: Start Nginx service
  service:
    name: nginx
    state: started
    enabled: yes

handlers/main.yml

---
- name: restart nginx
  service:
    name: nginx
    state: restarted

defaults/main.yml

---
nginx_user: nginx
nginx_worker_connections: 1024
http_port: 80
document_root: /usr/share/nginx/html

meta/main.yml

---
galaxy_info:
  author: your-name
  description: Nginx web server role
  company: your-company
  license: MIT
  min_ansible_version: 2.9
  platforms:
    - name: EL
      versions:
        - 7
        - 8
    - name: Ubuntu
      versions:
        - 18.04
        - 20.04
  dependencies: []

10.3 Using a Role

---
- name: Deploy Web Server
  hosts: webservers
  roles:
    - nginx
    - mysql
    - php

- name: Deploy Web Server with variables
  hosts: webservers
  roles:
    - role: nginx
      nginx_user: www-data
      http_port: 8080

11. Vault Encryption

11.1 Creating Encrypted Files

# Create encrypted file
ansible-vault create secret.yml

# Edit encrypted file
ansible-vault edit secret.yml

# View encrypted file
ansible-vault view secret.yml

# Encrypt existing file
ansible-vault encrypt existing_file.yml

# Decrypt file
ansible-vault decrypt secret.yml

11.2 Using Encrypted Variables

Encrypted variable file (vault.yml)

---
vault_database_password: secretpassword123
vault_api_key: abc123def456

Use in playbook

---
- name: Use encrypted variables
  hosts: databases
  vars_files:
    - vault.yml
  tasks:
    - name: Create database user
      mysql_user:
        name: myapp
        password: "{{ vault_database_password }}"
        priv: "myapp.*:ALL"
        state: present

Run playbook

ansible-playbook -i inventory site.yml --ask-vault-pass
# Or using a password file
ansible-playbook -i inventory site.yml --vault-password-file ~/.vault_pass

12. Advanced Features

12.1 Strategy Control

---
- name: Strategy control example
  hosts: all
  strategy: free  # free strategy, each host runs independently
  tasks:
    - name: Long‑running task
      shell: sleep 30
    - name: Fast task
      debug:
        msg: "This task finishes quickly"

12.2 Delegation and Proxy

---
- name: Delegated execution example
  hosts: webservers
  tasks:
    - name: Check service status on monitoring server
      uri:
        url: "http://{{ inventory_hostname }}/health"
        method: GET
      delegate_to: monitor.example.com
    - name: Run command locally
      shell: echo "Running on control node"
      delegate_to: localhost
      run_once: true

12.3 Rolling Updates

---
- name: Rolling update example
  hosts: webservers
  serial: 2  # process two servers at a time
  tasks:
    - name: Remove from load balancer
      uri:
        url: "http://lb.example.com/remove/{{ inventory_hostname }}"
        method: POST
      delegate_to: localhost
    - name: Update application
      yum:
        name: myapp
        state: latest
      notify: restart myapp
    - name: Wait for service to start
      wait_for:
        port: 8080
        delay: 10
    - name: Add back to load balancer
      uri:
        url: "http://lb.example.com/add/{{ inventory_hostname }}"
        method: POST
      delegate_to: localhost
  handlers:
    - name: restart myapp
      service:
        name: myapp
        state: restarted

12.4 Asynchronous Tasks

---
- name: Asynchronous task example
  hosts: all
  tasks:
    - name: Long‑running task
      shell: /opt/long_running_script.sh
      async: 3600   # max runtime in seconds
      poll: 0       # do not wait for result
      register: long_task
    - name: Continue with other tasks
      debug:
        msg: "Continuing with other tasks"
    - name: Check async task status
      async_status:
        jid: "{{ long_task.ansible_job_id }}"
      register: job_result
      until: job_result.finished
      retries: 30
      delay: 10

13. Best Practices

13.1 Directory Structure

ansible-project/
├── inventories/
│   ├── production/
│   │   ├── hosts
│   │   └── group_vars/
│   └── staging/
│       ├── hosts
│       └── group_vars/
├── roles/
│   ├── common/
│   ├── nginx/
│   └── mysql/
├── playbooks/
│   ├── site.yml
│   ├── webservers.yml
│   └── databases.yml
├── group_vars/
│   └── all.yml
├── host_vars/
├── vault/
│   └── secrets.yml
├── files/
├── templates/
├── library/
├── filter_plugins/
└── ansible.cfg

13.2 Naming Conventions

---
# Use descriptive names
- name: Install and configure Nginx web server
  hosts: webservers
  tasks:
    - name: Install Nginx package
      yum:
        name: nginx
        state: present
    - name: Create Nginx configuration file
      template:
        src: nginx.conf.j2
        dest: /etc/nginx/nginx.conf
        notify: Restart Nginx service
    - name: Start and enable Nginx service
      service:
        name: nginx
        state: started
        enabled: yes
  handlers:
    - name: Restart Nginx service
      service:
        name: nginx
        state: restarted

13.3 Variable Management

# Use clear variable hierarchy
nginx_config:
  user: nginx
  worker_processes: auto
  worker_connections: 1024
  keepalive_timeout: 65

server:
  listen: 80
  server_name: example.com
  root: /var/www/html
  index: index.html

upstream:
  name: backend
  servers:
    - 192.168.1.10:8080
    - 192.168.1.11:8080

13.4 Error Handling

---
- name: Robust error handling
  hosts: all
  tasks:
    - name: Check system requirements
      assert:
        that:
          - ansible_distribution_major_version|int >= 7
          - ansible_memtotal_mb >= 1024
        fail_msg: "System does not meet minimum requirements"
    - name: Backup configuration file
      copy:
        src: /etc/nginx/nginx.conf
        dest: /etc/nginx/nginx.conf.bak
        remote_src: yes
      ignore_errors: yes
    - name: Update configuration
      template:
        src: nginx.conf.j2
        dest: /etc/nginx/nginx.conf
        validate: nginx -t -c %s
      notify: restart nginx

14. Monitoring and Logging

14.1 Log Configuration

# ansible.cfg
[defaults]
log_path = /var/log/ansible.log
display_skipped_hosts = False
display_ok_hosts = False

14.2 Callback Plugins

# callback_plugins/timer.py
from ansible.plugins.callback import CallbackBase
from datetime import datetime

class CallbackModule(CallbackBase):
    def __init__(self):
        super(CallbackModule, self).__init__()
        self.start_time = datetime.now()

    def v2_playbook_on_task_start(self, task, is_conditional):
        self.task_start_time = datetime.now()

    def v2_runner_on_ok(self, result):
        duration = datetime.now() - self.task_start_time
        self._display.display(f"Task completed in {duration.total_seconds():.2f}s")

14.3 Performance Monitoring

---
- name: Performance monitoring example
  hosts: all
  tasks:
    - name: Gather system information
      setup:
        gather_subset:
          - hardware
          - network
          - virtual
    - name: Record performance metrics
      debug:
        msg: |
          CPU cores: {{ ansible_processor_cores }}
          Memory: {{ ansible_memtotal_mb }}MB
          Disk usage: {{ ansible_mounts | selectattr('mount', 'equalto', '/') | map(attribute='size_available') | first }}

15. CI/CD Integration

15.1 GitLab CI Integration

# .gitlab-ci.yml
stages:
  - test
  - deploy

ansible-lint:
  stage: test
  script:
    - ansible-lint playbooks/site.yml
  only:
    - merge_requests

deploy-staging:
  stage: deploy
  script:
    - ansible-playbook -i inventories/staging/hosts playbooks/site.yml --vault-password-file $VAULT_PASSWORD_FILE
  environment:
    name: staging
    url: https://staging.example.com
  only:
    - develop

deploy-production:
  stage: deploy
  script:
    - ansible-playbook -i inventories/production/hosts playbooks/site.yml --vault-password-file $VAULT_PASSWORD_FILE
  environment:
    name: production
    url: https://production.example.com
  when: manual
  only:
    - master

15.2 Jenkins Integration

// Jenkinsfile
pipeline {
    agent any
    environment {
        ANSIBLE_HOST_KEY_CHECKING = 'False'
        VAULT_PASSWORD_FILE = credentials('ansible-vault-password')
    }
    stages {
        stage('Checkout') {
            steps { checkout scm }
        }
        stage('Syntax Check') {
            steps { sh 'ansible-playbook --syntax-check playbooks/site.yml' }
        }
        stage('Lint') {
            steps { sh 'ansible-lint playbooks/site.yml' }
        }
        stage('Deploy to Staging') {
            steps {
                sh '''
                    ansible-playbook -i inventories/staging/hosts \
                    playbooks/site.yml \
                    --vault-password-file $VAULT_PASSWORD_FILE
                '''
            }
            environment {
                name = 'staging'
                url = 'https://staging.example.com'
            }
        }
        stage('Deploy to Production') {
            when { branch 'master' }
            steps {
                input message: 'Deploy to Production?'
                sh '''
                    ansible-playbook -i inventories/production/hosts \
                    playbooks/site.yml \
                    --vault-password-file $VAULT_PASSWORD_FILE
                '''
            }
            environment { name = 'production' }
        }
    }
    post {
        always { archiveArtifacts artifacts: 'logs/*.log', allowEmptyArchive: true }
        failure { emailext subject: "Ansible Deployment Failed: ${env.JOB_NAME} - ${env.BUILD_NUMBER}", body: "Build failed. Check console output at ${env.BUILD_URL}", to: "${env.CHANGE_AUTHOR_EMAIL}" }
    }
}

15.3 GitHub Actions Integration

# .github/workflows/deploy.yml
name: Deploy with Ansible

on:
  push:
    branches: [ main, develop ]
  pull_request:
    branches: [ main ]

jobs:
  test:
    runs-on: ubuntu-latest
    steps:
      - uses: actions/checkout@v2
      - name: Set up Python
        uses: actions/setup-python@v2
        with: { python-version: '3.8' }
      - name: Install dependencies
        run: |
          pip install ansible ansible-lint
      - name: Run ansible-lint
        run: ansible-lint playbooks/site.yml
      - name: Syntax check
        run: ansible-playbook --syntax-check playbooks/site.yml

  deploy-staging:
    needs: test
    runs-on: ubuntu-latest
    if: github.ref == 'refs/heads/develop'
    steps:
      - uses: actions/checkout@v2
      - name: Deploy to Staging
        env:
          ANSIBLE_HOST_KEY_CHECKING: False
          VAULT_PASSWORD: ${{ secrets.VAULT_PASSWORD }}
        run: |
          echo "$VAULT_PASSWORD" > .vault_pass
          ansible-playbook -i inventories/staging/hosts playbooks/site.yml --vault-password-file .vault_pass
          rm .vault_pass

  deploy-production:
    needs: test
    runs-on: ubuntu-latest
    if: github.ref == 'refs/heads/main'
    environment: production
    steps:
      - uses: actions/checkout@v2
      - name: Deploy to Production
        env:
          ANSIBLE_HOST_KEY_CHECKING: False
          VAULT_PASSWORD: ${{ secrets.VAULT_PASSWORD }}
        run: |
          echo "$VAULT_PASSWORD" > .vault_pass
          ansible-playbook -i inventories/production/hosts playbooks/site.yml --vault-password-file .vault_pass
          rm .vault_pass

16. Troubleshooting

16.1 Common Issues and Solutions

SSH connection problems

# Check SSH connectivity
ansible all -m ping

# Verbose output
ansible all -m ping -vvv

# Skip host key checking
export ANSIBLE_HOST_KEY_CHECKING=False

Permission issues

# Use sudo
ansible all -m shell -a "systemctl status nginx" --become

# Specify sudo user
ansible all -m shell -a "systemctl status nginx" --become-user=root

Module not found

# Check module path
ansible-doc -l | grep module_name

# Install missing Python module
pip install required_module

16.2 Debugging Techniques

---
- name: Debug example
  hosts: all
  tasks:
    - name: Gather facts
      setup:
      register: facts
    - name: Show variable content
      debug:
        var: facts
      verbosity: 2
    - name: Conditional debug
      debug:
        msg: "Disk space low"
      when:
        - ansible_mounts | selectattr('mount', 'equalto', '/') | map(attribute='size_available') | first < 1000000000
    - name: Pause execution
      pause:
        prompt: "Check system status and press Enter to continue"
      when: debug_mode | default(false)

16.3 Performance Optimization

# ansible.cfg optimization
[defaults]
forks = 50  # increase parallelism
#gathering = smart
fact_caching = jsonfile
fact_caching_connection = /tmp/ansible_facts_cache
fact_caching_timeout = 3600

[ssh_connection]
# Enable SSH pipelining
pipelining = True
# Control connection reuse
ssh_args = -o ControlMaster=auto -o ControlPersist=300s
control_path = /tmp/ansible-ssh-%h-%p-%r

17. Security Considerations

17.1 Permission Control

---
- name: Security configuration example
  hosts: all
  become: yes
  become_method: sudo
  tasks:
    - name: Create dedicated user
      user:
        name: ansible
        groups: wheel
        shell: /bin/bash
        create_home: yes
    - name: Configure sudo permissions
      lineinfile:
        path: /etc/sudoers.d/ansible
        line: 'ansible ALL=(ALL) NOPASSWD: ALL'
        create: yes
        validate: 'visudo -cf %s'
    - name: Configure SSH key
      authorized_key:
        user: ansible
        key: "{{ lookup('file', '~/.ssh/id_rsa.pub') }}"
        state: present

17.2 Protecting Sensitive Information

---
- name: Sensitive data handling
  hosts: all
  vars:
    sensitive_data: "{{ vault_sensitive_data }}"
  tasks:
    - name: Use sensitive data
      shell: echo "{{ sensitive_data }}"
      no_log: true
    - name: Process password
      user:
        name: dbuser
        password: "{{ vault_db_password | password_hash('sha512') }}"
      no_log: true

17.3 Network Security

---
- name: Network security configuration
  hosts: all
  tasks:
    - name: Configure firewall
      firewalld:
        service: "{{ item }}"
        permanent: yes
        state: enabled
        immediate: yes
      loop:
        - ssh
        - http
        - https
    - name: Restrict SSH access
      lineinfile:
        path: /etc/ssh/sshd_config
        regexp: '^#?PasswordAuthentication'
        line: 'PasswordAuthentication no'
      notify: restart sshd
    - name: Configure SELinux
      selinux:
        policy: targeted
        state: enforcing
  handlers:
    - name: restart sshd
      service:
        name: sshd
        state: restarted

18. Enterprise Deployment Cases

18.1 LAMP Architecture Automation

---
- name: Deploy LAMP stack
  hosts: webservers
  become: yes
  vars:
    mysql_root_password: "{{ vault_mysql_root_password }}"
    app_user: webapp
    app_group: webapp
  roles:
    - common
    - apache
    - mysql
    - php
    - application
  post_tasks:
    - name: Verify deployment
      uri:
        url: "http://{{ inventory_hostname }}/health"
        method: GET
        status_code: 200
      delegate_to: localhost

18.2 Docker Container Deployment

---
- name: Docker container deployment
  hosts: docker_hosts
  become: yes
  tasks:
    - name: Install Docker
      yum:
        name: docker
        state: present
    - name: Start Docker service
      service:
        name: docker
        state: started
        enabled: yes
    - name: Pull application image
      docker_image:
        name: "{{ app_image }}"
        tag: "{{ app_version }}"
        source: pull
    - name: Deploy application container
      docker_container:
        name: "{{ app_name }}"
        image: "{{ app_image }}:{{ app_version }}"
        state: started
        restart_policy: always
        ports:
          - "80:8080"
        env:
          DB_HOST: "{{ db_host }}"
          DB_PASSWORD: "{{ vault_db_password }}"
    - name: Wait for application to start
      wait_for:
        port: 80
        delay: 10

18.3 Kubernetes Cluster Deployment

---
- name: Kubernetes cluster deployment
  hosts: k8s_masters
  become: yes
  tasks:
    - name: Initialize cluster
      shell: |
        kubeadm init --pod-network-cidr=10.244.0.0/16
      register: kubeadm_init
      run_once: true
    - name: Save join command
      set_fact:
        join_command: "{{ kubeadm_init.stdout_lines | select('match', 'kubeadm join.*') | first }}"
      run_once: true
    - name: Configure kubectl
      shell: |
        mkdir -p $HOME/.kube
        cp -i /etc/kubernetes/admin.conf $HOME/.kube/config
        chown $(id -u):$(id -g) $HOME/.kube/config
      run_once: true
    - name: Deploy network plugin
      shell: kubectl apply -f https://raw.githubusercontent.com/coreos/flannel/master/Documentation/kube-flannel.yml
      run_once: true

- name: Join worker nodes
  hosts: k8s_workers
  become: yes
  tasks:
    - name: Join cluster
      shell: "{{ hostvars[groups['k8s_masters'][0]]['join_command'] }}"

19. Conclusion

Ansible, as a crucial tool for modern IT infrastructure automation, provides operations engineers with powerful and flexible configuration management capabilities. This article demonstrates Ansible's advantages:

Simplicity : YAML syntax, low learning curve, no complex programming required.

Agent‑less architecture : No client installation on target hosts, reducing deployment complexity.

Rich module ecosystem : Thousands of built‑in modules covering all aspects of system management, with support for custom extensions.

Scalability : Suitable for both small‑scale and large‑scale infrastructure management.

DevOps integration : Seamlessly integrates with CI/CD pipelines, monitoring systems, container platforms, enabling end‑to‑end automation workflows.

In practice, operations engineers are advised to master Ansible progressively through the following steps:

Fundamental learning : Understand core concepts, module usage, and playbook authoring.

Hands‑on practice : Start with simple system configuration and advance to multi‑tier architecture deployments.

Best practices : Follow naming conventions, directory structures, security hardening, and other proven guidelines.

Advanced features : Explore Roles, Vault, strategy control, and other advanced capabilities.

Integration optimization : Incorporate Ansible into existing DevOps workflows for full automation.

As cloud computing, containerization, and micro‑service architectures continue to evolve, the importance of configuration management automation becomes increasingly evident. Ansible, with its simplicity, power, and flexibility, will play an even more pivotal role in future IT operations automation. Mastering Ansible not only improves operational efficiency but also lays a solid foundation for the professional growth of operations engineers.

We hope this guide helps readers fully understand and master Ansible configuration automation techniques, enabling flexible application in real‑world projects and elevating automation capabilities.

AutomationConfiguration ManagementdevopsAnsiblePlaybooks
Open Source Linux
Written by

Open Source Linux

Focused on sharing Linux/Unix content, covering fundamentals, system development, network programming, automation/operations, cloud computing, and related professional knowledge.

0 followers
Reader feedback

How this landed with the community

Sign in to like

Rate this article

Was this worth your time?

Sign in to rate
Discussion

0 Comments

Thoughtful readers leave field notes, pushback, and hard-won operational detail here.