tcp-proxy.py 8.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255
  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. from config_dev import *
  12. # Connect to:
  13. # HOST = "twgs"
  14. # PORT = 2002
  15. # Listen on:
  16. # LISTEN_PORT = 2002
  17. # LISTEN_ON = "0.0.0.0"
  18. version = check_output(
  19. [
  20. "git",
  21. "describe",
  22. "--long",
  23. "--tags",
  24. # "--dirty",
  25. "--always",
  26. "--match",
  27. "v[0-9]\.[0-9]\.[0-9]",
  28. ],
  29. universal_newlines=True,
  30. ).strip()
  31. cleaner = re.compile(r"\x1b\[[0-9;]*[A-Zmh]")
  32. makeNL = re.compile(r"\x1b\[[0-9;]*[J]")
  33. def treatAsNL(line):
  34. global makeNL
  35. return makeNL.sub("\n", line)
  36. def cleanANSI(line):
  37. global cleaner
  38. return cleaner.sub("", line)
  39. # return re.sub(r'\x1b\[([0-9,A-Z]{1,2}(;[0-9]{1,2})?(;[0-9]{3})?)?[m|K]?', '', line)
  40. class PlayerProtocol(protocol.Protocol):
  41. def __init__(self):
  42. self.user = None
  43. self.dbinit = False
  44. self.db = None
  45. self.buffer = ""
  46. def connectionMade(self):
  47. log.msg("Client: connected to peer")
  48. self.queue_twgs = self.factory.queue_twgs
  49. self.queue_twgs.get().addCallback(self.serverDataReceived)
  50. def serverDataReceived(self, chunk):
  51. # rlogin looks like this: \x00 password \x00 username \x00 terminal/speed \x00
  52. # b'\x00up2lat3\x00bugz\x00ansi-bbs/115200\x00'
  53. # We're looking for 4 \x00!
  54. # TODO: Line processing, and line cleaning (remove ANSI color codes)
  55. if chunk is False:
  56. self.queue_twgs = None
  57. log.msg("Client: disconnecting from peer")
  58. self.factory.continueTrying = False
  59. self.transport.loseConnection()
  60. else:
  61. # self.buffer += chunk.decode("utf-8", 'ignore')
  62. if self.user is None:
  63. self.buffer += chunk.decode("utf-8", "ignore")
  64. # Ok, process this
  65. # self.buffer += chunk.decode('utf-8')
  66. # We don't have the username yet
  67. parts = self.buffer.split("\x00")
  68. if len(parts) >= 5:
  69. # Got it!
  70. self.user = parts[1]
  71. log.msg("User: {0}".format(self.user))
  72. # Reset buffer -- remove everything before last \x00
  73. zpos = self.buffer.rindex("\x00")
  74. self.buffer = self.buffer[zpos + 1 :]
  75. self.buffer = ""
  76. # init sqlite db using the username
  77. self.factory.getUser(self.user)
  78. self.transport.write(chunk)
  79. self.queue_twgs.get().addCallback(self.serverDataReceived)
  80. def dataReceived(self, chunk):
  81. # log.msg("Client: %d bytes received from peer" % len(chunk))
  82. # clean, strip ANSI, etc.
  83. # log.msg("<<", chunk.decode("utf-8", "ignore"))
  84. # log.msg("<<", repr(chunk))
  85. self.factory.queue_client.put(chunk)
  86. def connectionLost(self, why):
  87. log.msg("connectionLost because: %s" % why)
  88. # if self.cli_queue:
  89. # self.cli_queue = None
  90. # log.msg("Client: peer disconnect unexpectedly")
  91. self.factory.queue_client.put(False)
  92. self.queue_twgs = None
  93. self.transport.loseConnection()
  94. class GlueFactory(protocol.ClientFactory):
  95. # class GlueFactory(protocol.Factory):
  96. maxDelay = 10
  97. protocol = PlayerProtocol
  98. def __init__(self, queue_client, queue_twgs, twgs):
  99. self.queue_client = queue_client
  100. self.queue_twgs = queue_twgs
  101. self.twgs = twgs
  102. def closeIt(self):
  103. log.msg("closeIt")
  104. self.queue_client.put(False)
  105. def getUser(self, user):
  106. log.msg("getUser( %s )" % user)
  107. self.twgs.logUser(user)
  108. # This was needed when I replaced ClientFactory with Factory.
  109. # def clientConnectionLost(self, connector, why):
  110. # log.msg("clientconnectionlost: %s" % why)
  111. # self.queue_client.put(False)
  112. def clientConnectionFailed(self, connector, why):
  113. log.msg("connectionFailed: %s" % why)
  114. self.queue_client.put(b"Sorry! I'm Unable to connect to the game server.\r\n")
  115. # syncterm gets cranky/locks up if we close this here.
  116. # (Because it is still sending rlogin information?)
  117. reactor.callLater(2, self.closeIt)
  118. # ProxyServer is created for each connection
  119. class TWGSServer(protocol.Protocol):
  120. def __init__(self):
  121. self.buffer = ""
  122. self.fpRaw = None
  123. self.fpLines = None
  124. def connectionMade(self):
  125. self.queue_twgs = defer.DeferredQueue()
  126. self.queue_client = defer.DeferredQueue()
  127. self.queue_client.get().addCallback(self.clientDataReceived)
  128. factory = GlueFactory(self.queue_client, self.queue_twgs, self)
  129. reactor.connectTCP(HOST, PORT, factory, 5)
  130. def logUser(self, user):
  131. now = pendulum.now()
  132. filename = now.format("YYYY-MM-DD_HHmm") + "-" + user.lower()
  133. # self.fpRaw = open(filename + ".raw", "ab")
  134. self.fpLines = open(filename + ".lines", "a")
  135. # print("Log created:", now.to_rss_string(), "\n", file=self.fpRaw)
  136. print("Log created:", now.to_rss_string(), "\n", file=self.fpLines)
  137. def gotLine(self, line):
  138. # log.msg(">>> [{0}]".format(line.decode("utf-8", "ignore")))
  139. log.msg(">>> [{0}]".format(line))
  140. if self.fpLines is not None:
  141. print(line, file=self.fpLines)
  142. if "TWGS v2.20b" in line:
  143. if "www.eisonline.com" in line:
  144. # Must not be unicode
  145. # Is there a way to NOT have this logged?
  146. self.queue_client.put(
  147. (
  148. b"TWGS Proxy build "
  149. + version.encode()
  150. + b" is active. \x1b[1;34m~\x1b[0m to activate.\n\r\n\r",
  151. )
  152. )
  153. def clientDataReceived(self, chunk):
  154. if chunk is False:
  155. self.transport.loseConnection()
  156. else:
  157. if type(chunk) is tuple:
  158. self.transport.write(chunk[0])
  159. self.queue_client.get().addCallback(self.clientDataReceived)
  160. else:
  161. if self.fpRaw is not None:
  162. self.fpRaw.write(chunk)
  163. self.buffer += chunk.decode("utf-8", "ignore")
  164. # Process any backspaces in the buffer
  165. while "\x08" in self.buffer:
  166. part = self.buffer.partition("\x08")
  167. self.buffer = part[0][:-1] + part[2]
  168. # Treat some ANSI codes as a newline (for purposes of Lines)
  169. self.buffer = treatAsNL(self.buffer)
  170. # Break the buffer into lines
  171. while "\n" in self.buffer:
  172. part = self.buffer.partition("\n")
  173. line = part[0].replace("\r", "")
  174. # Clean ANSI codes from the line
  175. line = cleanANSI(line)
  176. self.gotLine(line)
  177. self.buffer = part[2]
  178. # log.msg("Server: writing %d bytes to original client" % len(chunk))
  179. self.transport.write(chunk)
  180. self.queue_client.get().addCallback(self.clientDataReceived)
  181. def dataReceived(self, chunk):
  182. # log.msg("Server: %d bytes received" % len(chunk))
  183. if chunk == b"~":
  184. self.queue_client.put(
  185. (b"\r\n**********\r\nTWGS Proxy is almost awake...\r\n",)
  186. )
  187. else:
  188. self.queue_twgs.put(chunk)
  189. def connectionLost(self, why):
  190. log.msg("lost connection %s" % why)
  191. self.queue_twgs.put(False)
  192. if self.fpRaw is not None:
  193. self.fpRaw.close()
  194. self.fpRaw = None
  195. if self.fpLines is not None:
  196. if self.buffer != "":
  197. print(self.buffer, file=self.fpLines)
  198. self.fpLines.close()
  199. self.fpLines = None
  200. def connectionFailed(self, why):
  201. log.msg("connectionFailed: %s" % why)
  202. if __name__ == "__main__":
  203. log.startLogging(sys.stdout)
  204. log.msg("This is version: %s" % version)
  205. factory = protocol.Factory()
  206. factory.protocol = TWGSServer
  207. reactor.listenTCP(LISTEN_PORT, factory, interface=LISTEN_ON)
  208. reactor.run()