mirror of
https://github.com/MikroWizard/mikroman.git
synced 2025-06-21 03:55:39 +02:00
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
551 lines
26 KiB
Python
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)
|