123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360 |
- """
- -----------------File Info-----------------------
- Name: web.py
- Description: web api support
- Author: GentleCP
- Email: me@gentlecp.com
- Create Date: 2021/6/19
- -----------------End-----------------------------
- """
- import json
- import re
- import sys
- import threading
- import time
- import uuid
- from xml.etree.ElementTree import fromstring
- import requests
- import uvicorn
- from fastapi import FastAPI, Response, Request, BackgroundTasks, Body
- from WXBizMsgCrypt3 import WXBizMsgCrypt
- from commom import get_logger, request_id_context
- logger = get_logger()
- with open('cqrcb_config.json', 'r') as f:
- config = json.load(f)
- token = config['token']
- aeskey = config['aeskey']
- corpid = config['corpid']
- corpsecret = config['corpsecret']
- coze_access_token = config['coze_access_token']
- wxcpt = WXBizMsgCrypt(token, aeskey, corpid)
- app = FastAPI()
- def call_llm(prompt: str, bot_id: str,coze_access_token:str):
- req_head = {
- "Authorization":f"Bearer {coze_access_token}",
- "Content-Type": "application/json",
- }
- req_data ={
- "bot_id": bot_id,
- "user_id": "123456789",
- "stream": False,
- "auto_save_history": True,
- "additional_messages": [
- {
- "role": "user",
- "content": prompt,
- "content_type": "object_string"
- }
- ]
- }
- res_create = requests.post(" https://api.coze.cn/v1/conversation/create", headers=req_head)
- conversation_id = res_create.json()["data"]["id"]
- res_chat = requests.post(f" https://api.coze.cn/v3/chat?conversation_id={conversation_id}", headers=req_head,json=req_data)
- chat_id = res_chat.json()["data"]["id"]
- while True:
- res_retrieve = requests.get(f" https://api.coze.cn/v3/chat/retrieve?chat_id={chat_id}&conversation_id={conversation_id}", headers=req_head)
- res_json = res_retrieve.json()
-
- if res_retrieve.status_code != 200:
- logger.error(f"网络状态码失败,错误码:{res_retrieve.status_code }")
- coze_response = f"网络状态码失败,错误码:{res_retrieve.status_code }"
- return coze_response
-
- if res_json["code"] != 0 :
- logger.error(f"API调用失败,错误码:{res_json['code']}")
- coze_response = f"API调用失败,错误码:{res_json['code']}"
- return coze_response
-
- logger.info(res_json["data"]["status"])
- status = res_json["data"]["status"]
-
- error_statuses = {"failed", "requires_action", "canceled"}
- if status in error_statuses:
- error_message = res_json["data"]["last_error"]
- logger.error(f"对话错误,状态:{status},错误信息:{error_message}")
- coze_response = f"对话错误,状态:{status},错误信息:{error_message}"
- return coze_response
-
- if status == "completed":
- res_message = requests.get(
- f"https://api.coze.cn/v3/chat/message/list?chat_id={chat_id}&conversation_id={conversation_id}",
- headers=req_head)
-
- data = res_message.json()['data']
-
- last_answer_record = next((record for record in reversed(data) if record['type'] == 'answer'), None)
- coze_response = last_answer_record['content'].replace(" ", "")
-
- return coze_response
- time.sleep(1)
- def qiwei_get():
- res = requests.get(f"https://qyapi.weixin.qq.com/cgi-bin/gettoken?corpid={corpid}&corpsecret={corpsecret}")
- qw_access_token = res.json()["access_token"]
- return qw_access_token
- def qiwei_post(username: str, answer: str,agentid:str):
- req_data = {
- "touser": username,
- "toparty": "",
- "totag": "",
- "msgtype": "text",
- "agentid": agentid,
- "text": {"content": answer},
- "image": {
- "media_id": "MEDIA_ID"
- },
- "safe": 0,
- "enable_id_trans": 0,
- "enable_duplicate_check": 0,
- "duplicate_check_interval": 1800
- }
- res = requests.post(f"https://qyapi.weixin.qq.com/cgi-bin/message/send?access_token={qiwei_get()}", json=req_data)
-
- logger.info(res.json())
-
- def post_consumer_api(user_query, decrypt_data, request_id):
- data = {
- "user_query": user_query,
- "decrypt_data": decrypt_data,
- "request_id": request_id,
- }
- request_id_context.set(request_id)
- url = "https://101.126.81.2:18088/consumer"
- try:
- logger.info(f"post_consumer_api 被执行{user_query}")
- t1 = threading.Thread(target=requests.post, kwargs={"url": url, "json": data, "verify": False})
- t1.start()
-
-
-
- except requests.exceptions.RequestException as e:
- logger.error(f"post_consumer_api 请求失败: {e}")
- user_bot_id_mapping = {}
- user_welcome_status = {}
- @app.post("/consumer")
- def consumer(
- request_id: str = Body(...),
- user_query: str = Body(...),
- decrypt_data: dict = Body(...),
- ):
-
-
-
-
-
-
-
-
-
-
- request_id_context.set(request_id)
-
-
-
-
-
- username = decrypt_data.get('FromUserName', '')
- agentid = decrypt_data.get('AgentID', '')
- msgtype = decrypt_data.get('MsgType', '')
- picurl = decrypt_data.get('PicUrl', '')
- event = decrypt_data.get('Event')
- msg_type = decrypt_data.get('MsgType')
- event_key = decrypt_data.get('EventKey')
- print(f"event_key: {event_key}")
-
- if msg_type == 'event' and event == 'enter_agent':
- if not user_welcome_status.get(username, False):
- welcome_message = "Hi,我是小微AI助手~你可以在屏幕底部“产品选择”菜单栏选择想咨询的产品,我会随时为你解答问题~"
- qiwei_post(username, welcome_message, agentid)
- user_welcome_status[username] = True
- return Response(content="")
-
- if msg_type == 'event' and event == 'click' and event_key == '#sendmsg#_0_0#7599827339205934':
- bot_id = "7456977536891846697"
-
-
- user_bot_id_mapping[username] = bot_id
- return Response(content="")
- elif msg_type == 'event' and event == 'click' and event_key == '#sendmsg#_0_1#7599827339205935':
- bot_id = "7445101065005154313"
-
-
- user_bot_id_mapping[username] = bot_id
- return Response(content="")
- else:
-
- bot_id = user_bot_id_mapping.get(username, "7456977536891846697")
- qiwei_post(username, "我正在思考,请稍等...", agentid)
- logger.info("我正在思考,请稍等...")
- logger.info(f"consumer 请求:{user_query}")
- user_query = user_query if user_query else "回答图片中的问题"
- multimodal_content = [
- {"type": "text", "text": user_query},
- {"type": msgtype, "file_url": picurl},
-
-
- ]
- user_query = json.dumps(multimodal_content, ensure_ascii=False)
-
- coze_response = call_llm(prompt=user_query, bot_id=bot_id, coze_access_token=coze_access_token)
-
-
-
- image_counter = 1
-
- def replace_with_counter(match):
- nonlocal image_counter
- alt_text = match.group(1) or f"示例图片{image_counter}"
- url = match.group(2)
- replacement = f'<a href="{url}">{alt_text}</a>'
- image_counter += 1
- return replacement
-
- answer = re.sub(r'!\[(.*?)\]\((https?://[^)]+)\)', replace_with_counter, coze_response)
-
- logger.info(f"结果:{answer}")
-
- qiwei_post(username, answer, agentid)
-
-
- @app.get("/ok")
- async def ok():
- return "ok"
- @app.get("/bot")
- async def verify(msg_signature: str, timestamp: str, nonce: str, echostr: str):
- ret, sEchoStr = wxcpt.VerifyURL(msg_signature, timestamp, nonce, echostr)
- if ret == 0:
- return Response(content=sEchoStr.decode('utf-8'))
- else:
-
- logger.info(sEchoStr)
- @app.post("/bot")
- async def recv(msg_signature: str, timestamp: str, nonce: str, request: Request, background_tasks: BackgroundTasks):
-
- body = await request.body()
- request_id = str(uuid.uuid4())
- request_id_context.set(request_id)
- ret, sMsg = wxcpt.DecryptMsg(body.decode('utf-8'), msg_signature, timestamp, nonce)
- decrypt_data = {}
- for node in list(fromstring(sMsg.decode('utf-8'))):
- decrypt_data[node.tag] = node.text
-
- print(decrypt_data)
-
- user_query = decrypt_data.get('Content', '')
-
-
-
- logger.info(f"start: {user_query}")
-
- post_consumer_api(user_query, decrypt_data, request_id)
- return Response(content="")
-
-
-
-
-
-
-
-
-
-
-
- if __name__ == "__main__":
-
-
- try:
- port = sys.argv[1]
- int(port)
- except:
- port = 18066
- request_id_context.set("app start")
- logger.info(f'{port=}')
- print(f'{port=}')
- uvicorn.run("coze_bot_api:app", port=int(port), host='0.0.0.0', reload=False,ssl_keyfile="./key.pem", ssl_certfile="./cert.pem")
|