ansible-cmdb — Структуризация и визуализация ansible facts
Приветствую!

ansible-cmdb — программа на Python, которая структуризирует собранные Ansible facts и визуализирует их с помощью HTML с красивым форматированием.

Ansible facts — это данные о серверах (хостах Ansible), которые собираются с помощью модуля setup. Они представляют собой «факты» о системе, например данные об ОС, сетевых интерфейсах, блочных устройствах и других компонентах системы.

С помощью ansible-cmdb также можно экспортировать структуризированные данные в файлы следующих форматов: json, csv, sql, txt_table. Такие файлы удобно использовать для анализа, хранения или дальнейшей обработки. Но это тема для отдельной статьи😉.

В этой заметке я расскажу📋:

Предполагается, что вы знаете, что такое Ansible и умеете им пользоваться🙄. Если нет, то рекомендую к прочтению мои статьи:

Все примеры из статьи выполнялись в среде дистрибутива Linux Mint Debian Edition 6 🍃 (Debian 12). Во всех остальных дистрибутивах процесс выглядит +- аналогично.

Установка ansible-cmdb

Первым делом переходим в рабочую директорию Ansible, где у вас находятся файл инвентаризации и конфигурации (у меня это ~/ansible):

BASH
cd ~/ansible/
Нажмите, чтобы развернуть и увидеть больше

Как уже говорил ранее, ansible-cmdb, как и сам Ansible, написан на языке программирования Python🐍. Поэтому рекомендуется создать виртуальное окружение venv 📦 и уже внутри него выполнять установку ansible-cmdb:

Способ #1 ‐ venv, если у вас версия Python ниже 3.12

Выполняем в терминале:

BASH
mkdir -p ./cmdb/venv

python3 -m venv ./cmdb/venv/ansible-cmdb

./cmdb/venv/ansible-cmdb/bin/pip install ansible-cmdb
Нажмите, чтобы развернуть и увидеть больше

Способ #2 — docker, если у вас версия Python 3.12 и выше

К сожалению ansible-cmdb давненько не обновлялся, а в Python 3.12 библиотека imp стала deprecated. В таком случае для запуска ansible-cmdb необходимо установить более раннюю версию Python.

Универсальным способом сделать это является Docker:

BASH
mkdir -p ./cmdb/venv

docker run -u $(id -u):$(id -g) -v ./cmdb:/cmdb python:3.11-alpine python -m venv /cmdb/venv/ansible-cmdb

docker run -u $(id -u):$(id -g) -v ./cmdb:/cmdb python:3.11-alpine /cmdb/venv/ansible-cmdb/bin/pip install ansible-cmdb
Нажмите, чтобы развернуть и увидеть больше

В конце установки должно быть подобное сообщение:

PLAINTEXT
Installing collected packages: ushlex, jsonxs, pyyaml, MarkupSafe, mako, ansible-cmdb
Successfully installed MarkupSafe-3.0.2 ansible-cmdb-1.31 jsonxs-0.6 mako-1.3.10 pyyaml-6.0.2 ushlex-0.99.1
Нажмите, чтобы развернуть и увидеть больше

Чтобы в будущем каждый раз не активировать venv (или не заходить в docker) для работы с ansible-cmdb создадим небольшой скрипт.

