Initial commit

This commit is contained in:
Julian Rother 2025-05-21 16:26:00 +02:00
commit 0fdb74a9a2
Signed by: julian
GPG key ID: C19B924C0CD13341
7 changed files with 168 additions and 0 deletions

8
defaults/main.yml Normal file
View file

@ -0,0 +1,8 @@
prometheus_opnsense_exporter_version: 0.0.8
prometheus_opnsense_exporter_config:
OPNSENSE_EXPORTER_ARGS: "--web.listen-address=127.0.0.1:9097 --exporter.instance-label=opnsense --opnsense.insecure"
OPNSENSE_DHCP_EXPORTER_PORT: "9098"
#OPNSENSE_EXPORTER_OPS_API: "..."
OPNSENSE_EXPORTER_OPS_PROTOCOL: "https"
#OPNSENSE_EXPORTER_OPS_API_KEY: "..."
#OPNSENSE_EXPORTER_OPS_API_SECRET: "..."

36
files/opnsense-dhcp-exporter Executable file
View file

@ -0,0 +1,36 @@
#!/usr/bin/python3
import time
import urllib3
import requests
import json
import os
from prometheus_client import start_http_server, PROCESS_COLLECTOR, PLATFORM_COLLECTOR, GC_COLLECTOR
from prometheus_client.core import GaugeMetricFamily, CounterMetricFamily, REGISTRY
from prometheus_client.registry import Collector
urllib3.disable_warnings()
class CustomCollector(Collector):
def collect(self):
api_host = os.environ['OPNSENSE_EXPORTER_OPS_API']
api_key = os.environ['OPNSENSE_EXPORTER_OPS_API_KEY']
api_secret = os.environ['OPNSENSE_EXPORTER_OPS_API_SECRET']
result = requests.get(f'https://{api_host}/api/dhcpv4/leases/searchLease', verify=False, auth=(api_key, api_secret))
leases = {}
for lease in result.json()['rows']:
if lease['state'] == 'active' and lease['type'] == 'dynamic':
leases.setdefault(lease['if_descr'], set()).add(lease['mac'])
opnsense_dhcpv4_active_leases_count = GaugeMetricFamily('opnsense_dhcpv4_active_leases_count', 'Number of active DHCPv4 leases per interface', labels=['interface'])
for if_descr, macs in leases.items():
opnsense_dhcpv4_active_leases_count.add_metric([if_descr], len(macs))
yield opnsense_dhcpv4_active_leases_count
REGISTRY.unregister(PROCESS_COLLECTOR)
REGISTRY.unregister(PLATFORM_COLLECTOR)
REGISTRY.unregister(GC_COLLECTOR)
REGISTRY.register(CustomCollector())
if __name__ == '__main__':
start_http_server(addr='127.0.0.1', port=int(os.environ['OPNSENSE_DHCP_EXPORTER_PORT']))
while True:
time.sleep(100)

View file

@ -0,0 +1,33 @@
[Unit]
Description=Prometheus exporter for OPNsense DHCPv4 metrics
Requires=network-online.target
After=network-online.target
[Service]
DynamicUser=yes
User=opnsense-exporter
Restart=on-failure
EnvironmentFile=-/etc/default/prometheus-opnsense-exporter
ExecStart=/usr/local/sbin/opnsense-dhcp-exporter
LockPersonality=yes
MemoryDenyWriteExecute=yes
NoNewPrivileges=true
PrivateDevices=yes
PrivateTmp=disconnected
PrivateUsers=yes
ProcSubset=pid
ProtectClock=yes
ProtectControlGroups=strict
ProtectHome=yes
ProtectHostname=yes
ProtectKernelLogs=yes
ProtectKernelModules=yes
ProtectKernelTunables=yes
ProtectProc=invisible
ProtectSystem=strict
RestrictNamespaces=yes
RestrictRealtime=yes
[Install]
WantedBy=multi-user.target

View file

@ -0,0 +1,33 @@
[Unit]
Description=Prometheus exporter for OPNsense metrics
Requires=network-online.target
After=network-online.target
[Service]
DynamicUser=yes
User=opnsense-exporter
Restart=on-failure
EnvironmentFile=-/etc/default/prometheus-opnsense-exporter
ExecStart=/usr/local/sbin/opnsense-exporter $OPNSENSE_EXPORTER_ARGS
LockPersonality=yes
MemoryDenyWriteExecute=yes
NoNewPrivileges=true
PrivateDevices=yes
PrivateTmp=disconnected
PrivateUsers=yes
ProcSubset=pid
ProtectClock=yes
ProtectControlGroups=strict
ProtectHome=yes
ProtectHostname=yes
ProtectKernelLogs=yes
ProtectKernelModules=yes
ProtectKernelTunables=yes
ProtectProc=invisible
ProtectSystem=strict
RestrictNamespaces=yes
RestrictRealtime=yes
[Install]
WantedBy=multi-user.target

13
handlers/main.yml Normal file
View file

@ -0,0 +1,13 @@
- name: Restart prometheus-opnsense-exporter
ansible.builtin.systemd_service:
name: prometheus-opnsense-exporter
daemon_reload: true
state: restarted
enabled: yes
- name: Restart prometheus-opnsense-dhcp-exporter
ansible.builtin.systemd_service:
name: prometheus-opnsense-dhcp-exporter
daemon_reload: true
state: restarted
enabled: yes

42
tasks/main.yml Normal file
View file

@ -0,0 +1,42 @@
- name: Update config
ansible.builtin.template:
src: prometheus-opnsense-exporter.j2
dest: /etc/default/prometheus-opnsense-exporter
mode: "0640"
notify:
- Restart prometheus-opnsense-exporter
- Restart prometheus-opnsense-dhcp-exporter
- name: Download opnsense-exporter release tarball
ansible.builtin.get_url:
url: "https://github.com/AthennaMind/opnsense-exporter/releases/download/v{{ prometheus_opnsense_exporter_version }}/opnsense-exporter_Linux_x86_64.tar.gz"
dest: "/var/tmp/opnsense-exporter-{{ prometheus_opnsense_exporter_version }}.tar.gz"
mode: "0644"
- name: Extract binary from opnsense-exporter release tarball
ansible.builtin.unarchive:
remote_src: true
src: "/var/tmp/opnsense-exporter-{{ prometheus_opnsense_exporter_version }}.tar.gz"
dest: "/usr/local/sbin/"
include: "opnsense-exporter"
mode: "0755"
notify: Restart prometheus-opnsense-exporter
- name: Create opnsense-exporter systemd unit
ansible.builtin.copy:
src: prometheus-opnsense-exporter.service
dest: /etc/systemd/system/prometheus-opnsense-exporter.service
notify: Restart prometheus-opnsense-exporter
- name: Copy opnsense-dhcp-exporter script
ansible.builtin.copy:
src: opnsense-dhcp-exporter
dest: /usr/local/sbin/opnsense-dhcp-exporter
mode: "0755"
notify: Restart prometheus-opnsense-dhcp-exporter
- name: Create opnsense-dhcp-exporter systemd unit
ansible.builtin.copy:
src: prometheus-opnsense-dhcp-exporter.service
dest: /etc/systemd/system/prometheus-opnsense-dhcp-exporter.service
notify: Restart prometheus-opnsense-dhcp-exporter

View file

@ -0,0 +1,3 @@
{% for key, value in prometheus_opnsense_exporter_config.items() %}
{{ key }}="{{ value }}"
{% endfor %}