fix_bug_log-export_格式和乱码
This commit is contained in:
		
							parent
							
								
									0378834133
								
							
						
					
					
						commit
						772748eaae
					
				@ -93,16 +93,13 @@ def log_detail(log_id):
 | 
				
			|||||||
@permission_required('view_logs')  # 替代 @admin_required
 | 
					@permission_required('view_logs')  # 替代 @admin_required
 | 
				
			||||||
def export_logs():
 | 
					def export_logs():
 | 
				
			||||||
    """导出日志API"""
 | 
					    """导出日志API"""
 | 
				
			||||||
    import csv
 | 
					 | 
				
			||||||
    from io import StringIO
 | 
					 | 
				
			||||||
    from flask import Response
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    data = request.get_json()
 | 
					    data = request.get_json()
 | 
				
			||||||
    user_id = data.get('user_id')
 | 
					    user_id = data.get('user_id')
 | 
				
			||||||
    action = data.get('action')
 | 
					    action = data.get('action')
 | 
				
			||||||
    target_type = data.get('target_type')
 | 
					    target_type = data.get('target_type')
 | 
				
			||||||
    start_date_str = data.get('start_date')
 | 
					    start_date_str = data.get('start_date')
 | 
				
			||||||
    end_date_str = data.get('end_date')
 | 
					    end_date_str = data.get('end_date')
 | 
				
			||||||
 | 
					    export_format = data.get('format', 'csv')  # 获取导出格式参数
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    # 处理日期范围
 | 
					    # 处理日期范围
 | 
				
			||||||
    start_date = None
 | 
					    start_date = None
 | 
				
			||||||
@ -129,9 +126,35 @@ def export_logs():
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
    logs = query.all()
 | 
					    logs = query.all()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    # 生成CSV文件
 | 
					    try:
 | 
				
			||||||
    si = StringIO()
 | 
					        # 根据格式选择导出方法
 | 
				
			||||||
    csv_writer = csv.writer(si)
 | 
					        if export_format == 'xlsx':
 | 
				
			||||||
 | 
					            return export_as_xlsx(logs)
 | 
				
			||||||
 | 
					        else:
 | 
				
			||||||
 | 
					            return export_as_csv(logs)
 | 
				
			||||||
 | 
					    except Exception as e:
 | 
				
			||||||
 | 
					        # 记录错误以便调试
 | 
				
			||||||
 | 
					        import traceback
 | 
				
			||||||
 | 
					        error_details = traceback.format_exc()
 | 
				
			||||||
 | 
					        current_app.logger.error(f"Export error: {str(e)}\n{error_details}")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        return jsonify({
 | 
				
			||||||
 | 
					            'success': False,
 | 
				
			||||||
 | 
					            'message': f'导出失败: {str(e)}'
 | 
				
			||||||
 | 
					        }), 500
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def export_as_csv(logs):
 | 
				
			||||||
 | 
					    """导出为CSV格式"""
 | 
				
			||||||
 | 
					    import csv
 | 
				
			||||||
 | 
					    from io import StringIO
 | 
				
			||||||
 | 
					    import base64
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    # 创建CSV文件
 | 
				
			||||||
 | 
					    output = StringIO()
 | 
				
			||||||
 | 
					    output.write('\ufeff')  # 添加BOM标记,解决Excel中文乱码
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    csv_writer = csv.writer(output)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    # 写入标题行
 | 
					    # 写入标题行
 | 
				
			||||||
    csv_writer.writerow(['ID', '用户', '操作类型', '目标类型', '目标ID', 'IP地址', '描述', '创建时间'])
 | 
					    csv_writer.writerow(['ID', '用户', '操作类型', '目标类型', '目标ID', 'IP地址', '描述', '创建时间'])
 | 
				
			||||||
