tcp-proxy2.py 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193
  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. # from config import *
  19. # Connect to:
  20. # HOST = "twgs"
  21. # PORT = 2002
  22. # Listen on:
  23. # LISTEN_PORT = 2002
  24. # LISTEN_ON = "0.0.0.0"
  25. version = check_output(
  26. [
  27. "git",
  28. "describe",
  29. "--long",
  30. "--tags",
  31. # "--dirty",
  32. "--always",
  33. "--match",
  34. "v[0-9]\.[0-9]\.[0-9]",
  35. ],
  36. universal_newlines=True,
  37. ).strip()
  38. # Cleans all ANSI
  39. cleaner = re.compile(r"\x1b\[[0-9;]*[A-Zmh]")
  40. # Looks for ANSI (that should be considered to be a newline)
  41. makeNL = re.compile(r"\x1b\[[0-9;]*[J]")
  42. def treatAsNL(line):
  43. """ Replace any ANSI codes that would be better understood as newlines. """
  44. global makeNL
  45. return makeNL.sub("\n", line)
  46. def cleanANSI(line):
  47. """ Remove all ANSI codes. """
  48. global cleaner
  49. return cleaner.sub("", line)
  50. # return re.sub(r'\x1b\[([0-9,A-Z]{1,2}(;[0-9]{1,2})?(;[0-9]{3})?)?[m|K]?', '', line)
  51. class Game(protocol.Protocol):
  52. def __init__(self):
  53. self.buffer = ""
  54. def connectionMade(self):
  55. log.msg("Connected to Game Server")
  56. self.queue_player = self.factory.queue_player
  57. self.queue_game = self.factory.queue_game
  58. self.setPlayerReceived()
  59. # self.queue_twgs.get().addCallback(self.serverDataReceived)
  60. def setPlayerReceived(self):
  61. """ Get deferred from client queue, callback clientDataReceived. """
  62. self.queue_player.get().addCallback(self.playerDataReceived)
  63. def playerDataReceived(self, chunk):
  64. if chunk is False:
  65. self.queue_game = None
  66. log.msg("Player: disconnected, close connection to game")
  67. self.factory.continueTrying = False
  68. self.transport.loseConnection()
  69. else:
  70. # Pass received data to the server
  71. self.transport.write(chunk)
  72. self.setPlayerReceived()
  73. def dataReceived(self, chunk):
  74. self.queue_game.put(chunk)
  75. def connectionLost(self, why):
  76. log.msg("Game connectionLost because: %s" % why)
  77. self.queue_game.put(False)
  78. self.transport.loseConnection()
  79. class GlueFactory(protocol.ClientFactory):
  80. # class GlueFactory(protocol.Factory):
  81. maxDelay = 10
  82. protocol = Game
  83. def __init__(self, player):
  84. self.player = player
  85. self.queue_player = player.queue_player
  86. self.queue_game = player.queue_game
  87. def closeIt(self):
  88. log.msg("closeIt")
  89. self.queue_game.put(False)
  90. def getUser(self, user):
  91. log.msg("getUser( %s )" % user)
  92. self.twgs.logUser(user)
  93. # This was needed when I replaced ClientFactory with Factory.
  94. # def clientConnectionLost(self, connector, why):
  95. # log.msg("clientconnectionlost: %s" % why)
  96. # self.queue_client.put(False)
  97. def clientConnectionFailed(self, connector, why):
  98. log.msg("connection to game failed: %s" % why)
  99. self.queue_game.put(b"Sorry! I'm Unable to connect to the game server.\r\n")
  100. # syncterm gets cranky/locks up if we close this here.
  101. # (Because it is still sending rlogin information?)
  102. reactor.callLater(2, self.closeIt)
  103. class Player(protocol.Protocol):
  104. def __init__(self):
  105. self.buffer = ""
  106. self.fpRaw = None
  107. self.fpLines = None
  108. self.action = None
  109. self.username = ""
  110. self.game = ""
  111. self.passon = True
  112. def connectionMade(self):
  113. """ connected, setup queues.
  114. queue_player is data from player.
  115. queue_game is data to player. (possibly from game)
  116. """
  117. self.queue_player = defer.DeferredQueue()
  118. self.queue_game = defer.DeferredQueue()
  119. self.setGameReceived()
  120. # Connect GlueFactory to this Player object.
  121. factory = GlueFactory(self)
  122. # Make connection to the game server
  123. reactor.connectTCP(HOST, PORT, factory, 5)
  124. def setGameReceived(self):
  125. """ Get deferred from client queue, callback clientDataReceived. """
  126. self.queue_game.get().addCallback(self.gameDataReceived)
  127. def gameDataReceived(self, chunk):
  128. """ Data received from the client/player. """
  129. if chunk is False:
  130. self.transport.loseConnection()
  131. else:
  132. self.transport.write(chunk)
  133. self.setGameReceived()
  134. def dataReceived(self, chunk):
  135. self.queue_player.put(chunk)
  136. def connectionLost(self, why):
  137. log.msg("lost connection %s" % why)
  138. self.queue_player.put(False)
  139. def connectionFailed(self, why):
  140. log.msg("connectionFailed: %s" % why)
  141. if __name__ == "__main__":
  142. if LOGFILE:
  143. log.startLogging(DailyLogFile("proxy.log", "."))
  144. else:
  145. log.startLogging(sys.stdout)
  146. log.msg("This is version: %s" % version)
  147. factory = protocol.Factory()
  148. factory.protocol = Player
  149. reactor.listenTCP(LISTEN_PORT, factory, interface=LISTEN_ON)
  150. reactor.run()