messages.py 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390
  1. from flask import Flask, render_template, make_response
  2. from flask_paginate import Pagination, get_page_parameter, get_page_args
  3. from flask_caching import Cache
  4. from flask import request
  5. import pendulum
  6. import subprocess
  7. import base64
  8. import os
  9. import textwrap
  10. import sys
  11. import re
  12. import json
  13. def rot47(s):
  14. x = ''
  15. for c in s:
  16. j = ord(c)
  17. if j >= 33 and j <= 126:
  18. x += chr(33+ ((j+14) % 94))
  19. else:
  20. x += c
  21. return x
  22. base_path = "/messagebase"
  23. app = Flask(__name__, static_url_path=base_path + "/static")
  24. @app.template_filter("datefmt")
  25. def format_datetime(value):
  26. # dt = pendulum.from_timestamp(value, tz=pendulum.tz.local_timezone())
  27. dt = pendulum.from_timestamp(value)
  28. return dt.to_datetime_string()
  29. # Check Configuring Flask-Caching section for more details
  30. # cache = Cache(app, config={"CACHE_TYPE": "filesystem", "CACHE_DIR": "cache"})
  31. cache = Cache(
  32. app, config={"CACHE_TYPE": "redis", "CACHE_REDIS_HOST": "redis"}
  33. )
  34. # cache = Cache(app, config={"CACHE_TYPE": "redis", "CACHE_REDIS_HOST": "olympus"})
  35. # import jammin
  36. import sqlite3
  37. # Should this be in the actual calls?
  38. # (So I don't keep a connection open all the time?)
  39. dbconnect = sqlite3.connect("db/message.sqlite3")
  40. dbc = dbconnect.cursor()
  41. bases = {
  42. "FSXNET-General": "fsx_gen",
  43. "FSXNET-Ads": "fsx_ads",
  44. "FSXNET-BBS": "fsx_bbs",
  45. "FSXNET-BOT": "fsx_bot",
  46. "FSXNET-Encryption": "fsx_cry",
  47. "FSXNET-Network": "fsx_net",
  48. "FSXNET-Ham Radio": "fsx_ham",
  49. # "FSXNET-Magicka": "msgs/fsx_mag",
  50. "FSXNET-Magicka": "fsx_mag",
  51. "FSXNET-Mystic": "fsx_mys",
  52. "FSXNET-Enigma": "fsx_eng",
  53. "FSXNET-Gaming": "fsx_gaming",
  54. "FSXNET-Space & Astronomy": "fsx_space",
  55. "FSXNET-Sports": "fsx_sports",
  56. "FSXNET-Retro Computing": "fsx_retro",
  57. "FSXNET-Transport": "fsx_transport",
  58. "FSXNET-Video": "fsx_video",
  59. "FSXNET-Music": "fsx_music",
  60. "FSXNET-Do It Yourself": "fsx_diy",
  61. "FSXNET-Food": "fsx_food",
  62. "FSXNET-Gardening": "fsx_gardening",
  63. "FSXNET-Arts": "fsx_arts",
  64. "FSXNET-Data": "fsx_dat",
  65. # "HappyNet-General": "msgs/hpy_gen",
  66. }
  67. def bbs_get_messages(area):
  68. global dbc
  69. messages = []
  70. for row in dbc.execute(
  71. # "SELECT message_id, to_user_name, from_user_name, subject, modified_timestamp from message WHERE area_tag=?",
  72. "SELECT message_id, to_user_name, from_user_name, subject, modified_timestamp from message WHERE area_tag=? ORDER BY message_id;",
  73. (area,),
  74. ):
  75. stamp = pendulum.parse(row[4]).timestamp()
  76. messages.append(
  77. {
  78. "MsgNum": row[0],
  79. "number": row[0],
  80. "to": row[1],
  81. "from": row[2],
  82. "subject": row[3],
  83. "written": stamp,
  84. # // written
  85. # // received
  86. # // processed
  87. }
  88. )
  89. return messages
  90. MATCH1 = re.compile(">>> BEGIN(.*)>>> END", re.DOTALL)
  91. def bbs_message(area, msgno):
  92. global dbc
  93. messages = []
  94. dbc.execute(
  95. "SELECT message_id, to_user_name, from_user_name, subject, modified_timestamp, message from message WHERE message_id=?",
  96. (msgno,),
  97. )
  98. row = dbc.fetchone()
  99. if (not row):
  100. return
  101. stamp = pendulum.parse(row[4]).timestamp()
  102. data = {
  103. "MsgNum": row[0],
  104. "number": row[0],
  105. "to": row[1],
  106. "from": row[2],
  107. "subject": row[3],
  108. "written": stamp,
  109. "received": stamp,
  110. "processed": stamp,
  111. "text": row[5], # .decode("cp437"),
  112. "bytes": row[5].encode("cp437")
  113. # // written
  114. # // received
  115. # // processed
  116. }
  117. if (area == 'fsx_dat') and ('>>> BEGIN' in row[5]):
  118. body = row[5] + "\n"
  119. result = MATCH1.search(body)
  120. if result:
  121. data['rot47'] = rot47(result.group(1)).lstrip("\n").replace("\n", "<br />")
  122. return data
  123. # bases = {"FSX_BOT": "fsx_bot"}
  124. # @cache.memoize(timeout=5 * 60, key_prefix="messages")
  125. @cache.memoize(timeout=5 * 60)
  126. def get_messages(base):
  127. messages = bbs_get_messages(base)
  128. messages.reverse()
  129. return messages
  130. @cache.memoize(timeout=60)
  131. def get_message(base, msgno):
  132. # message = jammin.read_message(base, msgno)
  133. message = bbs_message(base, msgno)
  134. return message
  135. @app.errorhandler(404)
  136. def not_found(e):
  137. return render_template("404.html")
  138. @app.route(base_path + "/list")
  139. def list_bases():
  140. return render_template(
  141. "list.html", bases=bases, base_path=base_path, title="Message Areas"
  142. )
  143. # return 'Here would be a listing of message bases'
  144. @app.route(base_path + "/clear")
  145. def clear_cache():
  146. cache.clear()
  147. return "Cache Cleared. Back to hitting refresh!"
  148. @app.route(base_path + "/messages/<area>")
  149. def display_messages(area):
  150. if area not in bases:
  151. return render_template(
  152. "missing-area.html", base_path=base_path, title="Missing Area"
  153. )
  154. # messages = jammin.get_messages(bases[area])
  155. messages = get_messages(bases[area])
  156. # messages.reverse() # cached.reverse()
  157. page = request.args.get(get_page_parameter(), type=int, default=1)
  158. # get_page_arg defaults to page 1, per_page of 10
  159. PER_PAGE = 50
  160. total = len(messages)
  161. pagination = Pagination(
  162. page=page,
  163. total=total,
  164. css_framework="foundation",
  165. record_name="messages",
  166. per_page=PER_PAGE,
  167. )
  168. page, per_page, offset = get_page_args()
  169. start = (page - 1) * PER_PAGE
  170. end = start + PER_PAGE
  171. # messages = messages[(page-1) * PER_PAGE:offset+PER_PAGE]
  172. messages = messages[start:end]
  173. return render_template(
  174. "messages.html",
  175. messages=messages,
  176. area=area,
  177. pagination=pagination,
  178. base_path=base_path,
  179. title="Messages for " + bases[area],
  180. )
  181. @cache.memoize(timeout=60)
  182. def ansi_to_png(raw_ansi_bytes, idx):
  183. pid = os.getppid()
  184. ansifile = "{0}-{1}.ans".format(idx, pid)
  185. pngfile = "{0}-{1}.png".format(idx, pid)
  186. with open(ansifile, "wb") as fp:
  187. fp.write(raw_ansi_bytes)
  188. subprocess.run(["./ansilove", "-d", "-o", pngfile, ansifile])
  189. with open(pngfile, "rb") as fp:
  190. png = fp.read()
  191. os.unlink(ansifile)
  192. os.unlink(pngfile)
  193. return png
  194. def ansi_to_png64(raw_ansi_bytes, idx):
  195. png = ansi_to_png(raw_ansi_bytes, idx)
  196. return base64.b64encode(png).decode("utf-8")
  197. @app.route(base_path + "/image/<area>/<int:msgno>.png")
  198. def display_ansi(area, msgno):
  199. if area not in bases:
  200. return "RATS", 404
  201. message = get_message(bases[area], msgno)
  202. if not message:
  203. return "RATS", 404
  204. if not "text" in message:
  205. return "RATS", 404
  206. # png = ansi_to_png(message["bytes"].replace(b"\r", b"\n"), msgno)
  207. png = ansi_to_png(message["bytes"], msgno)
  208. # png = ansi_to_png(message["bytes"].replace("\r", "\n"), msgno)
  209. response = make_response(png)
  210. response.headers.set("Content-Type", "image/png")
  211. return response
  212. # <img alt="My Image" src="data:image/png;base64,
  213. @app.route(base_path + "/read/<area>/<int:msgno>")
  214. def display_message(area, msgno):
  215. if area not in bases:
  216. return render_template(
  217. "missing-area.html", base_path=base_path, title="Missing Area"
  218. )
  219. # message = jammin.read_message(bases[area], msgno)
  220. message = get_message(bases[area], msgno)
  221. if not message:
  222. return render_template(
  223. "missing-message.html",
  224. base_path=base_path,
  225. area=area,
  226. title="Missing Message",
  227. )
  228. messages = get_messages(bases[area])
  229. # prevmsg and nextmsg are completely different now.
  230. prevmsg = None
  231. nextmsg = None
  232. total = len(messages)
  233. for idx, msg in enumerate(messages):
  234. if msg["MsgNum"] == msgno:
  235. # Ok, found what we're looking for
  236. if idx > 0:
  237. prevmsg = messages[idx - 1]["MsgNum"]
  238. if idx + 1 < total:
  239. nextmsg = messages[idx + 1]["MsgNum"]
  240. # prevmsg = None
  241. # nextmsg = None
  242. # if msgno > 1:
  243. # prevmsg = msgno - 1
  244. # if msgno < total:
  245. # nextmsg = msgno + 1
  246. if "text" in message:
  247. if "\x1b" in message["text"]:
  248. # Ok, the message contains ANSI CODES -- Convert
  249. message["png"] = True
  250. # message["png"] = ansi_to_png64(
  251. # message["bytes"].replace(b"\r", b"\n"), msgno
  252. # )
  253. else:
  254. text = message["text"].replace("\r", "\n")
  255. # Ok, latest changes aren't doing word-wrap for us, so do it here.
  256. text = "\n".join(
  257. [
  258. textwrap.fill(txt, width=78, replace_whitespace=False)
  259. for txt in text.splitlines()
  260. ]
  261. )
  262. message["text"] = text
  263. # message["text"].replace("\r", "\n") # <br >\n")
  264. return render_template(
  265. "message.html",
  266. message=message,
  267. area=area,
  268. msgnumber=msgno,
  269. prevmsg=prevmsg,
  270. nextmsg=nextmsg,
  271. base_path=base_path,
  272. title="Message {0}".format(msgno),
  273. )
  274. # LAST CALLERS PROCESSING
  275. def time_duration(time_delta):
  276. if (time_delta.in_seconds() < 60):
  277. return "{0} seconds ago".format(time_delta.in_seconds())
  278. if (time_delta.in_minutes() < 60):
  279. return "{0} minutes ago".format(time_delta.in_minutes())
  280. if (time_delta.in_hours() < 24):
  281. return "{0} hours ago".format(time_delta.in_hours())
  282. return "{0} days ago".format(time_delta.in_days())
  283. # in_months, in_years ...
  284. @cache.memoize(timeout=60)
  285. def last_bbs_callers():
  286. dbsystem = sqlite3.connect("db/system.sqlite3")
  287. dbsys = dbsystem.cursor()
  288. dbuser = sqlite3.connect("db/user.sqlite3")
  289. dbusr = dbuser.cursor()
  290. # step 1: get list of last 25 callers
  291. users = []
  292. lookup = set()
  293. now = pendulum.now()
  294. for row in dbsys.execute('select id,timestamp,log_value from system_event_log where log_name="user_login_history" order by id desc limit 25;'):
  295. # Ok!
  296. # row[0], row[1], row[2]
  297. jdata = json.loads(row[2])
  298. called_when = pendulum.parse(row[1])
  299. long_ago = time_duration(now - called_when)
  300. caller = { 'logid': row[0], 'timestamp': row[1], 'userId': jdata['userId'], 'ago': long_ago }
  301. lookup.add(jdata['userId'])
  302. users.append(caller)
  303. # Ok, we have a list of userIds to look up.
  304. # just look all 10 of them up. :P
  305. for row in dbusr.execute('select U.id,U.user_name, (SELECT prop_value FROM user_property AS UP WHERE U.id=user_id AND prop_name="location") as location from user as U;'):
  306. (userid, username, location) = row
  307. if userid in lookup:
  308. # Ok, we have something!
  309. for u in users:
  310. if u['userId'] == userid:
  311. u['username'] = username
  312. u['location'] = location
  313. return users;
  314. @app.route("/lastcallers")
  315. def display_lastcallers():
  316. users=last_bbs_callers()
  317. return render_template(
  318. "lastcallers.html",
  319. users=users,
  320. title="Last Callers"
  321. )