Compare commits
	
		
			93 Commits
		
	
	
		
			v1.3.35
			...
			c342877558
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| c342877558 | |||
| f5743cbd7b | |||
| 8a62597705 | |||
| 374721d1e5 | |||
| ea6f708c6e | |||
| 78169dfdb1 | |||
| 074bfb658d | |||
| 989076e794 | |||
| aa0d056d10 | |||
| cd619b8f2a | |||
| 6d8358cbb9 | |||
| 1f3a67634f | |||
| 09969b644e | |||
| deb7abd102 | |||
| 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 | |||
| 04a7c2cce3 | |||
| 78f54b72fd | |||
| f4eee9af91 | |||
| cad14b3bc2 | |||
| 312f75fc5f | |||
| b8714e93e2 | |||
| cd9da0fe4f | |||
| 2b620ef5ed | |||
| 3f63a01b8b | |||
| 22bb16b6a4 | |||
| 53ceee7816 | |||
| d48b002806 | |||
| dd905b6c6e | |||
| 77b9eda110 | |||
| 32a6e9dcd3 | |||
| 6cd5539e60 | |||
| 903b697912 | |||
| 72c2fb70c2 | |||
| f2f3f0ab9f | |||
| c07692c218 | |||
| a184903b66 | |||
| af1640383d | |||
| c00e54b145 | |||
| f6c92c686b | |||
| b8db01529b | |||
| 55db6d76ab | |||
| a18749a1ff | |||
| 1811fd9159 | |||
| b550760427 | 
							
								
								
									
										63
									
								
								.github/workflows/gitea-release.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										63
									
								
								.github/workflows/gitea-release.yml
									
									
									
									
										vendored
									
									
								
							| @@ -12,6 +12,8 @@ jobs: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|     - uses: actions/checkout@v4 | ||||