@ -150,25 +173,88 @@ def export_logs():
 | 
				
			|||||||
            log.created_at.strftime('%Y-%m-%d %H:%M:%S')
 | 
					            log.created_at.strftime('%Y-%m-%d %H:%M:%S')
 | 
				
			||||||
        ])
 | 
					        ])
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    # 设置响应头,使浏览器将其识别为下载文件
 | 
					    # 获取CSV字符串并进行Base64编码
 | 
				
			||||||
 | 
					    csv_string = output.getvalue()
 | 
				
			||||||
 | 
					    csv_bytes = csv_string.encode('utf-8')
 | 
				
			||||||
 | 
					    b64_encoded = base64.b64encode(csv_bytes).decode('utf-8')
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    # 设置文件名
 | 
				
			||||||
    filename = f"system_logs_{datetime.now().strftime('%Y%m%d_%H%M%S')}.csv"
 | 
					    filename = f"system_logs_{datetime.now().strftime('%Y%m%d_%H%M%S')}.csv"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    output = si.getvalue()
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    # 返回Base64编码的CSV数据
 | 
					 | 
				
			||||||
    import base64
 | 
					 | 
				
			||||||
    encoded_data = base64.b64encode(output.encode('utf-8')).decode('utf-8')
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    return jsonify({
 | 
					    return jsonify({
 | 
				
			||||||
        'success': True,
 | 
					        'success': True,
 | 
				
			||||||
        'message': f'已生成 {len(logs)} 条日志记录',
 | 
					        'message': f'已生成 {len(logs)} 条日志记录',
 | 
				
			||||||
        'count': len(logs),
 | 
					        'count': len(logs),
 | 
				
			||||||
        'filename': filename,
 | 
					        'filename': filename,
 | 
				
			||||||
        'filedata': encoded_data,
 | 
					        'filedata': b64_encoded,
 | 
				
			||||||
        'filetype': 'text/csv'
 | 
					        'filetype': 'text/csv'
 | 
				
			||||||
    })
 | 
					    })
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def export_as_xlsx(logs):
 | 
				
			||||||
 | 
					    """导出为XLSX格式"""
 | 
				
			||||||
 | 
					    import base64
 | 
				
			||||||
 | 
					    from io import BytesIO
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    try:
 | 
				
			||||||
 | 
					        # 动态导入openpyxl,如果不存在则抛出异常
 | 
				
			||||||
 | 
					        import openpyxl
 | 
				
			||||||
 | 
					    except ImportError:
 | 
				
			||||||
 | 
					        raise Exception("未安装openpyxl库,无法导出Excel格式。请安装后重试: pip install openpyxl")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    # 创建工作簿和工作表
 | 
				
			||||||
 | 
					    wb = openpyxl.Workbook()
 | 
				
			||||||
 | 
					    ws = wb.active
 | 
				
			||||||
 | 
					    ws.title = "系统日志"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    # 写入标题行
 | 
				
			||||||
 | 
					    headers = ['ID', '用户', '操作类型', '目标类型', '目标ID', 'IP地址', '描述', '创建时间']
 | 
				
			||||||
 | 
					    for col_idx, header in enumerate(headers, 1):
 | 
				
			||||||
 | 
					        ws.cell(row=1, column=col_idx, value=header)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    # 写入数据行
 | 
				
			||||||
 | 
					    for row_idx, log in enumerate(logs, 2):
 | 
				
			||||||
 | 
					        username = log.user.username if log.user else "未登录"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        ws.cell(row=row_idx, column=1, value=log.id)
 | 
				
			||||||
 | 
					        ws.cell(row=row_idx, column=2, value=username)
 | 
				
			||||||
 | 
					        ws.cell(row=row_idx, column=3, value=log.action)
 | 
				
			||||||
 | 
					        ws.cell(row=row_idx, column=4, value=log.target_type or '')
 | 
				
			||||||
 | 
					        ws.cell(row=row_idx, column=5, value=log.target_id or '')
 | 
				
			||||||
 | 
					        ws.cell(row=row_idx, column=6, value=log.ip_address or '')
 | 
				
			||||||
 | 
					        ws.cell(row=row_idx, column=7, value=log.description or '')
 | 
				
			||||||
 | 
					        ws.cell(row=row_idx, column=8, value=log.created_at.strftime('%Y-%m-%d %H:%M:%S'))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    # 调整列宽
 | 
				
			||||||
 | 
					    for col_idx, header in enumerate(headers, 1):
 | 
				
			||||||
 | 
					        column_letter = openpyxl.utils.get_column_letter(col_idx)
 | 
				
			||||||
 | 
					        if header == '描述':
 | 
				
			||||||
 | 
					            ws.column_dimensions[column_letter].width = 40
 | 
				
			||||||
 | 
					        else:
 | 
				
			||||||
 | 
					            ws.column_dimensions[column_letter].width = 15
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    # 保存到内存
 | 
				
			||||||
 | 
					    output = BytesIO()
 | 
				
			||||||
 | 
					    wb.save(output)
 | 
				
			||||||
 | 
					    output.seek(0)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    # 编码为Base64
 | 
				
			||||||
 | 
					    xlsx_data = output.getvalue()
 | 
				
			||||||
 | 
					    b64_encoded = base64.b64encode(xlsx_data).decode('utf-8')
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    # 设置文件名
 | 
				
			||||||
 | 
					    filename = f"system_logs_{datetime.now().strftime('%Y%m%d_%H%M%S')}.xlsx"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    return jsonify({
 | 
				
			||||||
 | 
					        'success': True,
 | 
				
			||||||
 | 
					        'message': f'已生成 {len(logs)} 条日志记录',
 | 
				
			||||||
 | 
					        'count': len(logs),
 | 
				
			||||||
 | 
					        'filename': filename,
 | 
				
			||||||
 | 
					        'filedata': b64_encoded,
 | 
				
			||||||
 | 
					        'filetype': 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet'
 | 
				
			||||||
 | 
					    })
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@log_bp.route('/api/clear', methods=['POST'])
 | 
					@log_bp.route('/api/clear', methods=['POST'])
 | 
				
			||||||
@login_required
 | 
					@login_required
 | 
				
			||||||
@permission_required('view_logs')  # 替代 @admin_required
 | 
					@permission_required('view_logs')  # 替代 @admin_required
 | 
				
			||||||
 | 
				
			|||||||
@ -14,3 +14,4 @@ flask-login
 | 
				
			|||||||
openpyxl
 | 
					openpyxl
 | 
				
			||||||
xlrd
 | 
					xlrd
 | 
				
			||||||
xlsxwriter
 | 
					xlsxwriter
 | 
				
			||||||
 | 
					openpyxl
 | 
				
			||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user