MikroWizard.mikroman/py/api/api_logs.py
sepehr 70dc0ddc55 Bugs:
Fixed Firmware download from the Mikrotik website when there are multiple npk available
Fixed Mikrowizard system permission error when it is set to None
Fixed user device group permissions
Some minor UI improvements
Fix IP scan for one IP scan / Fix not scanning the last IP in the range
Fix manual snippet execution not working when device groups are selected
Some minor bug fixes and improvements

New:
Show background tasks and be able to stop them while running in the background (like an IP scanner)
Add support for manual MikroWizard update dashboard/settings page
update to version 1.0.5

Enhancement:
Show permission error in some pages when the  user doesn't have permission for that page/action
show better charts/graphs in the dashboard and device interface details
show more info on the dashboard about update and version information and license
2025-01-02 20:12:00 +03:00

551 lines
26 KiB
Python

#!/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! <a target=\"_blank\" href=\"https://mikrowizard.com/plan-your-project-with-your-software/\">Read More</a>",
"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]+" ... " + '<a target="_blank" href="'+f['link']+'">Read More</a>'
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)