Если устанавливали через venv (Способ #1):

BASH
cat << EOF > "${PWD}"/cmdb/ansible_cmdb.sh
#!/bin/bash
"${PWD}/cmdb/venv/ansible-cmdb/bin/python" "${PWD}/cmdb/venv/ansible-cmdb/lib/ansiblecmdb/ansible-cmdb.py" "\$@"
EOF
Нажмите, чтобы развернуть и увидеть больше

Если устанавливали в Docker (Способ #2):

BASH
cat << EOF > "${PWD}"/cmdb/ansible_cmdb.sh
#!/bin/bash
docker run -v "${PWD}/cmdb":"/cmdb" python:3.11-alpine /cmdb/venv/ansible-cmdb/bin/python /cmdb/venv/ansible-cmdb/lib/ansiblecmdb/ansible-cmdb.py "\$@"
EOF
Нажмите, чтобы развернуть и увидеть больше

Сделаем его исполняемым и добавим символьную ссылку на него в директории ~/.local/bin для удобства:

BASH
chmod +x ./cmdb/ansible_cmdb.sh

ln -s "${PWD}"/cmdb/ansible_cmdb.sh ~/.local/bin/ansible-cmdb
Нажмите, чтобы развернуть и увидеть больше

Проверяем:

BASH
ansible-cmdb --help
Нажмите, чтобы развернуть и увидеть больше
PLAINTEXT
Usage: ansible-cmdb.py [option] <dir> > output.html

Options:
  --version             show program's version number and exit
  -h, --help            show this help message and exit
  -t TEMPLATE, --template=TEMPLATE
                        Template to use. Default is 'html_fancy'
  -i INVENTORY, --inventory=INVENTORY
                        Inventory to read extra info from
  -f, --fact-cache      <dir> contains fact-cache files
  -p PARAMS, --params=PARAMS
                        Params to send to template
  -d, --debug           Show debug output
  -q, --quiet           Don't report warnings
  -c COLUMNS, --columns=COLUMNS
                        Show only given columns
  -C CUST_COLS, --cust-cols=CUST_COLS
                        Path to a custom columns definition file
  -l LIMIT, --limit=LIMIT
                        Limit hosts to pattern
  --exclude-cols=EXCLUDE_COLUMNS
                        Exclude cols from output
Нажмите, чтобы развернуть и увидеть больше

Пример работы ansible-cmdb

Создаем директорию для базовых фактов и запускаем их сбор с помощью ansible и модуля setup:

BASH
mkdir -p ./cmdb/facts/basic

ansible -m setup --tree ./cmdb/facts/basic all &> /dev/null
Нажмите, чтобы развернуть и увидеть больше

Теперь с помощью ansible-cmdb генерируем HTML файл на основе собранных данных:

BASH
ansible-cmdb -p collapsed=1 ./cmdb/facts/basic > ./cmdb/cmdb.html
Нажмите, чтобы развернуть и увидеть больше

Теперь просто открываем полученный файл в браузере:

Аккуратно и удобно. На странице есть интерактивные элементы: поиск, фильтрация и пр. Очень удобно😏.

Для генерации json | csv | sql | txt_table | markdown команда будет примерно такой:

BASH
ansible-cmdb -t csv ./cmdb/facts/basic > ./cmdb/cmdb.csv
Нажмите, чтобы развернуть и увидеть больше
PLAINTEXT
"Name","OS","IP","Arch","Mem","MemFree","MemUsed","CPUs","Virt","Disk avail"
"host1","Debian 12.7","10.1.1.10","x86_64/x86_64","8g","0g","1g","4","kvm/guest","8.3g, 0.2g"
"host2","Debian 12.10","0.1.1.11","x86_64/x86_64","8g","2g","2g",6","kvm/guest","7.8g, 0.3g"
"host3","Debian 12.8","0.1.1.12","x86_64/x86_64","8g","0g","3g","4","kvm/guest","1.9g"
"host4","Debian 12.6","0.1.1.13","x86_64/x86_64","8g","0g","1g","8","kvm/guest","8.7g, 0.2g"
"host5","Debian 12.7","0.1.1.14","x86_64/x86_64","8g","0g","1g","2","kvm/guest","9.0g, 0.2g"
"host6","Debian 12.6","0.1.1.15","x86_64/x86_64","8g","0g","1g","2","kvm/guest","0.8g, 3.7g, 3.7g, 0.4g"
Нажмите, чтобы развернуть и увидеть больше

Стоит предупредить, что полный список фактов доступен только в шаблоне JSON. Остальные форматы имеют ограниченное количество полей, опеределенных в шаблонах, которые расположены по пути: ./cmdb/venv/ansible-cmdb/lib/ansiblecmdb/data/tpl. При необходимости их можно кастомизировать под себя.

Добавление кастомных фактов

ansible-cmdb предоставляет гибкие возможности по кастомизации итогового файла👨🎨.

Покажу на примере добавления своего факта в виде списка маршрутов хоста, который Ansible по умолчанию не собирает🤷♂️.

Моя реализация выполнена с помощью скрипта на Bash, который в выводе дает список маршрутов хоста в JSON формате📃.

Приступим👨🏭. Создаем файл скрипта:

BASH
nvim ./cmdb/routes_to_json.sh
Нажмите, чтобы развернуть и увидеть больше

Наполняем его таким содержимым:

BASH
#!/bin/bash

# Получаем маршруты в массив
mapfile -t routes < <(ip r)

# Начало JSON
RESULT='{"custom_facts": {"IP Routes": ['

# Добавляем маршруты
for i in "${!routes[@]}"; do
    # Экранируем спецсимволы
    line="${routes[$i]//\\/\\\\}"
    line="${line//\"/\\\"}"
    
    # Добавляем маршрут как отдельный объект
    RESULT+="{\"#$((i+1))\": \"$line\"}"
    
    # Добавляем запятую между элементами (кроме последнего)
    [ $i -lt $((${#routes[@]} - 1)) ] && RESULT+=","
done

# Завершаем JSON
RESULT+=']}}'

# Выводим результат
echo "$RESULT"
Нажмите, чтобы развернуть и увидеть больше

Делаем исполняемым:

BASH
chmod +x ./cmdb/routes_to_json.sh
Нажмите, чтобы развернуть и увидеть больше

Проверяем:

BASH
./cmdb/routes_to_json.sh | python3 -m json.tool
Нажмите, чтобы развернуть и увидеть больше

Вывод должен быть примерно таким:

JSON
{
    "custom_facts": {
        "IP Routes": [
            {
                "#1": "default dev vpn0 proto static scope link metric 50 "
            },
            {
                "#2": "default via 10.10.10.1 dev eth0 proto dhcp src 10.10.10.30 metric 600 "
            },
            {
                "#3": "10.11.11.0/24 dev vpn0 proto kernel scope link src 10.11.11.15 metric 50 "
            }
        ]
    }
}
Нажмите, чтобы развернуть и увидеть больше

Теперь создаем новый плейбук, который будет собирать наш кастомный факт (запуск скрипта выше на удалённых хостах и сбор вывода в локальные файлы):

BASH
nvim ./cmdb/get_custom_facts.yml
Нажмите, чтобы развернуть и увидеть больше

Наполняем:

BASH
---

- name: Get custom facts
  hosts: all
  gather_facts: false

  tasks:

    - name: Run script to get info about ip routes
      script: "~/ansible/cmdb/routes_to_json.sh"
      register: script_output

    - name: Create local output files with ip routes info
      local_action: copy content="{{ script_output.stdout }}" dest="~/ansible/cmdb/facts/custom/{{ inventory_hostname }}"
Нажмите, чтобы развернуть и увидеть больше

Теперь создаем отдельную директорию под кастомные факты, собираем их с помощью ansible-playbook и генерируем обновленный HTML с помощью ansible-cmdb:

BASH
mkdir -p ./cmdb/facts/custom

ansible-playbook ./cmdb/get_custom_facts.yml

ansible-cmdb -p collapsed=1 ./cmdb/facts/{basic,custom} > ./cmdb/cmdb.html
Нажмите, чтобы развернуть и увидеть больше

В итоговом HTML появится новый раздел «Custom facts». Как по мне выглядит неплохо😌:

ansible-cmdb позволяет настраивать свои столбцы в HTML и много чего еще. Рекомендую подробнее изучить документацию📚 (ссылки в конце статьи).

Доступ к cmdb.html по урлу в Nginx

Если вам необходимо предоставлять данную информацию кому-то еще, то можно добавить итоговый HTML в виде отдельной страницы на вашем веб сервере, например Nginx 🌐.

Если веб сервер работает от ограниченного в правах пользователя, проще всего будет создать хардлинк на ваш файл в корне сайта Nginx (условно корень /var/www/html:

BASH
sudo ln "${PWD}"/cmdb/cmdb.html /var/www/html/
Нажмите, чтобы развернуть и увидеть больше

А в конфиге Nginx, добавьте, например, такой location:

BASH
server {
    listen 80;
    server_name localhost;

    root /var/www/html;
    index index.html index.htm;

    location / {
        try_files $uri $uri.html $uri/ =404;
    }
}
Нажмите, чтобы развернуть и увидеть больше

Как то так:

Автоматизация сбора/обновления фактов

Без автоматизации в мире администрирования никуда🙂↔️. Напишем скрипт сбора/обновления фактов и настроим его ежедневный запуск в 01:00 по таймеру systemd в пространстве нашего пользователя (--user).

Создаем файл скрипта:

BASH
nvim ./cmdb/update_cmdb.sh
Нажмите, чтобы развернуть и увидеть больше

Наполняем:

BASH
#!/usr/bin/env bash

set -Eeuo pipefail

export PATH="$PATH:$HOME/.local/bin"

WORK_DIR="$HOME/ansible/"

echo "Script started"

cd "$WORK_DIR"

echo "Get basic facts"
ansible -m setup --tree ./cmdb/facts/basic all &> /dev/null || true

echo "Get custom facts"
ansible-playbook ./cmdb/get_custom_facts.yml &> /dev/null || true

echo "Generate cmdb html file"
ansible-cmdb -p collapsed=1 ./cmdb/facts/{basic,custom} > ./cmdb/cmdb.html

echo "Script done"
echo "--------------------------------------"
Нажмите, чтобы развернуть и увидеть больше

Сохраняем, делаем файл исполняемым, создаем линк в директорию ~/.local/bin для удобства и проверяем:

BASH
chmod +x ./cmdb/update_cmdb.sh

ln -s "${PWD}"/cmdb/update_cmdb.sh ~/.local/bin/update-cmdb

update-cmdb

# время изменения
stat -c '%y' ./cmdb/cmdb.html
Нажмите, чтобы развернуть и увидеть больше

Далее создаем systemd юнит сервиса:

BASH
systemctl --user edit --full --force update-cmdb.service
Нажмите, чтобы развернуть и увидеть больше

Наполняем:

BASH
[Unit]
Description=Update ansible cmdb

[Service]
WorkingDirectory=%h/ansible
ExecStart=%h/.local/bin/update-cmdb
Нажмите, чтобы развернуть и увидеть больше

Теперь создаем юнит таймера:

BASH
systemctl --user edit --full --force update-cmdb.timer
Нажмите, чтобы развернуть и увидеть больше

Наполняем:

BASH
[Unit]
Description=Update ansible cmdb daily at 01:00

[Timer]
OnCalendar=*-*-* 01:00:00
Persistent=true

[Install]
WantedBy=timers.target
Нажмите, чтобы развернуть и увидеть больше

Активируем таймер:

BASH
systemctl --user enable --now update-cmdb.timer
Нажмите, чтобы развернуть и увидеть больше

Проверяем:

BASH
systemctl --user status update-cmdb.timer

systemctl --user status update-cmdb.service

journalctl --user -fu update-cmdb
Нажмите, чтобы развернуть и увидеть больше

Теперь наш HTML файл будет всегда содержать актуальные данные по хостам Ansible👍.

Используемые материалы

Авторские права

Автор: Иван Чёрный

Ссылка: https://r4ven.me/automation/ansible-cmdb-strukturizacziya-i-vizualizacziya-ansible-facts/

Лицензия: CC BY-NC-SA 4.0

Использование материалов блога разрешается при условии: указания авторства/источника, некоммерческого использования и сохранения лицензии.

Начать поиск

Введите ключевые слова для поиска статей

↑↓
ESC
⌘K Горячая клавиша