tcp-proxy.py 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327
  1. #!/usr/bin/env python3
  2. import sys
  3. import re
  4. from twisted.internet import defer
  5. from twisted.internet import protocol
  6. from twisted.internet import reactor
  7. from twisted.python import log
  8. from twisted.enterprise import adbapi
  9. import pendulum
  10. from subprocess import check_output
  11. RAW = True
  12. try:
  13. from config_dev import *
  14. except ModuleNotFoundError as E:
  15. from config import *
  16. # from config import *
  17. # from config_dev import *
  18. # Connect to:
  19. # HOST = "twgs"
  20. # PORT = 2002
  21. # Listen on:
  22. # LISTEN_PORT = 2002
  23. # LISTEN_ON = "0.0.0.0"
  24. version = check_output(
  25. [
  26. "git",
  27. "describe",
  28. "--long",
  29. "--tags",
  30. # "--dirty",
  31. "--always",
  32. "--match",
  33. "v[0-9]\.[0-9]\.[0-9]",
  34. ],
  35. universal_newlines=True,
  36. ).strip()
  37. cleaner = re.compile(r"\x1b\[[0-9;]*[A-Zmh]")
  38. makeNL = re.compile(r"\x1b\[[0-9;]*[J]")
  39. def treatAsNL(line):
  40. global makeNL
  41. return makeNL.sub("\n", line)
  42. def cleanANSI(line):
  43. global cleaner
  44. return cleaner.sub("", line)
  45. # return re.sub(r'\x1b\[([0-9,A-Z]{1,2}(;[0-9]{1,2})?(;[0-9]{3})?)?[m|K]?', '', line)
  46. class PlayerProtocol(protocol.Protocol):
  47. def __init__(self):
  48. self.user = None
  49. self.dbinit = False
  50. self.db = None
  51. self.buffer = ""
  52. def connectionMade(self):
  53. log.msg("Client: connected to peer")
  54. self.queue_twgs = self.factory.queue_twgs
  55. self.queue_twgs.get().addCallback(self.serverDataReceived)
  56. def serverDataReceived(self, chunk):
  57. # rlogin looks like this: \x00 password \x00 username \x00 terminal/speed \x00
  58. # b'\x00up2lat3\x00bugz\x00ansi-bbs/115200\x00'
  59. # We're looking for 4 \x00!
  60. # TODO: Line processing, and line cleaning (remove ANSI color codes)
  61. if chunk is False:
  62. self.queue_twgs = None
  63. log.msg("Client: disconnecting from peer")
  64. self.factory.continueTrying = False
  65. self.transport.loseConnection()
  66. else:
  67. # self.buffer += chunk.decode("utf-8", 'ignore')
  68. if self.user is None:
  69. self.buffer += chunk.decode("utf-8", "ignore")
  70. # Ok, process this
  71. # self.buffer += chunk.decode('utf-8')
  72. # We don't have the username yet
  73. parts = self.buffer.split("\x00")
  74. if len(parts) >= 5:
  75. # Got it!
  76. self.user = parts[1]
  77. log.msg("User: {0}".format(self.user))
  78. # Reset buffer -- remove everything before last \x00
  79. zpos = self.buffer.rindex("\x00")
  80. self.buffer = self.buffer[zpos + 1 :]
  81. self.buffer = ""
  82. # init sqlite db using the username
  83. self.factory.getUser(self.user)
  84. self.transport.write(chunk)
  85. self.queue_twgs.get().addCallback(self.serverDataReceived)
  86. def dataReceived(self, chunk):
  87. # log.msg("Client: %d bytes received from peer" % len(chunk))
  88. # clean, strip ANSI, etc.
  89. # log.msg("<<", chunk.decode("utf-8", "ignore"))
  90. # log.msg("<<", repr(chunk))
  91. self.factory.queue_client.put(chunk)
  92. def connectionLost(self, why):
  93. log.msg("connectionLost because: %s" % why)
  94. # if self.cli_queue:
  95. # self.cli_queue = None
  96. # log.msg("Client: peer disconnect unexpectedly")
  97. self.factory.queue_client.put(False)
  98. self.queue_twgs = None
  99. self.transport.loseConnection()
  100. class GlueFactory(protocol.ClientFactory):
  101. # class GlueFactory(protocol.Factory):
  102. maxDelay = 10
  103. protocol = PlayerProtocol
  104. def __init__(self, queue_client, queue_twgs, twgs):
  105. self.queue_client = queue_client
  106. self.queue_twgs = queue_twgs
  107. self.twgs = twgs
  108. def closeIt(self):
  109. log.msg("closeIt")
  110. self.queue_client.put(False)
  111. def getUser(self, user):
  112. log.msg("getUser( %s )" % user)
  113. self.twgs.logUser(user)
  114. # This was needed when I replaced ClientFactory with Factory.
  115. # def clientConnectionLost(self, connector, why):
  116. # log.msg("clientconnectionlost: %s" % why)
  117. # self.queue_client.put(False)
  118. def clientConnectionFailed(self, connector, why):
  119. log.msg("connectionFailed: %s" % why)
  120. self.queue_client.put(b"Sorry! I'm Unable to connect to the game server.\r\n")
  121. # syncterm gets cranky/locks up if we close this here.
  122. # (Because it is still sending rlogin information?)
  123. reactor.callLater(2, self.closeIt)
  124. # ProxyServer is created for each connection
  125. class ProxyAction:
  126. def __init__(self, queue_client, queue_twgs):
  127. self.active = False
  128. self.queue_client = queue_client
  129. self.queue_twgs = queue_twgs
  130. self.buffer = ""
  131. self.prompt = ""
  132. def isActive(self):
  133. return self.active
  134. def keepAlive(self):
  135. if self.active:
  136. self.queue_twgs.put(b" ")
  137. reactor.callLater(30, self.keepAlive)
  138. def activate(self, prompt):
  139. self.active = True
  140. self.prompt = prompt
  141. self.send("\r\n**********\r\nTWGS Proxy ready...\r\n")
  142. self.send("Prompt seen: [{0}] ...\r\n".format(self.prompt))
  143. self.send("(T) Display Time\r\n")
  144. self.send("(Q) Quit\r\n")
  145. self.send(" --==> ")
  146. reactor.callLater(30, self.keepAlive)
  147. def server(self, line):
  148. pass
  149. def send(self, text):
  150. self.queue_client.put((text.encode(),))
  151. def received(self, chunk):
  152. # self.buffer += chunk.encode('UTF-8', 'ignore')
  153. text = chunk.decode("utf-8", "ignore").upper()
  154. if text == "T":
  155. now = pendulum.now('America/New_York')
  156. self.send("\r\nThe time is: {0}.\r\n".format(now.to_rss_string()))
  157. if text == "Q":
  158. self.send("\r\nReturning to TWGS.\r\n{0}".format(self.prompt))
  159. self.active = False
  160. class TWGSServer(protocol.Protocol):
  161. def __init__(self):
  162. self.buffer = ""
  163. self.fpRaw = None
  164. self.fpLines = None
  165. self.action = None
  166. self.user = ""
  167. self.game = ""
  168. def connectionMade(self):
  169. self.queue_twgs = defer.DeferredQueue()
  170. self.queue_client = defer.DeferredQueue()
  171. self.queue_client.get().addCallback(self.clientDataReceived)
  172. self.action = ProxyAction(self.queue_client, self.queue_twgs)
  173. factory = GlueFactory(self.queue_client, self.queue_twgs, self)
  174. reactor.connectTCP(HOST, PORT, factory, 5)
  175. def logUser(self, user):
  176. now = pendulum.now()
  177. self.user = user
  178. filename = now.format("YYYY-MM-DD_HHmm") + "-" + user.lower()
  179. if RAW:
  180. self.fpRaw = open(filename + ".raw", "ab")
  181. self.fpLines = open(filename + ".lines", "a")
  182. # print("Log created:", now.to_rss_string(), "\n", file=self.fpRaw)
  183. print("Log created:", now.to_rss_string(), "\n", file=self.fpLines)
  184. def gotLine(self, line):
  185. # log.msg(">>> [{0}]".format(line.decode("utf-8", "ignore")))
  186. log.msg(">>> [{0}]".format(line))
  187. if self.fpLines is not None:
  188. print(line, file=self.fpLines)
  189. if "TWGS v2.20b" in line:
  190. if "www.eisonline.com" in line:
  191. # Must not be unicode
  192. # Is there a way to NOT have this logged?
  193. self.queue_client.put(
  194. (
  195. b"TWGS Proxy build "
  196. + version.encode()
  197. + b" is active. \x1b[1;34m~\x1b[0m to activate.\n\r\n\r",
  198. )
  199. )
  200. if "TradeWars Game Server" in line:
  201. if "Copyright (C) EIS" in line:
  202. # We are not in a game
  203. self.game = ""
  204. if "Selection (? for menu): " in line:
  205. game = line[-1]
  206. if game >= "A" and game < "Q":
  207. self.game = game
  208. log.msg("USER {0} ENTERED {1}".format(self.user, self.game))
  209. def clientDataReceived(self, chunk):
  210. if chunk is False:
  211. self.transport.loseConnection()
  212. else:
  213. if type(chunk) is tuple:
  214. self.transport.write(chunk[0])
  215. self.queue_client.get().addCallback(self.clientDataReceived)
  216. else:
  217. if self.fpRaw is not None:
  218. self.fpRaw.write(chunk)
  219. self.buffer += chunk.decode("utf-8", "ignore")
  220. # Process any backspaces in the buffer
  221. while "\x08" in self.buffer:
  222. part = self.buffer.partition("\x08")
  223. self.buffer = part[0][:-1] + part[2]
  224. # Treat some ANSI codes as a newline (for purposes of Lines)
  225. self.buffer = treatAsNL(self.buffer)
  226. # I think I need something else in here. When something enters or leaves the sector
  227. # The message isn't shown on it's own line. I think they are using ANSI codes to
  228. # clear the line. (Which certainly would be faster!)
  229. # Break the buffer into lines
  230. while "\n" in self.buffer:
  231. part = self.buffer.partition("\n")
  232. line = part[0].replace("\r", "")
  233. # Clean ANSI codes from the line
  234. line = cleanANSI(line)
  235. self.gotLine(line)
  236. self.buffer = part[2]
  237. # log.msg("Server: writing %d bytes to original client" % len(chunk))
  238. self.transport.write(chunk)
  239. self.queue_client.get().addCallback(self.clientDataReceived)
  240. def dataReceived(self, chunk):
  241. # log.msg("Server: %d bytes received" % len(chunk))
  242. if self.action and self.action.isActive():
  243. # Do something completely different here
  244. self.action.received(chunk)
  245. else:
  246. if chunk == b"~":
  247. self.action.activate(self.buffer)
  248. else:
  249. self.queue_twgs.put(chunk)
  250. def connectionLost(self, why):
  251. log.msg("lost connection %s" % why)
  252. self.queue_twgs.put(False)
  253. if self.fpRaw is not None:
  254. self.fpRaw.close()
  255. self.fpRaw = None
  256. if self.fpLines is not None:
  257. if self.buffer != "":
  258. print(self.buffer, file=self.fpLines)
  259. self.fpLines.close()
  260. self.fpLines = None
  261. def connectionFailed(self, why):
  262. log.msg("connectionFailed: %s" % why)
  263. if __name__ == "__main__":
  264. log.startLogging(sys.stdout)
  265. log.msg("This is version: %s" % version)
  266. factory = protocol.Factory()
  267. factory.protocol = TWGSServer
  268. reactor.listenTCP(LISTEN_PORT, factory, interface=LISTEN_ON)
  269. reactor.run()