klippy_apis: Add annotiations

Signed-off-by:  Eric Callahan <arksine.code@gmail.com>
This commit is contained in:
Arksine 2021-05-14 08:28:15 -04:00
parent ce7495ecce
commit 9bf4ade35b
1 changed files with 73 additions and 36 deletions

View File

@ -3,8 +3,27 @@
# Copyright (C) 2020 Eric Callahan <arksine.code@gmail.com> # Copyright (C) 2020 Eric Callahan <arksine.code@gmail.com>
# #
# This file may be distributed under the terms of the GNU GPLv3 license. # 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" INFO_ENDPOINT = "info"
ESTOP_ENDPOINT = "emergency_stop" ESTOP_ENDPOINT = "emergency_stop"
@ -15,17 +34,16 @@ SUBSCRIPTION_ENDPOINT = "objects/subscribe"
STATUS_ENDPOINT = "objects/query" STATUS_ENDPOINT = "objects/query"
OBJ_LIST_ENDPOINT = "objects/list" OBJ_LIST_ENDPOINT = "objects/list"
REG_METHOD_ENDPOINT = "register_remote_method" REG_METHOD_ENDPOINT = "register_remote_method"
SENTINEL = SentinelClass.get_instance()
class Sentinel: class KlippyAPI(Subscribable):
pass def __init__(self, config: ConfigHelper) -> None:
class KlippyAPI:
def __init__(self, config):
self.server = config.get_server() 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 # Maintain a subscription for all moonraker requests, as
# we do not want to overwrite them # we do not want to overwrite them
self.host_subscription = {} self.host_subscription: Subscription = {}
# Register GCode Aliases # Register GCode Aliases
self.server.register_endpoint( self.server.register_endpoint(
@ -41,42 +59,49 @@ class KlippyAPI:
self.server.register_endpoint( self.server.register_endpoint(
"/printer/firmware_restart", ['POST'], self._gcode_firmware_restart) "/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") 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") 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") return await self.run_gcode("CANCEL_PRINT")
async def _gcode_start_print(self, web_request): async def _gcode_start_print(self, web_request: WebRequest) -> str:
filename = web_request.get_str('filename') filename: str = web_request.get_str('filename')
return await self.start_print(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") 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") 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: try:
result = await self.server.make_request( result = await self.server.make_request(
WebRequest(method, params, conn=self)) WebRequest(method, params, conn=self))
except self.server.error as e: except self.server.error:
if default == Sentinel: if isinstance(default, SentinelClass):
raise raise
result = default result = default
return result return result
async def run_gcode(self, script, default=Sentinel): async def run_gcode(self,
script: str,
default: Any = SENTINEL
) -> str:
params = {'script': script} params = {'script': script}
result = await self._send_klippy_request( result = await self._send_klippy_request(
GCODE_ENDPOINT, params, default) GCODE_ENDPOINT, params, default)
return result return result
async def start_print(self, filename): async def start_print(self, filename: str) -> str:
# XXX - validate that file is on disk # XXX - validate that file is on disk
if filename[0] == '/': if filename[0] == '/':
filename = filename[1:] filename = filename[1:]
@ -85,7 +110,7 @@ class KlippyAPI:
script = f'SDCARD_PRINT_FILE FILENAME="{filename}"' script = f'SDCARD_PRINT_FILE FILENAME="{filename}"'
return await self.run_gcode(script) return await self.run_gcode(script)
async def do_restart(self, gc): async def do_restart(self, gc: str) -> str:
try: try:
result = await self.run_gcode(gc) result = await self.run_gcode(gc)
except self.server.error as e: except self.server.error as e:
@ -95,28 +120,38 @@ class KlippyAPI:
raise raise
return result 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( return await self._send_klippy_request(
LIST_EPS_ENDPOINT, {}, default) LIST_EPS_ENDPOINT, {}, default)
async def emergency_stop(self, default=Sentinel): async def emergency_stop(self) -> str:
return await self._send_klippy_request(ESTOP_ENDPOINT, {}, default) 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 = {} params = {}
if send_id: if send_id:
ver = utils.get_software_version() ver = self.version
params = {'client_info': {'program': "Moonraker", 'version': ver}} params = {'client_info': {'program': "Moonraker", 'version': ver}}
return await self._send_klippy_request(INFO_ENDPOINT, params, default) 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( result = await self._send_klippy_request(
OBJ_LIST_ENDPOINT, {}, default) OBJ_LIST_ENDPOINT, {}, default)
if isinstance(result, dict) and 'objects' in result: if isinstance(result, dict) and 'objects' in result:
return result['objects'] return result['objects']
return result 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} params = {'objects': objects}
result = await self._send_klippy_request( result = await self._send_klippy_request(
STATUS_ENDPOINT, params, default) STATUS_ENDPOINT, params, default)
@ -124,7 +159,10 @@ class KlippyAPI:
return result['status'] return result['status']
return result 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(): for obj, items in objects.items():
if obj in self.host_subscription: if obj in self.host_subscription:
prev = self.host_subscription[obj] prev = self.host_subscription[obj]
@ -142,20 +180,19 @@ class KlippyAPI:
return result['status'] return result['status']
return result return result
async def subscribe_gcode_output(self, default=Sentinel): async def subscribe_gcode_output(self) -> str:
template = {'response_template': template = {'response_template':
{'method': "process_gcode_response"}} {'method': "process_gcode_response"}}
return await self._send_klippy_request( return await self._send_klippy_request(GC_OUTPUT_ENDPOINT, template)
GC_OUTPUT_ENDPOINT, template, default)
async def register_method(self, method_name): async def register_method(self, method_name: str) -> str:
return await self._send_klippy_request( return await self._send_klippy_request(
REG_METHOD_ENDPOINT, REG_METHOD_ENDPOINT,
{'response_template': {"method": method_name}, {'response_template': {"method": method_name},
'remote_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) self.server.send_event("server:status_update", status)
def load_component(config): def load_component(config: ConfigHelper) -> KlippyAPI:
return KlippyAPI(config) return KlippyAPI(config)