groupme_driver.py 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170
  1. from __future__ import annotations
  2. from datetime import datetime
  3. import json
  4. import traceback
  5. import asyncio
  6. import logging
  7. import os
  8. import toml
  9. from fastapi import FastAPI, BackgroundTasks, Response
  10. from pydantic import BaseModel
  11. import rollbot
  12. from commands import config
  13. logging.config.fileConfig("logging.conf", disable_existing_loggers=False)
  14. with open(os.environ.get("SECRET_FILE", "secrets.toml"), "r") as sfile:
  15. secrets = toml.load(sfile)
  16. database_file = os.environ.get("DATABASE_FILE", secrets["database_file"])
  17. groupme_bots = secrets["groupme"]["bots"]
  18. groupme_token = secrets["groupme"]["token"]
  19. groupme_admins = secrets["admins"]["origin"]
  20. group_admins = secrets["admins"]["channel"]
  21. max_msg_len = 1000
  22. split_text = "\n..."
  23. msg_slice = max_msg_len - len(split_text)
  24. class GroupMeMessage(BaseModel):
  25. sender_id: str
  26. group_id: str
  27. name: str
  28. text: str
  29. created_at: int
  30. attachments: list[dict[str, str]]
  31. class GroupMeBot(rollbot.Rollbot[GroupMeMessage]):
  32. def __init__(self):
  33. super().__init__(config.extend(rollbot.CommandConfiguration(bangs=("!",))), database_file)
  34. def read_config(self, key: str) -> Any:
  35. cfg = secrets
  36. for part in key.split("."):
  37. cfg = cfg.get(part, None)
  38. if cfg is None:
  39. return None
  40. return cfg
  41. async def parse(self, msg: GroupMeMessage):
  42. return rollbot.Message(
  43. origin_id="GROUPME",
  44. channel_id=msg.group_id,
  45. sender_id=msg.sender_id,
  46. timestamp=datetime.fromtimestamp(msg.created_at),
  47. origin_admin=msg.sender_id in groupme_admins,
  48. channel_admin=msg.sender_id in group_admins.get(msg.group_id, ()),
  49. sender_name=msg.name,
  50. text=msg.text,
  51. attachments=[rollbot.Attachment(d["type"], json.dumps(d)) for d in msg.attachments],
  52. )
  53. async def upload_image(self, data: bytes):
  54. async with self.context.request.post(
  55. "https://image.groupme.com/pictures",
  56. headers={
  57. "Content-Type": "image/png",
  58. "X-Access-Token": groupme_token,
  59. },
  60. data=data,
  61. ) as upload:
  62. upload.raise_for_status()
  63. return (await upload.json())["payload"]["url"]
  64. async def post_message(self, bot_id: str, text: str, attachments: list[dict[str, str]]):
  65. result = await self.context.request.post(
  66. "https://api.groupme.com/v3/bots/post",
  67. json={
  68. "bot_id": bot_id,
  69. "text": text,
  70. "attachments": attachments,
  71. },
  72. timeout=10,
  73. )
  74. self.context.logger.info(f"{result.status} - {await result.json()}")
  75. async def respond(self, res: rollbot.Response):
  76. # sleep for a moment to make groupme not misorder messages
  77. await asyncio.sleep(0.5)
  78. if res.origin_id != "GROUPME":
  79. self.context.logger.error(f"Unable to respond to {res.origin_id}")
  80. return
  81. bot_id = groupme_bots.get(res.channel_id, None)
  82. if bot_id is None:
  83. self.context.logger.error(f"Unable to respond to group {res.channel_id} in GroupMe")
  84. return
  85. message = ""
  86. attachments = []
  87. try:
  88. if res.attachments is not None:
  89. for att in res.attachments:
  90. if att.name == "image":
  91. if isinstance(att.body, bytes):
  92. attachments.append(
  93. {
  94. "type": "image",
  95. "url": await self.upload_image(att.body),
  96. }
  97. )
  98. else:
  99. attachments.append({"type": "image", "url": att.body})
  100. if att.name == "reply":
  101. if not isinstance(att.body, str):
  102. raise ValueError("Invalid reply body type")
  103. message += att.body # append reply text to response
  104. attachments.append({
  105. "type": "reply",
  106. "base_reply_id": res.channel_id,
  107. "reply_id": res.channel_id,
  108. })
  109. except:
  110. self.context.debugging = "".join(traceback.format_exc())
  111. message += "Failed to upload one or more attachments!\n"
  112. self.context.logger.exception("Failed to upload attachment")
  113. if res.text is not None:
  114. message += res.text
  115. msgs = []
  116. while len(message) > max_msg_len:
  117. msgs.append(message[:msg_slice] + split_text)
  118. message = message[msg_slice:]
  119. msgs.append(message)
  120. await self.post_message(bot_id, msgs[0], attachments)
  121. for m in msgs[1:]:
  122. await asyncio.sleep(0.1)
  123. await self.post_message(bot_id, m, [])
  124. app = FastAPI()
  125. bot = GroupMeBot()
  126. @app.on_event("startup")
  127. async def startup():
  128. await bot.on_startup()
  129. @app.on_event("shutdown")
  130. async def shutdown():
  131. await bot.on_shutdown()
  132. @app.post("/", status_code=204)
  133. async def receive(message: GroupMeMessage, tasks: BackgroundTasks):
  134. tasks.add_task(bot.on_message, message)
  135. return Response(status_code=204)
  136. @app.get("/health", status_code=204)
  137. def health():
  138. return Response(status_code=204)