您最多选择25个主题 主题必须以字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符

233 行
10KB

  1. # -*- coding: utf-8 -*-
  2. import web
  3. import time
  4. import math
  5. import hashlib
  6. import textwrap
  7. from channel.chat_channel import ChatChannel
  8. import channel.wechatmp.reply as reply
  9. import channel.wechatmp.receive as receive
  10. from common.singleton import singleton
  11. from common.log import logger
  12. from config import conf
  13. from bridge.reply import *
  14. from bridge.context import *
  15. from plugins import *
  16. import traceback
  17. # If using SSL, uncomment the following lines, and modify the certificate path.
  18. # from cheroot.server import HTTPServer
  19. # from cheroot.ssl.builtin import BuiltinSSLAdapter
  20. # HTTPServer.ssl_adapter = BuiltinSSLAdapter(
  21. # certificate='/ssl/cert.pem',
  22. # private_key='/ssl/cert.key')
  23. # from concurrent.futures import ThreadPoolExecutor
  24. # thread_pool = ThreadPoolExecutor(max_workers=8)
  25. @singleton
  26. class WechatMPChannel(ChatChannel):
  27. def __init__(self):
  28. super().__init__()
  29. self.cache_dict = dict()
  30. self.query1 = dict()
  31. self.query2 = dict()
  32. self.query3 = dict()
  33. def startup(self):
  34. urls = (
  35. '/wx', 'SubsribeAccountQuery',
  36. )
  37. app = web.application(urls, globals())
  38. app.run()
  39. def send(self, reply: Reply, context: Context):
  40. reply_cnt = math.ceil(len(reply.content) / 600)
  41. receiver = context["receiver"]
  42. self.cache_dict[receiver] = (reply_cnt, reply.content)
  43. logger.debug("[send] reply to {} saved to cache: {}".format(receiver, reply))
  44. def verify_server():
  45. try:
  46. data = web.input()
  47. if len(data) == 0:
  48. return "None"
  49. signature = data.signature
  50. timestamp = data.timestamp
  51. nonce = data.nonce
  52. echostr = data.echostr
  53. token = conf().get('wechatmp_token') #请按照公众平台官网\基本配置中信息填写
  54. data_list = [token, timestamp, nonce]
  55. data_list.sort()
  56. sha1 = hashlib.sha1()
  57. # map(sha1.update, data_list) #python2
  58. sha1.update("".join(data_list).encode('utf-8'))
  59. hashcode = sha1.hexdigest()
  60. print("handle/GET func: hashcode, signature: ", hashcode, signature)
  61. if hashcode == signature:
  62. return echostr
  63. else:
  64. return ""
  65. except Exception as Argument:
  66. return Argument
  67. # This class is instantiated once per query
  68. class SubsribeAccountQuery():
  69. def GET(self):
  70. return verify_server()
  71. def POST(self):
  72. channel_instance = WechatMPChannel()
  73. try:
  74. query_time = time.time()
  75. webData = web.data()
  76. # logger.debug("[wechatmp] Receive request:\n" + webData.decode("utf-8"))
  77. wechat_msg = receive.parse_xml(webData)
  78. if wechat_msg.msg_type == 'text':
  79. from_user = wechat_msg.from_user_id
  80. to_user = wechat_msg.to_user_id
  81. message = wechat_msg.content.decode("utf-8")
  82. message_id = wechat_msg.msg_id
  83. logger.info("[wechatmp] {}:{} Receive post query {} {}: {}".format(web.ctx.env.get('REMOTE_ADDR'), web.ctx.env.get('REMOTE_PORT'), from_user, message_id, message))
  84. cache_key = from_user
  85. cache = channel_instance.cache_dict.get(cache_key)
  86. reply_text = ""
  87. # New request
  88. if cache == None:
  89. # The first query begin, reset the cache
  90. context = channel_instance._compose_context(ContextType.TEXT, message, isgroup=False, msg=wechat_msg)
  91. logger.debug("[wechatmp] context: {} {}".format(context, wechat_msg))
  92. if context:
  93. # set private openai_api_key
  94. # if from_user is not changed in itchat, this can be placed at chat_channel
  95. user_data = conf().get_user_data(from_user)
  96. context['openai_api_key'] = user_data.get('openai_api_key') # None or user openai_api_key
  97. channel_instance.cache_dict[cache_key] = (0, "")
  98. channel_instance.produce(context)
  99. else:
  100. trigger_prefix = conf().get('single_chat_prefix',[''])[0]
  101. if trigger_prefix:
  102. content = textwrap.dedent(f"""\
  103. 请输入'{trigger_prefix}'接你想说的话跟我说话。
  104. 例如:
  105. {trigger_prefix}你好,很高兴见到你。""")
  106. else:
  107. logger.error(f"[wechatmp] unknown error")
  108. content = textwrap.dedent("""\
  109. 未知错误,请稍后再试""")
  110. replyMsg = reply.TextMsg(wechat_msg.from_user_id, wechat_msg.to_user_id, content)
  111. return replyMsg.send()
  112. channel_instance.query1[cache_key] = False
  113. channel_instance.query2[cache_key] = False
  114. channel_instance.query3[cache_key] = False
  115. # Request again
  116. elif cache[0] == 0 and channel_instance.query1.get(cache_key) == True and channel_instance.query2.get(cache_key) == True and channel_instance.query3.get(cache_key) == True:
  117. channel_instance.query1[cache_key] = False #To improve waiting experience, this can be set to True.
  118. channel_instance.query2[cache_key] = False #To improve waiting experience, this can be set to True.
  119. channel_instance.query3[cache_key] = False
  120. elif cache[0] >= 1:
  121. # Skip the waiting phase
  122. channel_instance.query1[cache_key] = True
  123. channel_instance.query2[cache_key] = True
  124. channel_instance.query3[cache_key] = True
  125. cache = channel_instance.cache_dict.get(cache_key)
  126. if channel_instance.query1.get(cache_key) == False:
  127. # The first query from wechat official server
  128. logger.debug("[wechatmp] query1 {}".format(cache_key))
  129. channel_instance.query1[cache_key] = True
  130. cnt = 0
  131. while cache[0] == 0 and cnt < 45:
  132. cnt = cnt + 1
  133. time.sleep(0.1)
  134. cache = channel_instance.cache_dict.get(cache_key)
  135. if cnt == 45:
  136. # waiting for timeout (the POST query will be closed by wechat official server)
  137. time.sleep(5)
  138. # and do nothing
  139. return
  140. else:
  141. pass
  142. elif channel_instance.query2.get(cache_key) == False:
  143. # The second query from wechat official server
  144. logger.debug("[wechatmp] query2 {}".format(cache_key))
  145. channel_instance.query2[cache_key] = True
  146. cnt = 0
  147. while cache[0] == 0 and cnt < 45:
  148. cnt = cnt + 1
  149. time.sleep(0.1)
  150. cache = channel_instance.cache_dict.get(cache_key)
  151. if cnt == 45:
  152. # waiting for timeout (the POST query will be closed by wechat official server)
  153. time.sleep(5)
  154. # and do nothing
  155. return
  156. else:
  157. pass
  158. elif channel_instance.query3.get(cache_key) == False:
  159. # The third query from wechat official server
  160. logger.debug("[wechatmp] query3 {}".format(cache_key))
  161. channel_instance.query3[cache_key] = True
  162. cnt = 0
  163. while cache[0] == 0 and cnt < 45:
  164. cnt = cnt + 1
  165. time.sleep(0.1)
  166. cache = channel_instance.cache_dict.get(cache_key)
  167. if cnt == 45:
  168. # Have waiting for 3x5 seconds
  169. # return timeout message
  170. reply_text = "【正在响应中,回复任意文字尝试获取回复】"
  171. logger.info("[wechatmp] Three queries has finished For {}: {}".format(from_user, message_id))
  172. replyPost = reply.TextMsg(from_user, to_user, reply_text).send()
  173. return replyPost
  174. else:
  175. pass
  176. if float(time.time()) - float(query_time) > 4.8:
  177. logger.info("[wechatmp] Timeout for {} {}".format(from_user, message_id))
  178. return
  179. if cache[0] > 1:
  180. reply_text = cache[1][:600] + "\n【未完待续,回复任意文字以继续】" #wechatmp auto_reply length limit
  181. channel_instance.cache_dict[cache_key] = (cache[0] - 1, cache[1][600:])
  182. elif cache[0] == 1:
  183. reply_text = cache[1]
  184. channel_instance.cache_dict.pop(cache_key)
  185. logger.info("[wechatmp] {}:{} Do send {}".format(web.ctx.env.get('REMOTE_ADDR'), web.ctx.env.get('REMOTE_PORT'), reply_text))
  186. replyPost = reply.TextMsg(from_user, to_user, reply_text).send()
  187. return replyPost
  188. elif wechat_msg.msg_type == 'event':
  189. logger.info("[wechatmp] Event {} from {}".format(wechat_msg.Event, wechat_msg.from_user_id))
  190. trigger_prefix = conf().get('single_chat_prefix',[''])[0]
  191. content = textwrap.dedent(f"""\
  192. 感谢您的关注!
  193. 这里是ChatGPT,可以自由对话。
  194. 资源有限,回复较慢,请勿着急。
  195. 支持通用表情输入。
  196. 暂时不支持图片输入。
  197. 支持图片输出,画字开头的问题将回复图片链接。
  198. 支持角色扮演和文字冒险两种定制模式对话。
  199. 输入'{trigger_prefix}#帮助' 查看详细指令。""")
  200. replyMsg = reply.TextMsg(wechat_msg.from_user_id, wechat_msg.to_user_id, content)
  201. return replyMsg.send()
  202. else:
  203. logger.info("暂且不处理")
  204. return "success"
  205. except Exception as exc:
  206. logger.exception(exc)
  207. return exc