mirror of
https://github.com/markqvist/rnsh.git
synced 2025-01-24 21:16:39 -05:00
Remote command line options
This commit is contained in:
parent
5ce4c342bc
commit
bf61ce891c
7
.github/workflows/python-package.yml
vendored
7
.github/workflows/python-package.yml
vendored
@ -8,6 +8,13 @@ on:
|
|||||||
paths-ignore:
|
paths-ignore:
|
||||||
- 'README.md'
|
- 'README.md'
|
||||||
- '.github/**'
|
- '.github/**'
|
||||||
|
pull_request:
|
||||||
|
types:
|
||||||
|
- opened
|
||||||
|
branches:
|
||||||
|
- 'main'
|
||||||
|
paths-ignore:
|
||||||
|
- 'README.md'
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
|
|
||||||
|
26
README.md
26
README.md
@ -17,6 +17,20 @@ out.
|
|||||||
|
|
||||||
Anyway, there's a lot of room for improvement.
|
Anyway, there's a lot of room for improvement.
|
||||||
|
|
||||||
|
## New in v0.0.5
|
||||||
|
### Remote command line and pipe compatibility
|
||||||
|
Command line options have changed somewhat to allow the initiator
|
||||||
|
to supply a command line. This allows `rnsh` to function similarly
|
||||||
|
to SSH. You can pipe into or out of `rnsh` to send input through
|
||||||
|
remote commands or remote command output through other commands.
|
||||||
|
|
||||||
|
This behavior can be blocked on the listener with the `-C` option.
|
||||||
|
|
||||||
|
When the initiator does not supply a command, the listener uses
|
||||||
|
a default command specified on its command line. If a default
|
||||||
|
command is not specified, the listener falls back to the shell
|
||||||
|
of the user it is running under.
|
||||||
|
|
||||||
## Quickstart
|
## Quickstart
|
||||||
|
|
||||||
Tested (thus far) on Python 3.11 macOS 13.1 ARM64. Should
|
Tested (thus far) on Python 3.11 macOS 13.1 ARM64. Should
|
||||||
@ -68,9 +82,10 @@ Usage:
|
|||||||
rnsh [--config <configdir>] [-i <identityfile>] [-s <service_name>] [-l] -p
|
rnsh [--config <configdir>] [-i <identityfile>] [-s <service_name>] [-l] -p
|
||||||
rnsh -l [--config <configfile>] [-i <identityfile>] [-s <service_name>]
|
rnsh -l [--config <configfile>] [-i <identityfile>] [-s <service_name>]
|
||||||
[-v... | -q...] [-b <period>] (-n | -a <identity_hash> [-a <identity_hash>] ...)
|
[-v... | -q...] [-b <period>] (-n | -a <identity_hash> [-a <identity_hash>] ...)
|
||||||
[--] <program> [<arg> ...]
|
[-C] [[--] <program> [<arg> ...]]
|
||||||
rnsh [--config <configfile>] [-i <identityfile>] [-s <service_name>]
|
rnsh [--config <configfile>] [-i <identityfile>] [-s <service_name>]
|
||||||
[-v... | -q...] [-N] [-m] [-w <timeout>] <destination_hash>
|
[-v... | -q...] [-N] [-m] [-w <timeout>] <destination_hash>
|
||||||
|
[[--] <program> [<arg> ...]]
|
||||||
rnsh -h
|
rnsh -h
|
||||||
rnsh --version
|
rnsh --version
|
||||||
|
|
||||||
@ -79,12 +94,17 @@ Options:
|
|||||||
-i FILE --identity FILE Specific identity file to use
|
-i FILE --identity FILE Specific identity file to use
|
||||||
-s NAME --service NAME Listen on/connect to specific service name if not default
|
-s NAME --service NAME Listen on/connect to specific service name if not default
|
||||||
-p --print-identity Print identity information and exit
|
-p --print-identity Print identity information and exit
|
||||||
-l --listen Listen (server) mode
|
-l --listen Listen (server) mode. If supplied, <program> <arg>...will
|
||||||
|
be used as the command line when the initiator does not
|
||||||
|
provide one or when remote command is disabled. If
|
||||||
|
<program> is not supplied, the default shell of the
|
||||||
|
user rnsh is running under will be used.
|
||||||
-b --announce PERIOD Announce on startup and every PERIOD seconds
|
-b --announce PERIOD Announce on startup and every PERIOD seconds
|
||||||
Specify 0 for PERIOD to announce on startup only.
|
Specify 0 for PERIOD to announce on startup only.
|
||||||
-a HASH --allowed HASH Specify identities allowed to connect
|
-a HASH --allowed HASH Specify identities allowed to connect
|
||||||
-n --no-auth Disable authentication
|
-n --no-auth Disable authentication
|
||||||
-N --no-id Disable identify on connect
|
-N --no-id Disable identify on connect
|
||||||
|
-C --no-remote-command Disable executing command line from remote
|
||||||
-m --mirror Client returns with code of remote process
|
-m --mirror Client returns with code of remote process
|
||||||
-w TIME --timeout TIME Specify client connect and request timeout in seconds
|
-w TIME --timeout TIME Specify client connect and request timeout in seconds
|
||||||
-q --quiet Increase quietness (move level up), multiple increases effect
|
-q --quiet Increase quietness (move level up), multiple increases effect
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
[tool.poetry]
|
[tool.poetry]
|
||||||
name = "rnsh"
|
name = "rnsh"
|
||||||
version = "0.0.4"
|
version = "0.0.5"
|
||||||
description = "Shell over Reticulum"
|
description = "Shell over Reticulum"
|
||||||
authors = ["acehoss <acehoss@acehoss.net>"]
|
authors = ["acehoss <acehoss@acehoss.net>"]
|
||||||
license = "MIT"
|
license = "MIT"
|
||||||
|
109
rnsh/args.py
Normal file
109
rnsh/args.py
Normal file
@ -0,0 +1,109 @@
|
|||||||
|
from typing import TypeVar
|
||||||
|
import RNS
|
||||||
|
import importlib.metadata
|
||||||
|
import docopt
|
||||||
|
import sys
|
||||||
|
|
||||||
|
_T = TypeVar("_T")
|
||||||
|
|
||||||
|
|
||||||
|
def _split_array_at(arr: [_T], at: _T) -> ([_T], [_T]):
|
||||||
|
try:
|
||||||
|
idx = arr.index(at)
|
||||||
|
return arr[:idx], arr[idx + 1:]
|
||||||
|
except ValueError:
|
||||||
|
return arr, []
|
||||||
|
|
||||||
|
|
||||||
|
usage = \
|
||||||
|
'''
|
||||||
|
Usage:
|
||||||
|
rnsh [--config <configdir>] [-i <identityfile>] [-s <service_name>] [-l] -p
|
||||||
|
rnsh -l [--config <configfile>] [-i <identityfile>] [-s <service_name>]
|
||||||
|
[-v... | -q...] [-b <period>] (-n | -a <identity_hash> [-a <identity_hash>] ...)
|
||||||
|
[-C] [[--] <program> [<arg> ...]]
|
||||||
|
rnsh [--config <configfile>] [-i <identityfile>] [-s <service_name>]
|
||||||
|
[-v... | -q...] [-N] [-m] [-w <timeout>] <destination_hash>
|
||||||
|
[[--] <program> [<arg> ...]]
|
||||||
|
rnsh -h
|
||||||
|
rnsh --version
|
||||||
|
|
||||||
|
Options:
|
||||||
|
--config DIR Alternate Reticulum config directory to use
|
||||||
|
-i FILE --identity FILE Specific identity file to use
|
||||||
|
-s NAME --service NAME Listen on/connect to specific service name if not default
|
||||||
|
-p --print-identity Print identity information and exit
|
||||||
|
-l --listen Listen (server) mode. If supplied, <program> <arg>...will
|
||||||
|
be used as the command line when the initiator does not
|
||||||
|
provide one or when remote command is disabled. If
|
||||||
|
<program> is not supplied, the default shell of the
|
||||||
|
user rnsh is running under will be used.
|
||||||
|
-b --announce PERIOD Announce on startup and every PERIOD seconds
|
||||||
|
Specify 0 for PERIOD to announce on startup only.
|
||||||
|
-a HASH --allowed HASH Specify identities allowed to connect
|
||||||
|
-n --no-auth Disable authentication
|
||||||
|
-N --no-id Disable identify on connect
|
||||||
|
-C --no-remote-command Disable executing command line from remote
|
||||||
|
-m --mirror Client returns with code of remote process
|
||||||
|
-w TIME --timeout TIME Specify client connect and request timeout in seconds
|
||||||
|
-q --quiet Increase quietness (move level up), multiple increases effect
|
||||||
|
DEFAULT LOGGING LEVEL
|
||||||
|
CRITICAL (silent)
|
||||||
|
Initiator -> ERROR
|
||||||
|
WARNING
|
||||||
|
Listener -> INFO
|
||||||
|
DEBUG (insane)
|
||||||
|
-v --verbose Increase verbosity (move level down), multiple increases effect
|
||||||
|
--version Show version
|
||||||
|
-h --help Show this help
|
||||||
|
'''
|
||||||
|
|
||||||
|
|
||||||
|
class Args:
|
||||||
|
def __init__(self, argv: [str]):
|
||||||
|
global usage
|
||||||
|
try:
|
||||||
|
argv, program_args = _split_array_at(argv, "--")
|
||||||
|
if len(program_args) > 0:
|
||||||
|
argv.append(program_args[0])
|
||||||
|
self.program_args = program_args[1:]
|
||||||
|
|
||||||
|
args = docopt.docopt(usage, argv=argv[1:], version=f"rnsh {importlib.metadata.version('rnsh')}")
|
||||||
|
# json.dump(args, sys.stdout)
|
||||||
|
|
||||||
|
self.service_name = args.get("--service", None) or "default"
|
||||||
|
self.listen = args.get("--listen", None) or False
|
||||||
|
self.identity = args.get("--identity", None)
|
||||||
|
self.config = args.get("--config", None)
|
||||||
|
self.print_identity = args.get("--print-identity", None) or False
|
||||||
|
self.verbose = args.get("--verbose", None) or 0
|
||||||
|
self.quiet = args.get("--quiet", None) or 0
|
||||||
|
announce = args.get("--announce", None)
|
||||||
|
self.announce = None
|
||||||
|
try:
|
||||||
|
if announce:
|
||||||
|
self.announce = int(announce)
|
||||||
|
except ValueError:
|
||||||
|
print("Invalid value for --announce")
|
||||||
|
self.valid = False
|
||||||
|
self.no_auth = args.get("--no-auth", None) or False
|
||||||
|
self.allowed = args.get("--allowed", None) or []
|
||||||
|
self.no_remote_cmd = args.get("--no-remote-command", None) or False
|
||||||
|
self.program = args.get("<program>", None)
|
||||||
|
self.program_args = args.get("<arg>", None) or []
|
||||||
|
if self.program is not None:
|
||||||
|
self.program_args.insert(0, self.program)
|
||||||
|
self.program_args.extend(program_args)
|
||||||
|
self.no_id = args.get("--no-id", None) or False
|
||||||
|
self.mirror = args.get("--mirror", None) or False
|
||||||
|
self.timeout = args.get("--timeout", None) or RNS.Transport.PATH_REQUEST_TIMEOUT
|
||||||
|
self.destination = args.get("<destination_hash>", None)
|
||||||
|
self.help = args.get("--help", None) or False
|
||||||
|
except Exception as e:
|
||||||
|
print("Error parsing arguments: {e}")
|
||||||
|
print()
|
||||||
|
print(usage)
|
||||||
|
sys.exit(1)
|
||||||
|
|
||||||
|
if self.help:
|
||||||
|
sys.exit(0)
|
@ -477,7 +477,8 @@ class CallbackSubprocess:
|
|||||||
for c in p.connections(kind='all'):
|
for c in p.connections(kind='all'):
|
||||||
with exception.permit(SystemExit):
|
with exception.permit(SystemExit):
|
||||||
os.close(c.fd)
|
os.close(c.fd)
|
||||||
os.setpgrp()
|
# TODO: verify that skipping setpgrp fixes Operation not permitted on Manjaro
|
||||||
|
# os.setpgrp()
|
||||||
os.execvpe(program, self._command, env)
|
os.execvpe(program, self._command, env)
|
||||||
except Exception as err:
|
except Exception as err:
|
||||||
print(f"Child process error: {err}, command: {self._command}")
|
print(f"Child process error: {err}, command: {self._command}")
|
||||||
|
242
rnsh/rnsh.py
242
rnsh/rnsh.py
@ -30,6 +30,7 @@ import functools
|
|||||||
import importlib.metadata
|
import importlib.metadata
|
||||||
import logging as __logging
|
import logging as __logging
|
||||||
import os
|
import os
|
||||||
|
import shlex
|
||||||
import signal
|
import signal
|
||||||
import sys
|
import sys
|
||||||
import termios
|
import termios
|
||||||
@ -45,6 +46,8 @@ import rnsh.rnslogging as rnslogging
|
|||||||
import rnsh.hacks as hacks
|
import rnsh.hacks as hacks
|
||||||
import re
|
import re
|
||||||
import contextlib
|
import contextlib
|
||||||
|
import rnsh.args
|
||||||
|
import pwd
|
||||||
|
|
||||||
module_logger = __logging.getLogger(__name__)
|
module_logger = __logging.getLogger(__name__)
|
||||||
|
|
||||||
@ -59,12 +62,13 @@ _identity = None
|
|||||||
_reticulum = None
|
_reticulum = None
|
||||||
_allow_all = False
|
_allow_all = False
|
||||||
_allowed_identity_hashes = []
|
_allowed_identity_hashes = []
|
||||||
_cmd: [str] = None
|
_cmd: [str] | None = None
|
||||||
DATA_AVAIL_MSG = "data available"
|
DATA_AVAIL_MSG = "data available"
|
||||||
_finished: asyncio.Event = None
|
_finished: asyncio.Event = None
|
||||||
_retry_timer: retry.RetryThread | None = None
|
_retry_timer: retry.RetryThread | None = None
|
||||||
_destination: RNS.Destination | None = None
|
_destination: RNS.Destination | None = None
|
||||||
_loop: asyncio.AbstractEventLoop | None = None
|
_loop: asyncio.AbstractEventLoop | None = None
|
||||||
|
_no_remote_command = True
|
||||||
|
|
||||||
|
|
||||||
async def _check_finished(timeout: float = 0):
|
async def _check_finished(timeout: float = 0):
|
||||||
@ -110,11 +114,11 @@ def _print_identity(configdir, identitypath, service_name, include_destination:
|
|||||||
exit(0)
|
exit(0)
|
||||||
|
|
||||||
|
|
||||||
async def _listen(configdir, command, identitypath=None, service_name="default", verbosity=0, quietness=0,
|
async def _listen(configdir, command, identitypath=None, service_name="default", verbosity=0, quietness=0, allowed=None,
|
||||||
allowed=None, disable_auth=None, announce_period=900):
|
disable_auth=None, announce_period=900, no_remote_command=True):
|
||||||
global _identity, _allow_all, _allowed_identity_hashes, _reticulum, _cmd, _destination
|
global _identity, _allow_all, _allowed_identity_hashes, _reticulum, _cmd, _destination, _no_remote_command
|
||||||
log = _get_logger("_listen")
|
log = _get_logger("_listen")
|
||||||
_cmd = command
|
|
||||||
|
|
||||||
targetloglevel = RNS.LOG_INFO + verbosity - quietness
|
targetloglevel = RNS.LOG_INFO + verbosity - quietness
|
||||||
_reticulum = RNS.Reticulum(configdir=configdir, loglevel=targetloglevel)
|
_reticulum = RNS.Reticulum(configdir=configdir, loglevel=targetloglevel)
|
||||||
@ -122,6 +126,18 @@ async def _listen(configdir, command, identitypath=None, service_name="default",
|
|||||||
_prepare_identity(identitypath)
|
_prepare_identity(identitypath)
|
||||||
_destination = RNS.Destination(_identity, RNS.Destination.IN, RNS.Destination.SINGLE, APP_NAME, service_name)
|
_destination = RNS.Destination(_identity, RNS.Destination.IN, RNS.Destination.SINGLE, APP_NAME, service_name)
|
||||||
|
|
||||||
|
_cmd = command
|
||||||
|
if _cmd is None or len(_cmd) == 0:
|
||||||
|
shell = pwd.getpwuid(os.getuid()).pw_shell
|
||||||
|
log.info(f"Using {shell} for default command.")
|
||||||
|
_cmd = [shell]
|
||||||
|
else:
|
||||||
|
log.info(f"Using command {shlex.join(_cmd)}")
|
||||||
|
|
||||||
|
_no_remote_command = no_remote_command
|
||||||
|
if _cmd is None and _no_remote_command:
|
||||||
|
raise Exception(f"Unable to look up shell for {os.getlogin}, cannot proceed with -C and no <program>.")
|
||||||
|
|
||||||
if disable_auth:
|
if disable_auth:
|
||||||
_allow_all = True
|
_allow_all = True
|
||||||
else:
|
else:
|
||||||
@ -202,6 +218,7 @@ def _protocol_make_version(version: int):
|
|||||||
|
|
||||||
|
|
||||||
_PROTOCOL_VERSION_0 = _protocol_make_version(0)
|
_PROTOCOL_VERSION_0 = _protocol_make_version(0)
|
||||||
|
_PROTOCOL_VERSION_1 = _protocol_make_version(1)
|
||||||
|
|
||||||
|
|
||||||
def _protocol_split_version(version: int):
|
def _protocol_split_version(version: int):
|
||||||
@ -252,6 +269,7 @@ class Session:
|
|||||||
loop=loop,
|
loop=loop,
|
||||||
stdout_callback=self._stdout_data,
|
stdout_callback=self._stdout_data,
|
||||||
terminated_callback=terminated_callback)
|
terminated_callback=terminated_callback)
|
||||||
|
self._log.debug(f"Starting {cmd}")
|
||||||
self._data_buffer = bytearray()
|
self._data_buffer = bytearray()
|
||||||
self._lock = threading.RLock()
|
self._lock = threading.RLock()
|
||||||
self._data_available_cb = data_available_callback
|
self._data_available_cb = data_available_callback
|
||||||
@ -335,13 +353,14 @@ class Session:
|
|||||||
REQUEST_IDX_COLS = 5
|
REQUEST_IDX_COLS = 5
|
||||||
REQUEST_IDX_HPIX = 6
|
REQUEST_IDX_HPIX = 6
|
||||||
REQUEST_IDX_VPIX = 7
|
REQUEST_IDX_VPIX = 7
|
||||||
|
REQUEST_IDX_CMD = 8
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def default_request(stdin_fd: int | None) -> [any]:
|
def default_request(stdin_fd: int | None) -> [any]:
|
||||||
global _tr
|
global _tr
|
||||||
global _PROTOCOL_VERSION_0
|
global _PROTOCOL_VERSION_1
|
||||||
request: list[any] = [
|
request: list[any] = [
|
||||||
_PROTOCOL_VERSION_0, # 0 Protocol Version
|
_PROTOCOL_VERSION_1, # 0 Protocol Version
|
||||||
None, # 1 Stdin
|
None, # 1 Stdin
|
||||||
None, # 2 TERM variable
|
None, # 2 TERM variable
|
||||||
None, # 3 termios attributes or something
|
None, # 3 termios attributes or something
|
||||||
@ -349,6 +368,7 @@ class Session:
|
|||||||
None, # 5 terminal cols
|
None, # 5 terminal cols
|
||||||
None, # 6 terminal horizontal pixels
|
None, # 6 terminal horizontal pixels
|
||||||
None, # 7 terminal vertical pixels
|
None, # 7 terminal vertical pixels
|
||||||
|
None, # 8 Command to run
|
||||||
].copy()
|
].copy()
|
||||||
|
|
||||||
if stdin_fd is not None:
|
if stdin_fd is not None:
|
||||||
@ -406,10 +426,9 @@ class Session:
|
|||||||
RESPONSE_IDX_TMSTAMP = 5
|
RESPONSE_IDX_TMSTAMP = 5
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def default_response() -> [any]:
|
def default_response(version: int = _PROTOCOL_VERSION_1) -> [any]:
|
||||||
global _PROTOCOL_VERSION_0
|
|
||||||
response: list[any] = [
|
response: list[any] = [
|
||||||
_PROTOCOL_VERSION_0, # 0: Protocol version
|
version, # 0: Protocol version
|
||||||
False, # 1: Process running
|
False, # 1: Process running
|
||||||
None, # 2: Return value
|
None, # 2: Return value
|
||||||
0, # 3: Number of outstanding bytes
|
0, # 3: Number of outstanding bytes
|
||||||
@ -419,6 +438,14 @@ class Session:
|
|||||||
response[Session.RESPONSE_IDX_TMSTAMP] = time.time()
|
response[Session.RESPONSE_IDX_TMSTAMP] = time.time()
|
||||||
return response
|
return response
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def error_response(cls, msg: str, version: int = _PROTOCOL_VERSION_1) -> [any]:
|
||||||
|
response = cls.default_response(version)
|
||||||
|
response[Session.RESPONSE_IDX_STDOUT] = base64.b64encode( f"{msg}\r\n".encode("utf-8"))
|
||||||
|
response[Session.RESPONSE_IDX_RETCODE] = 255
|
||||||
|
response[Session.RESPONSE_IDX_RDYBYTE] = 0
|
||||||
|
return response
|
||||||
|
|
||||||
|
|
||||||
def _subproc_data_ready(link: RNS.Link, chars_available: int):
|
def _subproc_data_ready(link: RNS.Link, chars_available: int):
|
||||||
global _retry_timer
|
global _retry_timer
|
||||||
@ -497,13 +524,12 @@ def _subproc_terminated(link: RNS.Link, return_code: int):
|
|||||||
_loop.call_soon_threadsafe(cleanup)
|
_loop.call_soon_threadsafe(cleanup)
|
||||||
|
|
||||||
|
|
||||||
def _listen_start_proc(link: RNS.Link, remote_identity: str | None, term: str,
|
def _listen_start_proc(link: RNS.Link, remote_identity: str | None, term: str, cmd: str | None,
|
||||||
loop: asyncio.AbstractEventLoop) -> Session | None:
|
loop: asyncio.AbstractEventLoop) -> Session | None:
|
||||||
global _cmd
|
|
||||||
log = _get_logger("_listen_start_proc")
|
log = _get_logger("_listen_start_proc")
|
||||||
try:
|
try:
|
||||||
return Session(tag=link.link_id,
|
return Session(tag=link.link_id,
|
||||||
cmd=_cmd,
|
cmd=cmd,
|
||||||
term=term,
|
term=term,
|
||||||
remote_identity=remote_identity,
|
remote_identity=remote_identity,
|
||||||
mdu=link.MDU,
|
mdu=link.MDU,
|
||||||
@ -550,7 +576,7 @@ def _initiator_identified(link, identity):
|
|||||||
|
|
||||||
|
|
||||||
def _listen_request(path, data, request_id, link_id, remote_identity, requested_at):
|
def _listen_request(path, data, request_id, link_id, remote_identity, requested_at):
|
||||||
global _destination, _retry_timer, _loop
|
global _destination, _retry_timer, _loop, _cmd, _no_remote_command
|
||||||
log = _get_logger("_listen_request")
|
log = _get_logger("_listen_request")
|
||||||
log.debug(
|
log.debug(
|
||||||
f"listen_execute {path} {RNS.prettyhexrep(request_id)} {RNS.prettyhexrep(link_id)} {remote_identity}, {requested_at}")
|
f"listen_execute {path} {RNS.prettyhexrep(request_id)} {RNS.prettyhexrep(link_id)} {remote_identity}, {requested_at}")
|
||||||
@ -565,13 +591,19 @@ def _listen_request(path, data, request_id, link_id, remote_identity, requested_
|
|||||||
if not _protocol_check_magic(remote_version):
|
if not _protocol_check_magic(remote_version):
|
||||||
raise Exception("Request magic incorrect")
|
raise Exception("Request magic incorrect")
|
||||||
|
|
||||||
if not remote_version == _PROTOCOL_VERSION_0:
|
if not remote_version == _PROTOCOL_VERSION_0 and not remote_version == _PROTOCOL_VERSION_1:
|
||||||
response = Session.default_response()
|
return Session.error_response("Listener<->initiator version mismatch")
|
||||||
response[Session.RESPONSE_IDX_STDOUT] = base64.b64encode(
|
|
||||||
"Listener<->initiator version mismatch\r\n".encode("utf-8"))
|
cmd = _cmd
|
||||||
response[Session.RESPONSE_IDX_RETCODE] = 255
|
if remote_version == _PROTOCOL_VERSION_1:
|
||||||
response[Session.RESPONSE_IDX_RDYBYTE] = 0
|
remote_command = data[Session.REQUEST_IDX_CMD]
|
||||||
return response
|
if remote_command is not None and len(remote_command) > 0:
|
||||||
|
if _no_remote_command:
|
||||||
|
return Session.error_response("Listener does not permit initiator to provide command.")
|
||||||
|
cmd = remote_command
|
||||||
|
|
||||||
|
if not _no_remote_command and (cmd is None or len(cmd) == 0):
|
||||||
|
return Session.error_response("No command supplied and no default command available.")
|
||||||
|
|
||||||
session: Session | None = None
|
session: Session | None = None
|
||||||
try:
|
try:
|
||||||
@ -584,6 +616,7 @@ def _listen_request(path, data, request_id, link_id, remote_identity, requested_
|
|||||||
log.debug(f"Process not found for link {link}")
|
log.debug(f"Process not found for link {link}")
|
||||||
session = _listen_start_proc(link=link,
|
session = _listen_start_proc(link=link,
|
||||||
term=term,
|
term=term,
|
||||||
|
cmd=cmd,
|
||||||
remote_identity=RNS.hexrep(remote_identity.hash).replace(":", ""),
|
remote_identity=RNS.hexrep(remote_identity.hash).replace(":", ""),
|
||||||
loop=_loop)
|
loop=_loop)
|
||||||
|
|
||||||
@ -641,7 +674,8 @@ def _response_handler(request_receipt: RNS.RequestReceipt):
|
|||||||
|
|
||||||
|
|
||||||
async def _execute(configdir, identitypath=None, verbosity=0, quietness=0, noid=False, destination=None,
|
async def _execute(configdir, identitypath=None, verbosity=0, quietness=0, noid=False, destination=None,
|
||||||
service_name="default", stdin=None, timeout=RNS.Transport.PATH_REQUEST_TIMEOUT):
|
service_name="default", stdin=None, timeout=RNS.Transport.PATH_REQUEST_TIMEOUT,
|
||||||
|
cmd: [str] | None = None):
|
||||||
global _identity, _reticulum, _link, _destination, _remote_exec_grace, _tr, _new_data
|
global _identity, _reticulum, _link, _destination, _remote_exec_grace, _tr, _new_data
|
||||||
log = _get_logger("_execute")
|
log = _get_logger("_execute")
|
||||||
|
|
||||||
@ -699,6 +733,7 @@ async def _execute(configdir, identitypath=None, verbosity=0, quietness=0, noid=
|
|||||||
log.debug(f"Sending {len(stdin) or 0} bytes to listener")
|
log.debug(f"Sending {len(stdin) or 0} bytes to listener")
|
||||||
# log.debug(f"Sending {stdin} to listener")
|
# log.debug(f"Sending {stdin} to listener")
|
||||||
request[Session.REQUEST_IDX_STDIN] = (base64.b64encode(stdin) if stdin is not None else None)
|
request[Session.REQUEST_IDX_STDIN] = (base64.b64encode(stdin) if stdin is not None else None)
|
||||||
|
request[Session.REQUEST_IDX_CMD] = cmd
|
||||||
|
|
||||||
# TODO: Tune
|
# TODO: Tune
|
||||||
timeout = timeout + _link.rtt * 4 + _remote_exec_grace
|
timeout = timeout + _link.rtt * 4 + _remote_exec_grace
|
||||||
@ -741,7 +776,7 @@ async def _execute(configdir, identitypath=None, verbosity=0, quietness=0, noid=
|
|||||||
version = request_receipt.response[Session.RESPONSE_IDX_VERSION] or 0
|
version = request_receipt.response[Session.RESPONSE_IDX_VERSION] or 0
|
||||||
if not _protocol_check_magic(version):
|
if not _protocol_check_magic(version):
|
||||||
raise RemoteExecutionError("Protocol error")
|
raise RemoteExecutionError("Protocol error")
|
||||||
elif version != _PROTOCOL_VERSION_0:
|
elif version != _PROTOCOL_VERSION_0 and version != _PROTOCOL_VERSION_1:
|
||||||
raise RemoteExecutionError("Protocol version mismatch")
|
raise RemoteExecutionError("Protocol version mismatch")
|
||||||
|
|
||||||
running = request_receipt.response[Session.RESPONSE_IDX_RUNNING] or True
|
running = request_receipt.response[Session.RESPONSE_IDX_RUNNING] or True
|
||||||
@ -775,13 +810,18 @@ async def _execute(configdir, identitypath=None, verbosity=0, quietness=0, noid=
|
|||||||
|
|
||||||
return None
|
return None
|
||||||
|
|
||||||
|
_pre_input = bytearray()
|
||||||
|
|
||||||
|
|
||||||
async def _initiate(configdir: str, identitypath: str, verbosity: int, quietness: int, noid: bool, destination: str,
|
async def _initiate(configdir: str, identitypath: str, verbosity: int, quietness: int, noid: bool, destination: str,
|
||||||
service_name: str, timeout: float):
|
service_name: str, timeout: float, command: [str] | None = None):
|
||||||
global _new_data, _finished, _tr
|
global _new_data, _finished, _tr, _cmd, _pre_input
|
||||||
log = _get_logger("_initiate")
|
log = _get_logger("_initiate")
|
||||||
loop = asyncio.get_running_loop()
|
loop = asyncio.get_running_loop()
|
||||||
_new_data = asyncio.Event()
|
_new_data = asyncio.Event()
|
||||||
|
command = command
|
||||||
|
if command is not None and len(command) == 1:
|
||||||
|
command = shlex.split(command[0])
|
||||||
|
|
||||||
data_buffer = bytearray(sys.stdin.buffer.read()) if not os.isatty(sys.stdin.fileno()) else bytearray()
|
data_buffer = bytearray(sys.stdin.buffer.read()) if not os.isatty(sys.stdin.fileno()) else bytearray()
|
||||||
|
|
||||||
@ -827,6 +867,7 @@ async def _initiate(configdir: str, identitypath: str, verbosity: int, quietness
|
|||||||
service_name=service_name,
|
service_name=service_name,
|
||||||
stdin=stdin,
|
stdin=stdin,
|
||||||
timeout=timeout,
|
timeout=timeout,
|
||||||
|
cmd=command,
|
||||||
)
|
)
|
||||||
|
|
||||||
if first_loop:
|
if first_loop:
|
||||||
@ -854,17 +895,6 @@ async def _initiate(configdir: str, identitypath: str, verbosity: int, quietness
|
|||||||
await process.event_wait_any([_new_data, _finished], timeout=min(max(rtt * 50, 5), 120))
|
await process.event_wait_any([_new_data, _finished], timeout=min(max(rtt * 50, 5), 120))
|
||||||
|
|
||||||
|
|
||||||
_T = TypeVar("_T")
|
|
||||||
|
|
||||||
|
|
||||||
def _split_array_at(arr: [_T], at: _T) -> ([_T], [_T]):
|
|
||||||
try:
|
|
||||||
idx = arr.index(at)
|
|
||||||
return arr[:idx], arr[idx + 1:]
|
|
||||||
except ValueError:
|
|
||||||
return arr, []
|
|
||||||
|
|
||||||
|
|
||||||
def _loop_set_signal(sig, loop):
|
def _loop_set_signal(sig, loop):
|
||||||
loop.remove_signal_handler(sig)
|
loop.remove_signal_handler(sig)
|
||||||
loop.add_signal_handler(sig, functools.partial(_sigint_handler, sig, None))
|
loop.add_signal_handler(sig, functools.partial(_sigint_handler, sig, None))
|
||||||
@ -879,134 +909,58 @@ async def _rnsh_cli_main():
|
|||||||
_finished = asyncio.Event()
|
_finished = asyncio.Event()
|
||||||
_loop_set_signal(signal.SIGINT, _loop)
|
_loop_set_signal(signal.SIGINT, _loop)
|
||||||
_loop_set_signal(signal.SIGTERM, _loop)
|
_loop_set_signal(signal.SIGTERM, _loop)
|
||||||
usage = '''
|
|
||||||
Usage:
|
|
||||||
rnsh [--config <configdir>] [-i <identityfile>] [-s <service_name>] [-l] -p
|
|
||||||
rnsh -l [--config <configfile>] [-i <identityfile>] [-s <service_name>]
|
|
||||||
[-v... | -q...] [-b <period>] (-n | -a <identity_hash> [-a <identity_hash>] ...)
|
|
||||||
[--] <program> [<arg> ...]
|
|
||||||
rnsh [--config <configfile>] [-i <identityfile>] [-s <service_name>]
|
|
||||||
[-v... | -q...] [-N] [-m] [-w <timeout>] <destination_hash>
|
|
||||||
rnsh -h
|
|
||||||
rnsh --version
|
|
||||||
|
|
||||||
Options:
|
args = rnsh.args.Args(sys.argv)
|
||||||
--config DIR Alternate Reticulum config directory to use
|
|
||||||
-i FILE --identity FILE Specific identity file to use
|
|
||||||
-s NAME --service NAME Listen on/connect to specific service name if not default
|
|
||||||
-p --print-identity Print identity information and exit
|
|
||||||
-l --listen Listen (server) mode
|
|
||||||
-b --announce PERIOD Announce on startup and every PERIOD seconds
|
|
||||||
Specify 0 for PERIOD to announce on startup only.
|
|
||||||
-a HASH --allowed HASH Specify identities allowed to connect
|
|
||||||
-n --no-auth Disable authentication
|
|
||||||
-N --no-id Disable identify on connect
|
|
||||||
-m --mirror Client returns with code of remote process
|
|
||||||
-w TIME --timeout TIME Specify client connect and request timeout in seconds
|
|
||||||
-q --quiet Increase quietness (move level up), multiple increases effect
|
|
||||||
DEFAULT LOGGING LEVEL
|
|
||||||
CRITICAL (silent)
|
|
||||||
Initiator -> ERROR
|
|
||||||
WARNING
|
|
||||||
Listener -> INFO
|
|
||||||
DEBUG (insane)
|
|
||||||
-v --verbose Increase verbosity (move level down), multiple increases effect
|
|
||||||
--version Show version
|
|
||||||
-h --help Show this help
|
|
||||||
'''
|
|
||||||
|
|
||||||
argv, program_args = _split_array_at(sys.argv, "--")
|
if args.print_identity:
|
||||||
if len(program_args) > 0:
|
_print_identity(args.config, args.identity, args.service_name, args.listen)
|
||||||
argv.append(program_args[0])
|
|
||||||
program_args = program_args[1:]
|
|
||||||
|
|
||||||
args = docopt.docopt(usage, argv=argv[1:], version=f"rnsh {importlib.metadata.version('rnsh')}")
|
|
||||||
# json.dump(args, sys.stdout)
|
|
||||||
|
|
||||||
args_service_name = args.get("--service", None) or "default"
|
|
||||||
args_listen = args.get("--listen", None) or False
|
|
||||||
args_identity = args.get("--identity", None)
|
|
||||||
args_config = args.get("--config", None)
|
|
||||||
args_print_identity = args.get("--print-identity", None) or False
|
|
||||||
args_verbose = args.get("--verbose", None) or 0
|
|
||||||
args_quiet = args.get("--quiet", None) or 0
|
|
||||||
args_announce = args.get("--announce", None)
|
|
||||||
try:
|
|
||||||
if args_announce:
|
|
||||||
args_announce = int(args_announce)
|
|
||||||
except ValueError:
|
|
||||||
print("Invalid value for --announce")
|
|
||||||
return 1
|
|
||||||
args_no_auth = args.get("--no-auth", None) or False
|
|
||||||
args_allowed = args.get("--allowed", None) or []
|
|
||||||
args_program = args.get("<program>", None)
|
|
||||||
args_program_args = args.get("<arg>", None) or []
|
|
||||||
args_program_args.insert(0, args_program)
|
|
||||||
args_program_args.extend(program_args)
|
|
||||||
args_no_id = args.get("--no-id", None) or False
|
|
||||||
args_mirror = args.get("--mirror", None) or False
|
|
||||||
args_timeout = args.get("--timeout", None) or RNS.Transport.PATH_REQUEST_TIMEOUT
|
|
||||||
args_destination = args.get("<destination_hash>", None)
|
|
||||||
args_help = args.get("--help", None) or False
|
|
||||||
|
|
||||||
if args_help:
|
|
||||||
return 0
|
return 0
|
||||||
|
|
||||||
if args_print_identity:
|
if args.listen:
|
||||||
_print_identity(args_config, args_identity, args_service_name, args_listen)
|
|
||||||
return 0
|
|
||||||
|
|
||||||
if args_listen:
|
|
||||||
# log.info("command " + args.command)
|
# log.info("command " + args.command)
|
||||||
await _listen(
|
await _listen(configdir=args.config,
|
||||||
configdir=args_config,
|
command=args.program_args,
|
||||||
command=args_program_args,
|
identitypath=args.identity,
|
||||||
identitypath=args_identity,
|
service_name=args.service_name,
|
||||||
service_name=args_service_name,
|
verbosity=args.verbose,
|
||||||
verbosity=args_verbose,
|
quietness=args.quiet,
|
||||||
quietness=args_quiet,
|
allowed=args.allowed,
|
||||||
allowed=args_allowed,
|
disable_auth=args.no_auth,
|
||||||
disable_auth=args_no_auth,
|
announce_period=args.announce,
|
||||||
announce_period=args_announce,
|
no_remote_command=args.no_remote_cmd)
|
||||||
)
|
return 0
|
||||||
|
|
||||||
if args_destination is not None and args_service_name is not None:
|
if args.destination is not None and args.service_name is not None:
|
||||||
return_code = await _initiate(
|
return_code = await _initiate(
|
||||||
configdir=args_config,
|
configdir=args.config,
|
||||||
identitypath=args_identity,
|
identitypath=args.identity,
|
||||||
verbosity=args_verbose,
|
verbosity=args.verbose,
|
||||||
quietness=args_quiet,
|
quietness=args.quiet,
|
||||||
noid=args_no_id,
|
noid=args.no_id,
|
||||||
destination=args_destination,
|
destination=args.destination,
|
||||||
service_name=args_service_name,
|
service_name=args.service_name,
|
||||||
timeout=args_timeout,
|
timeout=args.timeout,
|
||||||
|
command=args.program_args
|
||||||
)
|
)
|
||||||
return return_code if args_mirror else 0
|
return return_code if args.mirror else 0
|
||||||
else:
|
else:
|
||||||
print("")
|
print("")
|
||||||
print(args)
|
print(args.usage)
|
||||||
print("")
|
print("")
|
||||||
|
return 1
|
||||||
|
|
||||||
def _noop():
|
|
||||||
pass
|
|
||||||
|
|
||||||
# RNS.exit = _noop
|
|
||||||
|
|
||||||
|
|
||||||
def rnsh_cli():
|
def rnsh_cli():
|
||||||
global _tr, _retry_timer
|
global _tr, _retry_timer, _pre_input
|
||||||
with contextlib.suppress(Exception):
|
with contextlib.suppress(Exception):
|
||||||
if not os.isatty(sys.stdin.fileno()):
|
if not os.isatty(sys.stdin.fileno()):
|
||||||
|
time.sleep(0.1) # attempting to deal with an issue with missing input
|
||||||
tty.setraw(sys.stdin.fileno(), termios.TCSANOW)
|
tty.setraw(sys.stdin.fileno(), termios.TCSANOW)
|
||||||
|
|
||||||
with process.TTYRestorer(sys.stdin.fileno()) as _tr, retry.RetryThread() as _retry_timer:
|
with process.TTYRestorer(sys.stdin.fileno()) as _tr, retry.RetryThread() as _retry_timer:
|
||||||
return_code = asyncio.run(_rnsh_cli_main())
|
return_code = asyncio.run(_rnsh_cli_main())
|
||||||
|
|
||||||
with exception.permit(SystemExit):
|
process.tty_unset_reader_callbacks(sys.stdin.fileno())
|
||||||
process.tty_unset_reader_callbacks(sys.stdin.fileno())
|
|
||||||
|
|
||||||
# RNS.Reticulum.exit_handler()
|
|
||||||
# time.sleep(0.5)
|
|
||||||
sys.exit(return_code if return_code is not None else 255)
|
sys.exit(return_code if return_code is not None else 255)
|
||||||
|
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user