-
Notifications
You must be signed in to change notification settings - Fork 5
/
Copy pathcontainer_exporter.py
executable file
·106 lines (84 loc) · 4.86 KB
/
container_exporter.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
from asyncio import gather, new_event_loop, wait
from aiodocker import Docker
from docker import from_env as docker_env
from stats import get_docker_stats as stat
from prometheus_client import Gauge, Counter
from prometheus_client.exposition import generate_latest
from flask import Flask, Response, request
from configs import config
app = Flask(__name__)
# TO-DO : handle init containers with better storage methods
# TO-DO : modulization
# Create Prometheus gauge metrics for status and stats
container_status = Gauge('cxp_container_status', 'Docker container status (1 = running, 0 = not running)', ['container_name'])
container_cpu_percentage = Gauge('cxp_cpu_percentage', 'Docker container cpu usage', ['container_name'])
container_memory_percentage = Gauge('cxp_memory_percentage', 'Docker container memory usage in percent', ['container_name'])
container_memory_bytes_total = Gauge('cxp_memory_bytes_total', 'Docker container memory usage in bytes', ['container_name'])
# Create Prometheus Counter metric for Disk I/O
disk_io_read_counter = Counter("cxp_disk_io_read_bytes_total", "Total number of bytes read from disk", ['container_name'])
disk_io_write_counter = Counter("cxp_disk_io_write_bytes_total", "Total number of bytes written to disk", ['container_name'])
# Create Prometheus Counter metric for Network I/O
network_rx_counter = Counter("cxp_network_rx_bytes_total", "Total number of bytes received over the network", ['container_name'])
network_tx_counter = Counter("cxp_network_tx_bytes_total", "Total number of bytes transmitted over the network", ['container_name'])
# get the data that relates to running containers at the first startup
def get_init_container():
client = docker_env()
return client.containers.list()
init_containers_names = [c.name for c in get_init_container()]
# get the data for all containers (killed exited stopped and running)
def get_all_container():
client = docker_env()
return client.containers.list(all=True)
def update_container_status():
# update the running container_names that is offline with the status of all containers
all_containers = get_all_container()
for container in all_containers:
if container.name in init_containers_names:
container_status.labels(container_name=container.name).set(1 if container.status == "running" else 0)
elif container.status == "running":
container_status.labels(container_name=container.name).set(1)
init_containers_names.append(container.name)
for removed_container_name in init_containers_names:
if removed_container_name not in [c.name for c in all_containers]:
container_status.labels(container_name=removed_container_name).set(0)
# get containers' stats and update their metrics in async mode
async def container_stats():
docker = Docker()
containers = await docker.containers.list()
tasks = [stat.get_container_stats(container) for container in containers]
all_stats = await gather(*tasks)
for stats in all_stats:
container_cpu_percentage.labels(container_name=stats[0]['name'][1:]).set(stat.calculate_cpu_percentage(stats[0]))
container_memory_percentage.labels(container_name=stats[0]['name'][1:]).set(stat.calculate_memory_percentage(stats[0]))
container_memory_bytes_total.labels(container_name=stats[0]['name'][1:]).set(stat.calculate_memory_bytes(stats[0]))
disk_io_read_counter.labels(container_name=stats[0]['name'][1:]).inc(stat.calculate_disk_io(stats[0])[0])
disk_io_write_counter.labels(container_name=stats[0]['name'][1:]).inc(stat.calculate_disk_io(stats[0])[1])
network_rx_counter.labels(container_name=stats[0]['name'][1:]).inc(stat.calculate_network_io(stats[0])[0])
network_tx_counter.labels(container_name=stats[0]['name'][1:]).inc(stat.calculate_network_io(stats[0])[1])
metrics_names = [container_cpu_percentage, container_memory_percentage , container_memory_bytes_total , disk_io_read_counter , disk_io_write_counter , network_rx_counter , network_tx_counter ]
def flush_metric_labels():
all_containers = get_all_container()
for container in all_containers:
if container.status != "running":
for m in metrics_names:
m.clear()
@app.route('/')
def index():
return "Welcome To CXP, Contianer Exporter For Prometheus."
@app.route('/metrics')
def metrics():
try:
update_container_status()
flush_metric_labels()
loop = new_event_loop()
t = [loop.create_task(container_stats())]
loop.run_until_complete(wait(t))
except Exception as e:
return f"Error running script: {str(e)}"
# generate the latest value of metrics
return Response(generate_latest(), mimetype='text/plain')
def create_app():
app.config.from_object(config.Config)
return app
if __name__ == "__main__":
app.run('0.0.0.0', 8000)