Compare commits
	
		
			9 Commits
		
	
	
		
			0.1.1
			...
			698f8f4151
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 698f8f4151 | |||
| 7a0293bac7 | |||
|  | d0fd4a5434 | ||
|  | 3c218a548d | ||
|  | 03af51608d | ||
|  | c00285b1b2 | ||
|  | 7f1ae5a24b | ||
|  | 5754e81b72 | ||
|  | cd4103cc71 | 
							
								
								
									
										3
									
								
								.github/FUNDING.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								.github/FUNDING.yml
									
									
									
									
										vendored
									
									
										Normal 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
									
								
							
							
						
						
									
										26
									
								
								.github/ISSUE_TEMPLATE/bug_report.md
									
									
									
									
										vendored
									
									
										Normal 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? | ||||
							
								
								
									
										20
									
								
								.github/ISSUE_TEMPLATE/feature_request.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								.github/ISSUE_TEMPLATE/feature_request.md
									
									
									
									
										vendored
									
									
										Normal 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
									
								
							
							
						
						
									
										16
									
								
								.github/stale.yml
									
									
									
									
										vendored
									
									
										Normal 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
									
								
							
							
						
						
									
										27
									
								
								.github/workflows/stale.yml
									
									
									
									
										vendored
									
									
										Normal 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 | ||||
