gns3-server/gns3server/utils/asyncio/raw_command_server.py

141 lines
4.6 KiB
Python
Raw Normal View History

2016-05-03 16:49:33 +02:00
#
# Copyright (C) 2014 GNS3 Technologies Inc.
#
# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.
import re
2016-12-09 16:41:15 +01:00
import copy
2016-05-03 16:49:33 +02:00
import asyncio
import asyncio.subprocess
import logging
2021-04-13 18:46:50 +09:30
2016-05-03 16:49:33 +02:00
log = logging.getLogger(__name__)
READ_SIZE = 4096
2016-05-03 18:01:23 +02:00
2016-05-03 16:49:33 +02:00
class AsyncioRawCommandServer:
"""
Expose a process on the network his stdoud and stdin will be forward
on network
"""
2016-05-03 18:01:23 +02:00
def __init__(self, command, replaces=[]):
"""
:param command: Command to run
:param replaces: List of tuple to replace in the output ex: [(b":8080", b":6000")]
"""
2016-05-03 16:49:33 +02:00
self._command = command
2016-05-03 18:01:23 +02:00
self._replaces = replaces
# We limit number of process
self._lock = asyncio.Semaphore(value=4)
2016-05-03 16:49:33 +02:00
async def run(self, network_reader, network_writer):
await self._lock.acquire()
2021-04-13 18:46:50 +09:30
process = await asyncio.subprocess.create_subprocess_exec(
*self._command,
stdout=asyncio.subprocess.PIPE,
stderr=asyncio.subprocess.STDOUT,
stdin=asyncio.subprocess.PIPE
)
2016-05-03 16:49:33 +02:00
try:
await self._process(network_reader, network_writer, process.stdout, process.stdin)
2016-05-03 16:49:33 +02:00
except ConnectionResetError:
network_writer.close()
if process.returncode is None:
process.kill()
await process.wait()
self._lock.release()
2016-05-03 16:49:33 +02:00
async def _process(self, network_reader, network_writer, process_reader, process_writer):
2016-12-09 16:41:15 +01:00
replaces = []
# Server host from the client point of view
host = network_writer.transport.get_extra_info("sockname")[0]
for replace in self._replaces:
2021-04-13 18:46:50 +09:30
if b"{{HOST}}" in replace[1]:
replaces.append(
(
replace[0],
replace[1].replace(b"{{HOST}}", host.encode()),
)
)
2016-12-09 16:41:15 +01:00
else:
2021-04-13 18:46:50 +09:30
replaces.append(
(
replace[0],
replace[1],
)
)
2016-12-09 16:41:15 +01:00
network_read = asyncio.ensure_future(network_reader.read(READ_SIZE))
reader_read = asyncio.ensure_future(process_reader.read(READ_SIZE))
timeout = 30
2016-05-03 16:49:33 +02:00
while True:
done, pending = await asyncio.wait(
2021-04-13 18:46:50 +09:30
[network_read, reader_read], timeout=timeout, return_when=asyncio.FIRST_COMPLETED
)
if len(done) == 0:
raise ConnectionResetError()
2016-05-03 16:49:33 +02:00
for coro in done:
data = coro.result()
if coro == network_read:
if network_reader.at_eof():
raise ConnectionResetError()
network_read = asyncio.ensure_future(network_reader.read(READ_SIZE))
2016-05-03 16:49:33 +02:00
process_writer.write(data)
await process_writer.drain()
2016-05-03 16:49:33 +02:00
elif coro == reader_read:
if process_reader.at_eof():
raise ConnectionResetError()
reader_read = asyncio.ensure_future(process_reader.read(READ_SIZE))
2016-05-03 16:49:33 +02:00
2016-12-09 16:41:15 +01:00
for replace in replaces:
2016-05-03 18:01:23 +02:00
data = data.replace(replace[0], replace[1])
2016-05-10 17:51:40 +02:00
timeout = 2 # We reduce the timeout when the process start to return stuff to avoid problem with server not closing the connection
2016-05-03 16:49:33 +02:00
network_writer.write(data)
await network_writer.drain()
2016-05-03 16:49:33 +02:00
2021-04-13 18:46:50 +09:30
if __name__ == "__main__":
2016-05-03 16:49:33 +02:00
logging.basicConfig(level=logging.DEBUG)
loop = asyncio.get_event_loop()
command = ["nc", "localhost", "80"]
2021-04-13 18:46:50 +09:30
server = AsyncioRawCommandServer(
command,
replaces=[
(
b"work",
b"{{HOST}}",
)
],
)
coro = asyncio.start_server(server.run, "0.0.0.0", 4444)
2016-05-03 16:49:33 +02:00
s = loop.run_until_complete(coro)
try:
loop.run_forever()
except KeyboardInterrupt:
pass
# Close the server
s.close()
loop.run_until_complete(s.wait_closed())
loop.close()