Files
gandalf/app.py
2025-02-07 21:33:55 -05:00

176 lines
5.5 KiB
Python

import logging
import json
import platform
import subprocess
import threading
import time
from datetime import datetime
from flask import Flask, render_template, jsonify
import requests
from urllib3.exceptions import InsecureRequestWarning
# Configure logging
logging.basicConfig(level=logging.DEBUG)
logger = logging.getLogger(__name__)
# Disable InsecureRequestWarning
requests.packages.urllib3.disable_warnings(InsecureRequestWarning)
# Initialize Flask app
app = Flask(__name__)
# Global state
device_status = {}
# Configuration functions
def load_config():
with open('config.json') as f:
return json.load(f)
# Network utility functions
def ping(host):
param = '-n' if platform.system().lower() == 'windows' else '-c'
command = ['ping', param, '1', host]
return subprocess.call(command, stdout=subprocess.DEVNULL) == 0
def send_webhook(device, status, diagnostics):
config = load_config()
webhook_data = {
"device": device,
"status": status,
"timestamp": datetime.now().isoformat(),
"diagnostics": diagnostics
}
requests.post(config['webhook_url'], json=webhook_data)
# UniFi API integration
class UnifiAPI:
def __init__(self, config):
self.base_url = config['unifi']['controller']
self.headers = {
'X-API-KEY': config['unifi']['api_key'],
'Accept': 'application/json'
}
self.site_id = config['unifi']['site_id']
self.session = requests.Session()
self.session.verify = False
def get_device_details(self, device_id):
try:
url = f"{self.base_url}/proxy/network/integration/v1/sites/{self.site_id}/devices/{device_id}"
response = self.session.get(url, headers=self.headers, timeout=5)
response.raise_for_status()
return response.json()
except requests.exceptions.RequestException as e:
logging.error(f"Failed to get device details: {e}")
return None
def get_device_diagnostics(self, device):
details = self.get_device_details(device['device_id'])
if not details:
return {'state': 'ERROR', 'error': 'Failed to fetch device details'}
diagnostics = {
'state': details['state'],
'firmware': {
'version': details['firmwareVersion'],
'updatable': details.get('firmwareUpdatable', False)
},
'network': {
'ip': details['ipAddress'],
'mac': details['macAddress']
},
'uptime': {
'adopted_at': details['adoptedAt'],
'provisioned_at': details['provisionedAt']
},
'interfaces': {}
}
if 'interfaces' in details:
diagnostics['interfaces'] = self._parse_interfaces(details['interfaces'])
return diagnostics
def _parse_interfaces(self, interfaces):
result = {
'ports': {},
'radios': {}
}
for port in interfaces.get('ports', []):
result['ports'][f"port_{port['idx']}"] = {
'state': port['state'],
'type': port['connector'],
'speed': {
'current': port['speedMbps'],
'max': port['maxSpeedMbps']
}
}
for radio in interfaces.get('radios', []):
result['radios'][f"{radio['frequencyGHz']}GHz"] = {
'standard': radio['wlanStandard'],
'channel': radio['channel'],
'width': f"{radio['channelWidthMHz']}MHz"
}
return result
# Monitoring functions
def run_diagnostics(device):
config = load_config()
unifi = UnifiAPI(config)
diagnostics = unifi.get_device_diagnostics(device)
if device['connection_type'] == 'fiber':
# Add fiber-specific diagnostics
sfp_data = get_sfp_diagnostics(device['ip'])
diagnostics['sfp'] = sfp_data
return diagnostics
def update_status():
while True:
config = load_config()
for device in config['devices']:
current_status = ping(device['ip'])
previous_status = device_status.get(device['name'], True)
if current_status != previous_status:
diagnostics = run_diagnostics(device)
send_webhook(device, current_status, diagnostics)
device_status[device['name']] = current_status
time.sleep(config['check_interval'])
# Flask routes
@app.route('/')
def home():
config = load_config()
logger.debug(f"Loaded devices: {config['devices']}")
devices = config['devices']
logger.debug(f"Rendering template with devices: {devices}")
return render_template('index.html', devices=devices)
@app.route('/api/status')
def status():
return jsonify(device_status)
@app.route('/api/diagnostics')
def get_diagnostics():
config = load_config()
diagnostics = {}
for device in config['devices']:
if device.get('device_id'):
logger.debug(f"Fetching diagnostics for device: {device['name']}")
diagnostics[device['name']] = run_diagnostics(device)
logger.debug(f"Returning diagnostics: {diagnostics}")
return jsonify(diagnostics)
# Application entry point
if __name__ == '__main__':
status_thread = threading.Thread(target=update_status, daemon=True)
status_thread.start()
app.run(debug=True)