MikroWizard.mikroman/py/api/api_logs.py

554 lines
26 KiB
Python
Raw Permalink Normal View History

#!/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:
2024-07-31 17:42:42 +03:30
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
else:
res['username']=False
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)