434 lines
16 KiB
Python
434 lines
16 KiB
Python
import logging
|
||
import traceback
|
||
from rest_framework import status
|
||
from rest_framework.decorators import api_view, permission_classes
|
||
from rest_framework.permissions import IsAuthenticated
|
||
from rest_framework.response import Response
|
||
|
||
from user_management.models import (
|
||
GmailTalentMapping, ChatHistory, ConversationSummary
|
||
)
|
||
from user_management.gmail_integration import GmailIntegration
|
||
|
||
from feishu.feishu_ai_chat import (
|
||
fetch_table_records, find_duplicate_email_creators,
|
||
process_duplicate_emails, auto_chat_session,
|
||
check_goal_achieved
|
||
)
|
||
|
||
logger = logging.getLogger(__name__)
|
||
|
||
@api_view(['POST'])
|
||
@permission_classes([IsAuthenticated])
|
||
def process_feishu_table(request):
|
||
"""
|
||
从飞书多维表格读取数据,处理重复邮箱
|
||
|
||
请求参数:
|
||
table_id: 表格ID
|
||
view_id: 视图ID
|
||
app_token: 飞书应用TOKEN (可选)
|
||
access_token: 用户访问令牌 (可选)
|
||
app_id: 应用ID (可选)
|
||
app_secret: 应用密钥 (可选)
|
||
goal_template: 目标内容模板 (可选)
|
||
auto_chat: 是否自动执行AI对话 (可选)
|
||
turns: 自动对话轮次 (可选)
|
||
"""
|
||
try:
|
||
# 检查用户权限 - 只允许组长使用
|
||
if request.user.role != 'leader':
|
||
return Response(
|
||
{"error": "只有组长角色的用户可以使用此功能"},
|
||
status=status.HTTP_403_FORBIDDEN
|
||
)
|
||
|
||
# 获取参数
|
||
table_id = request.data.get("table_id", "tbl3oikG3F8YYtVA") # 默认表格ID
|
||
view_id = request.data.get("view_id", "vewSOIsmxc") # 默认视图ID
|
||
app_token = request.data.get("app_token", "XYE6bMQUOaZ5y5svj4vcWohGnmg")
|
||
access_token = request.data.get("access_token", "u-fK0HvbXVte.G2xzYs5oxV6k1nHu1glvFgG00l0Ma24VD")
|
||
app_id = request.data.get("app_id", "cli_a5c97daacb9e500d")
|
||
app_secret = request.data.get("app_secret", "fdVeOCLXmuIHZVmSV0VbJh9wd0Kq1o5y")
|
||
goal_template = request.data.get(
|
||
"goal_template",
|
||
"与达人{handle}(邮箱:{email})建立联系并了解其账号情况,评估合作潜力,处理合作需求,最终目标是达成合作并签约。"
|
||
)
|
||
auto_chat = request.data.get("auto_chat", False)
|
||
turns = request.data.get("turns", 5)
|
||
|
||
logger.info(f"处理飞书表格数据: table_id={table_id}, view_id={view_id}, app_id={app_id}")
|
||
|
||
# 从飞书表格获取记录
|
||
records = fetch_table_records(
|
||
app_token,
|
||
table_id,
|
||
view_id,
|
||
access_token,
|
||
app_id,
|
||
app_secret
|
||
)
|
||
|
||
if not records:
|
||
logger.warning("未获取到任何记录,可能是表格ID或视图ID不正确,或无权限访问")
|
||
|
||
# 尝试使用SDK中的search方法直接获取
|
||
try:
|
||
import lark_oapi as lark
|
||
from lark_oapi.api.bitable.v1 import (
|
||
SearchAppTableRecordRequest,
|
||
SearchAppTableRecordRequestBody
|
||
)
|
||
|
||
# 创建client
|
||
client = lark.Client.builder() \
|
||
.enable_set_token(True) \
|
||
.log_level(lark.LogLevel.DEBUG) \
|
||
.build()
|
||
|
||
# 构造请求对象
|
||
request = SearchAppTableRecordRequest.builder() \
|
||
.app_token(app_token) \
|
||
.table_id(table_id) \
|
||
.page_size(20) \
|
||
.request_body(SearchAppTableRecordRequestBody.builder().build()) \
|
||
.build()
|
||
|
||
# 发起请求
|
||
option = lark.RequestOption.builder().user_access_token(access_token).build()
|
||
response = client.bitable.v1.app_table_record.search(request, option)
|
||
|
||
if not response.success():
|
||
logger.error(f"直接搜索请求失败: {response.code}, {response.msg}")
|
||
return Response(
|
||
{"message": f"未获取到任何记录,错误: {response.msg}"},
|
||
status=status.HTTP_404_NOT_FOUND
|
||
)
|
||
|
||
# 获取记录
|
||
records = response.data.items
|
||
if not records:
|
||
return Response(
|
||
{"message": "未获取到任何记录"},
|
||
status=status.HTTP_404_NOT_FOUND
|
||
)
|
||
except Exception as e:
|
||
logger.error(f"尝试直接搜索时出错: {str(e)}")
|
||
logger.error(traceback.format_exc())
|
||
return Response(
|
||
{"message": f"未获取到任何记录,错误: {str(e)}"},
|
||
status=status.HTTP_404_NOT_FOUND
|
||
)
|
||
|
||
# 查找重复邮箱的创作者
|
||
duplicate_emails = find_duplicate_email_creators(records)
|
||
|
||
if not duplicate_emails:
|
||
return Response(
|
||
{"message": "未发现重复邮箱"},
|
||
status=status.HTTP_200_OK
|
||
)
|
||
|
||
# 处理重复邮箱记录
|
||
results = process_duplicate_emails(duplicate_emails, goal_template)
|
||
|
||
# 如果需要自动对话
|
||
chat_results = []
|
||
if auto_chat and results['success'] > 0:
|
||
# 为每个成功创建的记录执行自动对话
|
||
for detail in results['details']:
|
||
if detail['status'] == 'success':
|
||
email = detail['email']
|
||
chat_result = auto_chat_session(request.user, email, max_turns=turns)
|
||
chat_results.append({
|
||
'email': email,
|
||
'result': chat_result
|
||
})
|
||
|
||
# 返回处理结果
|
||
return Response({
|
||
'status': 'success',
|
||
'records_count': len(records),
|
||
'duplicate_emails_count': len(duplicate_emails),
|
||
'processing_results': results,
|
||
'chat_results': chat_results
|
||
}, status=status.HTTP_200_OK)
|
||
|
||
except Exception as e:
|
||
logger.error(f"处理飞书表格时出错: {str(e)}")
|
||
logger.error(traceback.format_exc())
|
||
return Response(
|
||
{"error": str(e)},
|
||
status=status.HTTP_500_INTERNAL_SERVER_ERROR
|
||
)
|
||
|
||
@api_view(['POST'])
|
||
@permission_classes([IsAuthenticated])
|
||
def run_auto_chat(request):
|
||
"""
|
||
为指定邮箱执行自动对话
|
||
|
||
请求参数:
|
||
email: 达人邮箱
|
||
force_send: 是否强制发送新邮件(即使没有新回复)(可选)
|
||
subject: 邮件主题(可选,仅当force_send=true时使用)
|
||
content: 邮件内容(可选,仅当force_send=true时使用)
|
||
"""
|
||
try:
|
||
# 检查用户权限 - 只允许组长使用
|
||
if request.user.role != 'leader':
|
||
return Response(
|
||
{"error": "只有组长角色的用户可以使用此功能"},
|
||
status=status.HTTP_403_FORBIDDEN
|
||
)
|
||
|
||
# 获取参数
|
||
email = request.data.get("email")
|
||
force_send = request.data.get("force_send", False)
|
||
subject = request.data.get("subject")
|
||
content = request.data.get("content")
|
||
|
||
# 验证必要参数
|
||
if not email:
|
||
return Response(
|
||
{"error": "缺少参数email"},
|
||
status=status.HTTP_400_BAD_REQUEST
|
||
)
|
||
|
||
# 如果强制发送且没有提供内容
|
||
if force_send and not content:
|
||
return Response(
|
||
{"error": "当force_send=true时,必须提供content参数"},
|
||
status=status.HTTP_400_BAD_REQUEST
|
||
)
|
||
|
||
# 首先尝试同步最新邮件
|
||
try:
|
||
# 创建Gmail集成实例
|
||
gmail_integration = GmailIntegration(request.user)
|
||
|
||
# 同步最新邮件
|
||
logger.info(f"正在同步与 {email} 的最新邮件...")
|
||
sync_result = gmail_integration.sync_talent_emails(email)
|
||
|
||
if sync_result.get('status') == 'success':
|
||
logger.info(f"成功同步邮件: {sync_result.get('message', 'No message')}")
|
||
else:
|
||
logger.warning(f"同步邮件警告: {sync_result.get('message', 'Unknown warning')}")
|
||
|
||
except Exception as e:
|
||
logger.error(f"同步邮件出错: {str(e)}")
|
||
logger.error(traceback.format_exc())
|
||
# 仅记录错误,不中断流程
|
||
|
||
# 如果是强制发送模式
|
||
if force_send:
|
||
try:
|
||
# 获取知识库映射
|
||
mapping = GmailTalentMapping.objects.filter(
|
||
user=request.user,
|
||
talent_email=email,
|
||
is_active=True
|
||
).first()
|
||
|
||
if not mapping:
|
||
return Response(
|
||
{"error": f"找不到与邮箱 {email} 的映射关系"},
|
||
status=status.HTTP_404_NOT_FOUND
|
||
)
|
||
|
||
# 直接发送邮件
|
||
mail_subject = subject if subject else "关于合作的洽谈"
|
||
mail_result = gmail_integration.send_email(
|
||
to_email=email,
|
||
subject=mail_subject,
|
||
body=content,
|
||
conversation_id=mapping.conversation_id
|
||
)
|
||
|
||
if mail_result['status'] != 'success':
|
||
return Response(
|
||
{"error": f"邮件发送失败: {mail_result.get('message', 'Unknown error')}"},
|
||
status=status.HTTP_500_INTERNAL_SERVER_ERROR
|
||
)
|
||
|
||
# 保存发送的内容到对话历史
|
||
ChatHistory.objects.create(
|
||
user=request.user,
|
||
knowledge_base=mapping.knowledge_base,
|
||
conversation_id=mapping.conversation_id,
|
||
role='assistant',
|
||
content=content
|
||
)
|
||
|
||
return Response({
|
||
'status': 'success',
|
||
'message': f"已强制发送邮件到 {email}",
|
||
'email_sent': True,
|
||
'conversation_id': mapping.conversation_id
|
||
}, status=status.HTTP_200_OK)
|
||
|
||
except Exception as e:
|
||
logger.error(f"强制发送邮件时出错: {str(e)}")
|
||
logger.error(traceback.format_exc())
|
||
return Response(
|
||
{"error": str(e)},
|
||
status=status.HTTP_500_INTERNAL_SERVER_ERROR
|
||
)
|
||
|
||
# 执行自动对话
|
||
result = auto_chat_session(request.user, email)
|
||
|
||
# 返回结果
|
||
return Response(result, status=status.HTTP_200_OK)
|
||
|
||
except Exception as e:
|
||
logger.error(f"执行自动对话时出错: {str(e)}")
|
||
logger.error(traceback.format_exc())
|
||
return Response(
|
||
{"error": str(e)},
|
||
status=status.HTTP_500_INTERNAL_SERVER_ERROR
|
||
)
|
||
|
||
@api_view(['GET', 'POST'])
|
||
@permission_classes([IsAuthenticated])
|
||
def feishu_user_goal(request):
|
||
"""
|
||
设置或获取用户总目标
|
||
|
||
GET 请求:
|
||
获取当前用户总目标
|
||
|
||
POST 请求参数:
|
||
email: 达人邮箱
|
||
goal: 目标内容
|
||
"""
|
||
try:
|
||
# 检查用户权限 - 只允许组长使用
|
||
if request.user.role != 'leader':
|
||
return Response(
|
||
{"error": "只有组长角色的用户可以使用此功能"},
|
||
status=status.HTTP_403_FORBIDDEN
|
||
)
|
||
|
||
if request.method == 'GET':
|
||
# 创建Gmail集成实例
|
||
gmail_integration = GmailIntegration(request.user)
|
||
# 获取总目标
|
||
result = gmail_integration.manage_user_goal()
|
||
return Response(result, status=status.HTTP_200_OK)
|
||
|
||
elif request.method == 'POST':
|
||
# 获取参数
|
||
email = request.data.get("email")
|
||
goal = request.data.get("goal")
|
||
|
||
# 验证必要参数
|
||
if not email:
|
||
return Response(
|
||
{"error": "缺少参数email"},
|
||
status=status.HTTP_400_BAD_REQUEST
|
||
)
|
||
if not goal:
|
||
return Response(
|
||
{"error": "缺少参数goal"},
|
||
status=status.HTTP_400_BAD_REQUEST
|
||
)
|
||
|
||
# 设置用户总目标
|
||
gmail_integration = GmailIntegration(request.user)
|
||
result = gmail_integration.manage_user_goal(goal)
|
||
|
||
return Response(result, status=status.HTTP_200_OK)
|
||
|
||
except Exception as e:
|
||
logger.error(f"管理用户总目标时出错: {str(e)}")
|
||
logger.error(traceback.format_exc())
|
||
return Response(
|
||
{"error": str(e)},
|
||
status=status.HTTP_500_INTERNAL_SERVER_ERROR
|
||
)
|
||
|
||
@api_view(['GET'])
|
||
@permission_classes([IsAuthenticated])
|
||
def check_goal_status(request):
|
||
"""
|
||
检查目标完成状态
|
||
|
||
请求参数:
|
||
email: 达人邮箱
|
||
"""
|
||
try:
|
||
# 检查用户权限 - 只允许组长使用
|
||
if request.user.role != 'leader':
|
||
return Response(
|
||
{"error": "只有组长角色的用户可以使用此功能"},
|
||
status=status.HTTP_403_FORBIDDEN
|
||
)
|
||
|
||
# 获取参数
|
||
email = request.query_params.get("email")
|
||
|
||
# 验证必要参数
|
||
if not email:
|
||
return Response(
|
||
{"error": "缺少参数email"},
|
||
status=status.HTTP_400_BAD_REQUEST
|
||
)
|
||
|
||
# 查找Gmail映射关系
|
||
mapping = GmailTalentMapping.objects.filter(
|
||
user=request.user,
|
||
talent_email=email,
|
||
is_active=True
|
||
).first()
|
||
|
||
if not mapping:
|
||
return Response(
|
||
{"error": f"找不到与邮箱 {email} 的映射关系"},
|
||
status=status.HTTP_404_NOT_FOUND
|
||
)
|
||
|
||
# 获取对话历史中最后的AI回复
|
||
last_ai_message = ChatHistory.objects.filter(
|
||
user=request.user,
|
||
knowledge_base=mapping.knowledge_base,
|
||
conversation_id=mapping.conversation_id,
|
||
role='assistant',
|
||
is_deleted=False
|
||
).order_by('-created_at').first()
|
||
|
||
if not last_ai_message:
|
||
return Response(
|
||
{"error": f"找不到与邮箱 {email} 的对话历史"},
|
||
status=status.HTTP_404_NOT_FOUND
|
||
)
|
||
|
||
# 检查目标是否已达成
|
||
goal_achieved = check_goal_achieved(last_ai_message.content)
|
||
|
||
# 获取对话总结
|
||
summary = ConversationSummary.objects.filter(
|
||
user=request.user,
|
||
talent_email=email,
|
||
is_active=True
|
||
).order_by('-updated_at').first()
|
||
|
||
result = {
|
||
'status': 'success',
|
||
'email': email,
|
||
'goal_achieved': goal_achieved,
|
||
'last_message_time': last_ai_message.created_at.strftime('%Y-%m-%d %H:%M:%S'),
|
||
'last_message': last_ai_message.content,
|
||
'summary': summary.summary if summary else None
|
||
}
|
||
|
||
return Response(result, status=status.HTTP_200_OK)
|
||
|
||
except Exception as e:
|
||
logger.error(f"检查目标状态时出错: {str(e)}")
|
||
logger.error(traceback.format_exc())
|
||
return Response(
|
||
{"error": str(e)},
|
||
status=status.HTTP_500_INTERNAL_SERVER_ERROR
|
||
) |