#!/usr/bin/python # -*- coding: utf-8 -*- # api_firmware.py: API for managing logs and dashboard data # MikroWizard.com , Mikrotik router management solution # Author: sepehr.ha@gmail.com from flask import request import datetime from libs.db import db,db_syslog,db_device,db_AA,db_events,db_sysconfig,db_tasks from libs.webutil import app,buildResponse,login_required import logging import operator from libs import util from functools import reduce from libs.red import RedisDB import feedparser import requests import json log = logging.getLogger("logs") def peewee_sql_to_str(sql): return (sql[0] % tuple(sql[1])) @app.route('/api/auth/list', methods = ['POST']) @login_required(role='admin',perm={'authentication':'read'}) def list_auth_log(): """return all authentication data (default last 24H)""" input = request.json start_time=input.get('start_time',False) end_time=input.get('end_time',False) ip=input.get('ip',False) devip=input.get('devip',False) devid=input.get('devid',False) username=input.get('user',False) ltype=input.get('state',False) server=input.get('server',False) by=input.get('connection_type',False) auth=db_AA.Auth # build where query clauses = [] if ip and ip != "": clauses.append(auth.ip.contains(ip)) if username and username !="": clauses.append(auth.username.contains(username)) if ltype and ltype!='All': clauses.append(auth.ltype == ltype) if by and by !='All': clauses.append(auth.by == by) if devid and devid>0: clauses.append(auth.devid == devid) if start_time: start_time=start_time.split(".000Z")[0] start_time=datetime.datetime.strptime(start_time, "%Y-%m-%dT%H:%M:%S") clauses.append(auth.created >= start_time) else: #set start time to one day ago start_time=datetime.datetime.now()-datetime.timedelta(days=1) clauses.append(auth.created >= start_time) if end_time: end_time=end_time.split(".000Z")[0] end_time=datetime.datetime.strptime(end_time, "%Y-%m-%dT%H:%M:%S") clauses.append(auth.created <= end_time) else: end_time=datetime.datetime.now() clauses.append(auth.created<=end_time) if server and server !="All": if server=='Local': clauses.append(auth.sessionid.is_null(True)) else: clauses.append(auth.sessionid.is_null(False)) expr="" devs=db_device.Devices if devip and devip!="": clauses.append(devs.ip.contains(devip)) logs = [] selector=[auth.ip,auth.username,auth.started,auth.ended,auth.sessionid,auth.ltype,auth.by,auth.message,auth.created,devs.ip.alias('devip'),devs.name] try: if len(clauses): expr = reduce(operator.and_, clauses) query=auth.select(*selector).join(devs).where(expr) else: query=auth.select(*selector).join(devs) query=query.order_by(auth.id.desc()) logs=list(query.dicts()) except Exception as e: return buildResponse({"status":"failed", "err":str(e)},200) return buildResponse(logs,200) @app.route('/api/account/list', methods = ['POST']) @login_required(role='admin',perm={'accounting':'read'}) def list_account_log(): """return all accounting data (default last 24H)""" input = request.json devid=input.get('devid',False) username=input.get('user',False) action=input.get('action',False) section=input.get('section',False) message=input.get('message',False) start_time=input.get('start_time',False) end_time=input.get('end_time',False) config=input.get('config',False) ip=input.get('ip',False) acc=db_AA.Account # build where query clauses = [] clauses.append(acc.username!="unknown") if action and action!='All': clauses.append(acc.action.contains(action)) if username: clauses.append(acc.username.contains(username)) if section and section!='All': clauses.append(acc.section.contains(section)) if message: clauses.append(acc.message.contains(message)) if start_time: start_time=start_time.split(".000Z")[0] start_time=datetime.datetime.strptime(start_time, "%Y-%m-%dT%H:%M:%S") clauses.append(acc.created >= start_time) else: #set start time to one day ago start_time=datetime.datetime.now()-datetime.timedelta(days=1) clauses.append(acc.created >= start_time) if devid and devid>0: clauses.append(acc.devid == devid) if end_time: end_time=end_time.split(".000Z")[0] end_time=datetime.datetime.strptime(end_time, "%Y-%m-%dT%H:%M:%S") clauses.append(acc.created <= end_time) else: end_time=datetime.datetime.now() clauses.append(acc.created<=end_time) if config and config!="": clauses.append(acc.config.contains(config)) expr="" devs=db_device.Devices if ip and ip!="": clauses.append(devs.ip.contains(ip)) logs = [] selector=[acc.action,acc.username,acc.ctype,acc.address,acc.config,acc.section,acc.message,acc.created,devs.ip.alias('devip'),devs.name] try: if len(clauses): expr = reduce(operator.and_, clauses) query=acc.select(*selector).join(devs).where(expr) else: query=acc.select(*selector).join(devs) query=query.order_by(acc.id.desc()) logs=list(query.dicts()) except Exception as e: return buildResponse({"status":"failed", "err":str(e)},200) return buildResponse(logs,200) @app.route('/api/devlogs/list', methods = ['POST']) @login_required(role='admin', perm={'device':'read'}) def dev_events_list(): """return Device Events""" input = request.json devid=input.get('devid',False) event_start_time=input.get('start_time',False) event_end_time=input.get('end_time',False) event_type=input.get('event_type',False) status=input.get('status',"All") level=input.get('level',False) detail=input.get('detail',False) comment=input.get('comment',False) src=input.get('src', False) event=db_events.Events # build where query clauses = [] clauses2 = [] if event_start_time: event_start_time=event_start_time.split(".000Z")[0] event_start_time=datetime.datetime.strptime(event_start_time, "%Y-%m-%dT%H:%M:%S") clauses.append(event.eventtime >= event_start_time) else: clauses.append(event.eventtime >= datetime.datetime.now()-datetime.timedelta(days=1)) if event_end_time: event_end_time=event_end_time.split(".000Z")[0] event_end_time=datetime.datetime.strptime(event_end_time, "%Y-%m-%dT%H:%M:%S") clauses.append(event.eventtime <= event_end_time) else: clauses.append(event.eventtime <= datetime.datetime.now()) if event_type: clauses.append(event.eventtype == event_type) if status!="all": clauses.append(event.status == status) if level and level!='All': clauses.append(event.level == level) if detail: for d in detail: clauses2.append(event.detail.contains(d)) # clauses.append(event.detail.contains(detail)) if comment: clauses.append(event.comment.contains(comment)) if src: clauses.append(event.src == src) if devid: dev=db_device.get_device(devid) if not dev: return buildResponse({'status': 'failed'}, 200, error="Wrong Data") else: clauses.append(event.devid == devid) expr="" devs=db_device.Devices events=[] selector=[event.eventtime,event.eventtype,event.fixtime,event.status,event.level,event.detail,event.comment,event.src,event.id,devs.ip.alias('devip'),devs.name,devs.mac] try: if len(clauses): expr = reduce(operator.and_, clauses) query=event.select(*selector).join(devs).where(expr) if len(clauses2): expr2 = reduce(operator.or_, clauses2) query=query.where(expr2) else: query=event.select(*selector).join(devs) query=query.order_by(event.id.desc()) events=list(query.dicts()) except Exception as e: log.error(e) return buildResponse({"status":"failed", "err":str(e)}, 200) return buildResponse(events, 200) @app.route('/api/syslog/list', methods = ['POST']) @login_required(role='admin', perm={'settings':'read'}) def syslog_list(): """return MikroWizard innternal syslog""" input = request.json userid=input.get('userid',False) event_start_time=input.get('start_time',False) event_end_time=input.get('end_time',False) action=input.get('action',False) section=input.get('section',False) ip=input.get('ip',False) syslog=db_syslog.SysLog # build where query clauses = [] if event_start_time: event_start_time=event_start_time.split(".000Z")[0] event_start_time=datetime.datetime.strptime(event_start_time, "%Y-%m-%dT%H:%M:%S") clauses.append(syslog.created >= event_start_time) else: clauses.append(syslog.created >= datetime.datetime.now()-datetime.timedelta(days=1)) if event_end_time: event_end_time=event_end_time.split(".000Z")[0] event_end_time=datetime.datetime.strptime(event_end_time, "%Y-%m-%dT%H:%M:%S") clauses.append(syslog.created <= event_end_time) else: clauses.append(syslog.created <= datetime.datetime.now()) if action and action!='All': clauses.append(syslog.action == action) if section and section!='All': clauses.append(syslog.section == section) if ip and ip !="": clauses.append(syslog.ip.contains(ip)) if userid: user=db.get_user(userid) if not user: return buildResponse({'status': 'failed'}, 200, error="Wrong Data") else: clauses.append(syslog.user_id == user.id) expr="" users=db.User events=[] selector=[syslog.created,syslog.action,syslog.section,syslog.ip,syslog.agent,syslog.data,syslog.id,users.username,users.first_name,users.last_name] try: if len(clauses): expr = reduce(operator.and_, clauses) query=syslog.select(*selector).join(users).where(expr) else: query=syslog.select(*selector).join(users) query=query.order_by(syslog.id.desc()) events=list(query.dicts()) except Exception as e: log.error(e) return buildResponse({"status":"failed", "err":str(e)}, 200) return buildResponse(events, 200) @app.route('/api/devlogs/details/list', methods = ['POST']) @login_required(role='admin', perm={'device':'read'}) def dev_events_details_list(): """return list of event details(types) for filters""" input = request.json devid=input.get('devid', False) event=db_events.select(event.details) if devid: dev=db_device.get_device(devid) if not dev: return buildResponse({'status': 'failed'}, 200, error="Wrong Data") else: event=event.where(event.devid == dev.id) event=event.group_by(event.details).order_by(event.id.desc()) res=list(event.dicts()) return buildResponse(res, 200) @app.route('/api/dashboard/stats', methods = ['POST']) @login_required(role='admin', perm={'device':'read'}) def dashboard_stats(): """return dashboard data""" input = request.json versioncheck = input.get('versioncheck',False) front_version = input.get('front_version',False) VERSIONFILE="_version.py" from _version import __version__ res={} res['version']=__version__ # get past 24h failed logins and success logins from auth auth=db_AA.Auth res['FailedLogins']=auth.select().where(auth.ltype=='failed',auth.created>(datetime.datetime.now()-datetime.timedelta(days=1))).count() res['SuccessfulLogins']=auth.select().where(auth.ltype=='loggedin', auth.created>(datetime.datetime.now()-datetime.timedelta(days=1))).count() # get past 24h Critical and WARNING and info from events and also Total events event=db_events.Events res['Critical']=event.select().where(event.level=='Critical', event.eventtime>(datetime.datetime.now()-datetime.timedelta(days=1))).count() res['Warning']=event.select().where(event.level=='Warning', event.eventtime>(datetime.datetime.now()-datetime.timedelta(days=1))).count() res['Info']=event.select().where(event.level=='info', event.eventtime>(datetime.datetime.now()-datetime.timedelta(days=1))).count() res['Events']=event.select().count() interfaces = util.get_ethernet_wifi_interfaces() hwid = util.generate_serial_number(interfaces) install_date=False try: install_date=db_sysconfig.get_sysconfig('install_date') except: pass if not install_date or install_date=='': install_date=datetime.datetime.now() db_sysconfig.set_sysconfig('install_date',install_date.strftime("%Y-%m-%d %H:%M:%S")) install_date=install_date.strftime("%Y-%m-%d %H:%M:%S") if install_date: res['serial']=hwid+"-"+datetime.datetime.strptime(install_date, "%Y-%m-%d %H:%M:%S").strftime("%Y%m%d") else: res['serial']=False # get total users , Total devices , total auth , total acc acc=db_AA.Account devs=db_device.Devices res['Users']=db.User.select().count() - 1 res['Devices']=devs.select().count() res['Auth']=auth.select().count() res['Acc']=acc.select().count() res['license']=False username=False internet_connection=True # check for internet connection before getting data from website feedurl="https://mikrowizard.com/tag/Blog/feed/?orderby=latest" test_url="https://google.com" update_mode=db_sysconfig.get_sysconfig('update_mode') update_mode=json.loads(update_mode) res['update_mode']=update_mode['mode'] try: req = requests.get(test_url, timeout=(0.5,1)) req.raise_for_status() except Exception as e: log.error(e) internet_connection=False pass try: username = db_sysconfig.get_sysconfig('username') params={ "serial_number": res['serial'], "username": username.strip(), "version": __version__ } if versioncheck: params['versioncheck'] = True url="https://mikrowizard.com/wp-json/mikrowizard/v1/get_update" # send post request to server mikrowizard.com with params in json try: if internet_connection: response = requests.post(url, json=params) response=response.json() # log.error(response) res['license']=response.get('license',False) res['update_available']=response.get('available',False) res['latest_version']=response.get('latest_version',False) res['update_inprogress']=update_mode['update_back'] else: res['license']='connection_error' res['update_available']=False res['latest_version']=False except: pass try: if front_version and internet_connection: params['version']=front_version params['front']=True response = requests.post(url, json=params) response=response.json() res['front_update_available']=response.get('available',False) res['front_latest_version']=response.get('latest_version',False) res['front_update_inprogress']=update_mode['update_front'] except: pass except: pass res['front_update_available']=True res['update_available']=True if username: res['username']=username res['blog']=[] noconnectiondata={ "content": "Unable to connect to mikrowizard.com! please check server connection", "media_content": "", "summery": "Unable to connect mikrowizard.com to get latest News! Read More", "title": "Connection Error" } try: if internet_connection: feed = feedparser.parse(feedurl)['entries'] else: feed = [] if len(feed) >0: for f in feed: tmp={} tmp['title']=f['title'] tmp['content']=f['content'][0]['value'] tmp['summery']=f['summary'][0:100]+" ... " + 'Read More' tmp['media_content']=f['media_content'][0]['url'] res['blog'].append(tmp) else: res['blog'].append(noconnectiondata) except: res['blog'].append(noconnectiondata) pass return buildResponse(res, 200) @app.route('/api/get_version', methods = ['POST','GET']) def get_version(): """return version info and serial in crypted format for front updater service""" VERSIONFILE="_version.py" log.error("front_update_request") from _version import __version__ res={} res['version']=__version__ try: res['username']=db_sysconfig.get_sysconfig('username') except: res['username']=False interfaces = util.get_ethernet_wifi_interfaces() hwid = util.generate_serial_number(interfaces) install_date=False try: install_date=db_sysconfig.get_sysconfig('install_date') except: pass update_mode=db_sysconfig.get_sysconfig('update_mode') update_mode=json.loads(update_mode) if install_date: if update_mode['mode']=='manual': if not update_mode['update_front']: hwid=hwid+"MANUAL" else: update_mode['update_front']=False db_sysconfig.set_sysconfig('update_mode',json.dumps(update_mode)) res['serial'] = hwid + "-" + datetime.datetime.strptime(install_date, "%Y-%m-%d %H:%M:%S").strftime("%Y%m%d") if update_mode=='update_now': db_sysconfig.update_sysconfig('update_mode','manual') else: res['serial']=False log.error(res) res=util.crypt_data(json.dumps(res)) return buildResponse(res, 200) @app.route('/api/dashboard/traffic', methods = ['POST']) @login_required(role='admin', perm={'device':'read'}) def dashboard_traffic(): """return all devices traffic information""" input = request.json devid='all' chart_type=input.get('type','bps') delta=input.get('delta',"live") interface=input.get('interface','total') if delta not in ["5m","1h","daily","live"]: return buildResponse({'status': 'failed'},200,error="Wrong Data") if delta=="5m": start_time=datetime.datetime.now()-datetime.timedelta(minutes=5*24) elif delta=="1h": start_time=datetime.datetime.now()-datetime.timedelta(hours=24) elif delta=="daily": start_time=datetime.datetime.now()-datetime.timedelta(days=30) else: start_time=datetime.datetime.now()-datetime.timedelta(days=30) end_time=datetime.datetime.now() #Fix and change some data #Get data from redis try: res={ 'id':devid, 'sensors':['rx-total','tx-total'] } redopts={ "dev_id":res['id'], "keys":res['sensors'], "start_time":start_time, "end_time":end_time, "delta":delta, } colors={ 'backgroundColor': 'rgba(77,189,116,.2)', 'borderColor': '#4dbd74', 'pointHoverBackgroundColor': '#fff' } reddb=RedisDB(redopts) data=reddb.get_dev_data_keys() temp=[] ids=['yA','yB'] colors=['#4caf50','#ff9800'] bgcolor=['rgba(76, 175, 80, 0.2)','rgba(255, 152, 0, 0.2)'] datasets=[] lables=[] data_keys=['tx-{}'.format(interface),'rx-{}'.format(interface)] if chart_type=='bps': data_keys=['tx-{}'.format(interface),'rx-{}'.format(interface)] elif chart_type=='pps': data_keys=['txp-{}'.format(interface),'rxp-{}'.format(interface)] for idx, val in enumerate(data_keys): for d in data[val]: if len(lables) <= len(data[val]): lables.append(datetime.datetime.fromtimestamp(d[0]/1000)) temp.append(round(d[1],1)) datasets.append({'label':val,'borderColor': colors[idx],'type': 'line','yAxisID': ids[idx],'data':temp,'unit':val.split("-")[0],'backgroundColor': bgcolor[idx],'pointHoverBackgroundColor': '#fff','fill': True}) temp=[] res["data"]={'labels':lables,'datasets':datasets} except Exception as e: log.error(e) return buildResponse({'status': 'failed'}, 200, error=e) pass return buildResponse(res,200) @app.route('/api/dashboard/tasks/running', methods = ['POST']) @login_required(role='admin', perm={'settings':'read'}) def dashboard_tasks_running(): """return all running tasks""" input = request.json tasks=db_tasks.Tasks try: res=tasks.select().where(tasks.status=='running').dicts() except Exception as e: log.error(e) return buildResponse({'status': 'failed'}, 200, error=e) return buildResponse(res,200)