Prep for Web-based stats
This commit is contained in:
parent
49d7c1f484
commit
1fdd8d3697
|
@ -49,6 +49,7 @@ from twisted.internet import task
|
||||||
from binascii import b2a_hex as ahex
|
from binascii import b2a_hex as ahex
|
||||||
from time import time
|
from time import time
|
||||||
from importlib import import_module
|
from importlib import import_module
|
||||||
|
from cPickle import dump as pickle_dump
|
||||||
|
|
||||||
import sys
|
import sys
|
||||||
|
|
||||||
|
@ -72,7 +73,7 @@ TS_CLEAR_TIME = .2
|
||||||
|
|
||||||
# Build the conference bridging structure from the bridge file.
|
# Build the conference bridging structure from the bridge file.
|
||||||
#
|
#
|
||||||
def make_bridges(_confbridge_rules):
|
def make_bridge_config(_confbridge_rules):
|
||||||
try:
|
try:
|
||||||
bridge_file = import_module(_confbridge_rules)
|
bridge_file = import_module(_confbridge_rules)
|
||||||
logger.info('Bridge configuration file found and imported')
|
logger.info('Bridge configuration file found and imported')
|
||||||
|
@ -97,7 +98,7 @@ def make_bridges(_confbridge_rules):
|
||||||
_system['TIMEOUT'] = _system['TIMEOUT']*60
|
_system['TIMEOUT'] = _system['TIMEOUT']*60
|
||||||
_system['TIMER'] = time() + _system['TIMEOUT']
|
_system['TIMER'] = time() + _system['TIMEOUT']
|
||||||
|
|
||||||
return bridge_file.BRIDGES
|
return {'BRIDGE_CONF': bridge_file.BRIDGE_CONF, 'BRIDGES': bridge_file.BRIDGES}
|
||||||
|
|
||||||
|
|
||||||
# Import subscriber ACL
|
# Import subscriber ACL
|
||||||
|
@ -169,6 +170,14 @@ def rule_timer_loop():
|
||||||
else:
|
else:
|
||||||
logger.debug('Conference Bridge NO ACTION: System: %s, Bridge: %s, TS: %s, TGID: %s', _system['SYSTEM'], _bridge, _system['TS'], int_id(_system['TGID']))
|
logger.debug('Conference Bridge NO ACTION: System: %s, Bridge: %s, TS: %s, TGID: %s', _system['SYSTEM'], _bridge, _system['TS'], int_id(_system['TGID']))
|
||||||
|
|
||||||
|
if BRIDGE_CONF['REPORT']:
|
||||||
|
try:
|
||||||
|
with open(CONFIG['REPORTS']['REPORT_PATH']+'confbridge_stats.pickle', 'wb') as file:
|
||||||
|
pickle_dump(BRIDGES, file, 2)
|
||||||
|
file.close()
|
||||||
|
except IOError as detail:
|
||||||
|
_logger.error('I/O Error: %s', detail)
|
||||||
|
|
||||||
|
|
||||||
class confbridgeIPSC(IPSC):
|
class confbridgeIPSC(IPSC):
|
||||||
def __init__(self, _name, _config, _logger):
|
def __init__(self, _name, _config, _logger):
|
||||||
|
@ -410,8 +419,10 @@ if __name__ == '__main__':
|
||||||
if talkgroup_ids:
|
if talkgroup_ids:
|
||||||
logger.info('ID ALIAS MAPPER: talkgroup_ids dictionary is available')
|
logger.info('ID ALIAS MAPPER: talkgroup_ids dictionary is available')
|
||||||
|
|
||||||
# Build the routing rules file
|
# Build the routing rules and other configuration
|
||||||
BRIDGES = make_bridges('confbridge_rules')
|
CONFIG_DICT = make_bridge_config('confbridge_rules')
|
||||||
|
BRIDGE_CONF = CONFIG_DICT['BRIDGE_CONF']
|
||||||
|
BRIDGES = CONFIG_DICT['BRIDGES']
|
||||||
|
|
||||||
# Build the Access Control List
|
# Build the Access Control List
|
||||||
ACL = build_acl('sub_acl')
|
ACL = build_acl('sub_acl')
|
||||||
|
|
|
@ -28,6 +28,16 @@ configuration file.
|
||||||
|
|
||||||
'''
|
'''
|
||||||
|
|
||||||
|
# CONFIGURATION ITEMS SPECIFICALLY FOR confbridge.py
|
||||||
|
#
|
||||||
|
# REPORT:
|
||||||
|
# True or False. True if you want to write a pickle file of the current rule file
|
||||||
|
# state. This is useful (and necessary) for reporting features to be active.
|
||||||
|
# The path follows the reporting path in the main dmrlink.cfg file.
|
||||||
|
BRIDGE_CONF = {
|
||||||
|
'REPORT': True,
|
||||||
|
}
|
||||||
|
|
||||||
BRIDGES = {
|
BRIDGES = {
|
||||||
'WORLDWIDE': [
|
'WORLDWIDE': [
|
||||||
{'SYSTEM': 'MASTER-1', 'TS': 1, 'TGID': 1, 'ACTIVE': True, 'TIMEOUT': 2, 'TO_TYPE': 'ON', 'ON': [2,], 'OFF': [9,10]},
|
{'SYSTEM': 'MASTER-1', 'TS': 1, 'TGID': 1, 'ACTIVE': True, 'TIMEOUT': 2, 'TO_TYPE': 'ON', 'ON': [2,], 'OFF': [9,10]},
|
||||||
|
|
|
@ -0,0 +1,138 @@
|
||||||
|
#!/usr/bin/env python
|
||||||
|
#
|
||||||
|
###############################################################################
|
||||||
|
# Copyright (C) 2017 Cortney T. Buffington, N0MJS <n0mjs@me.com>
|
||||||
|
#
|
||||||
|
# This program is free software; you can redistribute it and/or modify
|
||||||
|
# it under the terms of tde GNU General Public License as published by
|
||||||
|
# the Free Software Foundation; eitder version 3 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.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU General Public License
|
||||||
|
# along with this program; if not, write to the Free Software Foundation,
|
||||||
|
# Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
||||||
|
###############################################################################
|
||||||
|
|
||||||
|
from __future__ import print_function
|
||||||
|
from cPickle import load
|
||||||
|
from pprint import pprint
|
||||||
|
from time import ctime
|
||||||
|
from twisted.internet import reactor
|
||||||
|
from twisted.internet import task
|
||||||
|
from binascii import b2a_hex as h
|
||||||
|
from dmr_utils.utils import int_id, get_alias
|
||||||
|
|
||||||
|
__autdor__ = 'Cortney T. Buffington, N0MJS'
|
||||||
|
__copyright__ = 'Copyright (c) 2017 Cortney T. Buffington, N0MJS'
|
||||||
|
__license__ = 'GNU GPLv3'
|
||||||
|
__maintainer__ = 'Cort Buffington, N0MJS'
|
||||||
|
__email__ = 'n0mjs@me.com'
|
||||||
|
|
||||||
|
|
||||||
|
# This is the only user-configuration necessary
|
||||||
|
# Tell the program where the pickle file is
|
||||||
|
# Tell the program where to write the html table file
|
||||||
|
# Tell the program how often to print a report -- should match dmrlink report period
|
||||||
|
stat_file = '../confbridge_stats.pickle'
|
||||||
|
html_table_file = '../confbridge_stats.html'
|
||||||
|
frequency = 30
|
||||||
|
|
||||||
|
def read_dict():
|
||||||
|
try:
|
||||||
|
with open(stat_file, 'rb') as file:
|
||||||
|
NETWORK = load(file)
|
||||||
|
return NETWORK
|
||||||
|
except IOError as detail:
|
||||||
|
print('I/O Error: {}'.format(detail))
|
||||||
|
except EOFError:
|
||||||
|
print('EOFError')
|
||||||
|
|
||||||
|
def write_file(_string):
|
||||||
|
try:
|
||||||
|
with open(html_table_file, 'w') as file:
|
||||||
|
file.write(_string)
|
||||||
|
file.close()
|
||||||
|
except IOError as detail:
|
||||||
|
print('I/O Error: {}'.format(detail))
|
||||||
|
except EOFError:
|
||||||
|
print('EOFError')
|
||||||
|
|
||||||
|
def build_table():
|
||||||
|
NETWORK = read_dict()
|
||||||
|
if NETWORK != 'None':
|
||||||
|
stuff = 'Last Update: {}'.format(ctime())
|
||||||
|
stuff += '<style>table, td, th {border: .5px solid black; padding: 2px; border-collapse: collapse}</style>'
|
||||||
|
|
||||||
|
for ipsc in NETWORK:
|
||||||
|
stat = NETWORK[ipsc]['MASTER']['STATUS']
|
||||||
|
master = NETWORK[ipsc]['LOCAL']['MASTER_PEER']
|
||||||
|
|
||||||
|
stuff += '<br><br><table style="width:90%">'
|
||||||
|
|
||||||
|
stuff += '<colgroup>\
|
||||||
|
<col style="width: 10%" />\
|
||||||
|
<col style="width: 20%" />\
|
||||||
|
<col style="width: 20%" />\
|
||||||
|
<col style="width: 10%" />\
|
||||||
|
<col style="width: 15%" />\
|
||||||
|
<col style="width: 15%" />\
|
||||||
|
<col style="width: 10%" />\
|
||||||
|
</colgroup>'
|
||||||
|
|
||||||
|
stuff += '<caption>{} '.format(ipsc)
|
||||||
|
if master:
|
||||||
|
stuff += '(master)'
|
||||||
|
else:
|
||||||
|
stuff += '(peer)'
|
||||||
|
stuff +='</caption'
|
||||||
|
|
||||||
|
stuff += '<tr><th rowspan="2">Type</th>\
|
||||||
|
<th rowspan="2">Radio ID</th>\
|
||||||
|
<th rowspan="2">IP Address</th>\
|
||||||
|
<th rowspan="2">Connected</th>\
|
||||||
|
<th colspan="3">Keep Alives</th></tr>\
|
||||||
|
<tr><th>Sent</th><th>Received</th><th>Missed</th></tr>'
|
||||||
|
|
||||||
|
if not master:
|
||||||
|
stuff += '<tr><td>Master</td><td>{}</td><td>{}</td><td>{}</td><td>{}</td><td>{}</td><td>{}</td></tr>'.format(\
|
||||||
|
str(int_id(NETWORK[ipsc]['MASTER']['RADIO_ID'])).rjust(8,'0'),\
|
||||||
|
NETWORK[ipsc]['MASTER']['IP'],\
|
||||||
|
stat['CONNECTED'],\
|
||||||
|
stat['KEEP_ALIVES_SENT'],\
|
||||||
|
stat['KEEP_ALIVES_RECEIVED'],\
|
||||||
|
stat['KEEP_ALIVES_MISSED'],)
|
||||||
|
|
||||||
|
if master:
|
||||||
|
for peer in NETWORK[ipsc]['PEERS']:
|
||||||
|
stat = NETWORK[ipsc]['PEERS'][peer]['STATUS']
|
||||||
|
stuff += '<tr><td>Peer</td><td>{}</td><td>{}</td><td>{}</td><td>n/a</td><td>{}</td><td>n/a</td></tr>'.format(\
|
||||||
|
str(int_id(peer)).rjust(8,'0'),\
|
||||||
|
NETWORK[ipsc]['PEERS'][peer]['IP'],\
|
||||||
|
stat['CONNECTED'],\
|
||||||
|
stat['KEEP_ALIVES_RECEIVED'])
|
||||||
|
|
||||||
|
else:
|
||||||
|
for peer in NETWORK[ipsc]['PEERS']:
|
||||||
|
stat = NETWORK[ipsc]['PEERS'][peer]['STATUS']
|
||||||
|
if peer != NETWORK[ipsc]['LOCAL']['RADIO_ID']:
|
||||||
|
stuff += '<tr><td>Peer</td><td>{}</td><td>{}</td><td>{}</td><td>{}</td><td>{}</td><td>{}</td></tr>'.format(\
|
||||||
|
str(int_id(peer)).rjust(8,'0'),\
|
||||||
|
NETWORK[ipsc]['PEERS'][peer]['IP'],\
|
||||||
|
stat['CONNECTED'],\
|
||||||
|
stat['KEEP_ALIVES_SENT'],\
|
||||||
|
stat['KEEP_ALIVES_RECEIVED'],\
|
||||||
|
stat['KEEP_ALIVES_MISSED'])
|
||||||
|
stuff += '</table>'
|
||||||
|
|
||||||
|
write_file(stuff)
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
output_stats = task.LoopingCall(build_table)
|
||||||
|
output_stats.start(frequency)
|
||||||
|
reactor.run()
|
Loading…
Reference in New Issue