archive
/
tcp-nat-proxy
Archived
1
Fork 0
This repository has been archived on 2023-03-26. You can view files and clone it, but cannot push or open issues or pull requests.
tcp-nat-proxy/tcp_nat_proxy.py

85 lines
2.3 KiB
Python
Raw Permalink Normal View History

2019-01-01 19:07:00 +00:00
import argparse
2019-01-01 18:44:40 +00:00
import asyncio
2019-01-01 20:26:36 +00:00
import logging
2019-01-01 19:07:00 +00:00
from collections import namedtuple
2019-01-01 19:58:47 +00:00
from functools import partial
2019-01-01 18:44:40 +00:00
2019-01-01 20:26:36 +00:00
logger = logging.getLogger(__name__)
2019-01-01 19:07:00 +00:00
Route = namedtuple("Route", ["listen_port", "destination_host", "destination_port"])
BUFFER_SIZE = 128 * 1024 # 64kb
2019-01-01 18:44:40 +00:00
2019-01-01 20:26:36 +00:00
FORMAT = "%(levelname)s: %(message)s"
logging.basicConfig(format=FORMAT)
2019-01-01 18:44:40 +00:00
2019-01-01 19:07:00 +00:00
def destination_host_display(route: Route):
return "{}:{}".format(route.destination_host, route.destination_port)
def parse_argument(value):
return Route(*value.split(":"))
2019-01-01 18:44:40 +00:00
async def pipe(reader, writer):
try:
while not reader.at_eof():
writer.write(await reader.read(BUFFER_SIZE))
2019-01-01 21:30:55 +00:00
await writer.drain()
2019-01-01 18:44:40 +00:00
finally:
writer.close()
2019-01-01 19:58:47 +00:00
async def handle_client(route, local_reader, local_writer):
try:
2019-01-01 21:17:56 +00:00
logger.debug(
"Openning connection to {}".format(destination_host_display(route))
)
2019-01-01 19:58:47 +00:00
remote_reader, remote_writer = await asyncio.open_connection(
route.destination_host, route.destination_port
)
await asyncio.gather(
pipe(local_reader, remote_writer), pipe(remote_reader, local_writer)
)
2019-01-01 20:40:33 +00:00
except (ConnectionRefusedError, OSError) as e:
logger.warning(
"Connection to {} refused: {}".format(destination_host_display(route), e)
)
2019-01-01 19:58:47 +00:00
pass
finally:
local_writer.close()
2019-01-01 19:07:00 +00:00
async def create_proxy_pipe(route: Route):
2019-01-01 19:58:47 +00:00
server = await asyncio.start_server(
partial(handle_client, route), "0.0.0.0", route.listen_port
)
2019-01-01 20:26:36 +00:00
logger.info(
2019-01-01 19:07:00 +00:00
"Routing from {} to {}".format(
route.listen_port, destination_host_display(route)
)
)
2019-01-01 18:44:40 +00:00
await server.wait_closed()
async def main():
2019-01-01 19:07:00 +00:00
parser = argparse.ArgumentParser()
parser.add_argument(
"-R", "--route", action="append", required=True, type=parse_argument
)
2019-01-01 20:26:36 +00:00
parser.add_argument("--verbose", action="store_true")
2019-01-01 19:07:00 +00:00
args = parser.parse_args()
2019-01-01 20:26:36 +00:00
logger.setLevel(logging.DEBUG if args.verbose else logging.INFO)
2019-01-01 19:07:00 +00:00
servers = [create_proxy_pipe(route) for route in args.route]
2019-01-01 20:26:36 +00:00
logger.debug("Starting servers...")
2019-01-01 19:07:00 +00:00
await asyncio.gather(*servers)
2019-01-01 18:44:40 +00:00
if __name__ == "__main__":
2019-01-01 19:07:00 +00:00
try:
asyncio.run(main())
except KeyboardInterrupt:
2019-01-01 20:26:36 +00:00
logger.error("Process terminated")