You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
377 lines
14 KiB
377 lines
14 KiB
1 month ago
|
import dash
|
||
|
import time
|
||
|
import uuid
|
||
|
import json
|
||
|
from dash import dcc
|
||
|
from dash.dependencies import Input, Output, State, ALL
|
||
|
from dash.exceptions import PreventUpdate
|
||
|
import feffery_utils_components as fuc
|
||
|
|
||
|
from server import app
|
||
|
from api.log import get_operation_log_list_api, get_operation_log_detail_api, delete_operation_log_api, clear_operation_log_api, export_operation_log_list_api
|
||
|
from api.dict import query_dict_data_list_api
|
||
|
|
||
|
|
||
|
@app.callback(
|
||
|
output=dict(
|
||
|
operation_log_table_data=Output('operation_log-list-table', 'data', allow_duplicate=True),
|
||
|
operation_log_table_pagination=Output('operation_log-list-table', 'pagination', allow_duplicate=True),
|
||
|
operation_log_table_key=Output('operation_log-list-table', 'key'),
|
||
|
operation_log_table_selectedrowkeys=Output('operation_log-list-table', 'selectedRowKeys'),
|
||
|
api_check_token_trigger=Output('api-check-token', 'data', allow_duplicate=True)
|
||
|
),
|
||
|
inputs=dict(
|
||
|
search_click=Input('operation_log-search', 'nClicks'),
|
||
|
refresh_click=Input('operation_log-refresh', 'nClicks'),
|
||
|
sorter=Input('operation_log-list-table', 'sorter'),
|
||
|
pagination=Input('operation_log-list-table', 'pagination'),
|
||
|
operations=Input('operation_log-operations-store', 'data')
|
||
|
),
|
||
|
state=dict(
|
||
|
title=State('operation_log-title-input', 'value'),
|
||
|
oper_name=State('operation_log-oper_name-input', 'value'),
|
||
|
business_type=State('operation_log-business_type-select', 'value'),
|
||
|
status_select=State('operation_log-status-select', 'value'),
|
||
|
oper_time_range=State('operation_log-oper_time-range', 'value'),
|
||
|
button_perms=State('operation_log-button-perms-container', 'data')
|
||
|
),
|
||
|
prevent_initial_call=True
|
||
|
)
|
||
|
def get_operation_log_table_data(search_click, refresh_click, sorter, pagination, operations, title, oper_name, business_type, status_select, oper_time_range, button_perms):
|
||
|
"""
|
||
|
获取操作日志表格数据回调(进行表格相关增删查改操作后均会触发此回调)
|
||
|
"""
|
||
|
|
||
|
oper_time_start = None
|
||
|
oper_time_end = None
|
||
|
if oper_time_range:
|
||
|
oper_time_start = oper_time_range[0]
|
||
|
oper_time_end = oper_time_range[1]
|
||
|
query_params = dict(
|
||
|
title=title,
|
||
|
oper_name=oper_name,
|
||
|
business_type=business_type,
|
||
|
status=status_select,
|
||
|
oper_time_start=oper_time_start,
|
||
|
oper_time_end=oper_time_end,
|
||
|
order_by_column=sorter.get('columns')[0] if sorter else None,
|
||
|
is_asc=sorter.get('orders')[0] if sorter else None,
|
||
|
page_num=1,
|
||
|
page_size=10
|
||
|
)
|
||
|
triggered_prop = dash.ctx.triggered[0].get('prop_id')
|
||
|
if triggered_prop == 'operation_log-list-table.pagination':
|
||
|
query_params = dict(
|
||
|
title=title,
|
||
|
oper_name=oper_name,
|
||
|
business_type=business_type,
|
||
|
status=status_select,
|
||
|
oper_time_start=oper_time_start,
|
||
|
oper_time_end=oper_time_end,
|
||
|
order_by_column=sorter.get('columns')[0] if sorter else None,
|
||
|
is_asc=sorter.get('orders')[0] if sorter else None,
|
||
|
page_num=pagination['current'],
|
||
|
page_size=pagination['pageSize']
|
||
|
)
|
||
|
if search_click or refresh_click or pagination or operations:
|
||
|
option_table = []
|
||
|
info = query_dict_data_list_api(dict_type='sys_oper_type')
|
||
|
if info.get('code') == 200:
|
||
|
data = info.get('data')
|
||
|
option_table = [dict(label=item.get('dict_label'), value=item.get('dict_value'), css_class=item.get('css_class')) for item in data]
|
||
|
option_dict = {item.get('value'): item for item in option_table}
|
||
|
|
||
|
table_info = get_operation_log_list_api(query_params)
|
||
|
if table_info['code'] == 200:
|
||
|
table_data = table_info['data']['rows']
|
||
|
table_pagination = dict(
|
||
|
pageSize=table_info['data']['page_size'],
|
||
|
current=table_info['data']['page_num'],
|
||
|
showSizeChanger=True,
|
||
|
pageSizeOptions=[10, 30, 50, 100],
|
||
|
showQuickJumper=True,
|
||
|
total=table_info['data']['total']
|
||
|
)
|
||
|
for item in table_data:
|
||
|
if item['status'] == 0:
|
||
|
item['status'] = dict(tag='成功', color='blue')
|
||
|
else:
|
||
|
item['status'] = dict(tag='失败', color='volcano')
|
||
|
if str(item.get('business_type')) in option_dict.keys():
|
||
|
item['business_type'] = dict(
|
||
|
tag=option_dict.get(str(item.get('business_type'))).get('label'),
|
||
|
color=json.loads(option_dict.get(str(item.get('business_type'))).get('css_class')).get('color')
|
||
|
)
|
||
|
item['key'] = str(item['oper_id'])
|
||
|
item['cost_time'] = f"{item['cost_time']}毫秒"
|
||
|
item['operation'] = [
|
||
|
{
|
||
|
'content': '详情',
|
||
|
'type': 'link',
|
||
|
'icon': 'antd-eye'
|
||
|
} if 'monitor:operlog:query' in button_perms else {},
|
||
|
]
|
||
|
|
||
|
return dict(
|
||
|
operation_log_table_data=table_data,
|
||
|
operation_log_table_pagination=table_pagination,
|
||
|
operation_log_table_key=str(uuid.uuid4()),
|
||
|
operation_log_table_selectedrowkeys=None,
|
||
|
api_check_token_trigger={'timestamp': time.time()}
|
||
|
)
|
||
|
|
||
|
return dict(
|
||
|
operation_log_table_data=dash.no_update,
|
||
|
operation_log_table_pagination=dash.no_update,
|
||
|
operation_log_table_key=dash.no_update,
|
||
|
operation_log_table_selectedrowkeys=dash.no_update,
|
||
|
api_check_token_trigger={'timestamp': time.time()}
|
||
|
)
|
||
|
|
||
|
raise PreventUpdate
|
||
|
|
||
|
|
||
|
# 重置操作日志搜索表单数据回调
|
||
|
app.clientside_callback(
|
||
|
'''
|
||
|
(reset_click) => {
|
||
|
if (reset_click) {
|
||
|
return [null, null, null, null, null, {'type': 'reset'}]
|
||
|
}
|
||
|
return window.dash_clientside.no_update;
|
||
|
}
|
||
|
''',
|
||
|
[Output('operation_log-title-input', 'value'),
|
||
|
Output('operation_log-oper_name-input', 'value'),
|
||
|
Output('operation_log-business_type-select', 'value'),
|
||
|
Output('operation_log-status-select', 'value'),
|
||
|
Output('operation_log-oper_time-range', 'value'),
|
||
|
Output('operation_log-operations-store', 'data')],
|
||
|
Input('operation_log-reset', 'nClicks'),
|
||
|
prevent_initial_call=True
|
||
|
)
|
||
|
|
||
|
|
||
|
# 隐藏/显示操作日志搜索表单回调
|
||
|
app.clientside_callback(
|
||
|
'''
|
||
|
(hidden_click, hidden_status) => {
|
||
|
if (hidden_click) {
|
||
|
return [
|
||
|
!hidden_status,
|
||
|
hidden_status ? '隐藏搜索' : '显示搜索'
|
||
|
]
|
||
|
}
|
||
|
return window.dash_clientside.no_update;
|
||
|
}
|
||
|
''',
|
||
|
[Output('operation_log-search-form-container', 'hidden'),
|
||
|
Output('operation_log-hidden-tooltip', 'title')],
|
||
|
Input('operation_log-hidden', 'nClicks'),
|
||
|
State('operation_log-search-form-container', 'hidden'),
|
||
|
prevent_initial_call=True
|
||
|
)
|
||
|
|
||
|
|
||
|
@app.callback(
|
||
|
output=dict(
|
||
|
modal_visible=Output('operation_log-modal', 'visible', allow_duplicate=True),
|
||
|
modal_title=Output('operation_log-modal', 'title'),
|
||
|
form_value=Output({'type': 'operation_log-form-value', 'index': ALL}, 'children'),
|
||
|
api_check_token_trigger=Output('api-check-token', 'data', allow_duplicate=True)
|
||
|
),
|
||
|
inputs=dict(
|
||
|
button_click=Input('operation_log-list-table', 'nClicksButton')
|
||
|
),
|
||
|
state=dict(
|
||
|
clicked_content=State('operation_log-list-table', 'clickedContent'),
|
||
|
recently_button_clicked_row=State('operation_log-list-table', 'recentlyButtonClickedRow')
|
||
|
),
|
||
|
prevent_initial_call=True
|
||
|
)
|
||
|
def add_edit_operation_log_modal(button_click, clicked_content, recently_button_clicked_row):
|
||
|
"""
|
||
|
显示操作日志详情弹窗回调
|
||
|
"""
|
||
|
if button_click:
|
||
|
# 获取所有输出表单项对应value的index
|
||
|
form_value_list = [x['id']['index'] for x in dash.ctx.outputs_list[-2]]
|
||
|
oper_id = int(recently_button_clicked_row['key'])
|
||
|
operation_log_info_res = get_operation_log_detail_api(oper_id=oper_id)
|
||
|
if operation_log_info_res['code'] == 200:
|
||
|
operation_log_info = operation_log_info_res['data']
|
||
|
oper_name = operation_log_info.get('oper_name') if operation_log_info.get('oper_name') else ''
|
||
|
oper_ip = operation_log_info.get('oper_ip') if operation_log_info.get('oper_ip') else ''
|
||
|
oper_location = operation_log_info.get('oper_location') if operation_log_info.get('oper_location') else ''
|
||
|
operation_log_info['login_info'] = f'{oper_name} / {oper_ip} / {oper_location}'
|
||
|
operation_log_info['status'] = '正常' if operation_log_info.get('status') == 0 else '失败'
|
||
|
operation_log_info['cost_time'] = f"{operation_log_info.get('cost_time')}毫秒"
|
||
|
return dict(
|
||
|
modal_visible=True,
|
||
|
modal_title='操作日志详情',
|
||
|
form_value=[operation_log_info.get(k) for k in form_value_list],
|
||
|
api_check_token_trigger={'timestamp': time.time()}
|
||
|
)
|
||
|
|
||
|
return dict(
|
||
|
modal_visible=dash.no_update,
|
||
|
modal_title=dash.no_update,
|
||
|
form_value=[dash.no_update] * len(form_value_list),
|
||
|
api_check_token_trigger={'timestamp': time.time()}
|
||
|
)
|
||
|
|
||
|
raise PreventUpdate
|
||
|
|
||
|
|
||
|
@app.callback(
|
||
|
Output({'type': 'operation_log-operation-button', 'index': 'delete'}, 'disabled'),
|
||
|
Input('operation_log-list-table', 'selectedRowKeys'),
|
||
|
prevent_initial_call=True
|
||
|
)
|
||
|
def change_operation_log_delete_button_status(table_rows_selected):
|
||
|
"""
|
||
|
根据选择的表格数据行数控制删除按钮状态回调
|
||
|
"""
|
||
|
outputs_list = dash.ctx.outputs_list
|
||
|
if outputs_list:
|
||
|
if table_rows_selected:
|
||
|
|
||
|
return False
|
||
|
|
||
|
return True
|
||
|
|
||
|
raise PreventUpdate
|
||
|
|
||
|
|
||
|
@app.callback(
|
||
|
[Output('operation_log-delete-text', 'children'),
|
||
|
Output('operation_log-delete-confirm-modal', 'visible'),
|
||
|
Output('operation_log-delete-ids-store', 'data')],
|
||
|
Input({'type': 'operation_log-operation-button', 'index': ALL}, 'nClicks'),
|
||
|
State('operation_log-list-table', 'selectedRowKeys'),
|
||
|
prevent_initial_call=True
|
||
|
)
|
||
|
def operation_log_delete_modal(operation_click, selected_row_keys):
|
||
|
"""
|
||
|
显示删除或清空操作日志二次确认弹窗回调
|
||
|
"""
|
||
|
trigger_id = dash.ctx.triggered_id
|
||
|
if trigger_id.index in ['delete', 'clear']:
|
||
|
if trigger_id.index == 'delete':
|
||
|
oper_ids = ','.join(selected_row_keys)
|
||
|
|
||
|
return [
|
||
|
f'是否确认删除日志编号为{oper_ids}的操作日志?',
|
||
|
True,
|
||
|
{'oper_type': 'delete', 'oper_ids': oper_ids}
|
||
|
]
|
||
|
|
||
|
elif trigger_id.index == 'clear':
|
||
|
return [
|
||
|
f'是否确认清除所有的操作日志?',
|
||
|
True,
|
||
|
{'oper_type': 'clear', 'oper_ids': ''}
|
||
|
]
|
||
|
|
||
|
raise PreventUpdate
|
||
|
|
||
|
|
||
|
@app.callback(
|
||
|
[Output('operation_log-operations-store', 'data', allow_duplicate=True),
|
||
|
Output('api-check-token', 'data', allow_duplicate=True),
|
||
|
Output('global-message-container', 'children', allow_duplicate=True)],
|
||
|
Input('operation_log-delete-confirm-modal', 'okCounts'),
|
||
|
State('operation_log-delete-ids-store', 'data'),
|
||
|
prevent_initial_call=True
|
||
|
)
|
||
|
def operation_log_delete_confirm(delete_confirm, oper_ids_data):
|
||
|
"""
|
||
|
删除或清空操作日志弹窗确认回调,实现删除或清空操作
|
||
|
"""
|
||
|
if delete_confirm:
|
||
|
|
||
|
oper_type = oper_ids_data.get('oper_type')
|
||
|
if oper_type == 'clear':
|
||
|
params = dict(oper_type=oper_ids_data.get('oper_type'))
|
||
|
clear_button_info = clear_operation_log_api(params)
|
||
|
if clear_button_info['code'] == 200:
|
||
|
return [
|
||
|
{'type': 'delete'},
|
||
|
{'timestamp': time.time()},
|
||
|
fuc.FefferyFancyMessage('清除成功', type='success')
|
||
|
]
|
||
|
|
||
|
return [
|
||
|
dash.no_update,
|
||
|
{'timestamp': time.time()},
|
||
|
fuc.FefferyFancyMessage('清除失败', type='error')
|
||
|
]
|
||
|
else:
|
||
|
params = dict(oper_ids=oper_ids_data.get('oper_ids'))
|
||
|
delete_button_info = delete_operation_log_api(params)
|
||
|
if delete_button_info['code'] == 200:
|
||
|
return [
|
||
|
{'type': 'delete'},
|
||
|
{'timestamp': time.time()},
|
||
|
fuc.FefferyFancyMessage('删除成功', type='success')
|
||
|
]
|
||
|
|
||
|
return [
|
||
|
dash.no_update,
|
||
|
{'timestamp': time.time()},
|
||
|
fuc.FefferyFancyMessage('删除失败', type='error')
|
||
|
]
|
||
|
|
||
|
raise PreventUpdate
|
||
|
|
||
|
|
||
|
@app.callback(
|
||
|
[Output('operation_log-export-container', 'data', allow_duplicate=True),
|
||
|
Output('operation_log-export-complete-judge-container', 'data'),
|
||
|
Output('api-check-token', 'data', allow_duplicate=True),
|
||
|
Output('global-message-container', 'children', allow_duplicate=True)],
|
||
|
Input('operation_log-export', 'nClicks'),
|
||
|
prevent_initial_call=True
|
||
|
)
|
||
|
def export_operation_log_list(export_click):
|
||
|
"""
|
||
|
导出操作日志信息回调
|
||
|
"""
|
||
|
if export_click:
|
||
|
export_operation_log_res = export_operation_log_list_api({})
|
||
|
if export_operation_log_res.status_code == 200:
|
||
|
export_operation_log = export_operation_log_res.content
|
||
|
|
||
|
return [
|
||
|
dcc.send_bytes(export_operation_log, f'操作日志信息_{time.strftime("%Y%m%d%H%M%S", time.localtime())}.xlsx'),
|
||
|
{'timestamp': time.time()},
|
||
|
{'timestamp': time.time()},
|
||
|
fuc.FefferyFancyMessage('导出成功', type='success')
|
||
|
]
|
||
|
|
||
|
return [
|
||
|
dash.no_update,
|
||
|
dash.no_update,
|
||
|
{'timestamp': time.time()},
|
||
|
fuc.FefferyFancyMessage('导出失败', type='error')
|
||
|
]
|
||
|
|
||
|
raise PreventUpdate
|
||
|
|
||
|
|
||
|
@app.callback(
|
||
|
Output('operation_log-export-container', 'data', allow_duplicate=True),
|
||
|
Input('operation_log-export-complete-judge-container', 'data'),
|
||
|
prevent_initial_call=True
|
||
|
)
|
||
|
def reset_operation_log_export_status(data):
|
||
|
"""
|
||
|
导出完成后重置下载组件数据回调,防止重复下载文件
|
||
|
"""
|
||
|
time.sleep(0.5)
|
||
|
if data:
|
||
|
|
||
|
return None
|
||
|
|
||
|
raise PreventUpdate
|