diff --git a/moonraker/components/klippy_apis.py b/moonraker/components/klippy_apis.py index af6b0c3..c136823 100644 --- a/moonraker/components/klippy_apis.py +++ b/moonraker/components/klippy_apis.py @@ -3,8 +3,27 @@ # Copyright (C) 2020 Eric Callahan # # This file may be distributed under the terms of the GNU GPLv3 license. -import utils -from websockets import WebRequest + +from __future__ import annotations +from utils import SentinelClass +from websockets import WebRequest, Subscribable + +# Annotation imports +from typing import ( + TYPE_CHECKING, + Any, + Union, + Optional, + Dict, + List, + TypeVar, + Mapping, +) +if TYPE_CHECKING: + from confighelper import ConfigHelper + from websockets import WebRequest + Subscription = Dict[str, Optional[List[Any]]] + _T = TypeVar("_T") INFO_ENDPOINT = "info" ESTOP_ENDPOINT = "emergency_stop" @@ -15,17 +34,16 @@ SUBSCRIPTION_ENDPOINT = "objects/subscribe" STATUS_ENDPOINT = "objects/query" OBJ_LIST_ENDPOINT = "objects/list" REG_METHOD_ENDPOINT = "register_remote_method" +SENTINEL = SentinelClass.get_instance() -class Sentinel: - pass - -class KlippyAPI: - def __init__(self, config): +class KlippyAPI(Subscribable): + def __init__(self, config: ConfigHelper) -> None: self.server = config.get_server() - + system_args = config['system_args'] + self.version = system_args.get('software_version') # Maintain a subscription for all moonraker requests, as # we do not want to overwrite them - self.host_subscription = {} + self.host_subscription: Subscription = {} # Register GCode Aliases self.server.register_endpoint( @@ -41,42 +59,49 @@ class KlippyAPI: self.server.register_endpoint( "/printer/firmware_restart", ['POST'], self._gcode_firmware_restart) - async def _gcode_pause(self, web_request): + async def _gcode_pause(self, web_request: WebRequest) -> str: return await self.run_gcode("PAUSE") - async def _gcode_resume(self, web_request): + async def _gcode_resume(self, web_request: WebRequest) -> str: return await self.run_gcode("RESUME") - async def _gcode_cancel(self, web_request): + async def _gcode_cancel(self, web_request: WebRequest) -> str: return await self.run_gcode("CANCEL_PRINT") - async def _gcode_start_print(self, web_request): - filename = web_request.get_str('filename') + async def _gcode_start_print(self, web_request: WebRequest) -> str: + filename: str = web_request.get_str('filename') return await self.start_print(filename) - async def _gcode_restart(self, web_request): + async def _gcode_restart(self, web_request: WebRequest) -> str: return await self.do_restart("RESTART") - async def _gcode_firmware_restart(self, web_request): + async def _gcode_firmware_restart(self, web_request: WebRequest) -> str: return await self.do_restart("FIRMWARE_RESTART") - async def _send_klippy_request(self, method, params, default=Sentinel): + async def _send_klippy_request(self, + method: str, + params: Dict[str, Any], + default: Any = SENTINEL + ) -> Any: try: result = await self.server.make_request( WebRequest(method, params, conn=self)) - except self.server.error as e: - if default == Sentinel: + except self.server.error: + if isinstance(default, SentinelClass): raise result = default return result - async def run_gcode(self, script, default=Sentinel): + async def run_gcode(self, + script: str, + default: Any = SENTINEL + ) -> str: params = {'script': script} result = await self._send_klippy_request( GCODE_ENDPOINT, params, default) return result - async def start_print(self, filename): + async def start_print(self, filename: str) -> str: # XXX - validate that file is on disk if filename[0] == '/': filename = filename[1:] @@ -85,7 +110,7 @@ class KlippyAPI: script = f'SDCARD_PRINT_FILE FILENAME="{filename}"' return await self.run_gcode(script) - async def do_restart(self, gc): + async def do_restart(self, gc: str) -> str: try: result = await self.run_gcode(gc) except self.server.error as e: @@ -95,28 +120,38 @@ class KlippyAPI: raise return result - async def list_endpoints(self, default=Sentinel): + async def list_endpoints(self, + default: Union[SentinelClass, _T] = SENTINEL + ) -> Union[_T, Dict[str, List[str]]]: return await self._send_klippy_request( LIST_EPS_ENDPOINT, {}, default) - async def emergency_stop(self, default=Sentinel): - return await self._send_klippy_request(ESTOP_ENDPOINT, {}, default) + async def emergency_stop(self) -> str: + return await self._send_klippy_request(ESTOP_ENDPOINT, {}) - async def get_klippy_info(self, send_id=False, default=Sentinel): + async def get_klippy_info(self, + send_id: bool = False, + default: Union[SentinelClass, _T] = SENTINEL + ) -> Union[_T, Dict[str, Any]]: params = {} if send_id: - ver = utils.get_software_version() + ver = self.version params = {'client_info': {'program': "Moonraker", 'version': ver}} return await self._send_klippy_request(INFO_ENDPOINT, params, default) - async def get_object_list(self, default=Sentinel): + async def get_object_list(self, + default: Union[SentinelClass, _T] = SENTINEL + ) -> Union[_T, List[str]]: result = await self._send_klippy_request( OBJ_LIST_ENDPOINT, {}, default) if isinstance(result, dict) and 'objects' in result: return result['objects'] return result - async def query_objects(self, objects, default=Sentinel): + async def query_objects(self, + objects: Mapping[str, Optional[List[str]]], + default: Union[SentinelClass, _T] = SENTINEL + ) -> Union[_T, Dict[str, Any]]: params = {'objects': objects} result = await self._send_klippy_request( STATUS_ENDPOINT, params, default) @@ -124,7 +159,10 @@ class KlippyAPI: return result['status'] return result - async def subscribe_objects(self, objects, default=Sentinel): + async def subscribe_objects(self, + objects: Mapping[str, Optional[List[str]]], + default: Union[SentinelClass, _T] = SENTINEL + ) -> Union[_T, Dict[str, Any]]: for obj, items in objects.items(): if obj in self.host_subscription: prev = self.host_subscription[obj] @@ -142,20 +180,19 @@ class KlippyAPI: return result['status'] return result - async def subscribe_gcode_output(self, default=Sentinel): + async def subscribe_gcode_output(self) -> str: template = {'response_template': {'method': "process_gcode_response"}} - return await self._send_klippy_request( - GC_OUTPUT_ENDPOINT, template, default) + return await self._send_klippy_request(GC_OUTPUT_ENDPOINT, template) - async def register_method(self, method_name): + async def register_method(self, method_name: str) -> str: return await self._send_klippy_request( REG_METHOD_ENDPOINT, {'response_template': {"method": method_name}, 'remote_method': method_name}) - def send_status(self, status): + def send_status(self, status: Dict[str, Any]) -> None: self.server.send_event("server:status_update", status) -def load_component(config): +def load_component(config: ConfigHelper) -> KlippyAPI: return KlippyAPI(config)