talker.py 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373
  1. #!/home/mystic/mystic/bbs/venv/bin/python
  2. import trio
  3. import sys
  4. from pprint import pprint
  5. import json
  6. from colorama import Fore, Back, Style
  7. import argparse
  8. import string
  9. def merge(color_string):
  10. """ Given a string of colorama ANSI, merge them if you can. """
  11. return color_string.replace("m\x1b[", ";")
  12. parser = argparse.ArgumentParser(description="TradeWars Score Bulletins Puller")
  13. parser.add_argument("--host", type=str, help="Hostname to contact", default="127.0.0.1")
  14. parser.add_argument("--port", type=int, help="Port number", default=2002)
  15. parser.add_argument("--username", type=str, help="RLogin Username", default="phil")
  16. parser.add_argument("--password", type=str, help="RLogin Password", default="phil")
  17. parser.add_argument("--games", type=str, help="TWGS Games to select")
  18. parser.add_argument("--name", type=str, help="Server Name for Report")
  19. parser.add_argument("--report", type=str, help="Space Reports base filename")
  20. parser.add_argument("--debug", action="store_true")
  21. parser.add_argument(
  22. "--prompt", type=str, help="TWGS Custom Menu Prompt", default="Quit"
  23. )
  24. # Unfortunately, this can't figure out the custom prison TWGS menus. :(
  25. # I would like this to allow you to select which reports it'll do.
  26. # So I could do the prison report with games A and B.
  27. #
  28. args = parser.parse_args()
  29. HOST = args.host
  30. # PORT=2003
  31. PORT = args.port
  32. state = 1
  33. options = {}
  34. pos = "A"
  35. scoreboard = {}
  36. settings = {}
  37. setup = {}
  38. if args.games is not None:
  39. options = {x: x for x in args.games}
  40. print("We will pull:", ",".join(options.keys()))
  41. # NOTE: --games does not work unless you are also using prompt
  42. if args.debug:
  43. print("ARGS:")
  44. pprint(args)
  45. print("=" * 40)
  46. import re
  47. # Cleans all ANSI
  48. cleaner = re.compile(r"\x1b\[[0-9;]*[A-Zmh]")
  49. def cleanANSI(line):
  50. """ Remove all ANSI codes. """
  51. global cleaner
  52. return cleaner.sub("", line)
  53. def ANSIsplit(line, pos=70):
  54. # Verify that we're not inside an ANSI sequence at this very moment.
  55. save = "\x1b[s"
  56. restore = "\x1b[u"
  57. ansi = line.rindex("\x1b[", 0, pos-1)
  58. start_ansi = ansi
  59. ansi += 2
  60. # to_check = string.digits + string.ascii_letters + ";"
  61. to_check = string.digits + ";"
  62. while line[ansi] in to_check:
  63. ansi += 1
  64. if line[ansi] in string.ascii_letters:
  65. ansi += 1
  66. if ansi > pos:
  67. # Ok, pos is inside an ANSI escape code. break at the start of the
  68. # ANSI code.
  69. ansi = start_ansi
  70. else:
  71. next_ansi = line.index("\x1b[", start_ansi + 1)
  72. if pos < next_ansi:
  73. ansi = pos
  74. else:
  75. ansi = next_ansi
  76. # else:
  77. # next_ansi = line.index("\x1b[", start_ansi + 1)
  78. # if pos < next_ansi:
  79. # ansi = pos
  80. # else:
  81. # ansi = next_ansi
  82. parts = (line[0:ansi] + save, restore + line[ansi:])
  83. return parts
  84. def output(ansi, bbs, line):
  85. cleaned = cleanANSI(line)
  86. if len(line) > 60:
  87. line = re.sub(
  88. r"\s{5,}", lambda spaces: "\x1b[{0}C".format(len(spaces.group(0))), line
  89. )
  90. if len(line) > 65:
  91. lines = ANSIsplit(line, 65)
  92. print(lines[0], file=ansi)
  93. print(lines[1], file=ansi)
  94. else:
  95. print(line, file=ansi)
  96. print(cleaned, file=bbs)
  97. async def space_report():
  98. global args, scoreboard, settings, setup
  99. # Clean up the scoreboard
  100. for k in scoreboard:
  101. l = scoreboard[k][0]
  102. # _, x, l = l.rpartition("\x1b[2J")
  103. _, x, l = l.rpartition("\x1b[H")
  104. scoreboard[k][0] = l
  105. nl = "\r\n"
  106. reset = Style.RESET_ALL
  107. cyan = merge(Fore.CYAN + Style.BRIGHT)
  108. white = merge(Fore.WHITE + Style.BRIGHT)
  109. green = merge(Fore.GREEN + Style.NORMAL)
  110. if args.report:
  111. # Ok, there's a base filename for the report.
  112. ansi = args.report + ".ans"
  113. bbs = args.report + ".bbs"
  114. aFP = open(ansi, "w")
  115. bFP = open(bbs, "w")
  116. output(aFP, bFP, "{0}Space Report for {1}{2}".format(cyan, white, args.name))
  117. output(aFP, bFP, "")
  118. for k in sorted(scoreboard):
  119. name = options[k]
  120. if name == k:
  121. name = ""
  122. output(
  123. aFP, bFP, "{0}Game {1} {2}{3}{4}".format(cyan, k, white, name, reset)
  124. )
  125. # Ok, what settings would I want displayed?
  126. # >> Age of game : 65 days Days since start : 65 days
  127. # >> Delete if idle : 30 days
  128. # >> Time per day : Unlimited Turns per day : 10000
  129. # >> Sectors in game : 25000 Display StarDock : Yes
  130. line = "{0}{1:16} : {2}{3:20}{0}{4:16} : {2}{5}{6}".format(
  131. green,
  132. "Age of game",
  133. cyan,
  134. setup[k]["Age of game"],
  135. "Days since start",
  136. setup[k]["Days since start"],
  137. reset,
  138. )
  139. output(aFP, bFP, line)
  140. line = "{0}{1:16} : {2}{3:20}{0}{4:16} : {2}{5}{6}".format(
  141. green,
  142. "Time per day",
  143. cyan,
  144. setup[k]["Time per day"],
  145. "Turns per day",
  146. setup[k]["Turns per day"],
  147. reset,
  148. )
  149. output(aFP, bFP, line)
  150. line = "{0}{1:16} : {2}{3:20}{0}{4:16} : {2}{5}{6}".format(
  151. green,
  152. "Delete if idle",
  153. cyan,
  154. setup[k]["Delete if idle"],
  155. "Sectors in game",
  156. setup[k]["Sectors in game"],
  157. reset,
  158. )
  159. output(aFP, bFP, line)
  160. # line = "{0}{1:16} : {2}{3}".format(
  161. # green, "Delete if idle", cyan, setup[k]["Delete if idle"], reset
  162. # )
  163. # output(aFP, bFP, line)
  164. output(aFP, bFP, "")
  165. for line in scoreboard[k]:
  166. output(aFP, bFP, line)
  167. aFP.close()
  168. bFP.close()
  169. for k in scoreboard:
  170. print(k, "\n".join(scoreboard[k]))
  171. # pprint(scoreboard)
  172. # for k in settings:
  173. # print(k, "\n".join(settings[k]))
  174. if False:
  175. for k in settings:
  176. print(k)
  177. for l in settings[k]:
  178. # if "[Pause]" in l:
  179. # break
  180. print(l)
  181. pprint(settings)
  182. # pprint(setup)
  183. if args.report:
  184. # Save all the configuration settings to json
  185. filename = args.report + ".json"
  186. print("Saving setup to:", filename)
  187. with open(filename, "w") as fp:
  188. json.dump(setup, fp)
  189. async def send(client, data):
  190. global args
  191. if args.debug:
  192. print("<<", data)
  193. await client.send_all(data.encode("latin-1"))
  194. async def readline(client_stream, line):
  195. global state, pos, settings, args
  196. if state == 1:
  197. if "<" in line:
  198. parts = {x[0]: x[2:].strip() for x in line.split("<") if x != ""}
  199. options.update(parts)
  200. # if "Quit" in line:
  201. if args.prompt in line:
  202. if args.debug:
  203. print("Prompt seen!")
  204. state += 1
  205. if pos in options:
  206. await send(client_stream, pos)
  207. else:
  208. await send(client_stream, "Q\r")
  209. elif state == 3:
  210. if pos not in scoreboard:
  211. scoreboard[pos] = []
  212. if (
  213. not scoreboard[pos]
  214. and (line == "")
  215. or (("Enter your choice" in line) or ("Ranking Traders" in line))
  216. ):
  217. return
  218. scoreboard[pos].append(line)
  219. elif state == 5:
  220. if pos not in settings:
  221. settings[pos] = []
  222. setup[pos] = {}
  223. if not settings[pos] and (line == "" or "Enter your choice" in line):
  224. return
  225. settings[pos].append(line)
  226. if ":" in line:
  227. line = cleanANSI(line)
  228. part2 = line[40:]
  229. if ":" in part2:
  230. # yes, this is a 2 parter
  231. parts = line[0:39].strip().split(":")
  232. else:
  233. parts = line.strip().split(":")
  234. part2 = ""
  235. setup[pos][parts[0].strip()] = parts[1].strip()
  236. if ":" in part2:
  237. parts = part2.strip().split(":")
  238. setup[pos][parts[0].strip()] = parts[1].strip()
  239. elif state == 6:
  240. # if "Quit" in line:
  241. if args.prompt in line:
  242. pos = chr(ord(pos) + 1)
  243. if pos in options:
  244. state = 2
  245. await send(client_stream, pos)
  246. else:
  247. await send(client_stream, "q\r")
  248. # elif state == 2:
  249. # if "[Pause]" in line:
  250. # await client_stream.send_all("\r".encode("latin-1"))
  251. if args.debug:
  252. print(">>", line)
  253. async def prompt(client_stream, line):
  254. global state, pos
  255. if state == 2:
  256. if "[Pause]" in line:
  257. # await client_stream.send_all("\r".encode("latin-1"))
  258. await send(client_stream, "\r")
  259. if "Enter your choice" in line:
  260. state = 3
  261. await send(client_stream, "H\r")
  262. elif state == 3:
  263. if "[Pause]" in line:
  264. state = 4
  265. # await client_stream.send_all("\r".encode("latin-1"))
  266. await send(client_stream, "\r")
  267. # pos = chr(ord(pos) + 1)
  268. elif state == 4:
  269. if "Enter your choice" in line:
  270. state = 5
  271. await send(client_stream, "S\r")
  272. elif state == 5:
  273. if "[Pause]" in line:
  274. await send(client_stream, "\r")
  275. if "Enter your choice" in line:
  276. state = 6
  277. await send(client_stream, "X\r")
  278. if args.debug:
  279. print(">>> {!r}".format(line))
  280. async def receiver(client_stream):
  281. global args
  282. if args.debug:
  283. print("receiver: started")
  284. # I need to send the rlogin connection information ...
  285. rlogin = "\x00{0}\x00{1}\x00ansi-bbs\x009600\x00".format(
  286. args.username, args.password
  287. )
  288. await client_stream.send_all(rlogin.encode("utf-8"))
  289. buffer = ""
  290. async for chunk in client_stream:
  291. buffer += chunk.decode("latin-1", "ignore")
  292. while "\n" in buffer:
  293. part = buffer.partition("\n")
  294. line = part[0].replace("\r", "")
  295. buffer = part[2]
  296. await readline(client_stream, line)
  297. buffer = buffer.replace("\r", "")
  298. if buffer != "":
  299. await prompt(client_stream, buffer)
  300. print("receiver: closed")
  301. # pprint(options)
  302. await space_report()
  303. async def start():
  304. print("connecting...")
  305. client_stream = await trio.open_tcp_stream(HOST, PORT)
  306. async with client_stream:
  307. async with trio.open_nursery() as nursery:
  308. nursery.start_soon(receiver, client_stream)
  309. trio.run(start)