2018-01-23 17:01:58 +03:00
|
|
|
# coding=utf-8
|
|
|
|
from __future__ import absolute_import
|
2018-01-23 21:30:55 +03:00
|
|
|
import datetime
|
2018-01-23 17:01:58 +03:00
|
|
|
import logging
|
|
|
|
import octoprint.plugin
|
|
|
|
import octoprint.plugin.core
|
2018-02-08 18:38:48 +03:00
|
|
|
from octoprint.util.comm import parse_firmware_line
|
2018-08-09 03:16:07 +03:00
|
|
|
import flask
|
2018-01-23 17:01:58 +03:00
|
|
|
|
|
|
|
class KlipperPlugin(
|
|
|
|
octoprint.plugin.StartupPlugin,
|
|
|
|
octoprint.plugin.TemplatePlugin,
|
|
|
|
octoprint.plugin.SettingsPlugin,
|
2018-02-08 18:38:48 +03:00
|
|
|
octoprint.plugin.AssetPlugin,
|
|
|
|
octoprint.plugin.EventHandlerPlugin):
|
2018-02-02 14:39:06 +03:00
|
|
|
|
2018-08-09 10:25:05 +03:00
|
|
|
_parsing_response = False
|
2018-02-02 16:05:21 +03:00
|
|
|
_message = ""
|
2018-02-08 18:38:48 +03:00
|
|
|
|
2018-08-09 03:16:07 +03:00
|
|
|
#-- Startup Plugin
|
2018-01-23 17:01:58 +03:00
|
|
|
|
|
|
|
def on_after_startup(self):
|
2018-08-09 10:25:05 +03:00
|
|
|
klipper_port = self._settings.get(["connection", "port"])
|
|
|
|
additional_ports = self._settings.global_get(["serial", "additionalPorts"])
|
2018-02-02 16:05:21 +03:00
|
|
|
|
2018-08-09 10:25:05 +03:00
|
|
|
if klipper_port not in additional_ports:
|
|
|
|
additional_ports.append(klipper_port)
|
|
|
|
self._settings.global_set(["serial", "additionalPorts"], additional_ports)
|
2018-02-02 16:05:21 +03:00
|
|
|
self._settings.save()
|
2018-08-09 10:25:05 +03:00
|
|
|
self._logger.info("Added klipper serial port {} to list of additional ports.".format(klipper_port))
|
2018-02-02 16:05:21 +03:00
|
|
|
|
2018-02-02 14:39:06 +03:00
|
|
|
#-- Settings Plugin
|
2018-08-09 03:16:07 +03:00
|
|
|
|
2018-01-23 17:01:58 +03:00
|
|
|
def get_settings_defaults(self):
|
|
|
|
return dict(
|
2018-08-09 10:25:05 +03:00
|
|
|
connection = dict(
|
|
|
|
port="/tmp/printer",
|
|
|
|
replace_connection_panel=True
|
|
|
|
),
|
|
|
|
macros = [dict(
|
|
|
|
name="E-Stop",
|
|
|
|
macro="M112",
|
|
|
|
sidebar=True,
|
|
|
|
tab=True
|
|
|
|
)],
|
|
|
|
probe = dict(
|
|
|
|
height=0,
|
|
|
|
lift=5,
|
|
|
|
speed_xy=1500,
|
|
|
|
speed_z=500,
|
|
|
|
points=[dict(
|
|
|
|
name="point-1",
|
|
|
|
x=0,
|
|
|
|
y=0
|
|
|
|
)]
|
|
|
|
),
|
|
|
|
configuration = dict(
|
|
|
|
path="/home/pi/printer.cfg"
|
|
|
|
)
|
2018-08-09 03:16:07 +03:00
|
|
|
)
|
2018-08-09 10:25:05 +03:00
|
|
|
|
2018-08-09 03:16:07 +03:00
|
|
|
def on_settings_load(self):
|
|
|
|
data = octoprint.plugin.SettingsPlugin.on_settings_load(self)
|
2018-08-09 10:25:05 +03:00
|
|
|
f = open(self._settings.get(["configuration", "path"]), "r")
|
2018-08-09 03:16:07 +03:00
|
|
|
if f:
|
|
|
|
data["config"] = f.read()
|
|
|
|
f.close()
|
|
|
|
else:
|
|
|
|
self._logger.info(
|
2018-08-09 10:25:05 +03:00
|
|
|
"Error: Klipper config file not found at: {}".format(self._settings.get(["config_path"]))
|
2018-08-09 03:16:07 +03:00
|
|
|
)
|
|
|
|
return data
|
|
|
|
|
|
|
|
def on_settings_save(self, data):
|
|
|
|
if "config" in data:
|
2018-08-09 10:25:05 +03:00
|
|
|
f = open(self._settings.get(["configuration", "path"]), "w")
|
2018-08-09 03:16:07 +03:00
|
|
|
if f:
|
|
|
|
f.write(data["config"])
|
|
|
|
f.close()
|
|
|
|
self._logger.info(
|
2018-08-09 10:25:05 +03:00
|
|
|
"Write Klipper config to {}".format(self._settings.get(["config_path"]))
|
2018-08-09 03:16:07 +03:00
|
|
|
)
|
|
|
|
# Restart klipply to reload config
|
|
|
|
self._printer.commands("RESTART")
|
|
|
|
self.logInfo("Reloading Klipper Configuration.")
|
|
|
|
else:
|
|
|
|
self._logger.info(
|
2018-08-09 10:25:05 +03:00
|
|
|
"Error: Couldn't write Klipper config file: {}".format(self._settings.get(["config_path"]))
|
2018-08-09 03:16:07 +03:00
|
|
|
)
|
2018-08-09 10:25:05 +03:00
|
|
|
data.pop("config", None) # we dont want to write the klipper conf to the octoprint settings
|
2018-08-09 03:16:07 +03:00
|
|
|
else:
|
|
|
|
octoprint.plugin.SettingsPlugin.on_settings_save(self, data)
|
|
|
|
|
|
|
|
def get_settings_restricted_paths(self):
|
|
|
|
return dict(
|
|
|
|
admin=[
|
2018-08-09 10:25:05 +03:00
|
|
|
["connection", "port"],
|
|
|
|
["configuration", "path"],
|
|
|
|
["configuration", "replace_connection_panel"]
|
2018-08-09 03:16:07 +03:00
|
|
|
],
|
|
|
|
user=[
|
|
|
|
["macros"],
|
2018-08-09 10:25:05 +03:00
|
|
|
["probe"]
|
2018-08-09 03:16:07 +03:00
|
|
|
]
|
|
|
|
)
|
|
|
|
|
2018-08-09 10:25:05 +03:00
|
|
|
def get_settings_version(self):
|
|
|
|
return 2
|
|
|
|
|
|
|
|
def on_settings_migrate(self, target, current):
|
|
|
|
if current is None:
|
|
|
|
settings = self._settings
|
|
|
|
|
|
|
|
if settings.has(["serialport"]):
|
|
|
|
settings.set(["connection", "port"], settings.get(["serialport"]) )
|
|
|
|
settings.remove(["serialport"])
|
|
|
|
|
|
|
|
if settings.has(["replace_connection_panel"]):
|
|
|
|
settings.set(
|
|
|
|
["connection", "replace_connection_panel"],
|
|
|
|
settings.get(["replace_connection_panel"])
|
|
|
|
)
|
|
|
|
settings.remove(["replace_connection_panel"])
|
|
|
|
|
|
|
|
if settings.has(["probeHeight"]):
|
|
|
|
settings.set(["probe", "height"], settings.get(["probeHeight"]))
|
|
|
|
settings.remove(["probeHeight"])
|
|
|
|
|
|
|
|
if settings.has(["probeLift"]):
|
|
|
|
settings.set(["probe", "lift"], settings.get(["probeLift"]))
|
|
|
|
settings.remove(["probeLift"])
|
|
|
|
|
|
|
|
if settings.has(["probeSpeedXy"]):
|
|
|
|
settings.set(["probe", "speed_xy"], settings.get(["probeSpeedXy"]))
|
|
|
|
settings.remove(["probeSpeedXy"])
|
|
|
|
|
|
|
|
if settings.has(["probeSpeedZ"]):
|
|
|
|
settings.set(["probe", "speed_z"], settings.get(["probeSpeedZ"]))
|
|
|
|
settings.remove(["probeSpeedZ"])
|
|
|
|
|
|
|
|
if settings.has(["probePoints"]):
|
|
|
|
points = settings.get(["probePoints"])
|
|
|
|
points_new = []
|
|
|
|
for p in points:
|
|
|
|
points_new.append(dict(name="", x=int(p["x"]), y=int(p["y"]), z=0))
|
|
|
|
settings.set(["probe", "points"], points_new)
|
|
|
|
settings.remove(["probePoints"])
|
|
|
|
|
|
|
|
if settings.has(["configPath"]):
|
|
|
|
settings.set(["config_path"], settings.get(["configPath"]))
|
|
|
|
settings.remove(["configPath"])
|
|
|
|
|
2018-02-02 14:39:06 +03:00
|
|
|
#-- Template Plugin
|
2018-08-09 03:16:07 +03:00
|
|
|
|
2018-01-23 17:01:58 +03:00
|
|
|
def get_template_configs(self):
|
|
|
|
return [
|
2018-08-09 10:25:05 +03:00
|
|
|
dict(type="navbar", custom_bindings=True),
|
|
|
|
dict(type="settings", custom_bindings=True),
|
|
|
|
dict(
|
|
|
|
type="generic",
|
|
|
|
name="Assisted Bed Leveling",
|
|
|
|
template="klipper_leveling_dialog.jinja2",
|
|
|
|
custom_bindings=True
|
|
|
|
),
|
|
|
|
dict(
|
|
|
|
type="generic",
|
|
|
|
name="PID Tuning",
|
|
|
|
template="klipper_pid_tuning_dialog.jinja2",
|
|
|
|
custom_bindings=True
|
|
|
|
),
|
|
|
|
dict(
|
|
|
|
type="generic",
|
|
|
|
name="Coordinate Offset",
|
|
|
|
template="klipper_offset_dialog.jinja2",
|
|
|
|
custom_bindings=True
|
|
|
|
),
|
|
|
|
dict(
|
|
|
|
type="tab",
|
|
|
|
name="Klipper",
|
|
|
|
template="klipper_tab_main.jinja2",
|
|
|
|
suffix="_main",
|
|
|
|
custom_bindings=True
|
|
|
|
),
|
|
|
|
dict(type="sidebar",
|
|
|
|
custom_bindings=True,
|
|
|
|
icon="rocket",
|
|
|
|
replaces= "connection" if self._settings.get_boolean(["connection", "replace_connection_panel"]) else ""
|
|
|
|
)
|
2018-01-23 17:01:58 +03:00
|
|
|
]
|
|
|
|
|
2018-02-02 14:39:06 +03:00
|
|
|
#-- Asset Plugin
|
|
|
|
|
2018-01-23 17:01:58 +03:00
|
|
|
def get_assets(self):
|
|
|
|
return dict(
|
2018-01-24 19:15:48 +03:00
|
|
|
js=["js/klipper.js",
|
|
|
|
"js/klipper_settings.js",
|
|
|
|
"js/klipper_leveling.js",
|
2018-04-26 22:06:09 +03:00
|
|
|
"js/klipper_pid_tuning.js",
|
|
|
|
"js/klipper_offset.js"],
|
2018-01-23 17:01:58 +03:00
|
|
|
css=["css/klipper.css"],
|
|
|
|
less=["css/klipper.less"]
|
|
|
|
)
|
2018-01-23 20:33:22 +03:00
|
|
|
|
2018-02-08 18:38:48 +03:00
|
|
|
#-- Event Handler Plugin
|
|
|
|
|
|
|
|
def on_event(self, event, payload):
|
|
|
|
if "Connecting" == event:
|
|
|
|
self.updateStatus("info", "Connecting ...")
|
|
|
|
elif "Connected" == event:
|
2018-02-26 13:46:33 +03:00
|
|
|
self.updateStatus("info", "Connected to host")
|
2018-08-09 03:16:07 +03:00
|
|
|
self.logInfo("Connected to host via {} @{}bps".format(payload["port"], payload["baudrate"]))
|
2018-02-08 18:38:48 +03:00
|
|
|
elif "Disconnected" == event:
|
2018-02-26 13:46:33 +03:00
|
|
|
self.updateStatus("info", "Disconnected from host")
|
2018-02-08 18:38:48 +03:00
|
|
|
elif "Error" == event:
|
|
|
|
self.updateStatus("error", "Error")
|
|
|
|
self.logError(payload["error"])
|
|
|
|
|
2018-02-02 14:39:06 +03:00
|
|
|
#-- GCODE Hook
|
|
|
|
|
|
|
|
def on_parse_gcode(self, comm, line, *args, **kwargs):
|
2018-02-08 18:38:48 +03:00
|
|
|
if "FIRMWARE_VERSION" in line:
|
|
|
|
printerInfo = parse_firmware_line(line)
|
|
|
|
if "FIRMWARE_VERSION" in printerInfo:
|
2018-08-09 03:16:07 +03:00
|
|
|
self.logInfo("Firmware version: {}".format(printerInfo["FIRMWARE_VERSION"]))
|
2018-02-08 18:38:48 +03:00
|
|
|
elif "//" in line:
|
2018-02-02 23:51:24 +03:00
|
|
|
self._message = self._message + line.strip('/')
|
2018-08-09 10:25:05 +03:00
|
|
|
if not self._parsing_response:
|
2018-08-09 03:16:07 +03:00
|
|
|
self.updateStatus("info", self._message)
|
2018-08-09 10:25:05 +03:00
|
|
|
self._parsing_response = True
|
2018-02-02 14:39:06 +03:00
|
|
|
else:
|
2018-08-09 10:25:05 +03:00
|
|
|
if self._parsing_response:
|
|
|
|
self._parsing_response = False
|
2018-08-09 03:16:07 +03:00
|
|
|
self.logInfo(self._message)
|
|
|
|
self._message = ""
|
2018-02-02 14:39:06 +03:00
|
|
|
if "!!" in line:
|
2018-08-09 03:16:07 +03:00
|
|
|
msg = line.strip('!')
|
|
|
|
self.updateStatus("error", msg)
|
|
|
|
self.logError(msg)
|
2018-02-02 14:39:06 +03:00
|
|
|
return line
|
2018-02-08 18:38:48 +03:00
|
|
|
|
2018-02-02 14:39:06 +03:00
|
|
|
#-- Helpers
|
2018-02-08 18:38:48 +03:00
|
|
|
def sendMessage(self, type, subtype, payload):
|
2018-08-09 10:25:05 +03:00
|
|
|
self._plugin_manager.send_plugin_message(
|
|
|
|
self._identifier,
|
|
|
|
dict(
|
|
|
|
time=datetime.datetime.now().strftime("%H:%M:%S"),
|
|
|
|
type=type,
|
|
|
|
payload=payload
|
|
|
|
)
|
|
|
|
)
|
2018-01-23 20:33:22 +03:00
|
|
|
|
2018-02-26 13:46:33 +03:00
|
|
|
def pollStatus(self):
|
2018-08-09 10:25:05 +03:00
|
|
|
self._printer.commands("STATUS")
|
|
|
|
|
2018-02-08 18:38:48 +03:00
|
|
|
def updateStatus(self, type, status):
|
2018-08-09 10:25:05 +03:00
|
|
|
self.sendMessage("status", type, status)
|
2018-02-08 18:38:48 +03:00
|
|
|
|
|
|
|
def logInfo(self, message):
|
2018-08-09 10:25:05 +03:00
|
|
|
self.sendMessage("log", "info", message)
|
2018-02-02 14:39:06 +03:00
|
|
|
|
2018-02-08 18:38:48 +03:00
|
|
|
def logError(self, error):
|
2018-08-09 10:25:05 +03:00
|
|
|
self.sendMessage("log", "error", error)
|
2018-02-02 14:39:06 +03:00
|
|
|
|
|
|
|
|
2018-01-23 17:01:58 +03:00
|
|
|
|
|
|
|
__plugin_name__ = "Klipper"
|
|
|
|
|
|
|
|
def __plugin_load__():
|
|
|
|
global __plugin_implementation__
|
|
|
|
global __plugin_hooks__
|
2018-08-09 10:25:05 +03:00
|
|
|
|
2018-01-23 17:01:58 +03:00
|
|
|
__plugin_implementation__ = KlipperPlugin()
|
|
|
|
__plugin_hooks__ = {
|
2018-02-08 18:38:48 +03:00
|
|
|
"octoprint.comm.protocol.gcode.received": __plugin_implementation__.on_parse_gcode
|
2018-01-23 17:01:58 +03:00
|
|
|
}
|
|
|
|
|