Compare commits
9 Commits
Author | SHA1 | Date | |
---|---|---|---|
c110fa140a | |||
3889efa67a | |||
cb4b345aa7 | |||
3d0cc26147 | |||
ff58636e41 | |||
f54ab5c29f | |||
7a4439c53e | |||
9eb8b0da65 | |||
ef969d3d3b |
@ -10,11 +10,14 @@ from .ftpsclient import IoTFTPSClient
|
|||||||
|
|
||||||
|
|
||||||
class BambuPrintPlugin(
|
class BambuPrintPlugin(
|
||||||
octoprint.plugin.SettingsPlugin, octoprint.plugin.TemplatePlugin
|
octoprint.plugin.SettingsPlugin, octoprint.plugin.TemplatePlugin, octoprint.plugin.AssetPlugin
|
||||||
):
|
):
|
||||||
|
|
||||||
|
|
||||||
|
def get_assets(self):
|
||||||
|
return {'js': ["js/bambu_printer.js"]}
|
||||||
def get_template_configs(self):
|
def get_template_configs(self):
|
||||||
return [{"type": "settings", "custom_bindings": False}]
|
return [{"type": "settings", "custom_bindings": False}] #, {"type": "generic", "custom_bindings": True, "template": "bambu_printer.jinja2"}]
|
||||||
|
|
||||||
def get_settings_defaults(self):
|
def get_settings_defaults(self):
|
||||||
return {"device_type": "X1C",
|
return {"device_type": "X1C",
|
||||||
@ -31,7 +34,9 @@ class BambuPrintPlugin(
|
|||||||
"local_mqtt": True,
|
"local_mqtt": True,
|
||||||
"region": "",
|
"region": "",
|
||||||
"email": "",
|
"email": "",
|
||||||
"auth_token": ""}
|
"auth_token": "",
|
||||||
|
"always_use_default_options": False
|
||||||
|
}
|
||||||
|
|
||||||
def support_3mf_files(self):
|
def support_3mf_files(self):
|
||||||
return {'machinecode': {'3mf': ["3mf"]}}
|
return {'machinecode': {'3mf': ["3mf"]}}
|
||||||
@ -50,7 +55,7 @@ class BambuPrintPlugin(
|
|||||||
elapsed = time.monotonic() - elapsed
|
elapsed = time.monotonic() - elapsed
|
||||||
sd_upload_succeeded(filename, filename, elapsed)
|
sd_upload_succeeded(filename, filename, elapsed)
|
||||||
# remove local file after successful upload to Bambu
|
# remove local file after successful upload to Bambu
|
||||||
self._file_manager.remove_file("local", filename)
|
# self._file_manager.remove_file("local", filename)
|
||||||
else:
|
else:
|
||||||
raise Exception("upload failed")
|
raise Exception("upload failed")
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
|
@ -125,7 +125,7 @@ class IoTFTPSClient:
|
|||||||
with open(dest, "wb") as file:
|
with open(dest, "wb") as file:
|
||||||
self.ftps_session.retrbinary(f"RETR {source}", file.write)
|
self.ftps_session.retrbinary(f"RETR {source}", file.write)
|
||||||
|
|
||||||
def upload_file(self, source: str, dest: str, callback=None):
|
def upload_file(self, source: str, dest: str, callback=None) -> bool:
|
||||||
"""upload a file to a path inside the FTPS server"""
|
"""upload a file to a path inside the FTPS server"""
|
||||||
|
|
||||||
file_size = os.path.getsize(source)
|
file_size = os.path.getsize(source)
|
||||||
@ -133,43 +133,50 @@ class IoTFTPSClient:
|
|||||||
block_size = max(file_size // 100, 8192)
|
block_size = max(file_size // 100, 8192)
|
||||||
rest = None
|
rest = None
|
||||||
|
|
||||||
# Taken from ftplib.storbinary but with custom ssl handling
|
try:
|
||||||
# due to the shitty bambu p1p ftps server TODO fix properly.
|
# Taken from ftplib.storbinary but with custom ssl handling
|
||||||
with open(source, "rb") as fp:
|
# due to the shitty bambu p1p ftps server TODO fix properly.
|
||||||
self.ftps_session.voidcmd('TYPE I')
|
with open(source, "rb") as fp:
|
||||||
|
self.ftps_session.voidcmd('TYPE I')
|
||||||
|
|
||||||
with self.ftps_session.transfercmd(f"STOR {dest}", rest) as conn:
|
with self.ftps_session.transfercmd(f"STOR {dest}", rest) as conn:
|
||||||
while 1:
|
while 1:
|
||||||
buf = fp.read(block_size)
|
buf = fp.read(block_size)
|
||||||
|
|
||||||
if not buf:
|
if not buf:
|
||||||
break
|
break
|
||||||
|
|
||||||
conn.sendall(buf)
|
conn.sendall(buf)
|
||||||
|
|
||||||
if callback:
|
if callback:
|
||||||
callback(buf)
|
callback(buf)
|
||||||
|
|
||||||
# shutdown ssl layer
|
# shutdown ssl layer
|
||||||
if ftplib._SSLSocket is not None and isinstance(conn, ftplib._SSLSocket):
|
if ftplib._SSLSocket is not None and isinstance(conn, ftplib._SSLSocket):
|
||||||
# Yeah this is suposed to be conn.unwrap
|
# Yeah this is suposed to be conn.unwrap
|
||||||
# But since we operate in prot p mode
|
# But since we operate in prot p mode
|
||||||
# we can close the connection always.
|
# we can close the connection always.
|
||||||
# This is cursed but it works.
|
# This is cursed but it works.
|
||||||
if "vsFTPd" in self.welcome:
|
if "vsFTPd" in self.welcome:
|
||||||
conn.unwrap()
|
conn.unwrap()
|
||||||
else:
|
else:
|
||||||
conn.shutdown(socket.SHUT_RDWR)
|
conn.shutdown(socket.SHUT_RDWR)
|
||||||
|
|
||||||
return self.ftps_session.voidresp()
|
return True
|
||||||
|
except Exception as ex:
|
||||||
|
print(f"unexpected exception occurred: {ex}")
|
||||||
|
pass
|
||||||
|
return False
|
||||||
|
|
||||||
# Old api call.
|
def delete_file(self, path: str) -> bool:
|
||||||
# self.ftps_session.storbinary(
|
|
||||||
# f"STOR {dest}", file, blocksize=block_size, callback=callback)
|
|
||||||
|
|
||||||
def delete_file(self, path: str):
|
|
||||||
"""delete a file from under a path inside the FTPS server"""
|
"""delete a file from under a path inside the FTPS server"""
|
||||||
self.ftps_session.delete(path)
|
try:
|
||||||
|
self.ftps_session.delete(path)
|
||||||
|
return True
|
||||||
|
except Exception as ex:
|
||||||
|
print(f"unexpected exception occurred: {ex}")
|
||||||
|
pass
|
||||||
|
return False
|
||||||
|
|
||||||
def move_file(self, source: str, dest: str):
|
def move_file(self, source: str, dest: str):
|
||||||
"""move a file inside the FTPS server to another path inside the FTPS server"""
|
"""move a file inside the FTPS server to another path inside the FTPS server"""
|
||||||
|
@ -4,26 +4,74 @@
|
|||||||
* Author: jneilliii
|
* Author: jneilliii
|
||||||
* License: AGPLv3
|
* License: AGPLv3
|
||||||
*/
|
*/
|
||||||
|
|
||||||
$(function() {
|
$(function() {
|
||||||
function Bambu_printerViewModel(parameters) {
|
function Bambu_printerViewModel(parameters) {
|
||||||
var self = this;
|
var self = this;
|
||||||
|
|
||||||
// assign the injected parameters, e.g.:
|
self.settingsViewModel = parameters[0];
|
||||||
// self.loginStateViewModel = parameters[0];
|
self.filesViewModel = parameters[1];
|
||||||
// self.settingsViewModel = parameters[1];
|
|
||||||
|
|
||||||
// TODO: Implement your plugin's view model here.
|
/*$('#files div.upload-buttons > span.fileinput-button:first, #files div.folder-button').remove();
|
||||||
|
$('#files div.upload-buttons > span.fileinput-button:first').removeClass('span6').addClass('input-block-level');
|
||||||
|
|
||||||
|
self.onBeforePrintStart = function(start_print_command) {
|
||||||
|
let confirmation_html = '' +
|
||||||
|
' <div class="row-fluid form-vertical">\n' +
|
||||||
|
' <div class="control-group">\n' +
|
||||||
|
' <label class="control-label">' + gettext("Plate Number") + '</label>\n' +
|
||||||
|
' <div class="controls">\n' +
|
||||||
|
' <input type="number" min="1" value="1" id="bambu_printer_plate_number" class="input-mini">\n' +
|
||||||
|
' </div>\n' +
|
||||||
|
' </div>\n' +
|
||||||
|
' </div>';
|
||||||
|
|
||||||
|
if(!self.settingsViewModel.settings.plugins.bambu_printer.always_use_default_options()){
|
||||||
|
confirmation_html += '\n' +
|
||||||
|
' <div class="row-fluid">\n' +
|
||||||
|
' <div class="span6">\n' +
|
||||||
|
' <label class="checkbox"><input id="bambu_printer_timelapse" type="checkbox"' + ((self.settingsViewModel.settings.plugins.bambu_printer.timelapse()) ? ' checked' : '') + '> ' + gettext("Enable timelapse") + '</label>\n' +
|
||||||
|
' <label class="checkbox"><input id="bambu_printer_bed_leveling" type="checkbox"' + ((self.settingsViewModel.settings.plugins.bambu_printer.bed_leveling()) ? ' checked' : '') + '> ' + gettext("Enable bed leveling") + '</label>\n' +
|
||||||
|
' <label class="checkbox"><input id="bambu_printer_flow_cali" type="checkbox"' + ((self.settingsViewModel.settings.plugins.bambu_printer.flow_cali()) ? ' checked' : '') + '> ' + gettext("Enable flow calibration") + '</label>\n' +
|
||||||
|
' </div>\n' +
|
||||||
|
' <div class="span6">\n' +
|
||||||
|
' <label class="checkbox"><input id="bambu_printer_vibration_cali" type="checkbox"' + ((self.settingsViewModel.settings.plugins.bambu_printer.vibration_cali()) ? ' checked' : '') + '> ' + gettext("Enable vibration calibration") + '</label>\n' +
|
||||||
|
' <label class="checkbox"><input id="bambu_printer_layer_inspect" type="checkbox"' + ((self.settingsViewModel.settings.plugins.bambu_printer.layer_inspect()) ? ' checked' : '') + '> ' + gettext("Enable first layer inspection") + '</label>\n' +
|
||||||
|
' <label class="checkbox"><input id="bambu_printer_use_ams" type="checkbox"' + ((self.settingsViewModel.settings.plugins.bambu_printer.use_ams()) ? ' checked' : '') + '> ' + gettext("Use AMS") + '</label>\n' +
|
||||||
|
' </div>\n' +
|
||||||
|
' </div>\n';
|
||||||
|
}
|
||||||
|
|
||||||
|
showConfirmationDialog({
|
||||||
|
title: "Bambu Print Options",
|
||||||
|
html: confirmation_html,
|
||||||
|
cancel: gettext("Cancel"),
|
||||||
|
proceed: [gettext("Print"), gettext("Always")],
|
||||||
|
onproceed: function (idx) {
|
||||||
|
if(idx === 1){
|
||||||
|
self.settingsViewModel.settings.plugins.bambu_printer.timelapse($('#bambu_printer_timelapse').is(':checked'));
|
||||||
|
self.settingsViewModel.settings.plugins.bambu_printer.bed_leveling($('#bambu_printer_bed_leveling').is(':checked'));
|
||||||
|
self.settingsViewModel.settings.plugins.bambu_printer.flow_cali($('#bambu_printer_flow_cali').is(':checked'));
|
||||||
|
self.settingsViewModel.settings.plugins.bambu_printer.vibration_cali($('#bambu_printer_vibration_cali').is(':checked'));
|
||||||
|
self.settingsViewModel.settings.plugins.bambu_printer.layer_inspect($('#bambu_printer_layer_inspect').is(':checked'));
|
||||||
|
self.settingsViewModel.settings.plugins.bambu_printer.use_ams($('#bambu_printer_use_ams').is(':checked'));
|
||||||
|
self.settingsViewModel.settings.plugins.bambu_printer.always_use_default_options(true);
|
||||||
|
self.settingsViewModel.saveData();
|
||||||
|
}
|
||||||
|
// replace this with our own print command API call?
|
||||||
|
start_print_command();
|
||||||
|
},
|
||||||
|
nofade: true
|
||||||
|
});
|
||||||
|
return false;
|
||||||
|
};*/
|
||||||
}
|
}
|
||||||
|
|
||||||
/* view model class, parameters for constructor, container to bind to
|
|
||||||
* Please see http://docs.octoprint.org/en/master/plugins/viewmodels.html#registering-custom-viewmodels for more details
|
|
||||||
* and a full list of the available options.
|
|
||||||
*/
|
|
||||||
OCTOPRINT_VIEWMODELS.push({
|
OCTOPRINT_VIEWMODELS.push({
|
||||||
construct: Bambu_printerViewModel,
|
construct: Bambu_printerViewModel,
|
||||||
// ViewModels your plugin depends on, e.g. loginStateViewModel, settingsViewModel, ...
|
// ViewModels your plugin depends on, e.g. loginStateViewModel, settingsViewModel, ...
|
||||||
dependencies: [ /* "loginStateViewModel", "settingsViewModel" */ ],
|
dependencies: [ "settingsViewModel", "filesViewModel" ],
|
||||||
// Elements to bind to, e.g. #settings_plugin_bambu_printer, #tab_plugin_bambu_printer, ...
|
// Elements to bind to, e.g. #settings_plugin_bambu_printer, #tab_plugin_bambu_printer, ...
|
||||||
elements: [ /* ... */ ]
|
elements: [ "#bambu_printer_print_options" ]
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
@ -27,7 +27,7 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="control-group">
|
<div class="control-group">
|
||||||
<label class="control-label">{{ _('Print Options') }}</label>
|
<label class="control-label">{{ _('Default Print Options') }}</label>
|
||||||
<div class="controls">
|
<div class="controls">
|
||||||
<label class="checkbox"><input type="checkbox" data-bind="checked: settings.plugins.bambu_printer.timelapse"> {{ _('Enable timelapse') }}</label>
|
<label class="checkbox"><input type="checkbox" data-bind="checked: settings.plugins.bambu_printer.timelapse"> {{ _('Enable timelapse') }}</label>
|
||||||
<label class="checkbox"><input type="checkbox" data-bind="checked: settings.plugins.bambu_printer.bed_leveling"> {{ _('Enable bed leveling') }}</label>
|
<label class="checkbox"><input type="checkbox" data-bind="checked: settings.plugins.bambu_printer.bed_leveling"> {{ _('Enable bed leveling') }}</label>
|
||||||
@ -37,4 +37,10 @@
|
|||||||
<label class="checkbox"><input type="checkbox" data-bind="checked: settings.plugins.bambu_printer.use_ams"> {{ _('Use AMS') }}</label>
|
<label class="checkbox"><input type="checkbox" data-bind="checked: settings.plugins.bambu_printer.use_ams"> {{ _('Use AMS') }}</label>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
{#<div class="control-group">
|
||||||
|
<label class="control-label">{{ _('Always Use Default') }}</label>
|
||||||
|
<div class="controls">
|
||||||
|
<label class="checkbox"><input type="checkbox" data-bind="checked: settings.plugins.bambu_printer.always_use_default_options"> </label>
|
||||||
|
</div>
|
||||||
|
</div>#}
|
||||||
</form>
|
</form>
|
||||||
|
@ -68,8 +68,10 @@ class BambuPrinter:
|
|||||||
self._sdCardReady = True
|
self._sdCardReady = True
|
||||||
self._sdPrinter = None
|
self._sdPrinter = None
|
||||||
self._sdPrinting = False
|
self._sdPrinting = False
|
||||||
|
self._sdPrintStarting = False
|
||||||
self._sdPrintingSemaphore = threading.Event()
|
self._sdPrintingSemaphore = threading.Event()
|
||||||
self._sdPrintingPausedSemaphore = threading.Event()
|
self._sdPrintingPausedSemaphore = threading.Event()
|
||||||
|
self._sdFileListCache = {}
|
||||||
self._selectedSdFile = None
|
self._selectedSdFile = None
|
||||||
self._selectedSdFileSize = 0
|
self._selectedSdFileSize = 0
|
||||||
self._selectedSdFilePos = 0
|
self._selectedSdFilePos = 0
|
||||||
@ -78,6 +80,7 @@ class BambuPrinter:
|
|||||||
self._busy_loop = None
|
self._busy_loop = None
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
import logging
|
import logging
|
||||||
|
|
||||||
self._logger = logging.getLogger(
|
self._logger = logging.getLogger(
|
||||||
@ -163,14 +166,20 @@ class BambuPrinter:
|
|||||||
self.bedTargetTemp = temperatures.get("target_bed_temp", 0.0)
|
self.bedTargetTemp = temperatures.get("target_bed_temp", 0.0)
|
||||||
self.chamberTemp = temperatures.get("chamber_temp", 0.0)
|
self.chamberTemp = temperatures.get("chamber_temp", 0.0)
|
||||||
|
|
||||||
if print_job.get("gcode_state") == "RUNNING":
|
if print_job.get("gcode_state") == "RUNNING" or print_job.get("gcode_state") == "PREPARE":
|
||||||
if not self._sdPrintingSemaphore.is_set():
|
if not self._sdPrintingSemaphore.is_set():
|
||||||
self._sdPrintingSemaphore.set()
|
self._sdPrintingSemaphore.set()
|
||||||
if self._sdPrintingPausedSemaphore.is_set():
|
if self._sdPrintingPausedSemaphore.is_set():
|
||||||
self._sdPrintingPausedSemaphore.clear()
|
self._sdPrintingPausedSemaphore.clear()
|
||||||
|
self._sdPrintStarting = False
|
||||||
if not self._sdPrinting:
|
if not self._sdPrinting:
|
||||||
filename = print_job.get("subtask_name")
|
filename = print_job.get("subtask_name")
|
||||||
# TODO: swap this out to use 8 dot 3 name based on long name/path
|
if not self._sdFileListCache.get(filename.lower()):
|
||||||
|
if self._sdFileListCache.get(f"{filename.lower()}.3mf"):
|
||||||
|
filename = f"{filename.lower()}.3mf"
|
||||||
|
elif self._sdFileListCache.get(f"{filename.lower()}.gcode.3mf"):
|
||||||
|
filename = f"{filename.lower()}.gcode.3mf"
|
||||||
|
|
||||||
self._selectSdFile(filename)
|
self._selectSdFile(filename)
|
||||||
self._startSdPrint(from_printer=True)
|
self._startSdPrint(from_printer=True)
|
||||||
|
|
||||||
@ -185,9 +194,12 @@ class BambuPrinter:
|
|||||||
self._send("// action:paused")
|
self._send("// action:paused")
|
||||||
self._sendPaused()
|
self._sendPaused()
|
||||||
|
|
||||||
if print_job.get("gcode_state") == "FINISH" and self._sdPrintingSemaphore.is_set():
|
if ( print_job.get("gcode_state") == "FINISH" or print_job.get("gcode_state") == "FAILED" ):
|
||||||
self._selectedSdFilePos = self._selectedSdFileSize
|
if self._sdPrintStarting is False:
|
||||||
self._finishSdPrint()
|
self._sdPrinting = False
|
||||||
|
if self._sdPrintingSemaphore.is_set():
|
||||||
|
self._selectedSdFilePos = self._selectedSdFileSize
|
||||||
|
self._finishSdPrint()
|
||||||
def _create_connection(self):
|
def _create_connection(self):
|
||||||
if (self._settings.get(["device_type"]) != "" and
|
if (self._settings.get(["device_type"]) != "" and
|
||||||
self._settings.get(["serial"]) != "" and
|
self._settings.get(["serial"]) != "" and
|
||||||
@ -241,6 +253,7 @@ class BambuPrinter:
|
|||||||
|
|
||||||
self._sdCardReady = True
|
self._sdCardReady = True
|
||||||
self._sdPrinting = False
|
self._sdPrinting = False
|
||||||
|
self._sdPrintStarting = False
|
||||||
if self._sdPrinter:
|
if self._sdPrinter:
|
||||||
self._sdPrinting = False
|
self._sdPrinting = False
|
||||||
self._sdPrintingSemaphore.clear()
|
self._sdPrintingSemaphore.clear()
|
||||||
@ -429,6 +442,14 @@ class BambuPrinter:
|
|||||||
else:
|
else:
|
||||||
self._sendOk()
|
self._sendOk()
|
||||||
|
|
||||||
|
if self.bambu.connected:
|
||||||
|
GCODE_COMMAND = commands.SEND_GCODE_TEMPLATE
|
||||||
|
GCODE_COMMAND['print']['param'] = data + "\n"
|
||||||
|
if self.bambu.publish(GCODE_COMMAND):
|
||||||
|
self._logger.info("command sent successfully")
|
||||||
|
self._sendOk()
|
||||||
|
continue
|
||||||
|
|
||||||
finally:
|
finally:
|
||||||
self._logger.debug(f"{data}")
|
self._logger.debug(f"{data}")
|
||||||
|
|
||||||
@ -474,13 +495,18 @@ class BambuPrinter:
|
|||||||
|
|
||||||
def _gcode_M524(self, data: str) -> bool:
|
def _gcode_M524(self, data: str) -> bool:
|
||||||
if self._sdCardReady:
|
if self._sdCardReady:
|
||||||
self._cancelSdPrint()
|
return self._cancelSdPrint()
|
||||||
return False
|
return False
|
||||||
|
|
||||||
def _gcode_M26(self, data: str) -> bool:
|
def _gcode_M26(self, data: str) -> bool:
|
||||||
self._logger.debug("ignoring M26 command.")
|
if data == "M26 S0":
|
||||||
self._send("M26 disabled for Bambu")
|
if self._sdCardReady:
|
||||||
return True
|
return self._cancelSdPrint()
|
||||||
|
return False
|
||||||
|
else:
|
||||||
|
self._logger.debug("ignoring M26 command.")
|
||||||
|
self._send("M26 disabled for Bambu")
|
||||||
|
return True
|
||||||
|
|
||||||
def _gcode_M27(self, data: str) -> bool:
|
def _gcode_M27(self, data: str) -> bool:
|
||||||
def report():
|
def report():
|
||||||
@ -625,7 +651,7 @@ class BambuPrinter:
|
|||||||
access_code = self._settings.get(["access_code"])
|
access_code = self._settings.get(["access_code"])
|
||||||
|
|
||||||
ftp = IoTFTPSClient(f"{host}", 990, "bblp", f"{access_code}", ssl_implicit=True)
|
ftp = IoTFTPSClient(f"{host}", 990, "bblp", f"{access_code}", ssl_implicit=True)
|
||||||
filelist = ftp.list_files("", ".3mf")
|
filelist = ftp.list_files("", ".3mf") or []
|
||||||
|
|
||||||
for entry in filelist:
|
for entry in filelist:
|
||||||
if entry.startswith("/"):
|
if entry.startswith("/"):
|
||||||
@ -643,21 +669,41 @@ class BambuPrinter:
|
|||||||
"size": filesize,
|
"size": filesize,
|
||||||
"timestamp": unix_timestamp_to_m20_timestamp(int(filedate))
|
"timestamp": unix_timestamp_to_m20_timestamp(int(filedate))
|
||||||
}
|
}
|
||||||
result[filename.lower()] = data
|
|
||||||
result[dosname.lower()] = filename.lower()
|
result[dosname.lower()] = filename.lower()
|
||||||
|
result[filename.lower()] = data
|
||||||
|
|
||||||
|
filelistcache = ftp.list_files("cache/", ".3mf") or []
|
||||||
|
|
||||||
|
for entry in filelistcache:
|
||||||
|
if entry.startswith("/"):
|
||||||
|
filename = entry[1:].replace("cache/", "")
|
||||||
|
else:
|
||||||
|
filename = entry.replace("cache/", "")
|
||||||
|
filesize = ftp.ftps_session.size(f"cache/{filename}")
|
||||||
|
date_str = ftp.ftps_session.sendcmd(f"MDTM cache/{filename}").replace("213 ", "")
|
||||||
|
filedate = datetime.datetime.strptime(date_str, "%Y%m%d%H%M%S").replace(tzinfo=datetime.timezone.utc).timestamp()
|
||||||
|
dosname = get_dos_filename(filename, existing_filenames=list(result.keys())).lower()
|
||||||
|
data = {
|
||||||
|
"dosname": dosname,
|
||||||
|
"name": filename,
|
||||||
|
"path": "cache/"+filename,
|
||||||
|
"size": filesize,
|
||||||
|
"timestamp": unix_timestamp_to_m20_timestamp(int(filedate))
|
||||||
|
}
|
||||||
|
result[dosname.lower()] = filename.lower()
|
||||||
|
result[filename.lower()] = data
|
||||||
|
|
||||||
return result
|
return result
|
||||||
|
|
||||||
def _getSdFileData(self, filename: str) -> Optional[Dict[str, Any]]:
|
def _getSdFileData(self, filename: str) -> Optional[Dict[str, Any]]:
|
||||||
files = self._mappedSdList()
|
data = self._sdFileListCache.get(filename.lower())
|
||||||
data = files.get(filename.lower())
|
|
||||||
if isinstance(data, str):
|
if isinstance(data, str):
|
||||||
data = files.get(data.lower())
|
data = self._sdFileListCache.get(data.lower())
|
||||||
return data
|
return data
|
||||||
|
|
||||||
def _getSdFiles(self) -> List[Dict[str, Any]]:
|
def _getSdFiles(self) -> List[Dict[str, Any]]:
|
||||||
files = self._mappedSdList()
|
self._sdFileListCache = self._mappedSdList()
|
||||||
return [x for x in files.values() if isinstance(x, dict)]
|
return [x for x in self._sdFileListCache.values() if isinstance(x, dict)]
|
||||||
|
|
||||||
def _selectSdFile(self, filename: str, check_already_open: bool = False) -> None:
|
def _selectSdFile(self, filename: str, check_already_open: bool = False) -> None:
|
||||||
if filename.startswith("/"):
|
if filename.startswith("/"):
|
||||||
@ -680,6 +726,7 @@ class BambuPrinter:
|
|||||||
if self._selectedSdFile is not None:
|
if self._selectedSdFile is not None:
|
||||||
if self._sdPrinter is None:
|
if self._sdPrinter is None:
|
||||||
self._sdPrinting = True
|
self._sdPrinting = True
|
||||||
|
self._sdPrintStarting = True
|
||||||
self._sdPrinter = threading.Thread(target=self._sdPrintingWorker, kwargs={"from_printer": from_printer})
|
self._sdPrinter = threading.Thread(target=self._sdPrintingWorker, kwargs={"from_printer": from_printer})
|
||||||
self._sdPrinter.start()
|
self._sdPrinter.start()
|
||||||
# self._sdPrintingSemaphore.set()
|
# self._sdPrintingSemaphore.set()
|
||||||
@ -699,18 +746,21 @@ class BambuPrinter:
|
|||||||
else:
|
else:
|
||||||
self._logger.info("print pause failed")
|
self._logger.info("print pause failed")
|
||||||
|
|
||||||
def _cancelSdPrint(self):
|
def _cancelSdPrint(self) -> bool:
|
||||||
if self.bambu.connected:
|
if self.bambu.connected:
|
||||||
if self.bambu.publish(commands.STOP):
|
if self.bambu.publish(commands.STOP):
|
||||||
self._logger.info("print cancelled")
|
self._logger.info("print cancelled")
|
||||||
|
self._finishSdPrint()
|
||||||
|
return True
|
||||||
else:
|
else:
|
||||||
self._logger.info("print cancel failed")
|
self._logger.info("print cancel failed")
|
||||||
|
return False
|
||||||
|
|
||||||
def _setSdPos(self, pos):
|
def _setSdPos(self, pos):
|
||||||
self._newSdFilePos = pos
|
self._newSdFilePos = pos
|
||||||
|
|
||||||
def _reportSdStatus(self):
|
def _reportSdStatus(self):
|
||||||
if self._sdPrinter is not None and (self._sdPrintingSemaphore.is_set() or self._sdPrintingPausedSemaphore.is_set()):
|
if ( self._sdPrinter is not None or self._sdPrintStarting is True ) and self._selectedSdFileSize > 0:
|
||||||
self._send(f"SD printing byte {self._selectedSdFilePos}/{self._selectedSdFileSize}")
|
self._send(f"SD printing byte {self._selectedSdFilePos}/{self._selectedSdFileSize}")
|
||||||
else:
|
else:
|
||||||
self._send("Not SD printing")
|
self._send("Not SD printing")
|
||||||
@ -799,6 +849,7 @@ class BambuPrinter:
|
|||||||
self._selectedSdFilePos = 0
|
self._selectedSdFilePos = 0
|
||||||
self._selectedSdFileSize = 0
|
self._selectedSdFileSize = 0
|
||||||
self._sdPrinting = False
|
self._sdPrinting = False
|
||||||
|
self._sdPrintStarting = False
|
||||||
self._sdPrinter = None
|
self._sdPrinter = None
|
||||||
|
|
||||||
def _deleteSdFile(self, filename: str) -> None:
|
def _deleteSdFile(self, filename: str) -> None:
|
||||||
|
2
setup.py
2
setup.py
@ -14,7 +14,7 @@ plugin_package = "octoprint_bambu_printer"
|
|||||||
plugin_name = "OctoPrint-BambuPrinter"
|
plugin_name = "OctoPrint-BambuPrinter"
|
||||||
|
|
||||||
# The plugin's version. Can be overwritten within OctoPrint's internal data via __plugin_version__ in the plugin module
|
# The plugin's version. Can be overwritten within OctoPrint's internal data via __plugin_version__ in the plugin module
|
||||||
plugin_version = "0.0.8"
|
plugin_version = "0.0.15"
|
||||||
|
|
||||||
# The plugin's description. Can be overwritten within OctoPrint's internal data via __plugin_description__ in the plugin
|
# The plugin's description. Can be overwritten within OctoPrint's internal data via __plugin_description__ in the plugin
|
||||||
# module
|
# module
|
||||||
|
Reference in New Issue
Block a user