tcp-proxy2.py 7.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256
  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.python.logfile import DailyLogFile
  9. # from twisted.enterprise import adbapi
  10. import pendulum
  11. from subprocess import check_output
  12. # This isn't the best configuration, but it's simple
  13. # and works. Mostly.
  14. try:
  15. from config_dev import *
  16. except ModuleNotFoundError:
  17. from config import *
  18. # Extract the version information from git.
  19. version = check_output(
  20. [
  21. "git",
  22. "describe",
  23. "--long",
  24. "--tags",
  25. # "--dirty",
  26. "--always",
  27. "--match",
  28. "v[0-9]\.[0-9]\.[0-9]",
  29. ],
  30. universal_newlines=True,
  31. ).strip()
  32. # Cleans all ANSI
  33. cleaner = re.compile(r"\x1b\[[0-9;]*[A-Zmh]")
  34. # Looks for ANSI (that should be considered to be a newline)
  35. # This needs to see what is send when something enters / leaves
  36. # the player's current sector. (That doesn't work/isn't
  37. # detected. NNY!)
  38. makeNL = re.compile(r"\x1b\[[0-9;]*[J]")
  39. def treatAsNL(line):
  40. """ Replace any ANSI codes that would be better understood as newlines. """
  41. global makeNL
  42. return makeNL.sub("\n", line)
  43. def cleanANSI(line):
  44. """ Remove all ANSI codes. """
  45. global cleaner
  46. return cleaner.sub("", line)
  47. # return re.sub(r'\x1b\[([0-9,A-Z]{1,2}(;[0-9]{1,2})?(;[0-9]{3})?)?[m|K]?', '', line)
  48. class Game(protocol.Protocol):
  49. def __init__(self):
  50. self.buffer = ""
  51. self.game = ""
  52. def connectionMade(self):
  53. log.msg("Connected to Game Server")
  54. self.queue_player = self.factory.queue_player
  55. self.queue_game = self.factory.queue_game
  56. self.setPlayerReceived()
  57. def setPlayerReceived(self):
  58. """ Get deferred from client queue, callback clientDataReceived. """
  59. self.queue_player.get().addCallback(self.playerDataReceived)
  60. def playerDataReceived(self, chunk):
  61. if chunk is False:
  62. self.queue_player = None
  63. log.msg("Player: disconnected, close connection to game")
  64. # I don't believe I need this if I'm using protocol.Factory
  65. self.factory.continueTrying = False
  66. self.transport.loseConnection()
  67. else:
  68. # Pass received data to the server
  69. self.transport.write(chunk)
  70. self.setPlayerReceived()
  71. def lineReceived(self, line):
  72. """ line received from the game. """
  73. log.msg(">> [{0}]".format(line))
  74. if "TWGS v2.20b" in line and "www.eisonline.com" in line:
  75. # Must not be unicode
  76. # Is there a way to NOT have this logged?
  77. self.queue_game.put(
  78. "TWGS Proxy build "
  79. + version
  80. + " is active. \x1b[1;34m~\x1b[0m to activate.\n\r\n\r",
  81. )
  82. if "TradeWars Game Server" in line and "Copyright (C) EIS" in line:
  83. # We are not in a game
  84. self.game = ""
  85. if "Selection (? for menu): " in line:
  86. game = line[-1]
  87. if game >= "A" and game < "Q":
  88. self.game = game
  89. log.msg("Game: {0}".format(self.game))
  90. self.game = game
  91. # self.setGame(game)
  92. def dataReceived(self, chunk):
  93. """ Data received from the Game.
  94. Remove backspaces.
  95. Treat some ANSI codes as NewLine.
  96. Remove ANSI.
  97. Break into lines.
  98. Trim out carriage returns.
  99. Call lineReceived().
  100. "Optionally" pass data to player.
  101. FUTURE: trigger on prompt. [cleanANSI(buffer)]
  102. """
  103. # Sequence error:
  104. # If I don't put the chunk(I received) to the player.
  105. # anything I display -- lineReceive() put() ... would
  106. # be out of order. (I'd be responding -- before it
  107. # was displayed to the user.)
  108. # Possibly: if self.passon ... send to player
  109. self.queue_game.put(chunk)
  110. self.buffer += chunk.decode("utf-8", "ignore")
  111. # Process any backspaces
  112. while "\x08" in self.buffer:
  113. part = self.buffer.partition("\x08")
  114. self.buffer = part[0][:-1] + part[2]
  115. # Treat some ANSI codes as a newline
  116. self.buffer = treatAsNL(self.buffer)
  117. # Break into lines
  118. while "\n" in self.buffer:
  119. part = self.buffer.partition("\n")
  120. line = part[0].replace("\r", "")
  121. # Clean ANSI codes from line
  122. line = cleanANSI(line)
  123. self.lineReceived(line)
  124. self.buffer = part[2]
  125. def connectionLost(self, why):
  126. log.msg("Game connectionLost because: %s" % why)
  127. self.queue_game.put(False)
  128. self.transport.loseConnection()
  129. class GlueFactory(protocol.ClientFactory):
  130. # class GlueFactory(protocol.Factory):
  131. maxDelay = 10
  132. protocol = Game
  133. def __init__(self, player):
  134. self.player = player
  135. self.queue_player = player.queue_player
  136. self.queue_game = player.queue_game
  137. def closeIt(self):
  138. log.msg("closeIt")
  139. self.queue_game.put(False)
  140. def getUser(self, user):
  141. log.msg("getUser( %s )" % user)
  142. self.twgs.logUser(user)
  143. # This was needed when I replaced ClientFactory with Factory.
  144. # def clientConnectionLost(self, connector, why):
  145. # log.msg("clientconnectionlost: %s" % why)
  146. # self.queue_client.put(False)
  147. def clientConnectionFailed(self, connector, why):
  148. log.msg("connection to game failed: %s" % why)
  149. self.queue_game.put(b"Sorry! I'm Unable to connect to the game server.\r\n")
  150. # syncterm gets cranky/locks up if we close this here.
  151. # (Because it is still sending rlogin information?)
  152. reactor.callLater(2, self.closeIt)
  153. class Player(protocol.Protocol):
  154. def __init__(self):
  155. self.buffer = ""
  156. self.fpRaw = None
  157. self.fpLines = None
  158. self.action = None
  159. self.username = ""
  160. self.game = ""
  161. self.passon = True
  162. def connectionMade(self):
  163. """ connected, setup queues.
  164. queue_player is data from player.
  165. queue_game is data to player. (possibly from game)
  166. """
  167. self.queue_player = defer.DeferredQueue()
  168. self.queue_game = defer.DeferredQueue()
  169. self.setGameReceived()
  170. # Connect GlueFactory to this Player object.
  171. factory = GlueFactory(self)
  172. # Make connection to the game server
  173. reactor.connectTCP(HOST, PORT, factory, 5)
  174. def setGameReceived(self):
  175. """ Get deferred from client queue, callback clientDataReceived. """
  176. self.queue_game.get().addCallback(self.gameDataReceived)
  177. def gameDataReceived(self, chunk):
  178. """ Data received from the client/player. """
  179. if chunk is False:
  180. self.transport.loseConnection()
  181. else:
  182. if type(chunk) == bytes:
  183. self.transport.write(chunk)
  184. elif type(chunk) == str:
  185. self.transport.write(chunk.encode())
  186. else:
  187. log.err("gameDataReceived: type (%s) given!", type(chunk))
  188. self.transport.write(chunk)
  189. self.setGameReceived()
  190. def dataReceived(self, chunk):
  191. self.queue_player.put(chunk)
  192. def connectionLost(self, why):
  193. log.msg("lost connection %s" % why)
  194. self.queue_player.put(False)
  195. def connectionFailed(self, why):
  196. log.msg("connectionFailed: %s" % why)
  197. if __name__ == "__main__":
  198. if LOGFILE:
  199. log.startLogging(DailyLogFile("proxy.log", "."))
  200. else:
  201. log.startLogging(sys.stdout)
  202. log.msg("This is version: %s" % version)
  203. factory = protocol.Factory()
  204. factory.protocol = Player
  205. reactor.listenTCP(LISTEN_PORT, factory, interface=LISTEN_ON)
  206. reactor.run()