tcp-proxy.py 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405
  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.internet import task
  8. from twisted.python import log
  9. from twisted.python.logfile import DailyLogFile
  10. import pendulum
  11. from subprocess import check_output
  12. import os
  13. from colorama import Fore, Back, Style
  14. from pprint import pformat
  15. import yaml
  16. def config_load(filename):
  17. global config
  18. with open(filename, 'r') as fp:
  19. config = yaml.safe_load(fp)
  20. if os.path.exists('config_dev.yaml'):
  21. config_load('config_dev.yaml')
  22. else:
  23. config_load('config.yaml')
  24. # Extract the version information from git.
  25. # The match gives us only tags starting with v[0-9]* Using anything else trips up on double digits.
  26. version = check_output(
  27. [
  28. "git",
  29. "describe",
  30. "--abbrev=8",
  31. "--long",
  32. "--tags",
  33. "--dirty",
  34. "--always",
  35. "--match",
  36. "v[0-9]*",
  37. ],
  38. universal_newlines=True,
  39. ).strip()
  40. def merge(color_string):
  41. """ Given a string of colorama ANSI, merge them if you can. """
  42. return color_string.replace("m\x1b[", ";")
  43. # https://en.wikipedia.org/wiki/ANSI_escape_code
  44. # Cleans all ANSI
  45. cleaner = re.compile(r"\x1b\[[0-9;]*[A-Zmh]")
  46. # Looks for ANSI (that should be considered to be a newline)
  47. # This needs to see what is send when something enters / leaves
  48. # the player's current sector. (That doesn't work/isn't
  49. # detected. NNY!) It is "\x1b[K" Erase in Line!
  50. makeNL = re.compile(r"\x1b\[[0-9;]*[JK]")
  51. def treatAsNL(line):
  52. """ Replace any ANSI codes that would be better understood as newlines. """
  53. global makeNL
  54. return makeNL.sub("\n", line)
  55. def cleanANSI(line):
  56. """ Remove all ANSI codes. """
  57. global cleaner
  58. return cleaner.sub("", line)
  59. # return re.sub(r'\x1b\[([0-9,A-Z]{1,2}(;[0-9]{1,2})?(;[0-9]{3})?)?[m|K]?', '', line)
  60. from observer import Observer
  61. from flexible import PlayerInput, ProxyMenu
  62. class Game(protocol.Protocol):
  63. def __init__(self):
  64. self.buffer = ""
  65. self.game = None
  66. self.usergame = (None, None)
  67. self.to_player = True
  68. def connectionMade(self):
  69. log.msg("Connected to Game Server")
  70. self.queue_player = self.factory.queue_player
  71. self.queue_game = self.factory.queue_game
  72. self.observer = self.factory.observer
  73. self.factory.game = self
  74. self.setPlayerReceived()
  75. self.observer.connect("user-game", self.show_game)
  76. def show_game(self, game):
  77. self.usergame = game
  78. log.msg("## User-Game:", game)
  79. if game[1] is None:
  80. if hasattr(self, "portdata"):
  81. log.msg("Clearing out old portdata.")
  82. self.portdata = {}
  83. if hasattr(self, "warpdata"):
  84. log.msg("Clearing out old warpdata.")
  85. self.warpdata = {}
  86. def setPlayerReceived(self):
  87. """ Get deferred from client queue, callback clientDataReceived. """
  88. self.queue_player.get().addCallback(self.playerDataReceived)
  89. def playerDataReceived(self, chunk):
  90. if chunk is False:
  91. self.queue_player = None
  92. log.msg("Player: disconnected, close connection to game")
  93. # I don't believe I need this if I'm using protocol.Factory
  94. self.factory.continueTrying = False
  95. self.transport.loseConnection()
  96. else:
  97. # Pass received data to the server
  98. if type(chunk) == str:
  99. self.transport.write(chunk.encode())
  100. log.msg(">> [{0}]".format(chunk))
  101. else:
  102. self.transport.write(chunk)
  103. log.msg(">> [{0}]".format(chunk.decode("utf-8", "ignore")))
  104. self.setPlayerReceived()
  105. def lineReceived(self, line):
  106. """ line received from the game. """
  107. if 'log_lines' in config and config['log_lines']:
  108. log.msg("<< [{0}]".format(line))
  109. # if "TWGS v2.20b" in line and "www.eisonline.com" in line:
  110. # I would still love to "inject" this into the stream
  111. # so it is consistent.
  112. # self.queue_game.put(
  113. # "TWGS Proxy build "
  114. # + version
  115. # + " is active. \x1b[1;34m~\x1b[0m to activate.\n\r\n\r",
  116. # )
  117. if "TradeWars Game Server" in line and "Copyright (C) EIS" in line:
  118. # We are not in a game
  119. if not self.game is None:
  120. # We were in a game.
  121. self.game = None
  122. self.observer.emit("user-game", (self.factory.player.user, self.game))
  123. if "Selection (? for menu): " in line:
  124. game = line[-1]
  125. if game >= "A" and game < "Q":
  126. self.game = game
  127. log.msg("Game: {0}".format(self.game))
  128. self.observer.emit("user-game", (self.factory.player.user, self.game))
  129. self.observer.emit("game-line", line)
  130. def getPrompt(self):
  131. """ Return the current prompt, stripped of ANSI. """
  132. return cleanANSI(self.buffer)
  133. def dataReceived(self, chunk):
  134. """ Data received from the Game.
  135. Remove backspaces.
  136. Treat some ANSI codes as NewLine.
  137. Remove ANSI.
  138. Break into lines.
  139. Trim out carriage returns.
  140. Call lineReceived().
  141. "Optionally" pass data to player.
  142. FUTURE: trigger on prompt. [cleanANSI(buffer)]
  143. """
  144. # Store the text into the buffer before we inject into it.
  145. self.buffer += chunk.decode("utf-8", "ignore")
  146. # log.msg("data: [{0}]".format(repr(chunk)))
  147. if b"TWGS v2.20b" in chunk and b"www.eisonline.com" in chunk:
  148. # Ok, we have a possible target.
  149. target = b"www.eisonline.com\n\r"
  150. pos = chunk.find(target)
  151. if pos != -1:
  152. # Found it! Inject!
  153. message = (
  154. "TWGS Proxy build " + version + ". ~ to activate in game.\n\r"
  155. )
  156. chunk = (
  157. chunk[0 : pos + len(target)]
  158. + message.encode()
  159. + chunk[pos + len(target) :]
  160. )
  161. # Sequence error:
  162. # If I don't put the chunk(I received) to the player.
  163. # anything I display -- lineReceive() put() ... would
  164. # be out of order. (I'd be responding -- before it
  165. # was displayed to the user.)
  166. if self.to_player:
  167. self.queue_game.put(chunk)
  168. # self.buffer += chunk.decode("utf-8", "ignore")
  169. #
  170. # Begin processing the buffer
  171. #
  172. # Process any backspaces
  173. while "\b" in self.buffer:
  174. part = self.buffer.partition("\b")
  175. self.buffer = part[0][:-1] + part[2]
  176. # Treat some ANSI codes as a newline
  177. self.buffer = treatAsNL(self.buffer)
  178. # Break into lines
  179. while "\n" in self.buffer:
  180. part = self.buffer.partition("\n")
  181. line = part[0].replace("\r", "")
  182. # Clean ANSI codes from line
  183. line = cleanANSI(line)
  184. self.lineReceived(line)
  185. self.buffer = part[2]
  186. self.observer.emit("prompt", self.getPrompt())
  187. def connectionLost(self, why):
  188. log.msg("Game connectionLost because: %s" % why)
  189. self.observer.emit("close", why)
  190. self.queue_game.put(False)
  191. self.transport.loseConnection()
  192. class GlueFactory(protocol.ClientFactory):
  193. # class GlueFactory(protocol.Factory):
  194. maxDelay = 10
  195. protocol = Game
  196. def __init__(self, player):
  197. self.player = player
  198. self.queue_player = player.queue_player
  199. self.queue_game = player.queue_game
  200. self.observer = player.observer
  201. self.game = None
  202. def closeIt(self):
  203. log.msg("closeIt")
  204. self.queue_game.put(False)
  205. def getUser(self, user):
  206. log.msg("getUser( %s )" % user)
  207. self.twgs.logUser(user)
  208. # This was needed when I replaced ClientFactory with Factory.
  209. # def clientConnectionLost(self, connector, why):
  210. # log.msg("clientconnectionlost: %s" % why)
  211. # self.queue_client.put(False)
  212. def clientConnectionFailed(self, connector, why):
  213. log.msg("connection to game failed: %s" % why)
  214. self.queue_game.put(b"Sorry! I'm Unable to connect to the game server.\r\n")
  215. # syncterm gets cranky/locks up if we close this here.
  216. # (Because it is still sending rlogin information?)
  217. reactor.callLater(2, self.closeIt)
  218. class Player(protocol.Protocol):
  219. def __init__(self):
  220. self.buffer = ""
  221. self.user = None
  222. self.observer = Observer()
  223. self.game = None
  224. self.glue = None
  225. def connectionMade(self):
  226. """ connected, setup queues.
  227. queue_player is data from player.
  228. queue_game is data to player. (possibly from game)
  229. """
  230. self.queue_player = defer.DeferredQueue()
  231. self.queue_game = defer.DeferredQueue()
  232. self.setGameReceived()
  233. # Connect GlueFactory to this Player object.
  234. factory = GlueFactory(self)
  235. self.glue = factory
  236. # Make connection to the game server
  237. reactor.connectTCP(config['host'], config['port'], factory, 5)
  238. def setGameReceived(self):
  239. """ Get deferred from client queue, callback clientDataReceived. """
  240. self.queue_game.get().addCallback(self.gameDataReceived)
  241. def gameDataReceived(self, chunk):
  242. """ Data received from the game. """
  243. # If we have received game data, it has to be connected.
  244. if self.game is None:
  245. self.game = self.glue.game
  246. if chunk is False:
  247. self.transport.loseConnection()
  248. else:
  249. if type(chunk) == bytes:
  250. self.transport.write(chunk)
  251. elif type(chunk) == str:
  252. self.transport.write(chunk.encode())
  253. else:
  254. log.err("gameDataReceived: type (%s) given!".format(type(chunk)))
  255. self.transport.write(chunk)
  256. self.setGameReceived()
  257. def dataReceived(self, chunk):
  258. if self.user is None:
  259. self.buffer += chunk.decode("utf-8", "ignore")
  260. parts = self.buffer.split("\x00")
  261. if len(parts) >= 5:
  262. # rlogin we have the username
  263. self.user = parts[1]
  264. log.msg("User: {0}".format(self.user))
  265. zpos = self.buffer.rindex("\x00")
  266. self.buffer = self.buffer[zpos + 1 :]
  267. # but I don't need the buffer anymore, so:
  268. self.buffer = ""
  269. # Pass user value on to whatever needs it.
  270. self.observer.emit("user", self.user)
  271. # Unfortunately, the ones interested in this don't exist yet.
  272. if not self.observer.emit("player", chunk):
  273. # Was not dispatched. Send to game.
  274. self.queue_player.put(chunk)
  275. else:
  276. # There's an observer. Don't continue.
  277. return
  278. if chunk == b"~":
  279. prompt = self.game.getPrompt()
  280. # Selection (? for menu): (the game server menu)
  281. # Enter your choice: (game menu)
  282. # Command [TL=00:00:00]:[1800] (?=Help)? : <- YES!
  283. # Computer command [TL=00:00:00]:[613] (?=Help)?
  284. # (and others I've yet to see...)
  285. if re.match(r"Command \[TL=.* \(\?=Help\)\? :", prompt):
  286. menu = ProxyMenu(self.game)
  287. else:
  288. nl = "\n\r"
  289. r = Style.RESET_ALL
  290. log.msg("NNY!")
  291. prompt = self.game.buffer
  292. self.queue_game.put(
  293. r
  294. + nl
  295. + Style.BRIGHT
  296. + "Proxy:"
  297. + Style.RESET_ALL
  298. + " I can't activate at this time."
  299. + nl
  300. )
  301. self.queue_game.put(prompt)
  302. self.queue_player.put("\a")
  303. # self.observer.emit("notyet", prompt)
  304. def connectionLost(self, why):
  305. log.msg("lost connection %s" % why)
  306. self.observer.emit("close", why)
  307. self.queue_player.put(False)
  308. def connectionFailed(self, why):
  309. log.msg("connectionFailed: %s" % why)
  310. if __name__ == "__main__":
  311. if 'logfile' in config and config['logfile']:
  312. log.startLogging(DailyLogFile("proxy.log", "."))
  313. else:
  314. log.startLogging(sys.stdout)
  315. log.msg("This is version: %s" % version)
  316. factory = protocol.Factory()
  317. factory.protocol = Player
  318. reactor.listenTCP(config['listen_port'], factory, interface=config['listen_on'])
  319. reactor.run()
  320. else:
  321. # I can't seem to get twistd -y tcp-proxy.py
  322. # to work. Missing imports?
  323. application = service.Application("TradeWarsGameServer-Proxy")
  324. factory = protocol.Factory()
  325. factory.protocol = Player
  326. internet.TCPServer(config['listen_port'], factory).setServiceParent(application)