Compare commits
	
		
			50 Commits
		
	
	
		
			v1.3.44
			...
			1b059c35f1
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 1b059c35f1 | |||
| e098d71f6f | |||
| 4b25b72b2e | |||
| 5c59016f94 | |||
| d2da501b94 | |||
| 4135073623 | |||
| fe7b57fe0e | |||
| c1ae6b7295 | |||
| 9eee89fac7 | |||
| 8c5e7e26ac | |||
| 7b52066378 | |||
| d5afa38ded | |||
| cf50baba2d | |||
| aa9e7da94b | |||
| 71cd3ba4fc | |||
| 73e240e879 | |||
| 0d34e1d718 | |||
| 84cc8beb9b | |||
| fd70e3179d | |||
| c553640ad8 | |||
| 807eca3c43 | |||
| b52730bf67 | |||
| 9a59b91e88 | |||
| a5af4013d8 | |||
| e54ce58ec4 | |||
| 142eafd232 | |||
| 63ab9e0993 | |||
| aaa5506d40 | |||
| 8037adc045 | |||
| 6e7c728cd8 | |||
| 3fe8271344 | |||
| f2bc6eab92 | |||
| 37df492339 | |||
| c4b425403f | |||
| 73244689dd | |||
| 27296104d2 | |||
| 5f99773897 | |||
| 7416285fb9 | |||
| 85928e358d | |||
| 092b4fd8ec | |||
| 399645a2b3 | |||
| 164bb241b7 | |||
| e564c6eeae | |||
| 4288dd0cd4 | |||
| 37d43b2d7d | |||
| adb354ddcd | |||
| 15d5e5edce | |||
| c6edf30245 | |||
| 65ac207f36 | |||
| 698abbd669 | 
							
								
								
									
										2
									
								
								.github/workflows/gitea-release.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/gitea-release.yml
									
									
									
									
										vendored
									
									
								
							| @@ -40,7 +40,7 @@ jobs: | ||||
|         # Copy firmware binary | ||||
|         cp .pio/build/esp32dev/firmware.bin .pio/build/esp32dev/upgrade_filaman_firmware_v${VERSION}.bin | ||||
|          | ||||
|         # Create SPIFFS binary (without header) | ||||
|         # Create SPIFFS binary - direct copy without header | ||||
|         cp .pio/build/esp32dev/spiffs.bin .pio/build/esp32dev/upgrade_filaman_website_v${VERSION}.bin | ||||
|          | ||||
|         # Create full binary | ||||
|   | ||||
							
								
								
									
										2
									
								
								.github/workflows/github-release.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/github-release.yml
									
									
									
									
										vendored
									
									
								
							| @@ -47,7 +47,7 @@ jobs: | ||||
|         # Copy firmware binary | ||||
|         cp .pio/build/esp32dev/firmware.bin .pio/build/esp32dev/upgrade_filaman_firmware_v${VERSION}.bin | ||||
|          | ||||
|         # Create SPIFFS binary (without header) | ||||
|         # Create SPIFFS binary - direct copy without header | ||||
|         cp .pio/build/esp32dev/spiffs.bin .pio/build/esp32dev/upgrade_filaman_website_v${VERSION}.bin | ||||
|          | ||||
|         # Create full binary (always) | ||||
|   | ||||
							
								
								
									
										98
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										98
									
								
								CHANGELOG.md
									
									
									
									
									
								
							| @@ -1,5 +1,103 @@ | ||||
