2020-08-30 01:23:22 +03:00
# <Octoprint Klipper Plugin>
2018-08-19 14:49:24 +03:00
2020-08-30 01:23:22 +03:00
# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU Affero General Public License as
# published by the Free Software Foundation, either version 3 of the
# License, or (at your option) any later version.
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU Affero General Public License for more details.
# You should have received a copy of the GNU Affero General Public License
# along with this program. If not, see <https://www.gnu.org/licenses/>.
2020-08-29 21:13:27 +03:00
2021-03-27 01:39:08 +03:00
from __future__ import absolute_import , division , print_function , unicode_literals
2018-01-23 17:01:58 +03:00
import logging
import octoprint . plugin
import octoprint . plugin . core
2018-08-13 19:22:58 +03:00
import glob
import os
2021-09-05 16:28:57 +03:00
import time
2020-09-28 00:46:47 +03:00
import sys
2021-09-05 16:28:57 +03:00
from octoprint . server import NO_CONTENT
from octoprint . util import is_hidden_path
from octoprint . util import get_formatted_size
from . import util , cfgUtils , logger
2018-02-08 18:38:48 +03:00
from octoprint . util . comm import parse_firmware_line
2021-05-30 16:01:41 +03:00
from octoprint . access . permissions import Permissions , ADMIN_GROUP
2018-08-13 19:22:58 +03:00
from . modules import KlipperLogAnalyzer
2021-09-05 16:28:57 +03:00
from octoprint . server . util . flask import restricted_access
2018-08-09 03:16:07 +03:00
import flask
2021-03-11 21:42:05 +03:00
from flask_babel import gettext
2018-01-23 17:01:58 +03:00
2021-04-03 01:49:40 +03:00
if sys . version_info [ 0 ] < 3 :
import StringIO
2021-09-05 16:28:57 +03:00
MAX_UPLOAD_SIZE = 5 * 1024 * 1024 # 5Mb
2018-01-23 17:01:58 +03:00
class KlipperPlugin (
2021-03-27 01:39:08 +03:00
octoprint . plugin . StartupPlugin ,
octoprint . plugin . TemplatePlugin ,
octoprint . plugin . SettingsPlugin ,
octoprint . plugin . AssetPlugin ,
octoprint . plugin . SimpleApiPlugin ,
2021-09-05 16:28:57 +03:00
octoprint . plugin . EventHandlerPlugin ,
octoprint . plugin . BlueprintPlugin ) :
2021-03-27 01:39:08 +03:00
_parsing_response = False
2021-04-07 01:00:42 +03:00
_parsing_check_response = True
2021-03-27 01:39:08 +03:00
_message = " "
2021-03-29 01:57:23 +03:00
def __init__ ( self ) :
self . _logger = logging . getLogger ( " octoprint.plugins.klipper " )
self . _octoklipper_logger = logging . getLogger ( " octoprint.plugins.klipper.debug " )
2021-03-27 01:39:08 +03:00
# -- Startup Plugin
2021-03-29 01:57:23 +03:00
def on_startup ( self , host , port ) :
from octoprint . logging . handlers import CleaningTimedRotatingFileHandler
octoklipper_logging_handler = CleaningTimedRotatingFileHandler (
self . _settings . get_plugin_logfile_path ( postfix = " debug " ) , when = " D " , backupCount = 3 )
octoklipper_logging_handler . setFormatter ( logging . Formatter ( " [ %(asctime)s ] %(levelname)s : %(message)s " ) )
octoklipper_logging_handler . setLevel ( logging . DEBUG )
self . _octoklipper_logger . addHandler ( octoklipper_logging_handler )
self . _octoklipper_logger . setLevel (
logging . DEBUG if self . _settings . get_boolean ( [ " debug_logging " ] ) else logging . INFO )
self . _octoklipper_logger . propagate = False
2021-03-27 01:39:08 +03:00
def on_after_startup ( self ) :
klipper_port = self . _settings . get ( [ " connection " , " port " ] )
additional_ports = self . _settings . global_get (
[ " serial " , " additionalPorts " ] )
if klipper_port not in additional_ports :
additional_ports . append ( klipper_port )
self . _settings . global_set (
[ " serial " , " additionalPorts " ] , additional_ports )
self . _settings . save ( )
2021-09-05 16:28:57 +03:00
logger . log_info (
self ,
" Added klipper serial port {} to list of additional ports. " . format ( klipper_port )
)
2021-03-27 01:39:08 +03:00
# -- Settings Plugin
def get_additional_permissions ( self , * args , * * kwargs ) :
return [
{
" key " : " CONFIG " ,
" name " : " Config Klipper " ,
" description " : gettext ( " Allows to config klipper " ) ,
" default_groups " : [ ADMIN_GROUP ] ,
" dangerous " : True ,
" roles " : [ " admin " ]
} ,
{
" key " : " MACRO " ,
" name " : " Use Klipper Macros " ,
" description " : gettext ( " Allows to use klipper macros " ) ,
" default_groups " : [ ADMIN_GROUP ] ,
" dangerous " : True ,
" roles " : [ " admin " ]
} ,
]
2020-08-29 21:13:27 +03:00
2021-03-27 01:39:08 +03:00
def get_settings_defaults ( self ) :
return dict (
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 (
2021-03-29 01:57:23 +03:00
debug_logging = False ,
2021-09-05 18:01:03 +03:00
configpath = " ~/ " ,
2021-09-21 02:54:05 +03:00
baseconfig = " printer.cfg " ,
2021-03-29 01:57:23 +03:00
old_config = " " ,
2021-03-27 01:39:08 +03:00
logpath = " /tmp/klippy.log " ,
reload_command = " RESTART " ,
2021-09-16 13:31:51 +03:00
restart_onsave = False ,
2021-05-04 02:01:57 +03:00
shortStatus_navbar = True ,
shortStatus_sidebar = True ,
2021-04-11 01:17:19 +03:00
parse_check = False ,
fontsize = 9
2021-03-27 01:39:08 +03:00
)
)
2018-02-08 18:38:48 +03:00
2021-03-27 01:39:08 +03:00
def on_settings_save ( self , data ) :
2021-03-29 01:57:23 +03:00
old_debug_logging = self . _settings . get_boolean ( [ " configuration " , " debug_logging " ] )
2021-03-27 01:39:08 +03:00
octoprint . plugin . SettingsPlugin . on_settings_save ( self , data )
2021-03-29 01:57:23 +03:00
new_debug_logging = self . _settings . get_boolean ( [ " configuration " , " debug_logging " ] )
if old_debug_logging != new_debug_logging :
if new_debug_logging :
self . _octoklipper_logger . setLevel ( logging . DEBUG )
else :
self . _octoklipper_logger . setLevel ( logging . INFO )
2021-03-27 01:39:08 +03:00
def get_settings_restricted_paths ( self ) :
return dict (
admin = [
[ " connection " , " port " ] ,
[ " configuration " , " configpath " ] ,
[ " configuration " , " replace_connection_panel " ]
] ,
user = [
[ " macros " ] ,
[ " probe " ]
]
)
2020-10-04 06:31:10 +03:00
2021-03-27 01:39:08 +03:00
def get_settings_version ( self ) :
2021-09-16 13:31:51 +03:00
# Settings_Versionhistory:
# 3 = add shortstatus on navbar. migrate the navbar setting for this
# 4 = -change of configpath to only path without filename
2021-09-21 02:54:05 +03:00
# -parse configpath into path and baseconfig if not standard printer.cfg
# -switch setting for 'restart on editor save' to true if it was not set to manually
# -remove old_config
# -remove config on root settingsdirectory
2021-09-05 18:01:03 +03:00
return 4
2020-10-04 06:31:10 +03:00
2021-09-05 17:51:52 +03:00
#migrate Settings
2021-03-27 01:39:08 +03:00
def on_settings_migrate ( self , target , current ) :
2021-09-05 17:51:52 +03:00
settings = self . _settings
2021-03-27 01:39:08 +03:00
if current is None :
2021-09-05 17:51:52 +03:00
self . migrate_old_settings ( settings )
2021-09-16 13:31:51 +03:00
2021-05-30 16:01:40 +03:00
if current is not None and current < 3 :
2021-09-16 13:31:51 +03:00
self . migrate_settings_configuration (
2021-09-05 17:51:52 +03:00
settings ,
" shortStatus_navbar " ,
" navbar " ,
)
2021-05-04 02:01:57 +03:00
2021-09-05 17:51:52 +03:00
if current is not None and current < 4 :
2021-09-21 02:54:05 +03:00
self . migrate_settings_4 (
settings
2021-09-05 17:51:52 +03:00
)
2021-09-16 13:31:51 +03:00
def migrate_old_settings ( self , settings ) :
'''
For Old settings
'''
2021-09-05 17:51:52 +03:00
self . migrate_settings ( settings , " serialport " , " connection " , " port " )
self . migrate_settings ( settings , " replace_connection_panel " , " connection " , " replace_connection_panel " )
self . migrate_settings ( settings , " probeHeight " , " probe " , " height " )
self . migrate_settings ( settings , " probeLift " , " probe " , " lift " )
self . migrate_settings ( settings , " probeSpeedXy " , " probe " , " speed_xy " )
self . migrate_settings ( settings , " probeSpeedZ " , " probe " , " speed_z " )
2021-09-16 13:31:51 +03:00
self . migrate_settings ( settings , " configPath " , " configuration " , " configpath " )
2021-09-05 17:51:52 +03:00
if settings . has ( [ " probePoints " ] ) :
points = settings . get ( [ " probePoints " ] )
points_new = [ dict ( name = " " , x = int ( p [ " x " ] ) , y = int ( p [ " y " ] ) , z = 0 ) for p in points ]
settings . set ( [ " probe " , " points " ] , points_new )
settings . remove ( [ " probePoints " ] )
def migrate_settings ( self , settings , old , new , new2 = " " ) - > None :
2021-09-16 13:31:51 +03:00
""" migrate setting to setting with additional group
Args :
settings ( any ) : instance of self . _settings
old ( str ) : the old setting to migrate
new ( str ) : group or only new setting if there is no new2
new2 ( str , optional ) : the new setting to migrate to . Defaults to " " .
""" ' ' ' ' ' '
2021-09-05 17:51:52 +03:00
if settings . has ( [ old ] ) :
if new2 != " " :
logger . log_info ( self , " migrate setting for ' " + old + " ' -> ' " + new + " / " + new2 + " ' " )
settings . set ( [ new , new2 ] , settings . get ( [ old ] ) )
else :
logger . log_info ( self , " migrate setting for ' " + old + " ' -> ' " + new + " ' " )
settings . set ( [ new ] , settings . get ( [ old ] ) )
settings . remove ( [ old ] )
2021-03-27 01:39:08 +03:00
2021-09-05 17:51:52 +03:00
def migrate_settings_configuration ( self , settings , new , old ) :
if settings . has ( [ " configuration " , old ] ) :
logger . log_info ( self , " migrate setting for ' configuration/ " + old + " ' -> ' configuration/ " + new + " ' " )
settings . set ( [ " configuration " , new ] , settings . get ( [ " configuration " , old ] ) )
settings . remove ( [ " configuration " , old ] )
2021-09-21 02:54:05 +03:00
def migrate_settings_4 ( self , settings ) :
cfg_path = settings . get ( [ " configuration " , " configpath " ] )
if cfg_path . find ( " printer.cfg " ) != - 1 :
new_cfg_path = cfg_path . replace ( " printer.cfg " , " " )
logger . log_info ( self , " migrate setting for ' configuration/configpath ' : " + cfg_path + " -> " + new_cfg_path )
settings . set ( [ " configuration " , " configpath " ] , new_cfg_path )
else :
new_cfg_path , baseconfig = os . path . split ( cfg_path )
logger . log_info ( self , " migrate setting for ' configuration/configpath ' : " + cfg_path + " -> " + new_cfg_path )
logger . log_info ( self , " migrate setting for ' configuration/baseconfig ' : printer.cfg -> " + baseconfig )
settings . set ( [ " configuration " , " configpath " ] , new_cfg_path )
settings . set ( [ " configuration " , " baseconfig " ] , baseconfig )
if settings . get ( [ " configuration " , " reload_command " ] ) != " manually " :
logger . log_info ( self , " migrate setting for ' configuration/restart_onsave ' : False -> True " )
settings . set ( [ " configuration " , " restart_onsave " ] , True )
if settings . has ( [ " config " ] ) :
logger . log_info ( self , " remove old setting for ' config ' " )
settings . remove ( [ " config " ] )
if settings . has ( [ " configuration " , " old_config " ] ) :
logger . log_info ( self , " remove old setting for ' configuration/old_config ' " )
settings . remove ( [ " configuration " , " old_config " ] )
2021-09-05 17:51:52 +03:00
# -- Template Plugin
2021-03-27 01:39:08 +03:00
def get_template_configs ( self ) :
return [
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 " ,
2021-09-05 16:28:57 +03:00
custom_bindings = True ,
icon = " rocket " ,
replaces = " connection " if self . _settings . get_boolean (
[ " connection " , " replace_connection_panel " ] ) else " "
) ,
2021-03-27 01:39:08 +03:00
dict (
type = " generic " ,
name = " Performance Graph " ,
template = " klipper_graph_dialog.jinja2 " ,
custom_bindings = True
) ,
2021-09-05 16:28:57 +03:00
dict (
type = " generic " ,
name = " Config Backups " ,
template = " klipper_backups_dialog.jinja2 " ,
custom_bindings = True
) ,
dict (
type = " generic " ,
name = " Config Editor " ,
template = " klipper_editor.jinja2 " ,
custom_bindings = True
) ,
2021-03-27 01:39:08 +03:00
dict (
type = " generic " ,
name = " Macro Dialog " ,
template = " klipper_param_macro_dialog.jinja2 " ,
custom_bindings = True
2018-08-09 03:16:07 +03:00
)
2021-03-27 01:39:08 +03:00
]
2021-09-05 16:28:57 +03:00
def get_template_vars ( self ) :
return {
" max_upload_size " : MAX_UPLOAD_SIZE ,
" max_upload_size_str " : get_formatted_size ( MAX_UPLOAD_SIZE ) ,
}
2021-03-27 01:39:08 +03:00
# -- Asset Plugin
def get_assets ( self ) :
return dict (
js = [ " js/klipper.js " ,
" js/klipper_settings.js " ,
" js/klipper_leveling.js " ,
" js/klipper_pid_tuning.js " ,
" js/klipper_offset.js " ,
" js/klipper_param_macro.js " ,
2021-09-05 16:28:57 +03:00
" js/klipper_graph.js " ,
" js/klipper_backup.js " ,
" js/klipper_editor.js "
] ,
clientjs = [ " clientjs/klipper.js " ] ,
2021-05-11 23:22:37 +03:00
css = [ " css/klipper.css " ]
2021-03-27 01:39:08 +03:00
)
# -- Event Handler Plugin
def on_event ( self , event , payload ) :
2021-09-05 17:51:52 +03:00
if event == " UserLoggedIn " :
2021-09-05 16:28:57 +03:00
util . update_status ( self , " info " , " Klipper: Standby " )
2021-09-05 17:51:52 +03:00
if event == " Connecting " :
2021-09-05 16:28:57 +03:00
util . update_status ( self , " info " , " Klipper: Connecting ... " )
2021-09-05 17:51:52 +03:00
elif event == " Connected " :
2021-09-05 16:28:57 +03:00
util . update_status ( self , " info " , " Klipper: Connected to host " )
logger . log_info (
self ,
2021-03-27 01:39:08 +03:00
" Connected to host via {} @ {} bps " . format ( payload [ " port " ] , payload [ " baudrate " ] ) )
2021-09-05 17:51:52 +03:00
elif event == " Disconnected " :
2021-09-05 16:28:57 +03:00
util . update_status ( self , " info " , " Klipper: Disconnected from host " )
2021-09-05 17:51:52 +03:00
elif event == " Error " :
2021-09-05 16:28:57 +03:00
util . update_status ( self , " error " , " Klipper: Error " )
logger . log_error ( self , payload [ " error " ] )
2021-03-27 01:39:08 +03:00
# -- GCODE Hook
def on_parse_gcode ( self , comm , line , * args , * * kwargs ) :
if " FIRMWARE_VERSION " in line :
printerInfo = parse_firmware_line ( line )
if " FIRMWARE_VERSION " in printerInfo :
2021-09-05 16:28:57 +03:00
logger . log_info ( self , " Firmware version: {} " . format (
2021-03-27 01:39:08 +03:00
printerInfo [ " FIRMWARE_VERSION " ] ) )
2021-04-11 01:17:19 +03:00
elif " // probe " in line or " // Failed to verify BLTouch " in line :
2021-04-07 01:00:42 +03:00
msg = line . strip ( ' / ' )
2021-09-05 16:28:57 +03:00
logger . log_info ( self , msg )
2021-05-25 14:20:27 +03:00
self . write_parsing_response_buffer ( )
2021-03-27 01:39:08 +03:00
elif " // " in line :
2021-04-07 01:00:42 +03:00
# add lines with // to a buffer
2021-03-27 01:39:08 +03:00
self . _message = self . _message + line . strip ( ' / ' )
if not self . _parsing_response :
2021-09-05 16:28:57 +03:00
util . update_status ( self , " info " , self . _message )
2021-03-27 01:39:08 +03:00
self . _parsing_response = True
2021-04-07 01:00:42 +03:00
elif " !! " in line :
msg = line . strip ( ' ! ' )
2021-09-05 16:28:57 +03:00
util . update_status ( self , " error " , msg )
logger . log_error ( self , msg )
2021-04-07 01:00:42 +03:00
self . write_parsing_response_buffer ( )
2021-03-27 01:39:08 +03:00
else :
2021-04-07 01:00:42 +03:00
self . write_parsing_response_buffer ( )
2021-03-27 01:39:08 +03:00
return line
2021-04-07 01:00:42 +03:00
def write_parsing_response_buffer ( self ) :
# write buffer with // lines after a gcode response without //
if self . _parsing_response :
self . _parsing_response = False
2021-09-05 16:28:57 +03:00
logger . log_info ( self , self . _message )
2021-04-07 01:00:42 +03:00
self . _message = " "
2021-03-27 01:39:08 +03:00
def get_api_commands ( self ) :
return dict (
listLogFiles = [ ] ,
2021-09-05 17:51:52 +03:00
getStats = [ " logFile " ]
2021-03-27 01:39:08 +03:00
)
def on_api_command ( self , command , data ) :
if command == " listLogFiles " :
files = [ ]
logpath = os . path . expanduser (
self . _settings . get ( [ " configuration " , " logpath " ] )
2018-08-15 02:17:12 +03:00
)
2021-09-05 16:28:57 +03:00
if util . file_exist ( self , logpath ) :
2021-03-27 01:39:08 +03:00
for f in glob . glob ( self . _settings . get ( [ " configuration " , " logpath " ] ) + " * " ) :
filesize = os . path . getsize ( f )
2021-09-05 16:28:57 +03:00
filemdate = time . strftime ( " %d . % m. % Y % H: % M " , time . localtime ( os . path . getctime ( f ) ) )
2021-03-27 01:39:08 +03:00
files . append ( dict (
2021-09-05 16:28:57 +03:00
name = os . path . basename ( f ) + " ( " + filemdate + " ) " ,
2021-03-27 01:39:08 +03:00
file = f ,
size = filesize
) )
2021-09-05 17:51:52 +03:00
return flask . jsonify ( data = files )
2021-03-27 01:39:08 +03:00
elif command == " getStats " :
if " logFile " in data :
log_analyzer = KlipperLogAnalyzer . KlipperLogAnalyzer (
data [ " logFile " ] )
return flask . jsonify ( log_analyzer . analyze ( ) )
2021-09-05 16:28:57 +03:00
def is_blueprint_protected ( self ) :
return False
def route_hook ( self , server_routes , * args , * * kwargs ) :
from octoprint . server . util . tornado import LargeResponseHandler , path_validation_factory
from octoprint . util import is_hidden_path
configpath = os . path . expanduser (
self . _settings . get ( [ " configuration " , " configpath " ] )
)
bak_path = os . path . join ( self . get_plugin_data_folder ( ) , " configs " , " " )
return [
2021-09-05 17:54:44 +03:00
( r " /download/configs/(.*) " , LargeResponseHandler , dict ( path = configpath ,
2021-09-05 16:28:57 +03:00
as_attachment = True ,
path_validation = path_validation_factory ( lambda path : not is_hidden_path ( path ) ,
status_code = 404 ) ) ) ,
2021-09-05 17:54:44 +03:00
( r " /download/backup/(.*) " , LargeResponseHandler , dict ( path = bak_path ,
2021-09-05 16:28:57 +03:00
as_attachment = True ,
path_validation = path_validation_factory ( lambda path : not is_hidden_path ( path ) ,
status_code = 404 ) ) )
]
2021-09-05 17:51:52 +03:00
# API for Backups
2021-09-05 16:28:57 +03:00
# Get Content of a Backupconfig
@octoprint.plugin.BlueprintPlugin.route ( " /backup/<filename> " , methods = [ " GET " ] )
@restricted_access
@Permissions.PLUGIN_KLIPPER_CONFIG.require ( 403 )
def get_backup ( self , filename ) :
data_folder = self . get_plugin_data_folder ( )
full_path = os . path . realpath ( os . path . join ( data_folder , " configs " , filename ) )
response = cfgUtils . get_cfg ( self , full_path )
return flask . jsonify ( response = response )
# Delete a Backupconfig
@octoprint.plugin.BlueprintPlugin.route ( " /backup/<filename> " , methods = [ " DELETE " ] )
@restricted_access
@Permissions.PLUGIN_KLIPPER_CONFIG.require ( 403 )
def delete_backup ( self , filename ) :
data_folder = self . get_plugin_data_folder ( )
full_path = os . path . realpath ( os . path . join ( data_folder , " configs " , filename ) )
if (
full_path . startswith ( data_folder )
and os . path . exists ( full_path )
and not is_hidden_path ( full_path )
) :
try :
os . remove ( full_path )
except Exception :
self . _octoklipper_logger . exception ( " Could not delete {} " . format ( filename ) )
raise
return NO_CONTENT
# Get a list of all backuped configfiles
@octoprint.plugin.BlueprintPlugin.route ( " /backup/list " , methods = [ " GET " ] )
@restricted_access
@Permissions.PLUGIN_KLIPPER_CONFIG.require ( 403 )
def list_backups ( self ) :
files = cfgUtils . list_cfg_files ( self , " backup " )
return flask . jsonify ( files = files )
# restore a backuped configfile
@octoprint.plugin.BlueprintPlugin.route ( " /backup/restore/<filename> " , methods = [ " GET " ] )
@restricted_access
@Permissions.PLUGIN_KLIPPER_CONFIG.require ( 403 )
def restore_backup ( self , filename ) :
configpath = os . path . expanduser (
self . _settings . get ( [ " configuration " , " configpath " ] )
)
data_folder = self . get_plugin_data_folder ( )
backupfile = os . path . realpath ( os . path . join ( data_folder , " configs " , filename ) )
return flask . jsonify ( restored = cfgUtils . copy_cfg ( self , backupfile , configpath ) )
2021-09-05 17:51:52 +03:00
# API for Configs
2021-09-05 16:28:57 +03:00
# Get Content of a Configfile
@octoprint.plugin.BlueprintPlugin.route ( " /config/<filename> " , methods = [ " GET " ] )
@restricted_access
@Permissions.PLUGIN_KLIPPER_CONFIG.require ( 403 )
def get_config ( self , filename ) :
cfg_path = os . path . expanduser (
self . _settings . get ( [ " configuration " , " configpath " ] )
)
full_path = os . path . realpath ( os . path . join ( cfg_path , filename ) )
response = cfgUtils . get_cfg ( self , full_path )
return flask . jsonify ( response = response )
# Delete a Configfile
@octoprint.plugin.BlueprintPlugin.route ( " /config/<filename> " , methods = [ " DELETE " ] )
@restricted_access
@Permissions.PLUGIN_KLIPPER_CONFIG.require ( 403 )
def delete_config ( self , filename ) :
cfg_path = os . path . expanduser (
self . _settings . get ( [ " configuration " , " configpath " ] )
)
full_path = os . path . realpath ( os . path . join ( cfg_path , filename ) )
if (
full_path . startswith ( cfg_path )
and os . path . exists ( full_path )
and not is_hidden_path ( full_path )
) :
try :
os . remove ( full_path )
except Exception :
self . _octoklipper_logger . exception ( " Could not delete {} " . format ( filename ) )
raise
return NO_CONTENT
# Get a list of all configfiles
@octoprint.plugin.BlueprintPlugin.route ( " /config/list " , methods = [ " GET " ] )
@restricted_access
@Permissions.PLUGIN_KLIPPER_CONFIG.require ( 403 )
def list_configs ( self ) :
files = cfgUtils . list_cfg_files ( self , " " )
return flask . jsonify ( files = files , max_upload_size = MAX_UPLOAD_SIZE )
# check syntax of a given data
@octoprint.plugin.BlueprintPlugin.route ( " /config/check " , methods = [ " POST " ] )
@restricted_access
@Permissions.PLUGIN_KLIPPER_CONFIG.require ( 403 )
def check_config ( self ) :
data = flask . request . json
data_to_check = data . get ( " DataToCheck " , [ ] )
response = cfgUtils . check_cfg ( self , data_to_check )
return flask . jsonify ( is_syntax_ok = response )
# save a configfile
@octoprint.plugin.BlueprintPlugin.route ( " /config/save " , methods = [ " POST " ] )
@restricted_access
@Permissions.PLUGIN_KLIPPER_CONFIG.require ( 403 )
def save_config ( self ) :
data = flask . request . json
filename = data . get ( " filename " , [ ] )
if filename == [ ] :
flask . abort (
400 ,
description = " Invalid request, the filename is not set " ,
)
2021-09-05 17:51:52 +03:00
Filecontent = data . get ( " DataToSave " , [ ] )
2021-09-05 16:28:57 +03:00
saved = cfgUtils . save_cfg ( self , Filecontent , filename )
if saved == True :
util . send_message ( self , " reload " , " configlist " , " " , " " )
return flask . jsonify ( saved = saved )
2021-09-16 13:31:51 +03:00
# restart klipper
@octoprint.plugin.BlueprintPlugin.route ( " /restart " , methods = [ " POST " ] )
@restricted_access
@Permissions.PLUGIN_KLIPPER_CONFIG.require ( 403 )
def restart_klipper ( self ) :
reload_command = self . _settings . get ( [ " configuration " , " reload_command " ] )
if reload_command != " manually " :
# Restart klippy to reload config
self . _printer . commands ( reload_command )
logger . log_info ( self , " Restarting Klipper. " )
return NO_CONTENT
2021-09-05 17:51:52 +03:00
# APIs end
2021-09-05 16:28:57 +03:00
2021-03-27 01:39:08 +03:00
def get_update_information ( self ) :
return dict (
klipper = dict (
displayName = self . _plugin_name ,
displayVersion = self . _plugin_version ,
type = " github_release " ,
current = self . _plugin_version ,
user = " thelastWallE " ,
repo = " OctoprintKlipperPlugin " ,
pip = " https://github.com/thelastWallE/OctoprintKlipperPlugin/archive/ {target_version} .zip " ,
stable_branch = dict (
name = " Stable " ,
branch = " master " ,
comittish = [ " master " ]
) ,
prerelease_branches = [
dict (
name = " Release Candidate " ,
branch = " rc " ,
comittish = [ " rc " , " master " ]
) ]
)
)
2020-08-29 21:13:27 +03:00
__plugin_name__ = " OctoKlipper "
__plugin_pythoncompat__ = " >=2.7,<4 "
2021-09-05 16:28:17 +03:00
__plugin_settings_overlay__ = {
' system ' : {
' actions ' : [ {
' action ' : ' octoklipper_restart ' ,
' command ' : ' sudo service klipper restart ' ,
2021-09-08 17:03:06 +03:00
' name ' : gettext ( ' Restart Klipper ' ) ,
2021-09-05 16:28:17 +03:00
' confirm ' : ' <h3><center><b> ' + gettext ( " You are about to restart Klipper! " ) + ' <br> ' + gettext ( " This will stop ongoing prints! " ) + ' </b></center></h3><br>Command = " sudo service klipper restart " '
} ]
}
}
2020-08-29 21:13:27 +03:00
2018-01-23 17:01:58 +03:00
def __plugin_load__ ( ) :
2021-03-27 01:39:08 +03:00
global __plugin_implementation__
global __plugin_hooks__
__plugin_implementation__ = KlipperPlugin ( )
__plugin_hooks__ = {
2021-09-05 16:28:57 +03:00
" octoprint.server.http.routes " : __plugin_implementation__ . route_hook ,
2021-03-27 01:39:08 +03:00
" octoprint.access.permissions " : __plugin_implementation__ . get_additional_permissions ,
" octoprint.comm.protocol.gcode.received " : __plugin_implementation__ . on_parse_gcode ,
" octoprint.plugin.softwareupdate.check_config " : __plugin_implementation__ . get_update_information
}