Você não pode selecionar mais de 25 tópicos Os tópicos devem começar com uma letra ou um número, podem incluir traços ('-') e podem ter até 35 caracteres.

201 linhas
8.1KB

  1. # -*- coding: utf-8 -*-
  2. # filename: main.py
  3. import web
  4. import time
  5. import math
  6. import hashlib
  7. import textwrap
  8. from channel.channel import Channel
  9. import channel.wechatmp.reply as reply
  10. import channel.wechatmp.receive as receive
  11. from common.log import logger
  12. from config import conf
  13. class WechatMPServer():
  14. def __init__(self):
  15. pass
  16. def startup(self):
  17. urls = (
  18. '/wx', 'WechatMPChannel',
  19. )
  20. app = web.application(urls, globals())
  21. app.run()
  22. from concurrent.futures import ThreadPoolExecutor
  23. thread_pool = ThreadPoolExecutor(max_workers=8)
  24. cache_dict = dict()
  25. query1 = dict()
  26. query2 = dict()
  27. query3 = dict()
  28. class WechatMPChannel(Channel):
  29. def GET(self):
  30. try:
  31. data = web.input()
  32. if len(data) == 0:
  33. return "hello, this is handle view"
  34. signature = data.signature
  35. timestamp = data.timestamp
  36. nonce = data.nonce
  37. echostr = data.echostr
  38. token = conf().get('wechatmp_token') #请按照公众平台官网\基本配置中信息填写
  39. data_list = [token, timestamp, nonce]
  40. data_list.sort()
  41. sha1 = hashlib.sha1()
  42. # map(sha1.update, data_list) #python2
  43. sha1.update("".join(data_list).encode('utf-8'))
  44. hashcode = sha1.hexdigest()
  45. print("handle/GET func: hashcode, signature: ", hashcode, signature)
  46. if hashcode == signature:
  47. return echostr
  48. else:
  49. return ""
  50. except Exception as Argument:
  51. return Argument
  52. def _do_build_reply(self, cache_key, fromUser, message):
  53. context = dict()
  54. context['session_id'] = fromUser
  55. reply_text = super().build_reply_content(message, context)
  56. # The query is done, record the cache
  57. logger.info("[threaded] Get reply for {}: {} \nA: {}".format(fromUser, message, reply_text))
  58. reply_cnt = math.ceil(len(reply_text) / 600)
  59. global cache_dict
  60. cache_dict[cache_key] = (reply_cnt, reply_text)
  61. def POST(self):
  62. try:
  63. queryTime = time.time()
  64. webData = web.data()
  65. # logger.debug("[wechatmp] Receive request:\n" + webData.decode("utf-8"))
  66. recMsg = receive.parse_xml(webData)
  67. if isinstance(recMsg, receive.Msg) and recMsg.MsgType == 'text':
  68. fromUser = recMsg.FromUserName
  69. toUser = recMsg.ToUserName
  70. createTime = recMsg.CreateTime
  71. message = recMsg.Content.decode("utf-8")
  72. message_id = recMsg.MsgId
  73. logger.info("[wechatmp] {}:{} Receive post query {} {}: {}".format(web.ctx.env.get('REMOTE_ADDR'), web.ctx.env.get('REMOTE_PORT'), fromUser, message_id, message))
  74. global cache_dict
  75. global query1
  76. global query2
  77. global query3
  78. cache_key = fromUser
  79. cache = cache_dict.get(cache_key)
  80. reply_text = ""
  81. # New request
  82. if cache == None:
  83. # The first query begin, reset the cache
  84. cache_dict[cache_key] = (0, "")
  85. thread_pool.submit(self._do_build_reply, cache_key, fromUser, message)
  86. query1[cache_key] = False
  87. query2[cache_key] = False
  88. query3[cache_key] = False
  89. # Request again
  90. elif cache[0] == 0 and query1.get(cache_key) == True and query2.get(cache_key) == True and query3.get(cache_key) == True:
  91. query1[cache_key] = False #To improve waiting experience, this can be set to True.
  92. query2[cache_key] = False #To improve waiting experience, this can be set to True.
  93. query3[cache_key] = False
  94. elif cache[0] >= 1:
  95. # Skip the waiting phase
  96. query1[cache_key] = True
  97. query2[cache_key] = True
  98. query3[cache_key] = True
  99. cache = cache_dict.get(cache_key)
  100. if query1.get(cache_key) == False:
  101. # The first query from wechat official server
  102. logger.debug("[wechatmp] query1 {}".format(cache_key))
  103. query1[cache_key] = True
  104. cnt = 0
  105. while cache[0] == 0 and cnt < 45:
  106. cnt = cnt + 1
  107. time.sleep(0.1)
  108. cache = cache_dict.get(cache_key)
  109. if cnt == 45:
  110. # waiting for timeout (the POST query will be closed by wechat official server)
  111. time.sleep(5)
  112. # and do nothing
  113. return
  114. else:
  115. pass
  116. elif query2.get(cache_key) == False:
  117. # The second query from wechat official server
  118. logger.debug("[wechatmp] query2 {}".format(cache_key))
  119. query2[cache_key] = True
  120. cnt = 0
  121. while cache[0] == 0 and cnt < 45:
  122. cnt = cnt + 1
  123. time.sleep(0.1)
  124. cache = cache_dict.get(cache_key)
  125. if cnt == 45:
  126. # waiting for timeout (the POST query will be closed by wechat official server)
  127. time.sleep(5)
  128. # and do nothing
  129. return
  130. else:
  131. pass
  132. elif query3.get(cache_key) == False:
  133. # The third query from wechat official server
  134. logger.debug("[wechatmp] query3 {}".format(cache_key))
  135. query3[cache_key] = True
  136. cnt = 0
  137. while cache[0] == 0 and cnt < 45:
  138. cnt = cnt + 1
  139. time.sleep(0.1)
  140. cache = cache_dict.get(cache_key)
  141. if cnt == 45:
  142. # Have waiting for 3x5 seconds
  143. # return timeout message
  144. reply_text = "【服务器有点忙,回复任意文字再次尝试】"
  145. logger.info("[wechatmp] Three queries has finished For {}: {}".format(fromUser, message_id))
  146. replyPost = reply.TextMsg(fromUser, toUser, reply_text).send()
  147. return replyPost
  148. else:
  149. pass
  150. if float(time.time()) - float(queryTime) > 4.8:
  151. logger.info("[wechatmp] Timeout for {} {}".format(fromUser, message_id))
  152. return
  153. if cache[0] > 1:
  154. reply_text = cache[1][:600] + " 【未完待续,回复任意文字以继续】" #wechatmp auto_reply length limit
  155. cache_dict[cache_key] = (cache[0] - 1, cache[1][600:])
  156. elif cache[0] == 1:
  157. reply_text = cache[1]
  158. cache_dict.pop(cache_key)
  159. logger.info("[wechatmp] {}:{} Do send {}".format(web.ctx.env.get('REMOTE_ADDR'), web.ctx.env.get('REMOTE_PORT'), reply_text))
  160. replyPost = reply.TextMsg(fromUser, toUser, reply_text).send()
  161. return replyPost
  162. elif isinstance(recMsg, receive.Event) and recMsg.MsgType == 'event':
  163. toUser = recMsg.FromUserName
  164. fromUser = recMsg.ToUserName
  165. content = textwrap.dedent("""\
  166. 感谢您的关注!
  167. 这里是ChatGPT,可以自由对话。
  168. 资源有限,回复较慢,请不要着急。
  169. 暂时不支持图片输入输出,但是支持通用表情输入。""")
  170. replyMsg = reply.TextMsg(toUser, fromUser, content)
  171. return replyMsg.send()
  172. else:
  173. print("暂且不处理")
  174. return "success"
  175. except Exception as Argment:
  176. print(Argment)
  177. return Argment