| # Changelog | ||||
|  | ||||
| ## [1.3.59] - 2025-02-22 | ||||
| ### Added | ||||
| - implement enhanced update progress handling and WebSocket notifications | ||||
| - improve update progress reporting and enhance WebSocket notifications | ||||
| - enhance update progress handling and add WebSocket closure notification | ||||
| - implement WebSocket for update progress and enhance update response handling | ||||
|  | ||||
| ### Changed | ||||
| - update webpages for version v1.3.59 | ||||
|  | ||||
|  | ||||
| ## [1.3.58] - 2025-02-22 | ||||
| ### Added | ||||
| - implement backup and restore functionality for Bambu credentials and Spoolman URL | ||||
|  | ||||
| ### Changed | ||||
| - update webpages for version v1.3.58 | ||||
| - update upgrade page message and improve progress display logic | ||||
|  | ||||
|  | ||||
| ## [1.3.57] - 2025-02-22 | ||||
| ### Changed | ||||
| - update webpages for version v1.3.57 | ||||
| - update header title to 'Filament Management Tool' in multiple HTML files | ||||
|  | ||||
|  | ||||
| ## [1.3.56] - 2025-02-22 | ||||
| ### Changed | ||||
| - update webpages for version v1.3.56 | ||||
| - update header title and improve SPIFFS update error handling | ||||
| - clarify comments in Gitea and GitHub release workflows | ||||
|  | ||||
|  | ||||
| ## [1.3.55] - 2025-02-22 | ||||
| ### Changed | ||||
| - update webpages for version v1.3.55 | ||||
| - update component descriptions in README files | ||||
|  | ||||
|  | ||||
| ## [1.3.54] - 2025-02-22 | ||||
| ### Changed | ||||
| - update webpages for version v1.3.54 | ||||
| - workflow: update SPIFFS binary creation to exclude header | ||||
|  | ||||
|  | ||||
| ## [1.3.53] - 2025-02-22 | ||||
| ### Changed | ||||
| - version: update to version 1.3.53 | ||||
| - update changelog for version 1.3.51 | ||||
| - update changelog for version 1.3.51 | ||||
| - workflow: update SPIFFS binary magic byte and revert version to 1.3.51 | ||||
|  | ||||
|  | ||||
| ## [1.3.52] - 2025-02-22 | ||||
| ### Changed | ||||
| - update webpages for version v1.3.52 | ||||
| - workflow: update SPIFFS binary creation to use correct chip revision (0xEB for Rev 3) | ||||
|  | ||||
|  | ||||
| ## [1.3.51] - 2025-02-22 | ||||
| ### Changed | ||||
| - update changelog for version 1.3.51 | ||||
| - workflow: update SPIFFS binary magic byte and revert version to 1.3.51 | ||||
|  | ||||
| ## [1.3.50] - 2025-02-22 | ||||
| ### Changed | ||||
| - update webpages for version v1.3.50 | ||||
|  | ||||
|  | ||||
| ## [1.3.49] - 2025-02-22 | ||||
| ### Changed | ||||
| - update webpages for version v1.3.49 | ||||
| - workflow: update SPIFFS binary header to use correct chip revision | ||||
|  | ||||
|  | ||||
| ## [1.3.48] - 2025-02-22 | ||||
| ### Changed | ||||
| - update webpages for version v1.3.48 | ||||
| - workflow: update SPIFFS binary header for firmware release | ||||
|  | ||||
|  | ||||
| ## [1.3.47] - 2025-02-22 | ||||
| ### Changed | ||||
| - update webpages for version v1.3.47 | ||||
| - workflow: optimize firmware and SPIFFS update process, improve progress handling and logging | ||||
|  | ||||
|  | ||||
| ## [1.3.46] - 2025-02-22 | ||||
| ### Changed | ||||
| - update webpages for version v1.3.46 | ||||
|  | ||||
|  | ||||
| ## [1.3.45] - 2025-02-22 | ||||
| ### Changed | ||||
| - update webpages for version v1.3.45 | ||||
| - workflow: update SPIFFS binary creation to include minimal header and adjust update validation logic | ||||
|  | ||||
|  | ||||
| ## [1.3.44] - 2025-02-22 | ||||
| ### Changed | ||||
| - update webpages for version v1.3.44 | ||||
|   | ||||
							
								
								
									
										16
									
								
								README.de.md
									
									
									
									
									
								
							
							
						
						
									
										16
									
								
								README.de.md
									
									
									
									
									
								
							| @@ -53,14 +53,14 @@ Deutsches Erklärvideo: [Youtube](https://youtu.be/uNDe2wh9SS8?si=b-jYx4I1w62zaO | ||||
| ### Komponenten | ||||
| - **ESP32 Entwicklungsboard:** Jede ESP32-Variante. | ||||
| [Amazon Link](https://amzn.eu/d/aXThslf) | ||||
| - **HX711 Wägezellen-Verstärker:** Für Gewichtsmessung. | ||||
| [Amazon Link](https://amzn.eu/d/1wZ4v0x) | ||||
| - **OLED Display:** 128x64 SSD1306. | ||||
| [Amazon Link](https://amzn.eu/d/dozAYDU) | ||||
| - **PN532 NFC Modul:** Für NFC-Tag-Operationen. | ||||
| [Amazon Link](https://amzn.eu/d/8205DDh) | ||||
| - **NFC-Tag:** NTAG215 | ||||
| [Amazon Link](https://amzn.eu/d/fywy4c4) | ||||
| - **HX711 5kg Wägezellen-Verstärker:** Für Gewichtsmessung. | ||||
| [Amazon Link](https://amzn.eu/d/06A0DLb) | ||||
| - **OLED 0.96 Zoll I2C weiß/gelb Display:** 128x64 SSD1306. | ||||
| [Amazon Link](https://amzn.eu/d/0AuBp2c) | ||||
| - **PN532 NFC NXP RFID-Modul V3:** Für NFC-Tag-Operationen. | ||||
| [Amazon Link](https://amzn.eu/d/jfIuQXb) | ||||
| - **NFC Tags Ntag215:** RFID Tag | ||||
| [Amazon Link](https://amzn.eu/d/9Z6mXc1) | ||||
|  | ||||
| ### Pin-Konfiguration | ||||
| | Komponente        | ESP32 Pin | | ||||
|   | ||||
							
								
								
									
										16
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										16
									
								
								README.md
									
									
									
									
									
								
							| @@ -56,14 +56,14 @@ german explanatory video: [Youtube](https://youtu.be/uNDe2wh9SS8?si=b-jYx4I1w62z | ||||
| ### Components | ||||
| - **ESP32 Development Board:** Any ESP32 variant. | ||||
| [Amazon Link](https://amzn.eu/d/aXThslf) | ||||
| - **HX711 Load Cell Amplifier:** For weight measurement. | ||||
| [Amazon Link](https://amzn.eu/d/1wZ4v0x) | ||||
| - **OLED Display:** 128x64 SSD1306. | ||||
| [Amazon Link](https://amzn.eu/d/dozAYDU) | ||||
| - **PN532 NFC Module:** For NFC tag operations. | ||||
| [Amazon Link](https://amzn.eu/d/8205DDh) | ||||
| - **NFC-Tag:** NTAG215 | ||||
| [Amazon Link](https://amzn.eu/d/fywy4c4) | ||||
| - **HX711 5kg Load Cell Amplifier:** For weight measurement. | ||||
| [Amazon Link](https://amzn.eu/d/06A0DLb) | ||||
| - **OLED 0.96 Zoll I2C white/yellow Display:** 128x64 SSD1306. | ||||
| [Amazon Link](https://amzn.eu/d/0AuBp2c) | ||||
| - **PN532 NFC NXP RFID-Modul V3:** For NFC tag operations. | ||||
| [Amazon Link](https://amzn.eu/d/jfIuQXb) | ||||
| - **NFC Tags Ntag215:** RFID Tag | ||||
| [Amazon Link](https://amzn.eu/d/9Z6mXc1) | ||||
|  | ||||
|  | ||||
| ### Pin Configuration | ||||
|   | ||||
| @@ -24,7 +24,7 @@ | ||||
|             <img src="/logo.png" alt="FilaMan Logo" class="logo"> | ||||
|             <div class="logo-text"> | ||||
|                 <h1>FilaMan<span class="version"></span></h1> | ||||
|                 <h4>Hollo Lollo Trollo</h4> | ||||
|                 <h4>Filament Management Tool</h4> | ||||
|             </div> | ||||
|         </div> | ||||
|         <nav style="display: flex; gap: 1rem;"> | ||||
|   | ||||
| @@ -154,18 +154,108 @@ | ||||
|         const progress = document.querySelector('.progress-bar'); | ||||
|         const progressContainer = document.querySelector('.progress-container'); | ||||
|         const status = document.querySelector('.status'); | ||||
|         let updateInProgress = false; | ||||
|         let lastReceivedProgress = 0; | ||||
|  | ||||
|         // WebSocket Handling | ||||
|         let ws = null; | ||||
|         let wsReconnectTimer = null; | ||||
|  | ||||
|         function connectWebSocket() { | ||||
|             ws = new WebSocket('ws://' + window.location.host + '/ws'); | ||||
|              | ||||
|             ws.onmessage = function(event) { | ||||
|                 try { | ||||
|                     const data = JSON.parse(event.data); | ||||
|                     if (data.type === "updateProgress" && updateInProgress) { | ||||
|                         // Zeige Fortschrittsbalken | ||||
|                         progressContainer.style.display = 'block'; | ||||
|                          | ||||
|                         // Aktualisiere den Fortschritt nur wenn er größer ist | ||||
|                         const newProgress = parseInt(data.progress); | ||||
|                         if (!isNaN(newProgress) && newProgress >= lastReceivedProgress) { | ||||
|                             progress.style.width = newProgress + '%'; | ||||
|                             progress.textContent = newProgress + '%'; | ||||
|                             lastReceivedProgress = newProgress; | ||||
|                         } | ||||
|                          | ||||
|                         // Zeige Status-Nachricht | ||||
|                         if (data.message || data.status) { | ||||
|                             status.textContent = data.message || getStatusMessage(data.status); | ||||
|                             status.className = 'status success'; | ||||
|                             status.style.display = 'block'; | ||||
|                              | ||||
|                             // Starte Reload wenn Update erfolgreich | ||||
|                             if (data.status === 'success' || lastReceivedProgress >= 98) { | ||||
|                                 clearTimeout(wsReconnectTimer); | ||||
|                                 setTimeout(() => { | ||||
|                                     window.location.href = '/'; | ||||
|                                 }, 30000); | ||||
|                             } | ||||
|                         } | ||||
|                     } | ||||
|                 } catch (e) { | ||||
|                     console.error('WebSocket message error:', e); | ||||
|                 } | ||||
|             }; | ||||
|  | ||||
|             ws.onclose = function() { | ||||
|                 if (updateInProgress) { | ||||
|                     // Wenn der Fortschritt hoch genug ist, gehen wir von einem erfolgreichen Update aus | ||||
|                     if (lastReceivedProgress >= 85) { | ||||
|                         status.textContent = "Update appears successful! Device is restarting... Page will reload in 30 seconds."; | ||||
|                         status.className = 'status success'; | ||||
|                         status.style.display = 'block'; | ||||
|                         clearTimeout(wsReconnectTimer); | ||||
|                         setTimeout(() => { | ||||
|                             window.location.href = '/'; | ||||
|                         }, 30000); | ||||
|                     } else { | ||||
|                         // Versuche Reconnect bei niedrigem Fortschritt | ||||
|                         wsReconnectTimer = setTimeout(connectWebSocket, 1000); | ||||
|                     } | ||||
|                 } | ||||
|             }; | ||||
|  | ||||
|             ws.onerror = function(err) { | ||||
|                 console.error('WebSocket error:', err); | ||||
|                 if (updateInProgress && lastReceivedProgress >= 85) { | ||||
|                     status.textContent = "Update appears successful! Device is restarting... Page will reload in 30 seconds."; | ||||
|                     status.className = 'status success'; | ||||
|                     status.style.display = 'block'; | ||||
|                     setTimeout(() => { | ||||
|                         window.location.href = '/'; | ||||
|                     }, 30000); | ||||
|                 } | ||||
|             }; | ||||
|         } | ||||
|  | ||||
|         // Initial WebSocket connection | ||||
|         connectWebSocket(); | ||||
|  | ||||
|         function getStatusMessage(status) { | ||||
|             switch(status) { | ||||
|                 case 'starting': return 'Starting update...'; | ||||
|                 case 'uploading': return 'Uploading...'; | ||||
|                 case 'finalizing': return 'Finalizing update...'; | ||||
|                 case 'restoring': return 'Restoring configurations...'; | ||||
|                 case 'preparing': return 'Preparing for restart...'; | ||||
|                 case 'success': return 'Update successful! Device is restarting... Page will reload in 30 seconds.'; | ||||
|                 default: return 'Updating...'; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         function handleUpdate(e) { | ||||
|             e.preventDefault(); | ||||
|             const form = e.target; | ||||
|             const file = form.update.files[0]; | ||||
|             const updateType = form.dataset.type; | ||||
|  | ||||
|              | ||||
|             if (!file) { | ||||
|                 alert('Please select a file.'); | ||||
|                 return; | ||||
|             } | ||||
|  | ||||
|              | ||||
|             // Validate file name pattern | ||||
|             if (updateType === 'firmware' && !file.name.startsWith('upgrade_filaman_firmware_')) { | ||||
|                 alert('Please select a valid firmware file (upgrade_filaman_firmware_*.bin)'); | ||||
| @@ -176,83 +266,39 @@ | ||||
|                 return; | ||||
|             } | ||||
|              | ||||
|             // Reset UI | ||||
|             updateInProgress = true; | ||||
|             progressContainer.style.display = 'block'; | ||||
|             status.style.display = 'none'; | ||||
|             status.className = 'status'; | ||||
|  | ||||
|             // Reset progress bar | ||||
|             progress.style.width = '0%'; | ||||
|             progress.textContent = '0%'; | ||||
|  | ||||
|             // Disable both forms during update | ||||
|              | ||||
|             // Disable submit buttons | ||||
|             document.querySelectorAll('form input[type=submit]').forEach(btn => btn.disabled = true); | ||||
|  | ||||
|             // Send update | ||||
|             const xhr = new XMLHttpRequest(); | ||||
|             xhr.open('POST', '/update', true); | ||||
|             xhr.upload.onprogress = (e) => { | ||||
|                 if (e.lengthComputable) { | ||||
|                     const percentComplete = (e.loaded / e.total) * 100; | ||||
|                     progress.style.width = percentComplete + '%'; | ||||
|                     progress.textContent = Math.round(percentComplete) + '%'; | ||||
|                 } | ||||
|             }; | ||||
|  | ||||
|              | ||||
|             xhr.onload = function() { | ||||
|                 try { | ||||
|                     let response = this.responseText; | ||||
|                     try { | ||||
|                         const jsonResponse = JSON.parse(response); | ||||
|                          | ||||
|                         // Handle progress updates | ||||
|                         if (jsonResponse.progress !== undefined) { | ||||
|                             const percent = jsonResponse.progress; | ||||
|                             progress.style.width = percent + '%'; | ||||
|                             progress.textContent = Math.round(percent) + '%'; | ||||
|                             return; | ||||
|                         } | ||||
|                          | ||||
|                         // Handle success/error messages | ||||
|                         response = jsonResponse.message; | ||||
|                          | ||||
|                         if (jsonResponse.restart) { | ||||
|                             status.textContent = response + " Redirecting in 20 seconds..."; | ||||
|                             let countdown = 20; | ||||
|                             const timer = setInterval(() => { | ||||
|                                 countdown--; | ||||
|                                 if (countdown <= 0) { | ||||
|                                     clearInterval(timer); | ||||
|                                     window.location.href = '/'; | ||||
|                                 } else { | ||||
|                                     status.textContent = response + ` Redirecting in ${countdown} seconds...`; | ||||
|                                 } | ||||
|                             }, 1000); | ||||
|                         } | ||||
|                     } catch (e) { | ||||
|                         console.error('JSON parse error:', e); | ||||
|                         status.textContent = 'Update failed: Invalid response from server'; | ||||
|                         status.classList.add('error'); | ||||
|                     } | ||||
|                      | ||||
|                     status.textContent = response; | ||||
|                     status.classList.add(xhr.status === 200 ? 'success' : 'error'); | ||||
|                     status.style.display = 'block'; | ||||
|                      | ||||
|                     if (xhr.status !== 200) { | ||||
|                         document.querySelectorAll('form input[type=submit]').forEach(btn => btn.disabled = false); | ||||
|                     } | ||||
|                 } catch (error) { | ||||
|                     status.textContent = 'Error: ' + error.message; | ||||
|                     status.classList.add('error'); | ||||
|                 if (xhr.status !== 200 && !progress.textContent.startsWith('100')) { | ||||
|                     status.textContent = "Update failed: " + (xhr.responseText || "Unknown error"); | ||||
|                     status.className = 'status error'; | ||||
|                     status.style.display = 'block'; | ||||
|                     updateInProgress = false; | ||||
|                     document.querySelectorAll('form input[type=submit]').forEach(btn => btn.disabled = false); | ||||
|                 } | ||||
|             }; | ||||
|  | ||||
|              | ||||
|             xhr.onerror = function() { | ||||
|                 status.textContent = 'Update failed: Network error'; | ||||
|                 status.classList.add('error'); | ||||
|                 status.style.display = 'block'; | ||||
|                 document.querySelectorAll('form input[type=submit]').forEach(btn => btn.disabled = false); | ||||
|                 if (!progress.textContent.startsWith('100')) { | ||||
|                     status.textContent = "Network error during update"; | ||||
|                     status.className = 'status error'; | ||||
|                     status.style.display = 'block'; | ||||
|                     updateInProgress = false; | ||||
|                     document.querySelectorAll('form input[type=submit]').forEach(btn => btn.disabled = false); | ||||
|                 } | ||||
|             }; | ||||
|  | ||||
|             const formData = new FormData(); | ||||
|   | ||||
| @@ -9,7 +9,7 @@ | ||||
| ; https://docs.platformio.org/page/projectconf.html | ||||
|  | ||||
| [common] | ||||
| version = "1.3.44" | ||||
| version = "1.3.59" | ||||
|  | ||||
| #test | ||||
|  | ||||
|   | ||||
							
								
								
									
										291
									
								
								src/website.cpp
									
									
									
									
									
								
							
							
						
						
									
										291
									
								
								src/website.cpp
									
									
									
									
									
								
							| @@ -23,6 +23,49 @@ AsyncWebSocket ws("/ws"); | ||||
| uint8_t lastSuccess = 0; | ||||
| uint8_t lastHasReadRfidTag = 0; | ||||
|  | ||||
| // Globale Variablen für Config Backups hinzufügen | ||||
| String bambuCredentialsBackup; | ||||
| String spoolmanUrlBackup; | ||||
|  | ||||
| // Globale Variable für den Update-Typ | ||||
| static int currentUpdateCommand = 0; | ||||
|  | ||||
| // Globale Update-Variablen | ||||
| static size_t updateTotalSize = 0; | ||||
| static size_t updateWritten = 0; | ||||
| static bool isSpiffsUpdate = false; | ||||
|  | ||||
| void sendUpdateProgress(int progress, const char* status = nullptr, const char* message = nullptr) { | ||||
|     static int lastSentProgress = -1; | ||||
|      | ||||
|     // Verhindere zu häufige Updates | ||||
|     if (progress == lastSentProgress && !status && !message) { | ||||
|         return; | ||||
|     } | ||||
|      | ||||
|     String progressMsg = "{\"type\":\"updateProgress\",\"progress\":" + String(progress); | ||||
|     if (status) { | ||||
|         progressMsg += ",\"status\":\"" + String(status) + "\""; | ||||
|     } | ||||
|     if (message) { | ||||
|         progressMsg += ",\"message\":\"" + String(message) + "\""; | ||||
|     } | ||||
|     progressMsg += "}"; | ||||
|      | ||||
|     // Sende die Nachricht mehrmals mit Verzögerung für wichtige Updates | ||||
|     if (status || abs(progress - lastSentProgress) >= 10 || progress == 100) { | ||||
|         for (int i = 0; i < 2; i++) { | ||||
|             ws.textAll(progressMsg); | ||||
|             delay(100);  // Längerer Delay zwischen Nachrichten | ||||
|         } | ||||
|     } else { | ||||
|         ws.textAll(progressMsg); | ||||
|         delay(50); | ||||
|     } | ||||
|      | ||||
|     lastSentProgress = progress; | ||||
| } | ||||
|  | ||||
| void onWsEvent(AsyncWebSocket *server, AsyncWebSocketClient *client, AwsEventType type, void *arg, uint8_t *data, size_t len) { | ||||
|     if (type == WS_EVT_CONNECT) { | ||||
|         Serial.println("Neuer Client verbunden!"); | ||||
| @@ -164,7 +207,109 @@ void sendAmsData(AsyncWebSocketClient *client) { | ||||
|     } | ||||
| } | ||||
|  | ||||
| void handleUpdate(AsyncWebServer &server) { | ||||
|     AsyncCallbackWebHandler* updateHandler = new AsyncCallbackWebHandler(); | ||||
|     updateHandler->setUri("/update"); | ||||
|     updateHandler->setMethod(HTTP_POST); | ||||
|      | ||||
|     updateHandler->onUpload([](AsyncWebServerRequest *request, String filename, | ||||
|                              size_t index, uint8_t *data, size_t len, bool final) { | ||||
|         if (!index) { | ||||
|             updateTotalSize = request->contentLength(); | ||||
|             updateWritten = 0; | ||||
|             isSpiffsUpdate = (filename.indexOf("website") > -1); | ||||
|              | ||||
|             if (isSpiffsUpdate) { | ||||
|                 // Backup vor dem Update | ||||
|                 sendUpdateProgress(0, "backup", "Backing up configurations..."); | ||||
|                 delay(200); | ||||
|                 backupJsonConfigs(); | ||||
|                 delay(200); | ||||
|                  | ||||
|                 const esp_partition_t *partition = esp_partition_find_first(ESP_PARTITION_TYPE_DATA, ESP_PARTITION_SUBTYPE_DATA_SPIFFS, NULL); | ||||
|                 if (!partition || !Update.begin(partition->size, U_SPIFFS)) { | ||||
|                     request->send(400, "application/json", "{\"success\":false,\"message\":\"Update initialization failed\"}"); | ||||
|                     return; | ||||
|                 } | ||||
|                 sendUpdateProgress(5, "starting", "Starting SPIFFS update..."); | ||||
|                 delay(200); | ||||
|             } else { | ||||
|                 if (!Update.begin(updateTotalSize)) { | ||||
|                     request->send(400, "application/json", "{\"success\":false,\"message\":\"Update initialization failed\"}"); | ||||
|                     return; | ||||
|                 } | ||||
|                 sendUpdateProgress(0, "starting", "Starting firmware update..."); | ||||
|                 delay(200); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         if (len) { | ||||
|             if (Update.write(data, len) != len) { | ||||
|                 request->send(400, "application/json", "{\"success\":false,\"message\":\"Write failed\"}"); | ||||
|                 return; | ||||
|             } | ||||
|              | ||||
|             updateWritten += len; | ||||
|             int currentProgress; | ||||
|              | ||||
|             // Berechne den Fortschritt basierend auf dem Update-Typ | ||||
|             if (isSpiffsUpdate) { | ||||
|                 // SPIFFS: 5-75% für Upload | ||||
|                 currentProgress = 5 + (updateWritten * 100) / updateTotalSize; | ||||
|             } else { | ||||
|                 // Firmware: 0-100% für Upload | ||||
|                 currentProgress = 1 + (updateWritten * 100) / updateTotalSize; | ||||
|             } | ||||
|              | ||||
|             static int lastProgress = -1; | ||||
|             if (currentProgress != lastProgress && (currentProgress % 10 == 0 || final)) { | ||||
|                 sendUpdateProgress(currentProgress, "uploading"); | ||||
|                 oledShowMessage("Update: " + String(currentProgress) + "%"); | ||||
|                 delay(50); | ||||
|                 lastProgress = currentProgress; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         if (final) { | ||||
|             if (Update.end(true)) { | ||||
|                 if (isSpiffsUpdate) { | ||||
|                     restoreJsonConfigs(); | ||||
|                 } | ||||
|             } else { | ||||
|                 request->send(400, "application/json", "{\"success\":false,\"message\":\"Update finalization failed\"}"); | ||||
|             } | ||||
|         } | ||||
|     }); | ||||
|  | ||||
|     updateHandler->onRequest([](AsyncWebServerRequest *request) { | ||||
|         if (Update.hasError()) { | ||||
|             request->send(400, "application/json", "{\"success\":false,\"message\":\"Update failed\"}"); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         // Erste 100% Nachricht | ||||
|         ws.textAll("{\"type\":\"updateProgress\",\"progress\":100,\"status\":\"success\",\"message\":\"Update successful! Restarting device...\"}"); | ||||
|         delay(2000);  // Längerer Delay für die erste Nachricht | ||||
|          | ||||
|         AsyncWebServerResponse *response = request->beginResponse(200, "application/json",  | ||||
|             "{\"success\":true,\"message\":\"Update successful! Restarting device...\"}"); | ||||
|         response->addHeader("Connection", "close"); | ||||
|         request->send(response); | ||||
|          | ||||
|         // Zweite 100% Nachricht zur Sicherheit | ||||
|         ws.textAll("{\"type\":\"updateProgress\",\"progress\":100,\"status\":\"success\",\"message\":\"Update successful! Restarting device...\"}"); | ||||
|         delay(3000);  // Noch längerer Delay vor dem Neustart | ||||
|          | ||||
|         ESP.restart(); | ||||
|     }); | ||||
|  | ||||
|     server.addHandler(updateHandler); | ||||
| } | ||||
|  | ||||
| void setupWebserver(AsyncWebServer &server) { | ||||
|     // Deaktiviere alle Debug-Ausgaben | ||||
|     Serial.setDebugOutput(false); | ||||
|      | ||||
|     // WebSocket-Optimierungen | ||||
|     ws.onEvent(onWsEvent); | ||||
|     ws.enable(true); | ||||
| @@ -363,102 +508,8 @@ void setupWebserver(AsyncWebServer &server) { | ||||
|         request->send(response); | ||||
|     }); | ||||
|  | ||||
|     // Update-Handler mit verbesserter Fehlerbehandlung | ||||
|     server.on("/update", HTTP_POST,  | ||||
|         [](AsyncWebServerRequest *request) { | ||||
|             // Nach Update-Abschluss | ||||
|             bool success = !Update.hasError(); | ||||
|             String message = success ? "Update successful" : String("Update failed: ") + Update.errorString(); | ||||
|             AsyncWebServerResponse *response = request->beginResponse( | ||||
|                 success ? 200 : 400, | ||||
|                 "application/json", | ||||
|                 "{\"success\":" + String(success ? "true" : "false") + ",\"message\":\"" + message + "\"}" | ||||
|             ); | ||||
|             response->addHeader("Connection", "close"); | ||||
|             request->send(response); | ||||
|              | ||||
|             if (success) { | ||||
|                 oledShowMessage("Upgrade successful Rebooting"); | ||||
|                 delay(500); | ||||
|                 ESP.restart(); | ||||
|             } | ||||
|             else { | ||||
|                 oledShowMessage("Upgrade failed"); | ||||
|             } | ||||
|         }, | ||||
|         [](AsyncWebServerRequest *request, const String& filename, size_t index, uint8_t *data, size_t len, bool final) { | ||||
|             static size_t updateSize = 0; | ||||
|             static int command = 0; | ||||
|  | ||||
|             oledShowMessage("Upgrade please wait"); | ||||
|  | ||||
|             if (!index) { | ||||
|                 updateSize = request->contentLength(); | ||||
|                 command = (filename.indexOf("spiffs") > -1) ? U_SPIFFS : U_FLASH; | ||||
|                 Serial.printf("Update Start: %s\nSize: %u\nCommand: %d\n", filename.c_str(), updateSize, command); | ||||
|                  | ||||
|                 // Überprüfe die SPIFFS-Größe | ||||
|                 if (command == U_SPIFFS && updateSize > 0x30000) { | ||||
|                     String errorMsg = "SPIFFS update too large. Maximum size is 192KB"; | ||||
|                     Serial.println(errorMsg); | ||||
|                     request->send(400, "application/json", "{\"success\":false,\"message\":\"" + errorMsg + "\"}"); | ||||
|                     return; | ||||
|                 } | ||||
|  | ||||
|                 if (command == U_SPIFFS) { | ||||
|                     Serial.println("Backup JSON configs..."); | ||||
|                     backupJsonConfigs(); | ||||
|                      | ||||
|                     if (!Update.begin(updateSize, command, false)) { | ||||
|                         Serial.printf("Update Begin Error: %s\n", Update.errorString()); | ||||
|                         Serial.println("Restoring JSON configs..."); | ||||
|                         restoreJsonConfigs(); | ||||
|                         String errorMsg = String("Update begin failed: ") + Update.errorString(); | ||||
|                         request->send(400, "application/json", "{\"success\":false,\"message\":\"" + errorMsg + "\"}"); | ||||
|                         return; | ||||
|                     } | ||||
|                 } else { | ||||
|                     if (!Update.begin(updateSize, command)) { | ||||
|                         Serial.printf("Update Begin Error: %s\n", Update.errorString()); | ||||
|                         String errorMsg = String("Update begin failed: ") + Update.errorString(); | ||||
|                         request->send(400, "application/json", "{\"success\":false,\"message\":\"" + errorMsg + "\"}"); | ||||
|                         return; | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             if (len) { | ||||
|                 if (Update.write(data, len) != len) { | ||||
|                     Serial.printf("Update Write Error: %s\n", Update.errorString()); | ||||
|                     if (command == U_SPIFFS) { | ||||
|                         Serial.println("Restoring JSON configs..."); | ||||
|                         restoreJsonConfigs(); | ||||
|                     } | ||||
|                     String errorMsg = String("Write failed: ") + Update.errorString(); | ||||
|                     request->send(400, "application/json", "{\"success\":false,\"message\":\"" + errorMsg + "\"}"); | ||||
|                     return; | ||||
|                 } | ||||
|                  | ||||
|                 Serial.printf("Progress: %u/%u\n", index + len, updateSize); | ||||
|                 String progress = "{\"progress\":" + String((index + len) * 100 / updateSize) + "}"; | ||||
|                 request->send(200, "application/json", progress); | ||||
|             } | ||||
|  | ||||
|             if (final) { | ||||
|                 if (!Update.end(true)) { | ||||
|                     Serial.printf("Update End Error: %s\n", Update.errorString()); | ||||
|                     if (command == U_SPIFFS) { | ||||
|                         Serial.println("Restoring JSON configs..."); | ||||
|                         restoreJsonConfigs(); | ||||
|                     } | ||||
|                     String errorMsg = String("Update end failed: ") + Update.errorString(); | ||||
|                     request->send(400, "application/json", "{\"success\":false,\"message\":\"" + errorMsg + "\"}"); | ||||
|                     return; | ||||
|                 } | ||||
|                 Serial.println("Update Success!"); | ||||
|             } | ||||
|         } | ||||
|     ); | ||||
|     // Update-Handler registrieren | ||||
|     handleUpdate(server); | ||||
|  | ||||
|     server.on("/api/version", HTTP_GET, [](AsyncWebServerRequest *request){ | ||||
|         String fm_version = VERSION; | ||||
| @@ -485,23 +536,47 @@ void setupWebserver(AsyncWebServer &server) { | ||||
|  | ||||
|  | ||||
| void backupJsonConfigs() { | ||||
|     const char* configs[] = {"/bambu_credentials.json", "/spoolman_url.json"}; | ||||
|     for (const char* config : configs) { | ||||
|         if (SPIFFS.exists(config)) { | ||||
|             String backupPath = String(config) + ".bak"; | ||||
|             SPIFFS.remove(backupPath); | ||||
|             SPIFFS.rename(config, backupPath); | ||||
|     // Bambu Credentials backup | ||||
|     if (SPIFFS.exists("/bambu_credentials.json")) { | ||||
|         File file = SPIFFS.open("/bambu_credentials.json", "r"); | ||||
|         if (file) { | ||||
|             bambuCredentialsBackup = file.readString(); | ||||
|             file.close(); | ||||
|             Serial.println("Bambu credentials backed up"); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     // Spoolman URL backup | ||||
|     if (SPIFFS.exists("/spoolman_url.json")) { | ||||
|         File file = SPIFFS.open("/spoolman_url.json", "r"); | ||||
|         if (file) { | ||||
|             spoolmanUrlBackup = file.readString(); | ||||
|             file.close(); | ||||
|             Serial.println("Spoolman URL backed up"); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| void restoreJsonConfigs() { | ||||
|     const char* configs[] = {"/bambu_credentials.json", "/spoolman_url.json"}; | ||||
|     for (const char* config : configs) { | ||||
|         String backupPath = String(config) + ".bak"; | ||||
|         if (SPIFFS.exists(backupPath)) { | ||||
|             SPIFFS.remove(config); | ||||
|             SPIFFS.rename(backupPath, config); | ||||
|     // Restore Bambu credentials | ||||
|     if (bambuCredentialsBackup.length() > 0) { | ||||
|         File file = SPIFFS.open("/bambu_credentials.json", "w"); | ||||
|         if (file) { | ||||
|             file.print(bambuCredentialsBackup); | ||||
|             file.close(); | ||||
|             Serial.println("Bambu credentials restored"); | ||||
|         } | ||||
|         bambuCredentialsBackup = ""; // Clear backup | ||||
|     } | ||||
|  | ||||
|     // Restore Spoolman URL | ||||
|     if (spoolmanUrlBackup.length() > 0) { | ||||
|         File file = SPIFFS.open("/spoolman_url.json", "w"); | ||||
|         if (file) { | ||||
|             file.print(spoolmanUrlBackup); | ||||
|             file.close(); | ||||
|             Serial.println("Spoolman URL restored"); | ||||
|         } | ||||
|         spoolmanUrlBackup = ""; // Clear backup | ||||
|     } | ||||
| } | ||||
|   | ||||
		Reference in New Issue
	
	Block a user