-
Notifications
You must be signed in to change notification settings - Fork 1
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Signed-off-by: Tobias Wolf <[email protected]>
- Loading branch information
1 parent
d92f3b6
commit 7c946d5
Showing
12 changed files
with
275 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -11,4 +11,5 @@ jobs: | |
steps: | ||
- uses: actions/checkout@v3 | ||
- uses: actions/setup-python@v3 | ||
- run: python -m pip install .[tests] | ||
- uses: pre-commit/[email protected] |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -3,3 +3,7 @@ platforms = any | |
|
||
[options] | ||
install_requires=file:requirements.txt | ||
|
||
[options.extras_require] | ||
tests = | ||
pytest==8.0.2 |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,29 @@ | ||
# -*- coding: utf-8 -*- | ||
|
||
import json | ||
from collections.abc import Callable | ||
from rookify.modules.module import ModuleException | ||
from typing import Any, Dict, List, Tuple | ||
|
||
|
||
class MockCeph(object): | ||
def __init__( | ||
self, | ||
config: Dict[str, Any], | ||
_callable: Callable[[str, bytes], Tuple[int, bytes, str]], | ||
): | ||
if not callable(_callable): | ||
raise RuntimeError("Handler function given is invalid") | ||
|
||
self._callback_handler = _callable | ||
|
||
def mon_command( | ||
self, command: str, inbuf: bytes, **kwargs: Any | ||
) -> Dict[str, Any] | List[Any]: | ||
ret, outbuf, outstr = self._callback_handler(command, inbuf, **kwargs) | ||
if ret != 0: | ||
raise ModuleException("Ceph did return an error: {0!r}".format(outbuf)) | ||
|
||
data = json.loads(outbuf) | ||
assert isinstance(data, dict) or isinstance(data, list) | ||
return data |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,23 @@ | ||
# -*- coding: utf-8 -*- | ||
|
||
from collections.abc import Callable | ||
from typing import Any, Dict, List | ||
|
||
|
||
class MockK8s(object): | ||
def __init__(self, _callable: Callable[[str], Any], name: str = "") -> None: | ||
if not callable(_callable): | ||
raise RuntimeError("Handler function given is invalid") | ||
|
||
self._callback_handler = _callable | ||
self._attr_name = name | ||
|
||
def __call__(self, *args: List[Any], **kwargs: Dict[Any, Any]) -> Any: | ||
return self._callback_handler(self._attr_name, *args, **kwargs) | ||
|
||
def __getattr__(self, name: str) -> Any: | ||
attr_name = ( | ||
name if self._attr_name == "" else "{0}.{1}".format(self._attr_name, name) | ||
) | ||
|
||
return MockK8s(self._callback_handler, attr_name) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,136 @@ | ||
# -*- coding: utf-8 -*- | ||
|
||
|
||
from collections.abc import Callable | ||
from socket import AF_INET, IPPROTO_TCP, SO_REUSEADDR, SOCK_STREAM, SOL_SOCKET, socket | ||
from threading import Event, RLock | ||
from typing import Any, Optional | ||
|
||
from paramiko import ( # type: ignore[attr-defined] | ||
AUTH_FAILED, | ||
AUTH_SUCCESSFUL, | ||
OPEN_FAILED_ADMINISTRATIVELY_PROHIBITED, | ||
OPEN_SUCCEEDED, | ||
AutoAddPolicy, | ||
Channel, | ||
PKey, | ||
RSAKey, | ||
ServerInterface, | ||
SSHClient, | ||
Transport, | ||
) | ||
|
||
|
||
class MockSSHServer(ServerInterface): | ||
"""An ssh server accepting the pre-generated key.""" | ||
|
||
ssh_username = "pytest" | ||
ssh_key = RSAKey.generate(4096) | ||
|
||
def __init__(self, _callable: Callable[[bytes, Channel], None]) -> None: | ||
if not callable(_callable): | ||
raise RuntimeError("Handler function given is invalid") | ||
|
||
ServerInterface.__init__(self) | ||
|
||
self._callback_handler = _callable | ||
self._channel: Any = None | ||
self._client: Optional[SSHClient] = None | ||
self._command: Optional[bytes] = None | ||
self.event = Event() | ||
self._server_transport: Optional[Transport] = None | ||
self._thread_lock = RLock() | ||
|
||
def __del__(self) -> None: | ||
self.close() | ||
|
||
@property | ||
def client(self) -> SSHClient: | ||
with self._thread_lock: | ||
if self._client is None: | ||
connection_event = Event() | ||
|
||
server_socket = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP) | ||
server_socket.setsockopt(SOL_SOCKET, SO_REUSEADDR, 1) | ||
server_socket.bind(("127.0.0.1", 0)) | ||
server_socket.listen() | ||
|
||
server_address = server_socket.getsockname() | ||
|
||
client_socket = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP) | ||
client_socket.connect(server_address) | ||
|
||
(transport_socket, _) = server_socket.accept() | ||
|
||
self._server_transport = Transport(transport_socket) | ||
self._server_transport.add_server_key(self.__class__.ssh_key) | ||
self._server_transport.start_server(connection_event, self) | ||
|
||
self._client = SSHClient() | ||
self._client.set_missing_host_key_policy(AutoAddPolicy()) | ||
|
||
self._client.connect( | ||
server_address[0], | ||
server_address[1], | ||
username=self.__class__.ssh_username, | ||
pkey=self.__class__.ssh_key, | ||
sock=client_socket, | ||
) | ||
|
||
connection_event.wait() | ||
|
||
return self._client | ||
|
||
def check_channel_request(self, kind: str, chanid: int) -> int: | ||
if kind == "session": | ||
return OPEN_SUCCEEDED # type: ignore[no-any-return] | ||
return OPEN_FAILED_ADMINISTRATIVELY_PROHIBITED # type: ignore[no-any-return] | ||
|
||
def check_auth_password(self, username: str, password: str) -> int: | ||
return AUTH_FAILED # type: ignore[no-any-return] | ||
|
||
def check_auth_publickey(self, username: str, key: PKey) -> int: | ||
if username == self.__class__.ssh_username and key == self.__class__.ssh_key: | ||
return AUTH_SUCCESSFUL # type: ignore[no-any-return] | ||
return AUTH_FAILED # type: ignore[no-any-return] | ||
|
||
def check_channel_exec_request(self, channel: Channel, command: bytes) -> bool: | ||
if self.event.is_set(): | ||
return False | ||
|
||
self.event.set() | ||
|
||
with self._thread_lock: | ||
self._channel = channel | ||
self._command = command | ||
|
||
if self._callback_handler is not None: | ||
self.handle_exec_request(self._callback_handler) | ||
|
||
return True | ||
|
||
def close(self) -> None: | ||
if self._server_transport is not None: | ||
self._server_transport.close() | ||
self._server_transport = None | ||
|
||
def get_allowed_auths(self, username: str) -> str: | ||
if username == self.__class__.ssh_username: | ||
return "publickey" | ||
return "" | ||
|
||
def handle_exec_request(self, _callable: Callable[[bytes, Channel], None]) -> None: | ||
if not callable(_callable): | ||
raise RuntimeError("Handler function given is invalid") | ||
|
||
_callable(self._command, self._channel) # type: ignore[arg-type] | ||
|
||
if self._channel.recv_ready() is not True: | ||
self._channel.send( | ||
bytes("Command {0!r} invalid\n".format(self._command), "utf-8") | ||
) | ||
|
||
self._channel = None | ||
self._client = None | ||
|
||
self.event.clear() |
Empty file.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,11 @@ | ||
# -*- coding: utf-8 -*- | ||
|
||
import pytest | ||
|
||
from rookify.modules.example.main import ExampleHandler | ||
from rookify.modules.module import ModuleException | ||
|
||
|
||
def test_preflight() -> None: | ||
with pytest.raises(ModuleException): | ||
ExampleHandler({}, {}, "").preflight() |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,22 @@ | ||
# -*- coding: utf-8 -*- | ||
|
||
from typing import Any, Dict, Tuple | ||
from unittest import TestCase | ||
|
||
from .mock_ceph import MockCeph | ||
|
||
|
||
class TestMockCeph(TestCase): | ||
def setUp(self) -> None: | ||
self.ceph = MockCeph({}, self._command_callback) | ||
|
||
def _command_callback( | ||
self, command: str, inbuf: bytes, **kwargs: Dict[Any, Any] | ||
) -> Tuple[int, bytes, str]: | ||
if command == "test": | ||
return 0, b'["ok"]', "" | ||
return -1, b'["Command not found"]', "" | ||
|
||
def test_self(self) -> None: | ||
res = self.ceph.mon_command("test", b"") | ||
self.assertEqual(res, ["ok"]) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,22 @@ | ||
# -*- coding: utf-8 -*- | ||
|
||
from typing import Any, Dict, List | ||
from unittest import TestCase | ||
|
||
from .mock_k8s import MockK8s | ||
|
||
|
||
class TestMockK8s(TestCase): | ||
def setUp(self) -> None: | ||
self.k8s = MockK8s(self._request_callback) | ||
|
||
def _request_callback( | ||
self, method: str, *args: List[Any], **kwargs: Dict[Any, Any] | ||
) -> Any: | ||
if method == "core_v1_api.test": | ||
return True | ||
return None | ||
|
||
def test_self(self) -> None: | ||
res = self.k8s.core_v1_api.test() | ||
self.assertEqual(res, True) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,23 @@ | ||
# -*- coding: utf-8 -*- | ||
|
||
from paramiko import Channel | ||
from unittest import TestCase | ||
|
||
from .mock_ssh_server import MockSSHServer | ||
|
||
|
||
class TestMockSSHServer(TestCase): | ||
def setUp(self) -> None: | ||
self.ssh_server = MockSSHServer(self._command_callback) | ||
self.ssh_client = self.ssh_server.client | ||
|
||
def tearDown(self) -> None: | ||
self.ssh_server.close() | ||
|
||
def _command_callback(self, command: bytes, channel: Channel) -> None: | ||
if command == b"test": | ||
channel.send(b"ok\n") | ||
|
||
def test_self(self) -> None: | ||
_, stdout, _ = self.ssh_client.exec_command("test") | ||
self.assertEqual(stdout.readline(), "ok\n") |