Compare commits

...

7 Commits

Author SHA1 Message Date
d0fd4a5434 0.1.7
add back missing PREPARE printing state and associate printing status
2024-09-27 09:38:23 -04:00
3c218a548d add issue templates, funding, and stale bot 2024-09-12 19:56:40 -04:00
03af51608d 0.1.6
* replace 0 with 1 bytes during reporting print status to trigger state change in OctoPrint sooner.
2024-09-06 01:39:48 -04:00
c00285b1b2 0.1.5
* adjust M220 feed rate modifier calculations
2024-09-05 22:35:34 -04:00
7f1ae5a24b 0.1.4 (#43)
* fix stuck Printing from SD state when canceled in slicer or on printer, #42
2024-09-04 16:48:16 -04:00
5754e81b72 0.1.3
fix file uploads
2024-08-25 14:20:45 -04:00
cd4103cc71 0.1.2 (#40)
* fix issues related to 8dot3 filenames used in M23 command, #39 
* switch to auto reporting temp and sd status
2024-08-18 01:06:57 -04:00
12 changed files with 184 additions and 20 deletions

3
.github/FUNDING.yml vendored Normal file
View File

@ -0,0 +1,3 @@
github: [jneilliii]
patreon: jneilliii
custom: ['https://www.paypal.me/jneilliii']

26
.github/ISSUE_TEMPLATE/bug_report.md vendored Normal file
View File

@ -0,0 +1,26 @@
---
name: Bug report
about: Please make sure to check other issues, including closed ones, prior to submitting a bug report. Debug logs are required and any bug report submitted without them will be ignored and closed.
title: "[BUG]: "
labels: ''
assignees: ''
---
**Describe the Bug**
<!-- A clear and concise description of what the bug is. -->
**Expected Behavior**
<!-- A clear and concise description of what you expected to happen. -->
**Debug Logs**
<!-- If logs are not included in your bug report it will be closed. Enable debug logging for octoprint.plugins.bambu_printer in OctoPrint's logging section of settings and recreate the issue then attach octoprint.log and plugin_bambu_printer_serial.log to this bug report. -->
**Screenshots**
<!-- Please share any relevant screenshots related to the issue. -->
**Printer and Plugin Setting Details**
* Printer model?
* Is your printer connected to Bambu Cloud?
* Is the plugin configured for local access only?

View File

@ -0,0 +1,20 @@
---
name: Feature request
about: Create a feature request for an improvement or change you'd like implemented.
title: "[FR]: "
labels: ''
assignees: ''
---
**Is your feature request related to a problem? Please describe.**
<!-- A clear and concise description of what the problem is. Ex. I'm always frustrated when [...] -->
**Describe the solution you'd like**
<!-- A clear and concise description of what you want to happen. -->
**Describe alternatives you've considered**
<!-- A clear and concise description of any alternative solutions or features you've considered. -->
**Additional context**
<!-- Add any other context or screenshots about the feature request here. -->

16
.github/stale.yml vendored Normal file
View File

@ -0,0 +1,16 @@
# Number of days of inactivity before an issue becomes stale
daysUntilStale: 14
# Number of days of inactivity before a stale issue is closed
daysUntilClose: 7
# Issues with these labels will never be considered stale
exemptLabels:
- enhancement
- bug
# Label to use when marking an issue as stale
staleLabel: stale
# Comment to post when marking an issue as stale. Set to `false` to disable
markComment: >
This issue has been automatically marked as stale because it has not had
activity in 14 days. It will be closed if no further activity occurs in 7 days.
# Comment to post when closing a stale issue. Set to `false` to disable
closeComment: false

27
.github/workflows/stale.yml vendored Normal file
View File

@ -0,0 +1,27 @@
name: Mark Stale Issues
on:
workflow_dispatch:
schedule:
- cron: "0 0 * * *"
permissions:
actions: write
jobs:
stale:
runs-on: ubuntu-latest
steps:
- uses: actions/stale@v9
with:
repo-token: ${{ secrets.GITHUB_TOKEN }}
stale-issue-message: 'This issue has been automatically marked as stale because it has not had activity in 14 days. It will be closed if no further activity occurs in 7 days'
days-before-stale: 14
days-before-close: 7
stale-issue-label: 'stale'
days-before-issue-stale: 14
days-before-pr-stale: -1
days-before-issue-close: 7
days-before-pr-close: -1
exempt-issue-labels: 'bug,enhancement'
- uses: actions/checkout@v4
- uses: gautamkrishnar/keepalive-workflow@v2
with:
use_api: true

View File

@ -72,6 +72,7 @@ class BambuVirtualPrinter:
self._running = True self._running = True
self._print_status_reporter = None self._print_status_reporter = None
self._print_temp_reporter = None
self._printer_thread = threading.Thread( self._printer_thread = threading.Thread(
target=self._printer_worker, target=self._printer_worker,
name="octoprint.plugins.bambu_printer.printer_state", name="octoprint.plugins.bambu_printer.printer_state",
@ -192,7 +193,7 @@ class BambuVirtualPrinter:
or print_job_state == "FAILED" or print_job_state == "FAILED"
): ):
self.change_state(self._state_idle) self.change_state(self._state_idle)
elif print_job_state == "RUNNING": elif print_job_state == "RUNNING" or print_job_state == "PREPARE":
self.change_state(self._state_printing) self.change_state(self._state_printing)
elif print_job_state == "PAUSE": elif print_job_state == "PAUSE":
self.change_state(self._state_paused) self.change_state(self._state_paused)
@ -367,8 +368,10 @@ class BambuVirtualPrinter:
interval = int(matchS.group(1)) interval = int(matchS.group(1))
if interval > 0: if interval > 0:
self.start_continuous_status_report(interval) self.start_continuous_status_report(interval)
return False
else: else:
self.stop_continuous_status_report() self.stop_continuous_status_report()
return False
self.report_print_job_status() self.report_print_job_status()
return True return True
@ -403,10 +406,39 @@ class BambuVirtualPrinter:
self._processTemperatureQuery() self._processTemperatureQuery()
return True return True
@gcode_executor.register("M155")
def _auto_report_temperatures(self, data: str) -> bool:
matchS = re.search(r"S([0-9]+)", data)
if matchS:
interval = int(matchS.group(1))
if interval > 0:
self.start_continuous_temp_report(interval)
else:
self.stop_continuous_temp_report()
self.report_print_job_status()
return True
def start_continuous_temp_report(self, interval: int):
if self._print_temp_reporter is not None:
self._print_temp_reporter.cancel()
self._print_temp_reporter = RepeatedTimer(
interval, self._processTemperatureQuery
)
self._print_temp_reporter.start()
def stop_continuous_temp_report(self):
if self._print_temp_reporter is not None:
self._print_temp_reporter.cancel()
self._print_temp_reporter = None
# noinspection PyUnusedLocal # noinspection PyUnusedLocal
@gcode_executor.register_no_data("M115") @gcode_executor.register_no_data("M115")
def _report_firmware_info(self) -> bool: def _report_firmware_info(self) -> bool:
self.sendIO("Bambu Printer Integration") self.sendIO("Bambu Printer Integration")
self.sendIO("Cap:AUTOREPORT_SD_STATUS:1")
self.sendIO("Cap:AUTOREPORT_TEMP:1")
self.sendIO("Cap:EXTENDED_M20:1") self.sendIO("Cap:EXTENDED_M20:1")
self.sendIO("Cap:LFN_WRITE:1") self.sendIO("Cap:LFN_WRITE:1")
return True return True
@ -442,19 +474,51 @@ class BambuVirtualPrinter:
gcode_command = commands.SEND_GCODE_TEMPLATE gcode_command = commands.SEND_GCODE_TEMPLATE
percent = int(data.replace("M220 S", "")) percent = int(data.replace("M220 S", ""))
if percent is None or percent < 1 or percent > 166: def speed_fraction(speed_percent):
return True return math.floor(10000 / speed_percent) / 100
speed_fraction = 100 / percent def acceleration_magnitude(speed_percent):
acceleration = math.exp((speed_fraction - 1.0191) / -0.814) return math.exp((speed_fraction(speed_percent) - 1.0191) / -0.8139)
feed_rate = (
2.1645 * (acceleration**3) def feed_rate(speed_percent):
- 5.3247 * (acceleration**2) return 6.426e-5 * speed_percent ** 2 - 2.484e-3 * speed_percent + 0.654
+ 4.342 * acceleration
- 0.181 def linear_interpolate(x, x_points, y_points):
) if x <= x_points[0]: return y_points[0]
speed_level = 1.539 * (acceleration**2) - 0.7032 * acceleration + 4.0834 if x >= x_points[-1]: return y_points[-1]
speed_command = f"M204.2 K${acceleration:.2f} \nM220 K${feed_rate:.2f} \nM73.2 R${speed_fraction:.2f} \nM1002 set_gcode_claim_speed_level ${speed_level:.0f}\n" for i in range(len(x_points) - 1):
if x_points[i] <= x < x_points[i + 1]:
t = (x - x_points[i]) / (x_points[i + 1] - x_points[i])
return y_points[i] * (1 - t) + y_points[i + 1] * t
def scale_to_data_points(func, data_points):
data_points.sort(key=lambda x: x[0])
speeds, values = zip(*data_points)
scaling_factors = [v / func(s) for s, v in zip(speeds, values)]
return lambda x: func(x) * linear_interpolate(x, speeds, scaling_factors)
def speed_adjust(speed_percentage):
if not 30 <= speed_percentage <= 180:
speed_percentage = 100
bambu_params = {
"speed": [50, 100, 124, 166],
"acceleration": [0.3, 1.0, 1.4, 1.6],
"feed_rate": [0.7, 1.0, 1.4, 2.0]
}
acc_mag_scaled = scale_to_data_points(acceleration_magnitude,
list(zip(bambu_params["speed"], bambu_params["acceleration"])))
feed_rate_scaled = scale_to_data_points(feed_rate,
list(zip(bambu_params["speed"], bambu_params["feed_rate"])))
speed_frac = speed_fraction(speed_percentage)
acc_mag = acc_mag_scaled(speed_percentage)
feed = feed_rate_scaled(speed_percentage)
# speed_level = 1.539 * (acc_mag**2) - 0.7032 * acc_mag + 4.0834
return f"M204.2 K{acc_mag:.2f}\nM220 K{feed:.2f}\nM73.2 R{speed_frac:.2f}\n" # M1002 set_gcode_claim_speed_level ${speed_level:.0f}\n
speed_command = speed_adjust(percent)
gcode_command["print"]["param"] = speed_command gcode_command["print"]["param"] = speed_command
if self.bambu_client.publish(gcode_command): if self.bambu_client.publish(gcode_command):
@ -516,8 +580,9 @@ class BambuVirtualPrinter:
def report_print_job_status(self): def report_print_job_status(self):
if self.current_print_job is not None: if self.current_print_job is not None:
file_position = 1 if self.current_print_job.file_position == 0 else self.current_print_job.file_position
self.sendIO( self.sendIO(
f"SD printing byte {self.current_print_job.file_position}" f"SD printing byte {file_position}"
f"/{self.current_print_job.file_info.size}" f"/{self.current_print_job.file_info.size}"
) )
else: else:

View File

@ -85,7 +85,7 @@ class CachedFileView:
return file_data return file_data
def _get_file_by_stem_cached(self, file_stem: str, allowed_suffixes: list[str]): def _get_file_by_stem_cached(self, file_stem: str, allowed_suffixes: list[str]):
for file_path_str in self._file_data_cache.keys(): for file_path_str in list(self._file_data_cache.keys()) + list(self._file_alias_cache.keys()):
file_path = Path(file_path_str) file_path = Path(file_path_str)
if file_stem == file_path.with_suffix("").stem and all( if file_stem == file_path.with_suffix("").stem and all(
suffix in allowed_suffixes for suffix in file_path.suffixes suffix in allowed_suffixes for suffix in file_path.suffixes

View File

@ -117,7 +117,7 @@ class IoTFTPSConnection:
# 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.ftps_session.welcome:
conn.unwrap() conn.unwrap()
else: else:
conn.shutdown(socket.SHUT_RDWR) conn.shutdown(socket.SHUT_RDWR)

View File

@ -1,8 +1,6 @@
from __future__ import annotations from __future__ import annotations
from pathlib import Path
from octoprint_bambu_printer.printer.file_system.file_info import FileInfo from octoprint_bambu_printer.printer.file_system.file_info import FileInfo
from octoprint_bambu_printer.printer.print_job import PrintJob
from octoprint_bambu_printer.printer.states.a_printer_state import APrinterState from octoprint_bambu_printer.printer.states.a_printer_state import APrinterState
@ -26,7 +24,7 @@ class IdleState(APrinterState):
# URL to print. Root path, protocol can vary. E.g., if sd card, "ftp:///myfile.3mf", "ftp:///cache/myotherfile.3mf" # URL to print. Root path, protocol can vary. E.g., if sd card, "ftp:///myfile.3mf", "ftp:///cache/myotherfile.3mf"
filesystem_root = ( filesystem_root = (
"file:///mnt/sdcard/" "file:///mnt/sdcard/"
if self._printer._settings.get_boolean(["device_type"]) in ["X1", "X1C"] if self._printer._settings.get(["device_type"]) in ["X1", "X1C"]
else "file:///" else "file:///"
) )

View File

@ -22,6 +22,7 @@ class PrintingState(APrinterState):
def __init__(self, printer: BambuVirtualPrinter) -> None: def __init__(self, printer: BambuVirtualPrinter) -> None:
super().__init__(printer) super().__init__(printer)
self._current_print_job = None
self._is_printing = False self._is_printing = False
self._sd_printing_thread = None self._sd_printing_thread = None
@ -36,6 +37,7 @@ class PrintingState(APrinterState):
self._is_printing = False self._is_printing = False
self._sd_printing_thread.join() self._sd_printing_thread.join()
self._sd_printing_thread = None self._sd_printing_thread = None
self._printer.current_print_job = None
def _start_worker_thread(self): def _start_worker_thread(self):
if self._sd_printing_thread is None: if self._sd_printing_thread is None:

View File

@ -7,3 +7,10 @@
### ###
. .
pytest~=7.4.4
pybambu~=1.0.1
OctoPrint~=1.10.2
setuptools~=70.0.0
pyserial~=3.5
Flask~=2.2.5

View File

@ -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.1.1" plugin_version = "0.1.7"
# 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