|       with: | ||||
|         fetch-depth: 0 | ||||
|      | ||||
|     - name: Set up Python | ||||
|       uses: actions/setup-python@v4 | ||||
| @@ -40,11 +42,8 @@ jobs: | ||||
|         # Copy firmware binary | ||||
|         cp .pio/build/esp32dev/firmware.bin .pio/build/esp32dev/upgrade_filaman_firmware_v${VERSION}.bin | ||||
|          | ||||
|         # Create SPIFFS binary with ESP32 image header | ||||
|         # Create 16-byte ESP32 image header: (magic byte 0xE9, segment count 0x01, SPI mode DIO, SPI speed 40MHz, chip revision v3.1) | ||||
|         echo -ne '\xE9\x01\x00\x00\x46\x97\x00\x00\x00\x60\x00\x00\x03\x01\x00\x00' > .pio/build/esp32dev/upgrade_filaman_website_v${VERSION}.bin | ||||
|         # Append the actual SPIFFS data | ||||
|         cat .pio/build/esp32dev/spiffs.bin >> .pio/build/esp32dev/upgrade_filaman_website_v${VERSION}.bin | ||||
|         # 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 | ||||
|         (cd .pio/build/esp32dev &&  | ||||
| @@ -57,7 +56,7 @@ jobs: | ||||
|           0x1000 bootloader.bin \ | ||||
|           0x8000 partitions.bin \ | ||||
|           0x10000 firmware.bin \ | ||||
|           0x410000 spiffs.bin) | ||||
|           0x3D0000 spiffs.bin) | ||||
|          | ||||
|         # Verify file sizes | ||||
|         echo "File sizes:" | ||||
| @@ -69,14 +68,50 @@ jobs: | ||||
|         VERSION=$(grep '^version = ' platformio.ini | cut -d'"' -f2) | ||||
|         echo "VERSION=$VERSION" >> $GITHUB_OUTPUT | ||||
|        | ||||
|     - name: Read CHANGELOG.md | ||||
|       id: changelog | ||||
|     - name: Generate Release Notes | ||||
|       id: release_notes | ||||
|       run: | | ||||
|         VERSION=${{ steps.get_version.outputs.VERSION }} | ||||
|         CHANGELOG=$(awk "/## \\[$VERSION\\]/{p=1;print;next} /## \\[/{p=0} p" CHANGELOG.md) | ||||
|         echo "CHANGES<<EOF" >> $GITHUB_OUTPUT | ||||
|         echo "$CHANGELOG" >> $GITHUB_OUTPUT | ||||
|         echo "EOF" >> $GITHUB_OUTPUT | ||||
|         # Get all tags sorted by version | ||||
|         TAGS=($(git tag -l 'v*' --sort=-v:refname)) | ||||
|         CURRENT_TAG="${TAGS[0]}" | ||||
|          | ||||
|         if [ ${#TAGS[@]} -gt 1 ]; then | ||||
|           PREVIOUS_TAG="${TAGS[1]}" | ||||
|           echo "CHANGES<<EOF" >> $GITHUB_OUTPUT | ||||
|           echo "Changes since $PREVIOUS_TAG:" >> $GITHUB_OUTPUT | ||||
|           echo "" >> $GITHUB_OUTPUT | ||||
|            | ||||
|           # Get commits between previous and current tag | ||||
|           echo "### Added" >> $GITHUB_OUTPUT | ||||
|           git log ${PREVIOUS_TAG}..${CURRENT_TAG} --pretty=format:%s | grep -iE '^(feat|add|new)' | sed 's/^feat: /- /' >> $GITHUB_OUTPUT || true | ||||
|           echo "" >> $GITHUB_OUTPUT | ||||
|            | ||||
|           echo "### Fixed" >> $GITHUB_OUTPUT | ||||
|           git log ${PREVIOUS_TAG}..${CURRENT_TAG} --pretty=format:%s | grep -iE '^fix' | sed 's/^fix: /- /' >> $GITHUB_OUTPUT || true | ||||
|           echo "" >> $GITHUB_OUTPUT | ||||
|            | ||||
|           echo "### Changed" >> $GITHUB_OUTPUT | ||||
|           git log ${PREVIOUS_TAG}..${CURRENT_TAG} --pretty=format:%s | grep -ivE '^(feat|fix|add|new)' | sed 's/^/- /' >> $GITHUB_OUTPUT || true | ||||
|           echo "EOF" >> $GITHUB_OUTPUT | ||||
|         else | ||||
|           # First release or no previous tag | ||||
|           echo "CHANGES<<EOF" >> $GITHUB_OUTPUT | ||||
|           echo "Initial Release" >> $GITHUB_OUTPUT | ||||
|            | ||||
|           # Add all commits for initial release | ||||
|           echo "" >> $GITHUB_OUTPUT | ||||
|           echo "### Added" >> $GITHUB_OUTPUT | ||||
|           git log --pretty=format:%s | grep -iE '^(feat|add|new)' | sed 's/^feat: /- /' >> $GITHUB_OUTPUT || true | ||||
|           echo "" >> $GITHUB_OUTPUT | ||||
|            | ||||
|           echo "### Fixed" >> $GITHUB_OUTPUT | ||||
|           git log --pretty=format:%s | grep -iE '^fix' | sed 's/^fix: /- /' >> $GITHUB_OUTPUT || true | ||||
|           echo "" >> $GITHUB_OUTPUT | ||||
|            | ||||
|           echo "### Changed" >> $GITHUB_OUTPUT | ||||
|           git log --pretty=format:%s | grep -ivE '^(feat|fix|add|new)' | sed 's/^/- /' >> $GITHUB_OUTPUT || true | ||||
|           echo "EOF" >> $GITHUB_OUTPUT | ||||
|         fi | ||||
|  | ||||
|     - name: Determine Gitea URL | ||||
|       id: gitea_url | ||||
| @@ -125,7 +160,7 @@ jobs: | ||||
|          | ||||
|         # Erstelle zuerst den Release ohne Dateien | ||||
|         echo "Debug: Creating release..." | ||||
|         RELEASE_DATA="{\"tag_name\":\"v${VERSION}\",\"name\":\"v${VERSION}\",\"body\":\"${{ steps.changelog.outputs.CHANGES }}\"}" | ||||
|         RELEASE_DATA="{\"tag_name\":\"v${VERSION}\",\"name\":\"v${VERSION}\",\"body\":\"${{ steps.release_notes.outputs.CHANGES }}\"}" | ||||
|          | ||||
|         RELEASE_RESPONSE=$(curl -s -w "\n%{http_code}" \ | ||||
|           -X POST \ | ||||
|   | ||||
							
								
								
									
										61
									
								
								.github/workflows/github-release.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										61
									
								
								.github/workflows/github-release.yml
									
									
									
									
										vendored
									
									
								
							| @@ -47,11 +47,8 @@ jobs: | ||||
|         # Copy firmware binary | ||||
|         cp .pio/build/esp32dev/firmware.bin .pio/build/esp32dev/upgrade_filaman_firmware_v${VERSION}.bin | ||||
|          | ||||
|         # Create SPIFFS binary with ESP32 image header | ||||
|         # Create 16-byte ESP32 image header: (magic byte 0xE9, segment count 0x01, SPI mode DIO, SPI speed 40MHz, chip revision v3.1) | ||||
|         echo -ne '\xE9\x01\x00\x00\x46\x97\x00\x00\x00\x60\x00\x00\x03\x01\x00\x00' > .pio/build/esp32dev/upgrade_filaman_website_v${VERSION}.bin | ||||
|         # Append the actual SPIFFS data | ||||
|         cat .pio/build/esp32dev/spiffs.bin >> .pio/build/esp32dev/upgrade_filaman_website_v${VERSION}.bin | ||||
|         # 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) | ||||
|         (cd .pio/build/esp32dev &&  | ||||
| @@ -64,7 +61,7 @@ jobs: | ||||
|           0x1000 bootloader.bin \ | ||||
|           0x8000 partitions.bin \ | ||||
|           0x10000 firmware.bin \ | ||||
|           0x410000 spiffs.bin) | ||||
|           0x3D0000 spiffs.bin) | ||||
|          | ||||
|         # Verify file sizes | ||||
|         echo "File sizes:" | ||||
| @@ -76,14 +73,50 @@ jobs: | ||||
|         VERSION=$(grep '^version = ' platformio.ini | cut -d'"' -f2) | ||||
|         echo "VERSION=$VERSION" >> $GITHUB_OUTPUT | ||||
|        | ||||
|     - name: Read CHANGELOG.md | ||||
|       id: changelog | ||||
|     - name: Generate Release Notes | ||||
|       id: release_notes | ||||
|       run: | | ||||
|         VERSION=${{ steps.get_version.outputs.VERSION }} | ||||
|         CHANGELOG=$(awk "/## \\[$VERSION\\]/{p=1;print;next} /## \\[/{p=0} p" CHANGELOG.md) | ||||
|         echo "CHANGES<<EOF" >> $GITHUB_OUTPUT | ||||
|         echo "$CHANGELOG" >> $GITHUB_OUTPUT | ||||
|         echo "EOF" >> $GITHUB_OUTPUT | ||||
|         # Get all tags sorted by version | ||||
|         TAGS=($(git tag -l 'v*' --sort=-v:refname)) | ||||
|         CURRENT_TAG="${TAGS[0]}" | ||||
|          | ||||
|         if [ ${#TAGS[@]} -gt 1 ]; then | ||||
|           PREVIOUS_TAG="${TAGS[1]}" | ||||
|           echo "CHANGES<<EOF" >> $GITHUB_OUTPUT | ||||
|           echo "Changes since $PREVIOUS_TAG:" >> $GITHUB_OUTPUT | ||||
|           echo "" >> $GITHUB_OUTPUT | ||||
|            | ||||
|           # Get commits between previous and current tag | ||||
|           echo "### Added" >> $GITHUB_OUTPUT | ||||
|           git log ${PREVIOUS_TAG}..${CURRENT_TAG} --pretty=format:%s | grep -iE '^(feat|add|new)' | sed 's/^feat: /- /' >> $GITHUB_OUTPUT || true | ||||
|           echo "" >> $GITHUB_OUTPUT | ||||
|            | ||||
|           echo "### Fixed" >> $GITHUB_OUTPUT | ||||
|           git log ${PREVIOUS_TAG}..${CURRENT_TAG} --pretty=format:%s | grep -iE '^fix' | sed 's/^fix: /- /' >> $GITHUB_OUTPUT || true | ||||
|           echo "" >> $GITHUB_OUTPUT | ||||
|            | ||||
|           echo "### Changed" >> $GITHUB_OUTPUT | ||||
|           git log ${PREVIOUS_TAG}..${CURRENT_TAG} --pretty=format:%s | grep -ivE '^(feat|fix|add|new)' | sed 's/^/- /' >> $GITHUB_OUTPUT || true | ||||
|           echo "EOF" >> $GITHUB_OUTPUT | ||||
|         else | ||||
|           # First release or no previous tag | ||||
|           echo "CHANGES<<EOF" >> $GITHUB_OUTPUT | ||||
|           echo "Initial Release" >> $GITHUB_OUTPUT | ||||
|            | ||||
|           # Add all commits for initial release | ||||
|           echo "" >> $GITHUB_OUTPUT | ||||
|           echo "### Added" >> $GITHUB_OUTPUT | ||||
|           git log --pretty=format:%s | grep -iE '^(feat|add|new)' | sed 's/^feat: /- /' >> $GITHUB_OUTPUT || true | ||||
|           echo "" >> $GITHUB_OUTPUT | ||||
|            | ||||
|           echo "### Fixed" >> $GITHUB_OUTPUT | ||||
|           git log --pretty=format:%s | grep -iE '^fix' | sed 's/^fix: /- /' >> $GITHUB_OUTPUT || true | ||||
|           echo "" >> $GITHUB_OUTPUT | ||||
|            | ||||
|           echo "### Changed" >> $GITHUB_OUTPUT | ||||
|           git log --pretty=format:%s | grep -ivE '^(feat|fix|add|new)' | sed 's/^/- /' >> $GITHUB_OUTPUT || true | ||||
|           echo "EOF" >> $GITHUB_OUTPUT | ||||
|         fi | ||||
|  | ||||
|     - name: Create GitHub Release | ||||
|       env: | ||||
| @@ -113,7 +146,7 @@ jobs: | ||||
|         if [ -n "$FILES_TO_UPLOAD" ]; then | ||||
|           gh release create "v${VERSION}" \ | ||||
|             --title "Release ${VERSION}" \ | ||||
|             --notes "${{ steps.changelog.outputs.CHANGES }}" \ | ||||
|             --notes "${{ steps.release_notes.outputs.CHANGES }}" \ | ||||
|             $FILES_TO_UPLOAD | ||||
|         else | ||||
|           echo "Error: No files found to upload" | ||||
|   | ||||
							
								
								
									
										192
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										192
									
								
								CHANGELOG.md
									
									
									
									
									
								
							| @@ -1,5 +1,197 @@ | ||||
| # Changelog | ||||
|  | ||||
| ## [1.3.63] - 2025-02-22 | ||||
| ### Added | ||||
| - update update-form background and add glass border effect | ||||
|  | ||||
| ### Changed | ||||
| - update webpages for version v1.3.63 | ||||
| - update release note generation for initial release handling | ||||
|  | ||||
|  | ||||
| ## [1.3.62] - 2025-02-22 | ||||
| ### Changed | ||||
| - update webpages for version v1.3.62 | ||||
| - update background colors and improve layout for update sections | ||||
|  | ||||
|  | ||||
| ## [1.3.61] - 2025-02-22 | ||||
| ### Added | ||||
| - update release notes generation to use previous tag for changes | ||||
|  | ||||
| ### Changed | ||||
| - update webpages for version v1.3.61 | ||||
|  | ||||
|  | ||||
| ## [1.3.60] - 2025-02-22 | ||||
| ### Added | ||||
| - remove automatic git push from changelog update script | ||||
| - implement release notes generation with categorized changes since last tag | ||||
|  | ||||
| ### Changed | ||||
| - update webpages for version v1.3.60 | ||||
|  | ||||
|  | ||||
| ## [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 | ||||
| - update header title to 'Hollo Lollo Trollo' | ||||
| - update header title to 'Filament Management Tool' and improve update response messages | ||||
|  | ||||
|  | ||||
| ## [1.3.43] - 2025-02-22 | ||||
| ### Changed | ||||
| - update webpages for version v1.3.43 | ||||
| - update header title to 'Hollo Lollo Trollo' | ||||
|  | ||||
|  | ||||
| ## [1.3.42] - 2025-02-22 | ||||
| ### Changed | ||||
| - update webpages for version v1.3.42 | ||||
|  | ||||
| ### Fixed | ||||
| - correct path for SPIFFS binary creation in Gitea release workflow | ||||
|  | ||||
|  | ||||
| ## [1.3.41] - 2025-02-22 | ||||
| ### Changed | ||||
| - update webpages for version v1.3.41 | ||||
|  | ||||
| ### Fixed | ||||
| - remove redundant buffer size setting in NFC initialization | ||||
| - update SPIFFS binary creation and enhance NFC buffer size | ||||
|  | ||||
|  | ||||
| ## [1.3.40] - 2025-02-22 | ||||
| ### Changed | ||||
| - update webpages for version v1.3.40 | ||||
|  | ||||
| ### Fixed | ||||
| - update SPIFFS binary header and enhance WebSocket error handling | ||||
|  | ||||
|  | ||||
| ## [1.3.39] - 2025-02-22 | ||||
| ### Changed | ||||
| - update webpages for version v1.3.39 | ||||
| - workflow: update SPIFFS binary creation to set chip version to max supported | ||||
|  | ||||
|  | ||||
| ## [1.3.38] - 2025-02-22 | ||||
| ### Changed | ||||
| - update webpages for version v1.3.38 | ||||
| - workflow: update SPIFFS binary creation with minimal ESP32 image header | ||||
|  | ||||
|  | ||||
| ## [1.3.37] - 2025-02-22 | ||||
| ### Changed | ||||
| - update webpages for version v1.3.37 | ||||
| - workflow: update ESP32-WROOM image header for SPIFFS binary creation | ||||
|  | ||||
|  | ||||
| ## [1.3.36] - 2025-02-22 | ||||
| ### Changed | ||||
| - update webpages for version v1.3.36 | ||||
| - partition: update SPIFFS binary header and offsets in workflow files | ||||
|  | ||||
|  | ||||
| ## [1.3.35] - 2025-02-22 | ||||
| ### Changed | ||||
| - update webpages for version v1.3.35 | ||||
|   | ||||
							
								
								
									
										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 | ||||
|   | ||||
| @@ -1051,9 +1051,10 @@ input[type="submit"]:disabled, | ||||
| } | ||||
| .update-form { | ||||
|     background: var(--primary-color); | ||||
|     box-shadow: 0 4px 8px rgba(0, 0, 0, 0.05); | ||||
|     border: var(--glass-border); | ||||
|     padding: 20px; | ||||
|     border-radius: 8px; | ||||
|     box-shadow: 0 2px 4px rgba(0,0,0,0.1); | ||||
|     margin: 0 auto; | ||||
|     width: 400px; | ||||
|     text-align: center; | ||||
| @@ -1064,7 +1065,7 @@ input[type="submit"]:disabled, | ||||
|     padding: 8px; | ||||
|     border: 1px solid #ddd; | ||||
|     border-radius: 4px; | ||||
|     background: white; | ||||
|     background-color: #4CAF50; | ||||
| } | ||||
| .update-form input[type="submit"] { | ||||
|     background-color: #4CAF50; | ||||
| @@ -1086,10 +1087,66 @@ input[type="submit"]:disabled, | ||||
| .warning { | ||||
|     background-color: var(--primary-color); | ||||
|     border: 1px solid #ffe0b2; | ||||
|     color: white; | ||||
|     padding: 15px; | ||||
|     margin: 20px auto; | ||||
|     border-radius: 4px; | ||||
|     max-width: 600px; | ||||
|     text-align: center; | ||||
|     color: #e65100; | ||||
|     padding: 15px; | ||||
| } | ||||
|  | ||||
| .update-options { | ||||
|     display: flex; | ||||
|     gap: 2rem; | ||||
|     margin: 2rem 0; | ||||
| } | ||||
| .update-section { | ||||
|     flex: 1; | ||||
|     background: var(--background-green); | ||||
|     padding: 1.5rem; | ||||
|     border-radius: 8px; | ||||
| } | ||||
| .update-section h2 { | ||||
|     margin-top: 0; | ||||
|     color: #333; | ||||
| } | ||||
| .update-section p { | ||||
|     color: #666; | ||||
|     margin-bottom: 1rem; | ||||
| } | ||||
| .progress-container { | ||||
|     margin: 20px 0; | ||||
|     background: #f0f0f0; | ||||
|     border-radius: 4px; | ||||
|     overflow: hidden; | ||||
| } | ||||
| .progress-bar { | ||||
|     width: 0; | ||||
|     height: 20px; | ||||
|     background: #4CAF50; | ||||
|     transition: width 0.3s ease-in-out; | ||||
|     text-align: center; | ||||
|     line-height: 20px; | ||||
|     color: white; | ||||
| } | ||||
| .status { | ||||
|     margin-top: 20px; | ||||
|     padding: 10px; | ||||
|     border-radius: 4px; | ||||
|     display: none; | ||||
| } | ||||
| .status.success { | ||||
|     background: #e8f5e9; | ||||
|     color: #2e7d32; | ||||
| } | ||||
| .status.error { | ||||
|     background: #ffebee; | ||||
|     color: #c62828; | ||||
| } | ||||
| .warning { | ||||
|     background: #fff3e0; | ||||
|     color: #e65100; | ||||
|     padding: 15px; | ||||
|     border-radius: 4px; | ||||
|     margin-bottom: 20px; | ||||
| } | ||||
| @@ -86,64 +86,6 @@ | ||||
|         <div class="status"></div> | ||||
|     </div> | ||||
|  | ||||
|     <style> | ||||
|         .update-options { | ||||
|             display: flex; | ||||
|             gap: 2rem; | ||||
|             margin: 2rem 0; | ||||
|         } | ||||
|         .update-section { | ||||
|             flex: 1; | ||||
|             background: #f5f5f5; | ||||
|             padding: 1.5rem; | ||||
|             border-radius: 8px; | ||||
|         } | ||||
|         .update-section h2 { | ||||
|             margin-top: 0; | ||||
|             color: #333; | ||||
|         } | ||||
|         .update-section p { | ||||
|             color: #666; | ||||
|             margin-bottom: 1rem; | ||||
|         } | ||||
|         .progress-container { | ||||
|             margin: 20px 0; | ||||
|             background: #f0f0f0; | ||||
|             border-radius: 4px; | ||||
|             overflow: hidden; | ||||
|         } | ||||
|         .progress-bar { | ||||
|             width: 0; | ||||
|             height: 20px; | ||||
|             background: #4CAF50; | ||||
|             transition: width 0.3s ease-in-out; | ||||
|             text-align: center; | ||||
|             line-height: 20px; | ||||
|             color: white; | ||||
|         } | ||||
|         .status { | ||||
|             margin-top: 20px; | ||||
|             padding: 10px; | ||||
|             border-radius: 4px; | ||||
|             display: none; | ||||
|         } | ||||
|         .status.success { | ||||
|             background: #e8f5e9; | ||||
|             color: #2e7d32; | ||||
|         } | ||||
|         .status.error { | ||||
|             background: #ffebee; | ||||
|             color: #c62828; | ||||
|         } | ||||
|         .warning { | ||||
|             background: #fff3e0; | ||||
|             color: #e65100; | ||||
|             padding: 15px; | ||||
|             border-radius: 4px; | ||||
|             margin-bottom: 20px; | ||||
|         } | ||||
|     </style> | ||||
|  | ||||
|     <script> | ||||
|         // Hide status indicators during update | ||||
|         const statusContainer = document.querySelector('.status-container'); | ||||
| @@ -154,6 +96,96 @@ | ||||
|         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(); | ||||
| @@ -176,76 +208,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); | ||||
|                         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) { | ||||
|                         if (!isNaN(response)) { | ||||
|                             const percent = parseInt(response); | ||||
|                             progress.style.width = percent + '%'; | ||||
|                             progress.textContent = percent + '%'; | ||||
|                             return; | ||||
|                         } | ||||
|                     } | ||||
|                      | ||||
|                     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(); | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| # Name,   Type, SubType,    Offset,   Size,     Flags | ||||
| nvs,      data, nvs,       0x9000,   0x5000, | ||||
| otadata,  data, ota,       0xe000,   0x2000, | ||||
| app0,     app,  ota_0,     0x10000,  0x200000, | ||||
| app1,     app,  ota_1,     0x210000, 0x200000, | ||||
| spiffs,   data, spiffs,    0x410000, 0x60000, | ||||
| app0,     app,  ota_0,     0x10000,  0x1E0000, | ||||
| app1,     app,  ota_1,     0x1F0000, 0x1E0000, | ||||
| spiffs,   data, spiffs,    0x3D0000, 0x30000, | ||||
| 
 | 
| @@ -9,7 +9,7 @@ | ||||
| ; https://docs.platformio.org/page/projectconf.html | ||||
|  | ||||
| [common] | ||||
| version = "1.3.35" | ||||
| version = "1.3.63" | ||||
|  | ||||
| #test | ||||
|  | ||||
| @@ -54,12 +54,8 @@ build_flags = | ||||
|     -DCONFIG_ARDUHAL_LOG_COLORS=1 | ||||
|     -DOTA_DEBUG=1 | ||||
|     -DCONFIG_OPTIMIZATION_LEVEL_DEBUG=1 | ||||
|     -DCONFIG_ESP32_PANIC_PRINT_REBOOT | ||||
|     -DBOOT_APP_PARTITION_OTA_0=1 | ||||
|     -DCONFIG_LOG_DEFAULT_LEVEL=3 | ||||
|     -DCONFIG_LWIP_TCP_MSL=60000 | ||||
|     -DCONFIG_LWIP_TCP_WND_DEFAULT=8192 | ||||
|     -DCONFIG_LWIP_TCP_SND_BUF_DEFAULT=4096 | ||||
|     -DCONFIG_LWIP_TCP_RCV_BUF_DEFAULT=4096 | ||||
|     -DCONFIG_LWIP_MAX_ACTIVE_TCP=16 | ||||
|      | ||||
|   | ||||
| @@ -64,29 +64,10 @@ def get_changes_from_git(): | ||||
|      | ||||
|     return changes | ||||
|  | ||||
| def push_changes(version): | ||||
|     """Push changes to upstream""" | ||||
|     try: | ||||
|         # Stage the CHANGELOG.md | ||||
|         subprocess.run(['git', 'add', 'CHANGELOG.md'], check=True) | ||||
|          | ||||
|         # Commit the changelog | ||||
|         commit_msg = f"docs: update changelog for version {version}" | ||||
|         subprocess.run(['git', 'commit', '-m', commit_msg], check=True) | ||||
|          | ||||
|         # Push to origin (local) | ||||
|         subprocess.run(['git', 'push', 'origin'], check=True) | ||||
|         print("Successfully pushed to origin") | ||||
|          | ||||
|     except subprocess.CalledProcessError as e: | ||||
|         print(f"Error during git operations: {e}") | ||||
|         return False | ||||
|     return True | ||||
|  | ||||
| def update_changelog(): | ||||
|     print("Starting changelog update...")  # Add this line | ||||
|     print("Starting changelog update...") | ||||
|     version = get_version() | ||||
|     print(f"Current version: {version}")   # Add this line | ||||
|     print(f"Current version: {version}") | ||||
|     today = datetime.now().strftime('%Y-%m-%d') | ||||
|      | ||||
|     script_dir = os.path.dirname(os.path.abspath(__file__)) | ||||
| @@ -111,7 +92,7 @@ def update_changelog(): | ||||
|     if not os.path.exists(changelog_path): | ||||
|         with open(changelog_path, 'w') as f: | ||||
|             f.write(f"# Changelog\n\n{changelog_entry}") | ||||
|         push_changes(version) | ||||
|         print(f"Created new changelog file with version {version}") | ||||
|     else: | ||||
|         with open(changelog_path, 'r') as f: | ||||
|             content = f.read() | ||||
| @@ -120,7 +101,7 @@ def update_changelog(): | ||||
|             updated_content = content.replace("# Changelog\n", f"# Changelog\n\n{changelog_entry}") | ||||
|             with open(changelog_path, 'w') as f: | ||||
|                 f.write(updated_content) | ||||
|             push_changes(version) | ||||
|             print(f"Added new version {version} to changelog") | ||||
|         else: | ||||
|             # Version existiert bereits, aktualisiere die bestehenden Einträge | ||||
|             version_pattern = f"## \\[{version}\\] - \\d{{4}}-\\d{{2}}-\\d{{2}}" | ||||
| @@ -143,7 +124,6 @@ def update_changelog(): | ||||
|                  | ||||
|                 with open(changelog_path, 'w') as f: | ||||
|                     f.write(updated_content) | ||||
|                 push_changes(version) | ||||
|                 print(f"Updated entries for version {version}") | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|   | ||||
| @@ -19,6 +19,12 @@ | ||||
| void setup() { | ||||
|   Serial.begin(115200); | ||||
|  | ||||
|   uint64_t chipid; | ||||
|  | ||||
|   chipid = ESP.getEfuseMac(); //The chip ID is essentially its MAC address(length: 6 bytes). | ||||
|   Serial.printf("ESP32 Chip ID = %04X", (uint16_t)(chipid >> 32)); //print High 2 bytes | ||||
|   Serial.printf("%08X\n", (uint32_t)chipid); //print Low 4bytes. | ||||
|  | ||||
|   // Initialize SPIFFS | ||||
|   initializeSPIFFS(); | ||||
|  | ||||
|   | ||||
							
								
								
									
										314
									
								
								src/website.cpp
									
									
									
									
									
								
							
							
						
						
									
										314
									
								
								src/website.cpp
									
									
									
									
									
								
							| @@ -8,6 +8,7 @@ | ||||
| #include "scale.h" | ||||
| #include "esp_task_wdt.h" | ||||
| #include <Update.h> | ||||
| #include "display.h" | ||||
|  | ||||
| #ifndef VERSION | ||||
|   #define VERSION "1.1.0" | ||||
| @@ -22,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!"); | ||||
| @@ -32,6 +76,10 @@ void onWsEvent(AsyncWebSocket *server, AsyncWebSocketClient *client, AwsEventTyp | ||||
|         sendWriteResult(client, 3); | ||||
|     } else if (type == WS_EVT_DISCONNECT) { | ||||
|         Serial.println("Client getrennt."); | ||||
|     } else if (type == WS_EVT_ERROR) { | ||||
|         Serial.printf("WebSocket Client #%u error(%u): %s\n", client->id(), *((uint16_t*)arg), (char*)data); | ||||
|     } else if (type == WS_EVT_PONG) { | ||||
|         Serial.printf("WebSocket Client #%u pong\n", client->id()); | ||||
|     } else if (type == WS_EVT_DATA) { | ||||
|         String message = String((char*)data); | ||||
|         JsonDocument doc; | ||||
| @@ -159,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); | ||||
| @@ -358,66 +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(); | ||||
|             AsyncWebServerResponse *response = request->beginResponse( | ||||
|                 success ? 200 : 400, | ||||
|                 "application/json", | ||||
|                 success ? "{\"success\":true,\"message\":\"Update successful\"}"  | ||||
|                        : "{\"success\":false,\"message\":\"Update failed\"}" | ||||
|             ); | ||||
|             response->addHeader("Connection", "close"); | ||||
|             request->send(response); | ||||
|              | ||||
|             if (success) { | ||||
|                 delay(500); | ||||
|                 ESP.restart(); | ||||
|             } | ||||
|         }, | ||||
|         [](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; | ||||
|  | ||||
|             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); | ||||
|  | ||||
|                 if (!Update.begin(updateSize, command)) { | ||||
|                     Serial.printf("Update Begin Error: "); | ||||
|                     Update.printError(Serial); | ||||
|                     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: "); | ||||
|                     Update.printError(Serial); | ||||
|                     String errorMsg = String("Write failed: ") + Update.errorString(); | ||||
|                     request->send(400, "application/json", "{\"success\":false,\"message\":\"" + errorMsg + "\"}"); | ||||
|                     return; | ||||
|                 } | ||||
|                 Serial.printf("Progress: %u/%u\r", index + len, updateSize); | ||||
|             } | ||||
|  | ||||
|             if (final) { | ||||
|                 if (!Update.end(true)) { | ||||
|                     Serial.printf("Update End Error: "); | ||||
|                     Update.printError(Serial); | ||||
|                     String errorMsg = String("Update end failed: ") + Update.errorString(); | ||||
|                     request->send(400, "application/json", "{\"success\":false,\"message\":\"" + errorMsg + "\"}"); | ||||
|                     return; | ||||
|                 } | ||||
|                 Serial.printf("\nUpdate Success: %uB\n", index+len); | ||||
|             } | ||||
|         } | ||||
|     ); | ||||
|     // Update-Handler registrieren | ||||
|     handleUpdate(server); | ||||
|  | ||||
|     server.on("/api/version", HTTP_GET, [](AsyncWebServerRequest *request){ | ||||
|         String fm_version = VERSION; | ||||
| @@ -442,79 +534,49 @@ void setupWebserver(AsyncWebServer &server) { | ||||
|     Serial.println("Webserver gestartet"); | ||||
| } | ||||
|  | ||||
| void handleOTAUpload(AsyncWebServerRequest *request, const String& filename, size_t index, uint8_t *data, size_t len, bool final) { | ||||
|     static bool isSpiffsUpdate = false; | ||||
|     if (!index) { | ||||
|         // Start eines neuen Uploads | ||||
|         Serial.println("Update Start: " + filename); | ||||
|          | ||||
|         // Überprüfe den Dateityp basierend auf dem Dateinamen | ||||
|         bool isFirmware = filename.startsWith("filaman_"); | ||||
|         isSpiffsUpdate = filename.startsWith("webpage_"); | ||||
|          | ||||
|         if (!isFirmware && !isSpiffsUpdate) { | ||||
|             request->send(400, "application/json", "{\"message\":\"Invalid file type. File must start with 'filaman_' or 'webpage_'\"}"); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         // Wähle den Update-Typ basierend auf dem Dateinamen | ||||
|         if (isSpiffsUpdate) { | ||||
|             if (!Update.begin(SPIFFS.totalBytes(), U_SPIFFS)) { | ||||
|                 Update.printError(Serial); | ||||
|                 request->send(400, "application/json", "{\"message\":\"SPIFFS Update failed: " + String(Update.errorString()) + "\"}"); | ||||
|                 return; | ||||
|             } | ||||
|             // Backup JSON configs before SPIFFS update | ||||
|             backupJsonConfigs(); | ||||
|         } else { | ||||
|             if (!Update.begin(UPDATE_SIZE_UNKNOWN, U_FLASH)) { | ||||
|                 Update.printError(Serial); | ||||
|                 request->send(400, "application/json", "{\"message\":\"Firmware Update failed: " + String(Update.errorString()) + "\"}"); | ||||
|                 return; | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     if (Update.write(data, len) != len) { | ||||
|         Update.printError(Serial); | ||||
|         request->send(400, "application/json", "{\"message\":\"Write failed: " + String(Update.errorString()) + "\"}"); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     if (final) { | ||||
|         if (!Update.end(true)) { | ||||
|             Update.printError(Serial); | ||||
|             request->send(400, "application/json", "{\"message\":\"Update failed: " + String(Update.errorString()) + "\"}"); | ||||
|             return; | ||||
|         } | ||||
|         if (isSpiffsUpdate) { | ||||
|             // Restore JSON configs after SPIFFS update | ||||
|             restoreJsonConfigs(); | ||||
|         } | ||||
|         request->send(200, "application/json", "{\"message\":\"Update successful!\", \"restart\": true}"); | ||||
|         delay(500); | ||||
|         ESP.restart(); | ||||
|     } | ||||
| } | ||||
|  | ||||
| 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