move ssh cli to cli submodule
This commit is contained in:
@@ -1,8 +1,9 @@
|
|||||||
import argparse
|
import argparse
|
||||||
import sys
|
import sys
|
||||||
|
|
||||||
from . import admin, secrets, ssh
|
from . import admin, secrets
|
||||||
from .errors import ClanError
|
from .errors import ClanError
|
||||||
|
from .ssh import cli as ssh_cli
|
||||||
|
|
||||||
has_argcomplete = True
|
has_argcomplete = True
|
||||||
try:
|
try:
|
||||||
@@ -27,7 +28,7 @@ def main() -> None:
|
|||||||
# warn(f"The config command does not work in the nix sandbox: {e}")
|
# warn(f"The config command does not work in the nix sandbox: {e}")
|
||||||
|
|
||||||
parser_ssh = subparsers.add_parser("ssh", help="ssh to a remote machine")
|
parser_ssh = subparsers.add_parser("ssh", help="ssh to a remote machine")
|
||||||
ssh.register_parser(parser_ssh)
|
ssh_cli.register_parser(parser_ssh)
|
||||||
|
|
||||||
parser_secrets = subparsers.add_parser("secrets", help="manage secrets")
|
parser_secrets = subparsers.add_parser("secrets", help="manage secrets")
|
||||||
secrets.register_parser(parser_secrets)
|
secrets.register_parser(parser_secrets)
|
||||||
|
|||||||
@@ -144,7 +144,7 @@ class HostKeyCheck(Enum):
|
|||||||
NONE = 2
|
NONE = 2
|
||||||
|
|
||||||
|
|
||||||
class DeployHost:
|
class Host:
|
||||||
def __init__(
|
def __init__(
|
||||||
self,
|
self,
|
||||||
host: str,
|
host: str,
|
||||||
@@ -158,7 +158,7 @@ class DeployHost:
|
|||||||
verbose_ssh: bool = False,
|
verbose_ssh: bool = False,
|
||||||
) -> None:
|
) -> None:
|
||||||
"""
|
"""
|
||||||
Creates a DeployHost
|
Creates a Host
|
||||||
@host the hostname to connect to via ssh
|
@host the hostname to connect to via ssh
|
||||||
@port the port to connect to via ssh
|
@port the port to connect to via ssh
|
||||||
@forward_agent: wheter to forward ssh agent
|
@forward_agent: wheter to forward ssh agent
|
||||||
@@ -495,7 +495,7 @@ T = TypeVar("T")
|
|||||||
|
|
||||||
|
|
||||||
class HostResult(Generic[T]):
|
class HostResult(Generic[T]):
|
||||||
def __init__(self, host: DeployHost, result: Union[T, Exception]) -> None:
|
def __init__(self, host: Host, result: Union[T, Exception]) -> None:
|
||||||
self.host = host
|
self.host = host
|
||||||
self._result = result
|
self._result = result
|
||||||
|
|
||||||
@@ -518,12 +518,12 @@ class HostResult(Generic[T]):
|
|||||||
return self._result
|
return self._result
|
||||||
|
|
||||||
|
|
||||||
DeployResults = List[HostResult[subprocess.CompletedProcess[str]]]
|
Results = List[HostResult[subprocess.CompletedProcess[str]]]
|
||||||
|
|
||||||
|
|
||||||
def _worker(
|
def _worker(
|
||||||
func: Callable[[DeployHost], T],
|
func: Callable[[Host], T],
|
||||||
host: DeployHost,
|
host: Host,
|
||||||
results: List[HostResult[T]],
|
results: List[HostResult[T]],
|
||||||
idx: int,
|
idx: int,
|
||||||
) -> None:
|
) -> None:
|
||||||
@@ -534,15 +534,15 @@ def _worker(
|
|||||||
results[idx] = HostResult(host, e)
|
results[idx] = HostResult(host, e)
|
||||||
|
|
||||||
|
|
||||||
class DeployGroup:
|
class Group:
|
||||||
def __init__(self, hosts: List[DeployHost]) -> None:
|
def __init__(self, hosts: List[Host]) -> None:
|
||||||
self.hosts = hosts
|
self.hosts = hosts
|
||||||
|
|
||||||
def _run_local(
|
def _run_local(
|
||||||
self,
|
self,
|
||||||
cmd: Union[str, List[str]],
|
cmd: Union[str, List[str]],
|
||||||
host: DeployHost,
|
host: Host,
|
||||||
results: DeployResults,
|
results: Results,
|
||||||
stdout: FILE = None,
|
stdout: FILE = None,
|
||||||
stderr: FILE = None,
|
stderr: FILE = None,
|
||||||
extra_env: Dict[str, str] = {},
|
extra_env: Dict[str, str] = {},
|
||||||
@@ -569,8 +569,8 @@ class DeployGroup:
|
|||||||
def _run_remote(
|
def _run_remote(
|
||||||
self,
|
self,
|
||||||
cmd: Union[str, List[str]],
|
cmd: Union[str, List[str]],
|
||||||
host: DeployHost,
|
host: Host,
|
||||||
results: DeployResults,
|
results: Results,
|
||||||
stdout: FILE = None,
|
stdout: FILE = None,
|
||||||
stderr: FILE = None,
|
stderr: FILE = None,
|
||||||
extra_env: Dict[str, str] = {},
|
extra_env: Dict[str, str] = {},
|
||||||
@@ -621,8 +621,8 @@ class DeployGroup:
|
|||||||
check: bool = True,
|
check: bool = True,
|
||||||
verbose_ssh: bool = False,
|
verbose_ssh: bool = False,
|
||||||
timeout: float = math.inf,
|
timeout: float = math.inf,
|
||||||
) -> DeployResults:
|
) -> Results:
|
||||||
results: DeployResults = []
|
results: Results = []
|
||||||
threads = []
|
threads = []
|
||||||
for host in self.hosts:
|
for host in self.hosts:
|
||||||
fn = self._run_local if local else self._run_remote
|
fn = self._run_local if local else self._run_remote
|
||||||
@@ -662,7 +662,7 @@ class DeployGroup:
|
|||||||
check: bool = True,
|
check: bool = True,
|
||||||
verbose_ssh: bool = False,
|
verbose_ssh: bool = False,
|
||||||
timeout: float = math.inf,
|
timeout: float = math.inf,
|
||||||
) -> DeployResults:
|
) -> Results:
|
||||||
"""
|
"""
|
||||||
Command to run on the remote host via ssh
|
Command to run on the remote host via ssh
|
||||||
@stdout if not None stdout of the command will be redirected to this file i.e. stdout=subprocss.PIPE
|
@stdout if not None stdout of the command will be redirected to this file i.e. stdout=subprocss.PIPE
|
||||||
@@ -671,7 +671,7 @@ class DeployGroup:
|
|||||||
@verbose_ssh: Enables verbose logging on ssh connections
|
@verbose_ssh: Enables verbose logging on ssh connections
|
||||||
@timeout: Timeout in seconds for the command to complete
|
@timeout: Timeout in seconds for the command to complete
|
||||||
|
|
||||||
@return a lists of tuples containing DeployNode and the result of the command for this DeployNode
|
@return a lists of tuples containing Host and the result of the command for this Host
|
||||||
"""
|
"""
|
||||||
return self._run(
|
return self._run(
|
||||||
cmd,
|
cmd,
|
||||||
@@ -693,7 +693,7 @@ class DeployGroup:
|
|||||||
cwd: Union[None, str, Path] = None,
|
cwd: Union[None, str, Path] = None,
|
||||||
check: bool = True,
|
check: bool = True,
|
||||||
timeout: float = math.inf,
|
timeout: float = math.inf,
|
||||||
) -> DeployResults:
|
) -> Results:
|
||||||
"""
|
"""
|
||||||
Command to run locally for each host in the group in parallel
|
Command to run locally for each host in the group in parallel
|
||||||
@cmd the commmand to run
|
@cmd the commmand to run
|
||||||
@@ -703,7 +703,7 @@ class DeployGroup:
|
|||||||
@extra_env environment variables to override whe running the command
|
@extra_env environment variables to override whe running the command
|
||||||
@timeout: Timeout in seconds for the command to complete
|
@timeout: Timeout in seconds for the command to complete
|
||||||
|
|
||||||
@return a lists of tuples containing DeployNode and the result of the command for this DeployNode
|
@return a lists of tuples containing Host and the result of the command for this Host
|
||||||
"""
|
"""
|
||||||
return self._run(
|
return self._run(
|
||||||
cmd,
|
cmd,
|
||||||
@@ -717,7 +717,7 @@ class DeployGroup:
|
|||||||
)
|
)
|
||||||
|
|
||||||
def run_function(
|
def run_function(
|
||||||
self, func: Callable[[DeployHost], T], check: bool = True
|
self, func: Callable[[Host], T], check: bool = True
|
||||||
) -> List[HostResult[T]]:
|
) -> List[HostResult[T]]:
|
||||||
"""
|
"""
|
||||||
Function to run for each host in the group in parallel
|
Function to run for each host in the group in parallel
|
||||||
@@ -745,9 +745,9 @@ class DeployGroup:
|
|||||||
self._reraise_errors(results)
|
self._reraise_errors(results)
|
||||||
return results
|
return results
|
||||||
|
|
||||||
def filter(self, pred: Callable[[DeployHost], bool]) -> "DeployGroup":
|
def filter(self, pred: Callable[[Host], bool]) -> "Group":
|
||||||
"""Return a new DeployGroup with the results filtered by the predicate"""
|
"""Return a new Group with the results filtered by the predicate"""
|
||||||
return DeployGroup(list(filter(pred, self.hosts)))
|
return Group(list(filter(pred, self.hosts)))
|
||||||
|
|
||||||
|
|
||||||
@overload
|
@overload
|
||||||
@@ -3,7 +3,7 @@ import json
|
|||||||
import subprocess
|
import subprocess
|
||||||
from typing import Optional
|
from typing import Optional
|
||||||
|
|
||||||
from .nix import nix_shell
|
from ..nix import nix_shell
|
||||||
|
|
||||||
|
|
||||||
def ssh(
|
def ssh(
|
||||||
@@ -6,7 +6,7 @@ import pytest_subprocess.fake_process
|
|||||||
from environment import mock_env
|
from environment import mock_env
|
||||||
from pytest_subprocess import utils
|
from pytest_subprocess import utils
|
||||||
|
|
||||||
import clan_cli.ssh
|
from clan_cli.ssh import cli
|
||||||
|
|
||||||
|
|
||||||
def test_no_args(
|
def test_no_args(
|
||||||
@@ -40,7 +40,7 @@ def test_ssh_no_pass(fp: pytest_subprocess.fake_process.FakeProcess) -> None:
|
|||||||
fp.any(),
|
fp.any(),
|
||||||
]
|
]
|
||||||
fp.register(cmd)
|
fp.register(cmd)
|
||||||
clan_cli.ssh.ssh(
|
cli.ssh(
|
||||||
host=host,
|
host=host,
|
||||||
user=user,
|
user=user,
|
||||||
)
|
)
|
||||||
@@ -64,7 +64,7 @@ def test_ssh_with_pass(fp: pytest_subprocess.fake_process.FakeProcess) -> None:
|
|||||||
fp.any(),
|
fp.any(),
|
||||||
]
|
]
|
||||||
fp.register(cmd)
|
fp.register(cmd)
|
||||||
clan_cli.ssh.ssh(
|
cli.ssh(
|
||||||
host=host,
|
host=host,
|
||||||
user=user,
|
user=user,
|
||||||
password="XXX",
|
password="XXX",
|
||||||
@@ -75,5 +75,5 @@ def test_ssh_with_pass(fp: pytest_subprocess.fake_process.FakeProcess) -> None:
|
|||||||
def test_qrcode_scan(fp: pytest_subprocess.fake_process.FakeProcess) -> None:
|
def test_qrcode_scan(fp: pytest_subprocess.fake_process.FakeProcess) -> None:
|
||||||
cmd: list[Union[str, utils.Any]] = [fp.any()]
|
cmd: list[Union[str, utils.Any]] = [fp.any()]
|
||||||
fp.register(cmd, stdout="https://test.test")
|
fp.register(cmd, stdout="https://test.test")
|
||||||
result = clan_cli.ssh.qrcode_scan("test.png")
|
result = cli.qrcode_scan("test.png")
|
||||||
assert result == "https://test.test"
|
assert result == "https://test.test"
|
||||||
|
|||||||
Reference in New Issue
Block a user