Files
mktxp-no-cli/mktxp/datasource/connection_ds.py

77 lines
3.8 KiB
Python

# coding=utf8
## Copyright (c) 2020 Arseniy Kuznetsov
##
## This program is free software; you can redistribute it and/or
## modify it under the terms of the GNU General Public License
## as published by the Free Software Foundation; either version 2
## of the License, or (at your option) any later version.
##
## This program is distributed in the hope that it will be useful,
## but WITHOUT ANY WARRANTY; without even the implied warranty of
## MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
## GNU General Public License for more details.
from collections import namedtuple
from mktxp.datasource.base_ds import BaseDSProcessor
class IPConnectionDatasource:
''' IP connections data provider
'''
@staticmethod
def metric_records(router_entry, *, metric_labels = None):
if metric_labels is None:
metric_labels = []
try:
answer = router_entry.api_connection.router_api().get_binary_resource('/ip/firewall/connection/').call('print', {'count-only': b''})
# answer looks and feels like an empty list: [], but it has a special attribute `done_message`
done_message = answer.done_message
# `done_msg` is a dict with the return code as a key - which is the count that we are looking for
cnt = done_message['ret'].decode()
records = [{'count': cnt}]
return BaseDSProcessor.trimmed_records(router_entry, router_records = records, metric_labels = metric_labels)
except Exception as exc:
print(f'Error getting IP connection info from router{router_entry.router_name}@{router_entry.config_entry.hostname}: {exc}')
return None
class IPConnectionStatsDatasource:
''' IP connections stats data provider
'''
@staticmethod
def metric_records(router_entry, *, metric_labels = None, add_router_id = True):
if metric_labels is None:
metric_labels = []
try:
connection_records = router_entry.api_connection.router_api().get_resource('/ip/firewall/connection/').call('print', \
{'proplist':'src-address,dst-address,protocol'})
# calculate number of connections per src-address
connections_per_src_address = {}
for connection_record in connection_records:
#address, port = (connection_record['src-address'].split(':') + [None])[:2]
address = connection_record['src-address'].split(':')[0]
destination = f"{connection_record.get('dst-address')}({connection_record.get('protocol')})"
count, destinations = 0, set()
if connections_per_src_address.get(address):
count, destinations = connections_per_src_address[address]
count += 1
destinations.add(destination)
connections_per_src_address[address] = ConnStatsEntry(count, destinations)
# compile connections-per-interface records
records = []
for key, entry in connections_per_src_address.items():
record = {'src_address': key, 'connection_count': entry.count, 'dst_addresses': ', '.join(entry.destinations)}
if add_router_id:
for router_key, router_value in router_entry.router_id.items():
record[router_key] = router_value
records.append(record)
return records
except Exception as exc:
print(f'Error getting IP connection stats info from router{router_entry.router_name}@{router_entry.config_entry.hostname}: {exc}')
return None
ConnStatsEntry = namedtuple('ConnStatsEntry', ['count', 'destinations'])