You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

145 lines
7.1KB

  1. import web
  2. import time
  3. import channel.wechatmp.reply as reply
  4. import channel.wechatmp.receive as receive
  5. from config import conf
  6. from common.log import logger
  7. from bridge.context import *
  8. from channel.wechatmp.common import *
  9. from channel.wechatmp.wechatmp_channel import WechatMPChannel
  10. # This class is instantiated once per query
  11. class Query():
  12. def GET(self):
  13. return verify_server(web.input())
  14. def POST(self):
  15. # Make sure to return the instance that first created, @singleton will do that.
  16. channel_instance = WechatMPChannel()
  17. try:
  18. query_time = time.time()
  19. webData = web.data()
  20. # logger.debug("[wechatmp] Receive request:\n" + webData.decode("utf-8"))
  21. wechatmp_msg = receive.parse_xml(webData)
  22. if wechatmp_msg.msg_type == 'text':
  23. from_user = wechatmp_msg.from_user_id
  24. to_user = wechatmp_msg.to_user_id
  25. message = wechatmp_msg.content.decode("utf-8")
  26. message_id = wechatmp_msg.msg_id
  27. logger.info("[wechatmp] {}:{} Receive post query {} {}: {}".format(web.ctx.env.get('REMOTE_ADDR'), web.ctx.env.get('REMOTE_PORT'), from_user, message_id, message))
  28. cache_key = from_user
  29. cache = channel_instance.cache_dict.get(cache_key)
  30. reply_text = ""
  31. # New request
  32. if cache == None:
  33. # The first query begin, reset the cache
  34. channel_instance.cache_dict[cache_key] = (0, "")
  35. context = channel_instance._compose_context(ContextType.TEXT, message, isgroup=False, msg=wechatmp_msg)
  36. if context:
  37. # set private openai_api_key
  38. # if from_user is not changed in itchat, this can be placed at chat_channel
  39. user_data = conf().get_user_data(from_user)
  40. context['openai_api_key'] = user_data.get('openai_api_key') # None or user openai_api_key
  41. channel_instance.produce(context)
  42. channel_instance.query1[cache_key] = False
  43. channel_instance.query2[cache_key] = False
  44. channel_instance.query3[cache_key] = False
  45. # Request again
  46. 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:
  47. channel_instance.query1[cache_key] = False #To improve waiting experience, this can be set to True.
  48. channel_instance.query2[cache_key] = False #To improve waiting experience, this can be set to True.
  49. channel_instance.query3[cache_key] = False
  50. elif cache[0] >= 1:
  51. # Skip the waiting phase
  52. channel_instance.query1[cache_key] = True
  53. channel_instance.query2[cache_key] = True
  54. channel_instance.query3[cache_key] = True
  55. cache = channel_instance.cache_dict.get(cache_key)
  56. if channel_instance.query1.get(cache_key) == False:
  57. # The first query from wechat official server
  58. logger.debug("[wechatmp] query1 {}".format(cache_key))
  59. channel_instance.query1[cache_key] = True
  60. cnt = 0
  61. while cache[0] == 0 and cnt < 45:
  62. cnt = cnt + 1
  63. time.sleep(0.1)
  64. cache = channel_instance.cache_dict.get(cache_key)
  65. if cnt == 45:
  66. # waiting for timeout (the POST query will be closed by wechat official server)
  67. time.sleep(5)
  68. # and do nothing
  69. return
  70. else:
  71. pass
  72. elif channel_instance.query2.get(cache_key) == False:
  73. # The second query from wechat official server
  74. logger.debug("[wechatmp] query2 {}".format(cache_key))
  75. channel_instance.query2[cache_key] = True
  76. cnt = 0
  77. while cache[0] == 0 and cnt < 45:
  78. cnt = cnt + 1
  79. time.sleep(0.1)
  80. cache = channel_instance.cache_dict.get(cache_key)
  81. if cnt == 45:
  82. # waiting for timeout (the POST query will be closed by wechat official server)
  83. time.sleep(5)
  84. # and do nothing
  85. return
  86. else:
  87. pass
  88. elif channel_instance.query3.get(cache_key) == False:
  89. # The third query from wechat official server
  90. logger.debug("[wechatmp] query3 {}".format(cache_key))
  91. channel_instance.query3[cache_key] = True
  92. cnt = 0
  93. while cache[0] == 0 and cnt < 45:
  94. cnt = cnt + 1
  95. time.sleep(0.1)
  96. cache = channel_instance.cache_dict.get(cache_key)
  97. if cnt == 45:
  98. # Have waiting for 3x5 seconds
  99. # return timeout message
  100. reply_text = "【正在响应中,回复任意文字尝试获取回复】"
  101. logger.info("[wechatmp] Three queries has finished For {}: {}".format(from_user, message_id))
  102. replyPost = reply.TextMsg(from_user, to_user, reply_text).send()
  103. return replyPost
  104. else:
  105. pass
  106. if float(time.time()) - float(query_time) > 4.8:
  107. logger.info("[wechatmp] Timeout for {} {}".format(from_user, message_id))
  108. return
  109. if cache[0] > 1:
  110. reply_text = cache[1][:600] + "\n【未完待续,回复任意文字以继续】" #wechatmp auto_reply length limit
  111. channel_instance.cache_dict[cache_key] = (cache[0] - 1, cache[1][600:])
  112. elif cache[0] == 1:
  113. reply_text = cache[1]
  114. channel_instance.cache_dict.pop(cache_key)
  115. logger.info("[wechatmp] {}:{} Do send {}".format(web.ctx.env.get('REMOTE_ADDR'), web.ctx.env.get('REMOTE_PORT'), reply_text))
  116. replyPost = reply.TextMsg(from_user, to_user, reply_text).send()
  117. return replyPost
  118. elif wechatmp_msg.msg_type == 'event':
  119. logger.info("[wechatmp] Event {} from {}".format(wechatmp_msg.Event, wechatmp_msg.from_user_id))
  120. content = subscribe_msg()
  121. replyMsg = reply.TextMsg(wechatmp_msg.from_user_id, wechatmp_msg.to_user_id, content)
  122. return replyMsg.send()
  123. else:
  124. logger.info("暂且不处理")
  125. return "success"
  126. except Exception as exc:
  127. logger.exception(exc)
  128. return exc