| @@ -85,7 +85,7 @@ class BambuPrintPlugin( | ||||
|             "serial": "", | ||||
|             "host": "", | ||||
|             "access_code": "", | ||||
|             "username": "bblp", | ||||
|             "username": "octobambu", | ||||
|             "timelapse": False, | ||||
|             "bed_leveling": True, | ||||
|             "flow_cali": False, | ||||
| @@ -286,10 +286,10 @@ class BambuPrintPlugin( | ||||
|     def get_update_information(self): | ||||
|         return { | ||||
|             "bambu_printer": { | ||||
|                 "displayName": "Bambu Printer", | ||||
|                 "displayName": "Manus Bambu Printer", | ||||
|                 "displayVersion": self._plugin_version, | ||||
|                 "type": "github_release", | ||||
|                 "user": "jneilliii", | ||||
|                 "user": "ManuelW", | ||||
|                 "repo": "OctoPrint-BambuPrinter", | ||||
|                 "current": self._plugin_version, | ||||
|                 "stable_branch": { | ||||
| @@ -304,6 +304,6 @@ class BambuPrintPlugin( | ||||
|                         "comittish": ["rc", "master"], | ||||
|                     } | ||||
|                 ], | ||||
|                 "pip": "https://github.com/jneilliii/OctoPrint-BambuPrinter/archive/{target_version}.zip", | ||||
|                 "pip": "https://gitlab.fire-devils.org/3D-Druck/OctoPrint-BambuPrinter/archive/{target_version}.zip", | ||||
|             } | ||||
|         } | ||||
|   | ||||
| @@ -72,6 +72,7 @@ class BambuVirtualPrinter: | ||||
|  | ||||
|         self._running = True | ||||
|         self._print_status_reporter = None | ||||
|         self._print_temp_reporter = None | ||||
|         self._printer_thread = threading.Thread( | ||||
|             target=self._printer_worker, | ||||
|             name="octoprint.plugins.bambu_printer.printer_state", | ||||
| @@ -192,7 +193,7 @@ class BambuVirtualPrinter: | ||||
|             or print_job_state == "FAILED" | ||||
|         ): | ||||
|             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) | ||||
|         elif print_job_state == "PAUSE": | ||||
|             self.change_state(self._state_paused) | ||||
| @@ -243,11 +244,7 @@ class BambuVirtualPrinter: | ||||
|             device_type=self._settings.get(["device_type"]), | ||||
|             serial=self._settings.get(["serial"]), | ||||
|             host=self._settings.get(["host"]), | ||||
|             username=( | ||||
|                 "bblp" | ||||
|                 if self._settings.get_boolean(["local_mqtt"]) | ||||
|                 else self._settings.get(["username"]) | ||||
|             ), | ||||
|             username=("bambuocto"), | ||||
|             access_code=self._settings.get(["access_code"]), | ||||
|             local_mqtt=self._settings.get_boolean(["local_mqtt"]), | ||||
|             region=self._settings.get(["region"]), | ||||
| @@ -367,8 +364,10 @@ class BambuVirtualPrinter: | ||||
|             interval = int(matchS.group(1)) | ||||
|             if interval > 0: | ||||
|                 self.start_continuous_status_report(interval) | ||||
|                 return False | ||||
|             else: | ||||
|                 self.stop_continuous_status_report() | ||||
|                 return False | ||||
|  | ||||
|         self.report_print_job_status() | ||||
|         return True | ||||
| @@ -403,10 +402,39 @@ class BambuVirtualPrinter: | ||||
|         self._processTemperatureQuery() | ||||
|         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 | ||||
|     @gcode_executor.register_no_data("M115") | ||||
|     def _report_firmware_info(self) -> bool: | ||||
|         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:LFN_WRITE:1") | ||||
|         return True | ||||
| @@ -442,19 +470,51 @@ class BambuVirtualPrinter: | ||||
|             gcode_command = commands.SEND_GCODE_TEMPLATE | ||||
|             percent = int(data.replace("M220 S", "")) | ||||
|  | ||||
|             if percent is None or percent < 1 or percent > 166: | ||||
|                 return True | ||||
|             def speed_fraction(speed_percent): | ||||
|                 return math.floor(10000 / speed_percent) / 100 | ||||
|  | ||||
|             speed_fraction = 100 / percent | ||||
|             acceleration = math.exp((speed_fraction - 1.0191) / -0.814) | ||||
|             feed_rate = ( | ||||
|                 2.1645 * (acceleration**3) | ||||
|                 - 5.3247 * (acceleration**2) | ||||
|                 + 4.342 * acceleration | ||||
|                 - 0.181 | ||||
|             ) | ||||
|             speed_level = 1.539 * (acceleration**2) - 0.7032 * acceleration + 4.0834 | ||||
|             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" | ||||
|             def acceleration_magnitude(speed_percent): | ||||
|                 return math.exp((speed_fraction(speed_percent) - 1.0191) / -0.8139) | ||||
|  | ||||
|             def feed_rate(speed_percent): | ||||
|                 return 6.426e-5 * speed_percent ** 2 - 2.484e-3 * speed_percent + 0.654 | ||||
|  | ||||
|             def linear_interpolate(x, x_points, y_points): | ||||
|                 if x <= x_points[0]: return y_points[0] | ||||
|                 if x >= x_points[-1]: return y_points[-1] | ||||
|                 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 | ||||
|             if self.bambu_client.publish(gcode_command): | ||||
| @@ -516,8 +576,9 @@ class BambuVirtualPrinter: | ||||
|  | ||||
|     def report_print_job_status(self): | ||||
|         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( | ||||
|                 f"SD printing byte {self.current_print_job.file_position}" | ||||
|                 f"SD printing byte {file_position}" | ||||
|                 f"/{self.current_print_job.file_info.size}" | ||||
|             ) | ||||
|         else: | ||||
|   | ||||
| @@ -85,7 +85,7 @@ class CachedFileView: | ||||
|         return file_data | ||||
|  | ||||
|     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) | ||||
|             if file_stem == file_path.with_suffix("").stem and all( | ||||
|                 suffix in allowed_suffixes for suffix in file_path.suffixes | ||||
|   | ||||
| @@ -117,7 +117,7 @@ class IoTFTPSConnection: | ||||
|                         # But since we operate in prot p mode | ||||
|                         # we can close the connection always. | ||||
|                         # This is cursed but it works. | ||||
|                         if "vsFTPd" in self.welcome: | ||||
|                         if "vsFTPd" in self.ftps_session.welcome: | ||||
|                             conn.unwrap() | ||||
|                         else: | ||||
|                             conn.shutdown(socket.SHUT_RDWR) | ||||
|   | ||||
| @@ -1,8 +1,6 @@ | ||||
| from __future__ import annotations | ||||
| from pathlib import Path | ||||
|  | ||||
| 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 | ||||
|  | ||||
|  | ||||
| @@ -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" | ||||
|         filesystem_root = ( | ||||
|             "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:///" | ||||
|         ) | ||||
|  | ||||
|   | ||||
| @@ -22,6 +22,7 @@ class PrintingState(APrinterState): | ||||
|  | ||||
|     def __init__(self, printer: BambuVirtualPrinter) -> None: | ||||
|         super().__init__(printer) | ||||
|         self._current_print_job = None | ||||
|         self._is_printing = False | ||||
|         self._sd_printing_thread = None | ||||
|  | ||||
| @@ -36,6 +37,7 @@ class PrintingState(APrinterState): | ||||
|             self._is_printing = False | ||||
|             self._sd_printing_thread.join() | ||||
|             self._sd_printing_thread = None | ||||
|         self._printer.current_print_job = None | ||||
|  | ||||
|     def _start_worker_thread(self): | ||||
|         if self._sd_printing_thread is None: | ||||
|   | ||||
| @@ -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 | ||||
|   | ||||
							
								
								
									
										8
									
								
								setup.py
									
									
									
									
									
								
							
							
						
						
									
										8
									
								
								setup.py
									
									
									
									
									
								
							| @@ -14,20 +14,20 @@ plugin_package = "octoprint_bambu_printer" | ||||
| plugin_name = "OctoPrint-BambuPrinter" | ||||
|  | ||||
| # 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 = "1.0.0" | ||||
|  | ||||
| # The plugin's description. Can be overwritten within OctoPrint's internal data via __plugin_description__ in the plugin | ||||
| # module | ||||
| plugin_description = """Connects OctoPrint to BambuLabs printers.""" | ||||
|  | ||||
| # The plugin's author. Can be overwritten within OctoPrint's internal data via __plugin_author__ in the plugin module | ||||
| plugin_author = "jneilliii" | ||||
| plugin_author = "ManuelW" | ||||
|  | ||||
| # The plugin's author's mail address. | ||||
| plugin_author_email = "jneilliii+github@gmail.com" | ||||
| plugin_author_email = "manuelw@example.com" | ||||
|  | ||||
| # The plugin's homepage URL. Can be overwritten within OctoPrint's internal data via __plugin_url__ in the plugin module | ||||
| plugin_url = "https://github.com/jneilliii/OctoPrint-BambuPrinter" | ||||
| plugin_url = "https://gitlab.fire-devils.org/3D-Druck/OctoPrint-BambuPrinter" | ||||
|  | ||||
| # The plugin's license. Can be overwritten within OctoPrint's internal data via __plugin_license__ in the plugin module | ||||
| plugin_license = "AGPLv3" | ||||
|   | ||||
		Reference in New Issue
	
	Block a user