Compare commits
	
		
			459 Commits
		
	
	
		
			v1.2.61
			...
			2948a35fa8
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 2948a35fa8 | |||
| 730724fe58 | |||
| 714b7065e7 | |||
| 2d8aec515d | |||
| b245a206ce | |||
| f1489e75cc | |||
| d9ae829503 | |||
| 2247b8ed6c | |||
| d70b187bf9 | |||
| 1ade007473 | |||
| 0af14e2f7d | |||
| de67cdbff3 | |||
| 98fce15ccc | |||
| ab417ba64b | |||
| 320057bc49 | |||
| 9007a65fc2 | |||
| 2214f5f5de | |||
| 5c5846c52c | |||
| 517fa37a3d | |||
| aaa7a6ee9c | |||
| a0b8639488 | |||
| a16c05287e | |||
| ecb35a97bd | |||
| ba968611ec | |||
| 6bd11ddce3 | |||
| 3eb313e61a | |||
| aad35dc296 | |||
| 85ac636b1e | |||
| 6f1804c3fe | |||
| 89716920dc | |||
| 78b5078651 | |||
| 6098c3b052 | |||
| e7537f94d4 | |||
| 37717392d0 | |||
| c6da28ad6f | |||
| d6e38a4e73 | |||
| 4e0d9353c8 | |||
| 7059826659 | |||
| 41faa8bb1c | |||
| b38e3fa5ef | |||
| 5280d7e341 | |||
| 2f95c66d39 | |||
| df1b87465c | |||
| 84f1420999 | |||
| b14dd5475d | |||
| 975845421b | |||
| 044ddbe0eb | |||
| c385544d67 | |||
| c6cfd85687 | |||
| 84632322e2 | |||
| 86e55a8696 | |||
| d2b40daaca | |||
| 9d58cbc31c | |||
| d09aeaf47c | |||
| 9fb82fe51e | |||
| 5e0e2c5f6b | |||
| a8460503ff | |||
| 6700a1761f | |||
| 7207f36e06 | |||
| e79bee3381 | |||
| c3918f075b | |||
| 0c384219c5 | |||
| 42b9daf4be | |||
| 13a771682f | |||
| f79f87bf09 | |||
| 9fe3f6c0ff | |||
| 55e89948bb | |||
| 6c5e8c4d07 | |||
| 4f79700d74 | |||
| 1b4fecf409 | |||
| 89a6101d97 | |||
| ee45a74fee | |||
| db365aba3c | |||
| 63cdfaee6c | |||
| eb2e360c35 | |||
| 7d578640e2 | |||
| b006533a91 | |||
| 9fa7526623 | |||
| dfbb2fbd9b | |||
| 0302158449 | |||
| 68c385f9d7 | |||
| 9a8bd58cb3 | |||
| 0d8b8918c1 | |||
| a892b854b5 | |||
| 0f02f6c848 | |||
| 96c054827e | |||
| f93eedf775 | |||
| 68a10dfeb2 | |||
| 632b7a089e | |||
| c0e3650bf4 | |||
| 8e3dfc93f7 | |||
| 5016285dce | |||
| 9b1a232fde | |||
| 37e79b7a49 | |||
| 6bd23f31c1 | |||
| 3099e9ded9 | |||
| 4952ad3150 | |||
| 2055da9962 | |||
| 459a31cad3 | |||
| 4b1930209b | |||
| 7dde07b5ab | |||
| 33a5406248 | |||
| b016a31ff0 | |||
| 19bc4927e4 | |||
| cd55cb86ba | |||
| 8ab16b351b | |||
| 400a37d3ac | |||
| eb4f809435 | |||
| 1148947b8e | |||
| 3b01336999 | |||
| 44614b58dc | |||
| ed8d618272 | |||
| cd2ac54e98 | |||
| 92f675b24c | |||
| 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 | |||
| c5033acadc | |||
| 7de4189c83 | |||
| f43f2a15b2 | |||
| 858192c6cb | |||
| e2bd39922d | |||
| c86cc7173e | |||
| 16362e66a3 | |||
| 48d9ba8f71 | |||
| e2bea5a0c3 | |||
| 3e11f65188 | |||
| df59c42c8a | |||
| abe1d7c930 | |||
| ca614c3cc4 | |||
| 5153374093 | |||
| 66db4d7a85 | |||
| 90e71922b1 | |||
| e8e5c0bd3d | |||
| 7e53e1ccb0 | |||
| e49e812b13 | |||
| b1e0fcfadf | |||
| 31ef3ac8df | |||
| 8cf3f87c89 | |||
| c446188311 | |||
| 8e2a8d597d | |||
| 7d3b1c34f6 | |||
| b95c61118b | |||
| 0dfb158959 | |||
| 75c774bb24 | |||
| cf80adb43c | |||
| 36d50cbe7f | |||
| 9148d207c7 | |||
| 5f6fef9448 | |||
| 946202de0e | |||
| 41a3717347 | |||
| 255c820439 | |||
| aef3ba77ba | |||
| 2592c3a497 | |||
| a48c5dfef0 | |||
| 00554d0b09 | |||
| 05a91cd8d8 | |||
| 7cf113eaff | |||
| 44d27adab2 | |||
| e0a2dff5fe | |||
| 519a089684 | |||
| ef053bb2b6 | |||
| 0a91c7b269 | |||
| 875d9d2b70 | |||
| 52840b9b0b | |||
| da1fc7678f | |||
| 982bb5aa21 | |||
| 007737db13 | |||
| 17e5949201 | |||
| 6a57186091 | |||
| babd3f47a0 | |||
| 5372fe10fe | |||
| e0c9d90892 | |||
| e5f5d1961b | |||
| 31a960fb9e | |||
| 3c2e75b77a | |||
| 367143c456 | |||
| fbde4b764f | |||
| e57f4216d4 | |||
| b8beb992d6 | |||
| 4234b2254e | |||
| b8faf79163 | |||
| d35afaff46 | |||
| a8a00372b5 | |||
| 72f4eab588 | |||
| afa4eddc00 | |||
| b0888e7e63 | |||
| 238a84a8a2 | |||
| 59cc00ca13 | |||
| ab083f5f57 | |||
| c111573206 | |||
| 52b2494e52 | |||
| 069ec2d7a1 | |||
| 94e35ae86e | |||
| d71e3d8184 | |||
| bb166aa29f | |||
| 0d718023f8 | |||
| b16781043f | |||
| dff184ff25 | |||
| 0ce281221d | |||
| bc26c160e8 | |||
| c25f41db75 | |||
| e107c17f50 | |||
| 85b9d03ebd | |||
| 17b188626a | |||
| a534c5f872 | |||
| 93f7582790 | |||
| 46acc63756 | |||
| 67a9e1bdce | |||
| 2b75b64b4a | |||
| 8d003295e7 | |||
| f89500946a | |||
| 14e745ff06 | |||
| d058397fa2 | |||
| 622f5403a7 | |||
| 92b78a86dd | |||
| ec399390e8 | |||
| 909c4e9b5e | |||
| f4b20bfffd | |||
| 78464215a9 | |||
| 4365f0463a | |||
| 727bc0e760 | |||
| 04604013eb | |||
| cf5fc5f6f1 | |||
| 945a4ccce6 | |||
| 7cf9e2d145 | |||
| 9db4e338ea | |||
| dea6ca2c66 | |||
| e224e72e41 | |||
| 306c517da7 | |||
| 0337bbabe0 | |||
| bde14e50e0 | |||
| 9c656a9bd0 | |||
| eae552017d | |||
| a77918da41 | |||
| 262dad38a6 | |||
| cfc9f103cf | |||
| 0117302672 | |||
| 1de283b62f | |||
| f1eb78eb38 | |||
| 8a65b86475 | |||
| a3aef819c8 | |||
| a62b5ec933 | |||
| 1a8cf7a58f | |||
| b0b3d41c84 | |||
| 38b68aecfc | |||
| 4992f5f433 | |||
| 5cbbe1d231 | |||
| 9b29460d64 | |||
| dd14d475b7 | |||
| 9e6cd3b451 | |||
| c1be6ca582 | |||
| 265ff0c787 | |||
| 67eca82ac5 | |||
| 568db90db0 | |||
| 2dfd53d64a | |||
| 262a2fcbd4 | |||
| 3770de15d3 | |||
| 75a74ec9bd | |||
| 979adcbb14 | |||
| 2dd563a178 | |||
| 767c217c25 | |||
| c07689e15a | |||
| d6ca69fd19 | |||
| 60553255b8 | |||
| 8199b283c0 | |||
| d774ce0d09 | |||
| 4a44eda5c4 | |||
| c43ca20d8d | |||
| 21ba35cd19 | |||
| 62273320e9 | |||
| b8e4af4e4d | |||
| 513d74fdb0 | |||
| df884e7668 | |||
| 8182b5f684 | |||
| 4477537cec | |||
| 44ba7df34f | |||
| 54744a06dd | |||
| cefa81030b | |||
| 62052927d2 | |||
| 933a84f8ce | |||
| db3c19ff2e | |||
| ae9eb4cc6b | |||
| 89d40832c5 | |||
| c161216c04 | |||
| 6a016b6ac4 | |||
| 44dd485e17 | |||
| d41f0f3e67 | |||
| 484058515e | |||
| f552b492cf | |||
| c3040b3c29 | |||
| d7ba67085d | |||
| 48efb9e21a | |||
| e983ba6e44 | |||
| 37171d6eca | |||
| ebb08a7a66 | |||
| b5330af351 | |||
| 4919d34484 | |||
| 2da641d604 | |||
| ce413965c7 | |||
| 3fafed930e | |||
| e1c604ee8d | |||
| 11bbfb7db6 | |||
| 71d8f7ec5a | |||
| f4518e4a36 | |||
| 62d9596d08 | |||
| e27e95d291 | |||
| b7651ad50d | |||
| f1937e2977 | |||
| ad5ddf713c | |||
| ccb494f843 | |||
| 17307d8f03 | |||
| e5240a9572 | |||
| 06ebf105cf | |||
| 118e099fc5 | |||
| 8edd50f786 | |||
| b85325a747 | |||
| e1e0352beb | |||
| 8a93cccfce | |||
| c374069f36 | |||
| 59cd7c177d | |||
| 45088b5838 | |||
| 1b9c79b559 | |||
| 37e1e861d3 | |||
| cce39319d9 | |||
| 6391054c23 | |||
| 52cf46d7f8 | |||
| 84b05e48ce | |||
| 5c41d864c1 | |||
| 5dc3563da6 | |||
| 9e1b2943d6 | |||
| 7b89b04621 | |||
| e140f8e003 | |||
| 3d0bdde476 | |||
| 3ac7d6b4f7 | |||
| 5f52775984 | |||
| 463eaf4b6f | |||
| 4bf6b11d3a | |||
| b0c4af7c4e | |||
| 249e896ea4 | |||
| c74f587fff | |||
| 7a7ee72585 | |||
| 3dd5fbc585 | |||
| ed9c1487ed | |||
| d8756421a1 | |||
| d92c78f9d0 | |||
| 2d19ea745f | |||
| 13779cc9d7 | |||
| b6d5a8a00b | |||
| f6319e79f0 | |||
| 6f24630a7d | |||
| 4475d21218 | |||
| 01a926a38d | |||
| 6b966c02b3 | |||
| 1450e1ad2e | |||
| 3102a6c217 | |||
| d5b2b2746d | |||
| 7e776d4816 | |||
| e84b2973c5 | |||
| 5793dc1a1f | |||
| 1732491c48 | |||
| 0500bb6951 | |||
| ef9ef7257a | |||
| e86fd229dc | |||
| b940a166da | |||
| c857e16de2 | |||
| 8b2a537b72 | |||
| 88ec151c4c | |||
| 0a203f02eb | 
							
								
								
									
										208
									
								
								.github/workflows/gitea-release.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										208
									
								
								.github/workflows/gitea-release.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,208 @@ | ||||
| name: Gitea Release | ||||
|  | ||||
| on: | ||||
|   workflow_call: | ||||
|     secrets: | ||||
|       GITEA_TOKEN: | ||||
|         description: 'Token für Gitea API-Zugriff' | ||||
|         required: true | ||||
|  | ||||
|     outputs: | ||||
|       version: | ||||
|         description: 'The version that was released' | ||||
|         value: ${{ jobs.create-release.outputs.version }} | ||||
|  | ||||
| jobs: | ||||
|   create-release: | ||||
|     runs-on: ubuntu-latest | ||||
|     outputs: | ||||
|       version: ${{ steps.get_version.outputs.VERSION }} | ||||
|     steps: | ||||
|     - uses: actions/checkout@v4 | ||||
|       with: | ||||
|         fetch-depth: 0 | ||||
|      | ||||
|     - name: Set up Python | ||||
|       uses: actions/setup-python@v4 | ||||
|       with: | ||||
|         python-version: '3.x' | ||||
|      | ||||
|     - name: Install PlatformIO | ||||
|       run: | | ||||
|         python -m pip install --upgrade pip | ||||
|         pip install --upgrade platformio esptool | ||||
|      | ||||
|     - name: Install xxd | ||||
|       run: | | ||||
|         sudo apt-get update | ||||
|         sudo apt-get install xxd | ||||
|      | ||||
|     - name: Build Firmware | ||||
|       run: | | ||||
|         VERSION=$(grep '^version = ' platformio.ini | cut -d'"' -f2) | ||||
|          | ||||
|         # Build firmware and SPIFFS | ||||
|         echo "Building firmware and SPIFFS..." | ||||
|         pio run -e esp32dev | ||||
|         pio run -t buildfs | ||||
|          | ||||
|         # Copy firmware binary | ||||
|         cp .pio/build/esp32dev/firmware.bin .pio/build/esp32dev/upgrade_filaman_firmware_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 &&  | ||||
|         esptool.py --chip esp32 merge_bin \ | ||||
|           --fill-flash-size 4MB \ | ||||
|           --flash_mode dio \ | ||||
|           --flash_freq 40m \ | ||||
|           --flash_size 4MB \ | ||||
|           -o filaman_full_${VERSION}.bin \ | ||||
|           0x1000 bootloader.bin \ | ||||
|           0x8000 partitions.bin \ | ||||
|           0x10000 firmware.bin \ | ||||
|           0x3D0000 spiffs.bin) | ||||
|          | ||||
|         # Verify file sizes | ||||
|         echo "File sizes:" | ||||
|         (cd .pio/build/esp32dev && ls -lh *.bin) | ||||
|      | ||||
|     - name: Get version from platformio.ini | ||||
|       id: get_version | ||||
|       run: | | ||||
|         VERSION=$(grep '^version = ' platformio.ini | cut -d'"' -f2) | ||||
|         echo "VERSION=$VERSION" >> $GITHUB_OUTPUT | ||||
|        | ||||
|     - name: Generate Release Notes | ||||
|       id: release_notes | ||||
|       run: | | ||||
|         # Get the latest tag | ||||
|         LATEST_TAG=$(git for-each-ref --sort=-creatordate --format '%(refname:short)' refs/tags | sed -n '2p') | ||||
|          | ||||
|         if [ -n "$LATEST_TAG" ]; then | ||||
|           echo "CHANGES<<EOF" >> $GITHUB_OUTPUT | ||||
|           echo "Changes since ${LATEST_TAG}:" >> $GITHUB_OUTPUT | ||||
|           echo "" >> $GITHUB_OUTPUT | ||||
|            | ||||
|           # Get all commits since last release with commit hash and author | ||||
|           echo "### Added" >> $GITHUB_OUTPUT | ||||
|           git log ${LATEST_TAG}..HEAD --pretty=format:"%h - %s (%an)" | grep -iE '^[a-f0-9]+ - (feat|add|new)' | sed 's/^[a-f0-9]* - feat: /- /' >> $GITHUB_OUTPUT || true | ||||
|           echo "" >> $GITHUB_OUTPUT | ||||
|            | ||||
|           echo "### Fixed" >> $GITHUB_OUTPUT | ||||
|           git log ${LATEST_TAG}..HEAD --pretty=format:"%h - %s (%an)" | grep -iE '^[a-f0-9]+ - fix' | sed 's/^[a-f0-9]* - fix: /- /' >> $GITHUB_OUTPUT || true | ||||
|           echo "" >> $GITHUB_OUTPUT | ||||
|            | ||||
|           echo "### Changed" >> $GITHUB_OUTPUT | ||||
|           git log ${LATEST_TAG}..HEAD --pretty=format:"%h - %s (%an)" | grep -ivE '^[a-f0-9]+ - (feat|fix|add|new)' | sed 's/^[a-f0-9]* - /- /' >> $GITHUB_OUTPUT || true | ||||
|           echo "EOF" >> $GITHUB_OUTPUT | ||||
|         else | ||||
|           # First release | ||||
|           echo "CHANGES<<EOF" >> $GITHUB_OUTPUT | ||||
|           echo "Initial Release" >> $GITHUB_OUTPUT | ||||
|           echo "" >> $GITHUB_OUTPUT | ||||
|            | ||||
|           # Add all commits for initial release | ||||
|           echo "### Added" >> $GITHUB_OUTPUT | ||||
|           git log --pretty=format:"%h - %s (%an)" | grep -iE '^[a-f0-9]+ - (feat|add|new)' | sed 's/^[a-f0-9]* - feat: /- /' >> $GITHUB_OUTPUT || true | ||||
|           echo "" >> $GITHUB_OUTPUT | ||||
|            | ||||
|           echo "### Fixed" >> $GITHUB_OUTPUT | ||||
|           git log --pretty=format:"%h - %s (%an)" | grep -iE '^[a-f0-9]+ - fix' | sed 's/^[a-f0-9]* - fix: /- /' >> $GITHUB_OUTPUT || true | ||||
|           echo "" >> $GITHUB_OUTPUT | ||||
|            | ||||
|           echo "### Changed" >> $GITHUB_OUTPUT | ||||
|           git log --pretty=format:"%h - %s (%an)" | grep -ivE '^[a-f0-9]+ - (feat|fix|add|new)' | sed 's/^[a-f0-9]* - /- /' >> $GITHUB_OUTPUT || true | ||||
|           echo "EOF" >> $GITHUB_OUTPUT | ||||
|         fi | ||||
|  | ||||
|     - name: Determine Gitea URL | ||||
|       id: gitea_url | ||||
|       run: | | ||||
|         echo "Debug Environment:" | ||||
|         echo "GITHUB_SERVER_URL=${GITHUB_SERVER_URL:-not set}" | ||||
|         echo "GITEA_SERVER_URL=${GITEA_SERVER_URL:-not set}" | ||||
|         echo "GITHUB_REPOSITORY=${GITHUB_REPOSITORY:-not set}" | ||||
|         echo "GITEA_REPOSITORY=${GITEA_REPOSITORY:-not set}" | ||||
|         echo "RUNNER_NAME=${RUNNER_NAME:-not set}" | ||||
|          | ||||
|         # Set API URL based on environment | ||||
|         if [ -n "${GITEA_ACTIONS}" ] || [ -n "${GITEA_REPOSITORY}" ] || [[ "${RUNNER_NAME}" == *"gitea"* ]]; then | ||||
|           GITEA_API_URL="${GITHUB_SERVER_URL}" | ||||
|           GITEA_REPO=$(echo "${GITHUB_REPOSITORY}" | cut -d'/' -f2) | ||||
|           GITEA_OWNER=$(echo "${GITHUB_REPOSITORY}" | cut -d'/' -f1) | ||||
|         else | ||||
|           echo "Error: This workflow is only for Gitea" | ||||
|           exit 1 | ||||
|         fi | ||||
|          | ||||
|         echo "GITEA_API_URL=${GITEA_API_URL}" >> $GITHUB_OUTPUT | ||||
|         echo "GITEA_REPO=${GITEA_REPO}" >> $GITHUB_OUTPUT | ||||
|         echo "GITEA_OWNER=${GITEA_OWNER}" >> $GITHUB_OUTPUT | ||||
|  | ||||
|     - name: Create Gitea Release | ||||
|       env: | ||||
|         GITEA_TOKEN: ${{ secrets.GITEA_TOKEN }} | ||||
|         GITEA_API_URL: ${{ steps.gitea_url.outputs.GITEA_API_URL }} | ||||
|         GITEA_REPO: ${{ steps.gitea_url.outputs.GITEA_REPO }} | ||||
|         GITEA_OWNER: ${{ steps.gitea_url.outputs.GITEA_OWNER }} | ||||
|       run: | | ||||
|         # Debug Token (nur Länge ausgeben für Sicherheit) | ||||
|         echo "Debug: Token length: ${#GITEA_TOKEN}" | ||||
|         if [ -z "$GITEA_TOKEN" ]; then | ||||
|           echo "Error: GITEA_TOKEN is empty" | ||||
|           exit 1 | ||||
|         fi | ||||
|  | ||||
|         VERSION=${{ steps.get_version.outputs.VERSION }} | ||||
|         cd .pio/build/esp32dev | ||||
|          | ||||
|         # Debug-Ausgaben | ||||
|         echo "Debug: API URL: ${GITEA_API_URL}" | ||||
|         echo "Debug: Repository: ${GITEA_OWNER}/${GITEA_REPO}" | ||||
|          | ||||
|         # Erstelle zuerst den Release ohne Dateien | ||||
|         echo "Debug: Creating release..." | ||||
|         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 \ | ||||
|           -H "Authorization: token ${GITEA_TOKEN}" \ | ||||
|           -H "Content-Type: application/json" \ | ||||
|           -d "${RELEASE_DATA}" \ | ||||
|           "${GITEA_API_URL}/api/v1/repos/${GITEA_OWNER}/${GITEA_REPO}/releases") | ||||
|          | ||||
|         RELEASE_STATUS=$(echo "$RELEASE_RESPONSE" | tail -n1) | ||||
|         RELEASE_BODY=$(echo "$RELEASE_RESPONSE" | head -n -1) | ||||
|          | ||||
|         if [ "$RELEASE_STATUS" != "201" ]; then | ||||
|           echo "Error: Failed to create release" | ||||
|           echo "Response: $RELEASE_BODY" | ||||
|           exit 1 | ||||
|         fi | ||||
|          | ||||
|         # Extrahiere die Release-ID aus der Antwort | ||||
|         RELEASE_ID=$(echo "$RELEASE_BODY" | grep -o '"id":[0-9]*' | cut -d':' -f2) | ||||
|          | ||||
|         # Lade die Dateien einzeln hoch | ||||
|         for file in upgrade_filaman_firmware_v${VERSION}.bin upgrade_filaman_website_v${VERSION}.bin filaman_full_${VERSION}.bin; do | ||||
|           if [ -f "$file" ]; then | ||||
|             echo "Debug: Uploading $file..." | ||||
|             UPLOAD_RESPONSE=$(curl -s -w "\n%{http_code}" \ | ||||
|               -X POST \ | ||||
|               -H "Authorization: token ${GITEA_TOKEN}" \ | ||||
|               -H "Content-Type: application/octet-stream" \ | ||||
|               --data-binary @"$file" \ | ||||
|               "${GITEA_API_URL}/api/v1/repos/${GITEA_OWNER}/${GITEA_REPO}/releases/${RELEASE_ID}/assets?name=${file}") | ||||
|              | ||||
|             UPLOAD_STATUS=$(echo "$UPLOAD_RESPONSE" | tail -n1) | ||||
|             if [ "$UPLOAD_STATUS" != "201" ]; then | ||||
|               echo "Warning: Failed to upload $file" | ||||
|               echo "Response: $(echo "$UPLOAD_RESPONSE" | head -n -1)" | ||||
|             else | ||||
|               echo "Successfully uploaded $file" | ||||
|             fi | ||||
|           fi | ||||
|         done | ||||
							
								
								
									
										185
									
								
								.github/workflows/github-release.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										185
									
								
								.github/workflows/github-release.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,185 @@ | ||||
| name: GitHub Release | ||||
|  | ||||
| on: | ||||
|   workflow_call: | ||||
|     secrets: | ||||
|       RELEASE_TOKEN: | ||||
|         description: 'GitHub token for release creation' | ||||
|         required: true | ||||
|  | ||||
| permissions: | ||||
|   contents: write | ||||
|  | ||||
| jobs: | ||||
|   create-release: | ||||
|     runs-on: ubuntu-latest | ||||
|     permissions: | ||||
|       contents: write | ||||
|     steps: | ||||
|     - uses: actions/checkout@v4 | ||||
|       with: | ||||
|         fetch-depth: 0 | ||||
|      | ||||
|     - name: Set up Python | ||||
|       uses: actions/setup-python@v4 | ||||
|       with: | ||||
|         python-version: '3.x' | ||||
|      | ||||
|     - name: Install PlatformIO | ||||
|       run: | | ||||
|         python -m pip install --upgrade pip | ||||
|         pip install --upgrade platformio esptool | ||||
|      | ||||
|     - name: Install xxd | ||||
|       run: | | ||||
|         sudo apt-get update | ||||
|         sudo apt-get install xxd | ||||
|      | ||||
|     - name: Build Firmware | ||||
|       run: | | ||||
|         VERSION=$(grep '^version = ' platformio.ini | cut -d'"' -f2) | ||||
|          | ||||
|         # Always build firmware and SPIFFS | ||||
|         echo "Building firmware and SPIFFS..." | ||||
|         pio run -e esp32dev | ||||
|         pio run -t buildfs | ||||
|          | ||||
|         # Copy firmware binary | ||||
|         cp .pio/build/esp32dev/firmware.bin .pio/build/esp32dev/upgrade_filaman_firmware_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 &&  | ||||
|         esptool.py --chip esp32 merge_bin \ | ||||
|           --fill-flash-size 4MB \ | ||||
|           --flash_mode dio \ | ||||
|           --flash_freq 40m \ | ||||
|           --flash_size 4MB \ | ||||
|           -o filaman_full_${VERSION}.bin \ | ||||
|           0x1000 bootloader.bin \ | ||||
|           0x8000 partitions.bin \ | ||||
|           0x10000 firmware.bin \ | ||||
|           0x3D0000 spiffs.bin) | ||||
|          | ||||
|         # Verify file sizes | ||||
|         echo "File sizes:" | ||||
|         (cd .pio/build/esp32dev && ls -lh *.bin) | ||||
|      | ||||
|     - name: Get version from platformio.ini | ||||
|       id: get_version | ||||
|       run: | | ||||
|         VERSION=$(grep '^version = ' platformio.ini | cut -d'"' -f2) | ||||
|         echo "VERSION=$VERSION" >> $GITHUB_OUTPUT | ||||
|        | ||||
|     - name: Generate Release Notes | ||||
|       id: release_notes | ||||
|       run: | | ||||
|         # Get the latest tag | ||||
|         LATEST_TAG=$(git for-each-ref --sort=-creatordate --format '%(refname:short)' refs/tags | sed -n '2p') | ||||
|          | ||||
|         if [ -n "$LATEST_TAG" ]; then | ||||
|           echo "CHANGES<<EOF" >> $GITHUB_OUTPUT | ||||
|           echo "Changes since ${LATEST_TAG}:" >> $GITHUB_OUTPUT | ||||
|           echo "" >> $GITHUB_OUTPUT | ||||
|            | ||||
|           # Get all commits since last release with commit hash and author | ||||
|           echo "### Added" >> $GITHUB_OUTPUT | ||||
|           git log ${LATEST_TAG}..HEAD --pretty=format:"%h - %s (%an)" | grep -iE '^[a-f0-9]+ - (feat|add|new)' | sed 's/^[a-f0-9]* - feat: /- /' >> $GITHUB_OUTPUT || true | ||||
|           echo "" >> $GITHUB_OUTPUT | ||||
|            | ||||
|           echo "### Fixed" >> $GITHUB_OUTPUT | ||||
|           git log ${LATEST_TAG}..HEAD --pretty=format:"%h - %s (%an)" | grep -iE '^[a-f0-9]+ - fix' | sed 's/^[a-f0-9]* - fix: /- /' >> $GITHUB_OUTPUT || true | ||||
|           echo "" >> $GITHUB_OUTPUT | ||||
|            | ||||
|           echo "### Changed" >> $GITHUB_OUTPUT | ||||
|           git log ${LATEST_TAG}..HEAD --pretty=format:"%h - %s (%an)" | grep -ivE '^[a-f0-9]+ - (feat|fix|add|new)' | sed 's/^[a-f0-9]* - /- /' >> $GITHUB_OUTPUT || true | ||||
|           echo "EOF" >> $GITHUB_OUTPUT | ||||
|         else | ||||
|           # First release | ||||
|           echo "CHANGES<<EOF" >> $GITHUB_OUTPUT | ||||
|           echo "Initial Release" >> $GITHUB_OUTPUT | ||||
|           echo "" >> $GITHUB_OUTPUT | ||||
|            | ||||
|           # Add all commits for initial release | ||||
|           echo "### Added" >> $GITHUB_OUTPUT | ||||
|           git log --pretty=format:"%h - %s (%an)" | grep -iE '^[a-f0-9]+ - (feat|add|new)' | sed 's/^[a-f0-9]* - feat: /- /' >> $GITHUB_OUTPUT || true | ||||
|           echo "" >> $GITHUB_OUTPUT | ||||
|            | ||||
|           echo "### Fixed" >> $GITHUB_OUTPUT | ||||
|           git log --pretty=format:"%h - %s (%an)" | grep -iE '^[a-f0-9]+ - fix' | sed 's/^[a-f0-9]* - fix: /- /' >> $GITHUB_OUTPUT || true | ||||
|           echo "" >> $GITHUB_OUTPUT | ||||
|            | ||||
|           echo "### Changed" >> $GITHUB_OUTPUT | ||||
|           git log --pretty=format:"%h - %s (%an)" | grep -ivE '^[a-f0-9]+ - (feat|fix|add|new)' | sed 's/^[a-f0-9]* - /- /' >> $GITHUB_OUTPUT || true | ||||
|           echo "EOF" >> $GITHUB_OUTPUT | ||||
|         fi | ||||
|  | ||||
|     - name: Create GitHub Release | ||||
|       env: | ||||
|         GH_TOKEN: ${{ secrets.RELEASE_TOKEN }} | ||||
|       run: | | ||||
|         VERSION=${{ steps.get_version.outputs.VERSION }} | ||||
|         cd .pio/build/esp32dev | ||||
|          | ||||
|         # Create release with available files | ||||
|         FILES_TO_UPLOAD="" | ||||
|          | ||||
|         # Always add firmware | ||||
|         if [ -f "upgrade_filaman_firmware_v${VERSION}.bin" ]; then | ||||
|           FILES_TO_UPLOAD="$FILES_TO_UPLOAD upgrade_filaman_firmware_v${VERSION}.bin" | ||||
|         fi | ||||
|          | ||||
|         # Add SPIFFS and full binary only if they exist | ||||
|         if [ -f "upgrade_filaman_website_v${VERSION}.bin" ]; then | ||||
|           FILES_TO_UPLOAD="$FILES_TO_UPLOAD upgrade_filaman_website_v${VERSION}.bin" | ||||
|         fi | ||||
|          | ||||
|         if [ -f "filaman_full_${VERSION}.bin" ]; then | ||||
|           FILES_TO_UPLOAD="$FILES_TO_UPLOAD filaman_full_${VERSION}.bin" | ||||
|         fi | ||||
|          | ||||
|         # Create release with available files | ||||
|         if [ -n "$FILES_TO_UPLOAD" ]; then | ||||
|           gh release create "v${VERSION}" \ | ||||
|             --title "Release ${VERSION}" \ | ||||
|             --notes "${{ steps.release_notes.outputs.CHANGES }}" \ | ||||
|             $FILES_TO_UPLOAD | ||||
|         else | ||||
|           echo "Error: No files found to upload" | ||||
|           exit 1 | ||||
|         fi | ||||
|  | ||||
|     - name: Install lftp | ||||
|       run: sudo apt-get install -y lftp | ||||
|          | ||||
|     - name: Upload Firmware via FTP | ||||
|       if: success() | ||||
|       env: | ||||
|         FTP_PASSWORD: ${{ vars.FTP_PASSWORD }} | ||||
|         FTP_USER: ${{ vars.FTP_USER }} | ||||
|         FTP_HOST: ${{ vars.FTP_HOST }} | ||||
|         VERSION: ${{ steps.get_version.outputs.VERSION }} | ||||
|       run: | | ||||
|         echo "Environment variables:" | ||||
|         env | grep -E '^FTP_' | while read -r line; do | ||||
|           var_name=$(echo "$line" | cut -d= -f1) | ||||
|           var_value=$(echo "$line" | cut -d= -f2-) | ||||
|           echo "$var_name is $(if [ -n "$var_value" ]; then echo "set"; else echo "empty"; fi)" | ||||
|         done | ||||
|          | ||||
|         cd .pio/build/esp32dev | ||||
|         if [ -n "$FTP_USER" ] && [ -n "$FTP_PASSWORD" ] && [ -n "$FTP_HOST" ]; then | ||||
|           echo "All FTP credentials are present, attempting upload..." | ||||
|           lftp -c "set ssl:verify-certificate no; \ | ||||
|                     set ftp:ssl-protect-data true; \ | ||||
|                     set ftp:ssl-force true; \ | ||||
|                     set ssl:check-hostname false; \ | ||||
|                     set ftp:ssl-auth TLS; \ | ||||
|                     open -u $FTP_USER,$FTP_PASSWORD $FTP_HOST; \ | ||||
|                     put -O / filaman_full_${VERSION}.bin -o filaman_full.bin" | ||||
|         else | ||||
|           echo "Error: Some FTP credentials are missing" | ||||
|           exit 1 | ||||
|         fi | ||||
							
								
								
									
										134
									
								
								.github/workflows/providers/gitea-release.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										134
									
								
								.github/workflows/providers/gitea-release.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,134 +0,0 @@ | ||||
| name: Gitea Release | ||||
|  | ||||
| on: | ||||
|   workflow_call: | ||||
|     inputs: | ||||
|       gitea_ref_name: | ||||
|         description: 'Gitea ref name' | ||||
|         required: true | ||||
|         type: string | ||||
|       gitea_server_url: | ||||
|         description: 'Gitea server URL' | ||||
|         required: true | ||||
|         type: string | ||||
|       gitea_repository: | ||||
|         description: 'Gitea repository' | ||||
|         required: true | ||||
|         type: string | ||||
|     secrets: | ||||
|       GITEA_TOKEN: | ||||
|         required: true | ||||
|  | ||||
| jobs: | ||||
|   create-release: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|     - name: Checkout Repository | ||||
|       uses: actions/checkout@v4 | ||||
|  | ||||
|     - name: Install System Dependencies | ||||
|       run: | | ||||
|         sudo apt-get update | ||||
|         sudo apt-get install -y python3 python3-venv build-essential curl git | ||||
|  | ||||
|     - name: Set up Python Virtual Environment | ||||
|       run: | | ||||
|         python3 -m venv venv | ||||
|         source venv/bin/activate | ||||
|         pip install --upgrade pip | ||||
|         pip install platformio esptool | ||||
|          | ||||
|         echo "Verifying installations:" | ||||
|         platformio --version | ||||
|         python3 --version | ||||
|         esptool.py version | ||||
|  | ||||
|     - name: Build Firmware | ||||
|       run: | | ||||
|         source venv/bin/activate | ||||
|         echo "Building SPIFFS..." | ||||
|         platformio run -t buildfs | ||||
|          | ||||
|         echo "Building firmware..." | ||||
|         platformio run | ||||
|  | ||||
|     - name: Create Release Files | ||||
|       run: | | ||||
|         source venv/bin/activate | ||||
|         echo "Creating release files..." | ||||
|         esptool.py --chip esp32 merge_bin \ | ||||
|           --flash_mode dio \ | ||||
|           --flash_freq 40m \ | ||||
|           --flash_size 4MB \ | ||||
|           -o .pio/build/esp32dev/filaman_full.bin \ | ||||
|           0x1000 .pio/build/esp32dev/bootloader.bin \ | ||||
|           0x8000 .pio/build/esp32dev/partitions.bin \ | ||||
|           0x10000 .pio/build/esp32dev/firmware.bin \ | ||||
|           0x3D0000 .pio/build/esp32dev/spiffs.bin | ||||
|          | ||||
|         cp .pio/build/esp32dev/firmware.bin .pio/build/esp32dev/filaman_ota.bin | ||||
|  | ||||
|     - name: Read CHANGELOG.md | ||||
|       id: changelog | ||||
|       run: | | ||||
|         VERSION=$(echo "${{ inputs.gitea_ref_name }}" | sed 's/^v//') | ||||
|         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 | ||||
|         echo "CHANGELOG CONTENT:" | ||||
|         echo "$CHANGELOG" | ||||
|         if [ -z "$CHANGELOG" ]; then | ||||
|           echo "No changelog found for version $VERSION" | ||||
|           exit 1 | ||||
|         fi | ||||
|  | ||||
|     - name: Create Release | ||||
|       env: | ||||
|         TOKEN: ${{ secrets.GITEA_TOKEN }} | ||||
|         GITEA_REF_NAME: ${{ inputs.gitea_ref_name }} | ||||
|         GITEA_SERVER_URL: ${{ inputs.gitea_server_url }} | ||||
|         GITEA_REPOSITORY: ${{ inputs.gitea_repository }} | ||||
|         CHANGELOG: ${{ steps.changelog.outputs.CHANGES }} | ||||
|       run: | | ||||
|         echo "Debug environment:" | ||||
|         echo "GITEA_REF_NAME: ${GITEA_REF_NAME}" | ||||
|         echo "GITEA_SERVER_URL: ${GITEA_SERVER_URL}" | ||||
|         echo "GITEA_REPOSITORY: ${GITEA_REPOSITORY}" | ||||
|         echo "CHANGELOG: ${CHANGELOG}" | ||||
|          | ||||
|         TAG="${GITEA_REF_NAME}" | ||||
|         API_URL="${GITEA_SERVER_URL}/api/v1" | ||||
|         REPO="${GITEA_REPOSITORY}" | ||||
|          | ||||
|         echo "Creating release for ${TAG} on ${REPO}..." | ||||
|          | ||||
|         # Create release | ||||
|         RESPONSE=$(curl -k -s \ | ||||
|           -X POST \ | ||||
|           -H "Authorization: token ${TOKEN}" \ | ||||
|           -H "Content-Type: application/json" \ | ||||
|           -d "{\"tag_name\":\"${TAG}\",\"name\":\"Release ${TAG}\",\"body\":\"${CHANGELOG}\"}" \ | ||||
|           "${API_URL}/repos/${REPO}/releases") | ||||
|          | ||||
|         RELEASE_ID=$(echo "$RESPONSE" | grep -o '"id":[0-9]*' | cut -d':' -f2 | head -n1) | ||||
|         UPLOAD_URL=$(echo "$RESPONSE" | grep -o '"upload_url":"[^"]*' | cut -d':' -f2- | tr -d '"') | ||||
|          | ||||
|         if [ -n "$RELEASE_ID" ]; then | ||||
|           echo "Release created with ID: $RELEASE_ID" | ||||
|            | ||||
|           # Upload files | ||||
|           for file in "filaman_full.bin" "filaman_ota.bin"; do | ||||
|             echo "Uploading $file..." | ||||
|             curl -k -s \ | ||||
|               -X POST \ | ||||
|               -H "Authorization: token ${TOKEN}" \ | ||||
|               -H "Content-Type: application/octet-stream" \ | ||||
|               --data-binary "@.pio/build/esp32dev/$file" \ | ||||
|               "${UPLOAD_URL}?name=$file" | ||||
|           done | ||||
|         else | ||||
|           echo "Failed to create release. Response:" | ||||
|           echo "$RESPONSE" | ||||
|           exit 1 | ||||
|         fi | ||||
							
								
								
									
										71
									
								
								.github/workflows/providers/github-release.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										71
									
								
								.github/workflows/providers/github-release.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,71 +0,0 @@ | ||||
| name: GitHub Release | ||||
|  | ||||
| on: | ||||
|   workflow_call: | ||||
|  | ||||
| jobs: | ||||
|   create-release: | ||||
|     runs-on: ubuntu-latest | ||||
|     permissions: | ||||
|       contents: write | ||||
|     steps: | ||||
|     - uses: actions/checkout@v4 | ||||
|      | ||||
|     - name: Set up Python | ||||
|       uses: actions/setup-python@v4 | ||||
|       with: | ||||
|         python-version: '3.x' | ||||
|      | ||||
|     - name: Install PlatformIO | ||||
|       run: | | ||||
|         python -m pip install --upgrade pip | ||||
|         pip install --upgrade platformio | ||||
|      | ||||
|     - name: Build Firmware | ||||
|       run: | | ||||
|         pio run -t buildfs  # Build SPIFFS | ||||
|         pio run            # Build firmware | ||||
|          | ||||
|     - name: Install esptool | ||||
|       run: | | ||||
|         pip install esptool | ||||
|          | ||||
|     - name: Merge firmware and SPIFFS | ||||
|       run: | | ||||
|         esptool.py --chip esp32 merge_bin \ | ||||
|           --flash_mode dio \ | ||||
|           --flash_freq 40m \ | ||||
|           --flash_size 4MB \ | ||||
|           -o .pio/build/esp32dev/filaman_full.bin \ | ||||
|           0x1000 .pio/build/esp32dev/bootloader.bin \ | ||||
|           0x8000 .pio/build/esp32dev/partitions.bin \ | ||||
|           0x10000 .pio/build/esp32dev/firmware.bin \ | ||||
|           0x3D0000 .pio/build/esp32dev/spiffs.bin | ||||
|  | ||||
|     - name: Prepare OTA firmware | ||||
|       run: | | ||||
|         cp .pio/build/esp32dev/firmware.bin .pio/build/esp32dev/filaman_ota.bin | ||||
|      | ||||
|     - name: Get version from tag | ||||
|       id: get_version | ||||
|       run: | | ||||
|         echo "VERSION=${GITHUB_REF#refs/tags/v}" >> $GITHUB_OUTPUT | ||||
|        | ||||
|     - name: Read CHANGELOG.md | ||||
|       id: changelog | ||||
|       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 | ||||
|  | ||||
|     - name: Create GitHub Release | ||||
|       env: | ||||
|         GH_TOKEN: ${{ github.token }} | ||||
|       run: | | ||||
|         gh release create "${{ github.ref_name }}" \ | ||||
|           --title "Release ${{ steps.get_version.outputs.VERSION }}" \ | ||||
|           --notes "${{ steps.changelog.outputs.CHANGES }}" \ | ||||
|           .pio/build/esp32dev/filaman_full.bin \ | ||||
|           .pio/build/esp32dev/filaman_ota.bin | ||||
							
								
								
									
										59
									
								
								.github/workflows/release.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										59
									
								
								.github/workflows/release.yml
									
									
									
									
										vendored
									
									
								
							| @@ -5,66 +5,37 @@ on: | ||||
|     tags: | ||||
|       - 'v*' | ||||
|  | ||||
| permissions: | ||||
|   contents: write | ||||
|  | ||||
| jobs: | ||||
|   route: | ||||
|   detect-provider: | ||||
|     runs-on: ubuntu-latest | ||||
|     outputs: | ||||
|       provider: ${{ steps.provider.outputs.provider }} | ||||
|       gitea_ref_name: ${{ steps.provider.outputs.gitea_ref_name }} | ||||
|       gitea_server_url: ${{ steps.provider.outputs.gitea_server_url }} | ||||
|       gitea_repository: ${{ steps.provider.outputs.gitea_repository }} | ||||
|     steps: | ||||
|       - name: Checkout Repository | ||||
|         uses: actions/checkout@v3 | ||||
|  | ||||
|       - name: Debug Environment | ||||
|         run: | | ||||
|           echo "CI Environment Details:" | ||||
|           echo "GITHUB_ACTIONS=${GITHUB_ACTIONS:-not set}" | ||||
|           echo "GITEA_ACTIONS=${GITEA_ACTIONS:-not set}" | ||||
|           echo "GITEA_REPOSITORY=${GITEA_REPOSITORY:-not set}" | ||||
|           echo "GITEA_SERVER_URL=${GITEA_SERVER_URL:-not set}" | ||||
|           echo "RUNNER_NAME=${RUNNER_NAME:-not set}" | ||||
|  | ||||
|       - name: Determine CI Provider | ||||
|         id: provider | ||||
|         shell: bash | ||||
|         run: | | ||||
|           if [ -n "${GITEA_ACTIONS}" ] || [ -n "${GITEA_REPOSITORY}" ] || [[ "${RUNNER_NAME}" == *"gitea"* ]]; then | ||||
|             echo "provider=gitea" >> "$GITHUB_OUTPUT" | ||||
|             echo "gitea_ref_name=${GITHUB_REF_NAME}" >> "$GITHUB_OUTPUT" | ||||
|             echo "gitea_server_url=${GITHUB_SERVER_URL}" >> "$GITHUB_OUTPUT" | ||||
|             echo "gitea_repository=${GITHUB_REPOSITORY}" >> "$GITHUB_OUTPUT" | ||||
|           elif [ "${GITHUB_ACTIONS}" = "true" ]; then | ||||
|             echo "provider=github" >> "$GITHUB_OUTPUT" | ||||
|           else | ||||
|             echo "provider=unknown" >> "$GITHUB_OUTPUT" | ||||
|           fi | ||||
|  | ||||
|   verify-provider: | ||||
|     needs: route | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: Echo detected provider | ||||
|         run: | | ||||
|           echo "Detected CI Provider: ${{ needs.route.outputs.provider }}" | ||||
|           if [ "${{ needs.route.outputs.provider }}" = "unknown" ]; then | ||||
|             echo "::error::Failed to detect CI provider!" | ||||
|             exit 1 | ||||
|             echo "provider=github" >> "$GITHUB_OUTPUT" | ||||
|           fi | ||||
|  | ||||
|   github-release: | ||||
|     needs: [route, verify-provider] | ||||
|     if: needs.route.outputs.provider == 'github' | ||||
|     uses: ./.github/workflows/providers/github-release.yml | ||||
|     needs: detect-provider | ||||
|     permissions: | ||||
|       contents: write | ||||
|     if: needs.detect-provider.outputs.provider == 'github' | ||||
|     uses: ./.github/workflows/github-release.yml | ||||
|     secrets: | ||||
|       RELEASE_TOKEN: ${{ secrets.GITHUB_TOKEN }} | ||||
|  | ||||
|   gitea-release: | ||||
|     needs: [route, verify-provider] | ||||
|     if: needs.route.outputs.provider == 'gitea' | ||||
|     uses: ./.github/workflows/providers/gitea-release.yml | ||||
|     with: | ||||
|       gitea_ref_name: ${{ needs.route.outputs.gitea_ref_name }} | ||||
|       gitea_server_url: ${{ needs.route.outputs.gitea_server_url }} | ||||
|       gitea_repository: ${{ needs.route.outputs.gitea_repository }} | ||||
|     needs: detect-provider | ||||
|     if: needs.detect-provider.outputs.provider == 'gitea' | ||||
|     uses: ./.github/workflows/gitea-release.yml | ||||
|     secrets: | ||||
|       GITEA_TOKEN: ${{ secrets.GITEA_TOKEN }} | ||||
							
								
								
									
										1000
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										1000
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										55
									
								
								README.de.md
									
									
									
									
									
								
							
							
						
						
									
										55
									
								
								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 | | ||||
| @@ -71,10 +71,15 @@ Deutsches Erklärvideo: [Youtube](https://youtu.be/uNDe2wh9SS8?si=b-jYx4I1w62zaO | ||||
| | OLED SCL          | 22        | | ||||
| | PN532 IRQ         | 32        | | ||||
| | PN532 RESET       | 33        | | ||||
| | PN532 SCK         | 14        | | ||||
| | PN532 MOSI        | 13        | | ||||
| | PN532 MISO        | 12        | | ||||
| | PN532 CS/SS       | 15        | | ||||
| | PN532 SDA         | 21        | | ||||
| | PN532 SCL         | 22        | | ||||
|  | ||||
| **Achte darauf, dass am PN532 die DIP-Schalter auf I2C gestellt sind** | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
| ## Software-Abhängigkeiten | ||||
|  | ||||
| @@ -101,7 +106,31 @@ Deutsches Erklärvideo: [Youtube](https://youtu.be/uNDe2wh9SS8?si=b-jYx4I1w62zaO | ||||
|   - PN532 NFC Modul | ||||
|   - Verbindungskabel | ||||
|  | ||||
| ### Schritt-für-Schritt Installation | ||||
| ## Wichtiger Hinweis | ||||
| Du musst Spoolman auf DEBUG Modus setzten, da man bisher in Spoolman keine CORS Domains setzen kann! | ||||
|  | ||||
| ``` | ||||
| # Enable debug mode | ||||
| # If enabled, the client will accept requests from any host | ||||
| # This can be useful when developing, but is also a security risk | ||||
| # Default: FALSE | ||||
| #SPOOLMAN_DEBUG_MODE=TRUE | ||||
| ``` | ||||
|  | ||||
| ## Schritt-für-Schritt Installation | ||||
| ### Einfache Installation | ||||
| 1. **Gehe auf [FilaMan Installer](https://www.filaman.app/installer.html)** | ||||
|  | ||||
| 2. **Stecke dein ESP an den Rechner und klicke Connect** | ||||
|  | ||||
| 3. **Wähle dein Device Port und klicke Intall** | ||||
|  | ||||
| 4. **Ersteinrichtung:** | ||||
|     - Mit dem "FilaMan" WLAN-Zugangspunkt verbinden. | ||||
|     - WLAN-Einstellungen über das Konfigurationsportal vornehmen. | ||||
|     - Weboberfläche unter `http://filaman.local` oder der IP-Adresse aufrufen. | ||||
|  | ||||
| ### Compile by yourself | ||||
| 1. **Repository klonen:** | ||||
|     ```bash | ||||
|     git clone https://github.com/ManuelW77/Filaman.git | ||||
|   | ||||
							
								
								
									
										79
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										79
									
								
								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 | ||||
| @@ -75,10 +75,15 @@ german explanatory video: [Youtube](https://youtu.be/uNDe2wh9SS8?si=b-jYx4I1w62z | ||||
| | OLED SCL          | 22        | | ||||
| | PN532 IRQ         | 32        | | ||||
| | PN532 RESET       | 33        | | ||||
| | PN532 SCK  	    | 14        | | ||||
| | PN532 MOSI    	| 13        | | ||||
| | PN532 MISO       	| 12        | | ||||
| | PN532 CS/SS       | 15        | | ||||
| | PN532 SDA         | 21        | | ||||
| | PN532 SCL         | 22        | | ||||
|  | ||||
| **Make sure that the DIP switches on the PN532 are set to I2C** | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
| ## Software Dependencies | ||||
|  | ||||
| @@ -91,9 +96,9 @@ german explanatory video: [Youtube](https://youtu.be/uNDe2wh9SS8?si=b-jYx4I1w62z | ||||
| - `Adafruit_SSD1306`: OLED display control | ||||
| - `HX711`: Load cell communication | ||||
|  | ||||
| ## Installation | ||||
| ### Installation | ||||
|  | ||||
| ### Prerequisites | ||||
| ## Prerequisites | ||||
| - **Software:** | ||||
|   - [PlatformIO](https://platformio.org/) in VS Code | ||||
|   - [Spoolman](https://github.com/Donkie/Spoolman) instance | ||||
| @@ -105,7 +110,32 @@ german explanatory video: [Youtube](https://youtu.be/uNDe2wh9SS8?si=b-jYx4I1w62z | ||||
|   - PN532 NFC Module | ||||
|   - Connecting wires | ||||
|  | ||||
| ### Step-by-Step Installation | ||||
| ## Important Note | ||||
| You have to activate Spoolman in debug mode, because you are not able to set CORS Domains in Spoolman yet. | ||||
|  | ||||
| ``` | ||||
| # Enable debug mode | ||||
| # If enabled, the client will accept requests from any host | ||||
| # This can be useful when developing, but is also a security risk | ||||
| # Default: FALSE | ||||
| #SPOOLMAN_DEBUG_MODE=TRUE | ||||
| ``` | ||||
|  | ||||
|  | ||||
| ## Step-by-Step Installation | ||||
| ### Easy Installation | ||||
| 1. **Go to [FilaMan Installer](https://www.filaman.app/installer.html)** | ||||
|  | ||||
| 2. **Plug you device in and push Connect button** | ||||
|  | ||||
| 3. **Select your Device Port and push Intall** | ||||
|  | ||||
| 4. **Initial Setup:** | ||||
|     - Connect to the "FilaMan" WiFi access point. | ||||
|     - Configure WiFi settings through the captive portal. | ||||
|     - Access the web interface at `http://filaman.local` or the IP address. | ||||
|  | ||||
| ### Compile by yourself | ||||
| 1. **Clone the Repository:** | ||||
|     ```bash | ||||
|     git clone https://github.com/ManuelW77/Filaman.git | ||||
| @@ -124,25 +154,6 @@ german explanatory video: [Youtube](https://youtu.be/uNDe2wh9SS8?si=b-jYx4I1w62z | ||||
|     - Configure WiFi settings through the captive portal. | ||||
|     - Access the web interface at `http://filaman.local` or the IP address. | ||||
|  | ||||
| ## GitHub Actions Configuration | ||||
|  | ||||
| ### Required Secrets for Gitea Releases | ||||
|  | ||||
| When using Gitea as your repository host, you need to configure the following secrets in your repository: | ||||
|  | ||||
| - `GITEA_API_URL`: The base URL of your Gitea instance, including protocol (e.g., `https://git.example.com`) | ||||
| - `GITEA_TOKEN`: Your Gitea access token with permissions to create releases | ||||
| - `GITEA_REPOSITORY`: The repository name in format `owner/repo` (e.g., `username/filaman`) | ||||
|  | ||||
| Example values: | ||||
| ``` | ||||
| GITEA_API_URL=https://git.example.com | ||||
| GITEA_TOKEN=abcdef1234567890 | ||||
| GITEA_REPOSITORY=username/filaman | ||||
| ``` | ||||
|  | ||||
| Make sure to set these secrets in your repository settings under Settings > Secrets and Variables > Actions. | ||||
|  | ||||
| ## Documentation | ||||
|  | ||||
| ### Relevant Links | ||||
|   | ||||
| @@ -1,7 +1,31 @@ | ||||
| { | ||||
|     "GFU99": "Generic TPU", | ||||
|     "GFN99": "Generic PA", | ||||
|     "GFN98": "Generic PA-CF", | ||||
|     "GFU99": "TPU", | ||||
|     "GFN99": "PA", | ||||
|     "GFN98": "PA-CF", | ||||
|     "GFL99": "PLA", | ||||
|     "GFL96": "PLA Silk", | ||||
|     "GFL98": "PLA-CF", | ||||
|     "GFL95": "PLA High Speed", | ||||
|     "GFG99": "PETG", | ||||
|     "GFG98": "PETG-CF", | ||||
|     "GFG97": "PCTG", | ||||
|     "GFB99": "ABS", | ||||
|     "GFC99": "PC", | ||||
|     "GFB98": "ASA", | ||||
|     "GFS99": "PVA", | ||||
|     "GFS98": "HIPS", | ||||
|     "GFT98": "PPS-CF", | ||||
|     "GFT97": "PPS", | ||||
|     "GFN97": "PPA-CF", | ||||
|     "GFN96": "PPA-GF", | ||||
|     "GFP99": "PE", | ||||
|     "GFP98": "PE-CF", | ||||
|     "GFP97": "PP", | ||||
|     "GFP96": "PP-CF", | ||||
|     "GFP95": "PP-GF", | ||||
|     "GFR99": "EVA", | ||||
|     "GFR98": "PHA", | ||||
|     "GFS97": "BVOH", | ||||
|     "GFA01": "Bambu PLA Matte", | ||||
|     "GFA00": "Bambu PLA Basic", | ||||
|     "GFA09": "Bambu PLA Tough", | ||||
| @@ -13,15 +37,11 @@ | ||||
|     "GFL03": "eSUN PLA+", | ||||
|     "GFL01": "PolyTerra PLA", | ||||
|     "GFL00": "PolyLite PLA", | ||||
|     "GFL99": "Generic PLA", | ||||
|     "GFL96": "Generic PLA Silk", | ||||
|     "GFL98": "Generic PLA-CF", | ||||
|     "GFA50": "Bambu PLA-CF", | ||||
|     "GFS02": "Bambu Support For PLA", | ||||
|     "GFA11": "Bambu PLA Aero", | ||||
|     "GFL04": "Overture PLA", | ||||
|     "GFL05": "Overture Matte PLA", | ||||
|     "GFL95": "Generic PLA High Speed", | ||||
|     "GFA12": "Bambu PLA Glow", | ||||
|     "GFA13": "Bambu PLA Dynamic", | ||||
|     "GFA15": "Bambu PLA Galaxy", | ||||
| @@ -30,41 +50,21 @@ | ||||
|     "GFU00": "Bambu TPU 95A HF", | ||||
|     "GFG00": "Bambu PETG Basic", | ||||
|     "GFT01": "Bambu PET-CF", | ||||
|     "GFG99": "Generic PETG", | ||||
|     "GFG98": "Generic PETG-CF", | ||||
|     "GFG50": "Bambu PETG-CF", | ||||
|     "GFG60": "PolyLite PETG", | ||||
|     "GFG01": "Bambu PETG Translucent", | ||||
|     "GFG97": "Generic PCTG", | ||||
|     "GFB00": "Bambu ABS", | ||||
|     "GFB99": "Generic ABS", | ||||
|     "GFB60": "PolyLite ABS", | ||||
|     "GFB50": "Bambu ABS-GF", | ||||
|     "GFC00": "Bambu PC", | ||||
|     "GFC99": "Generic PC", | ||||
|     "GFB98": "Generic ASA", | ||||
|     "GFB01": "Bambu ASA", | ||||
|     "GFB61": "PolyLite ASA", | ||||
|     "GFB02": "Bambu ASA-Aero", | ||||
|     "GFS99": "Generic PVA", | ||||
|     "GFS04": "Bambu PVA", | ||||
|     "GFS01": "Bambu Support G", | ||||
|     "GFN03": "Bambu PA-CF", | ||||
|     "GFN04": "Bambu PAHT-CF", | ||||
|     "GFS03": "Bambu Support For PA/PET", | ||||
|     "GFN05": "Bambu PA6-CF", | ||||
|     "GFN08": "Bambu PA6-GF", | ||||
|     "GFS98": "Generic HIPS", | ||||
|     "GFT98": "Generic PPS-CF", | ||||
|     "GFT97": "Generic PPS", | ||||
|     "GFN97": "Generic PPA-CF", | ||||
|     "GFN96": "Generic PPA-GF", | ||||
|     "GFP99": "Generic PE", | ||||
|     "GFP98": "Generic PE-CF", | ||||
|     "GFP97": "Generic PP", | ||||
|     "GFP96": "Generic PP-CF", | ||||
|     "GFP95": "Generic PP-GF", | ||||
|     "GFR99": "Generic EVA", | ||||
|     "GFR98": "Generic PHA", | ||||
|     "GFS97": "Generic BVOH" | ||||
|     "GFN08": "Bambu PA6-GF" | ||||
| } | ||||
| @@ -6,13 +6,24 @@ | ||||
|     <title>FilaMan - Filament Management Tool</title> | ||||
|     <link rel="icon" type="image/png" href="/favicon.ico"> | ||||
|     <link rel="stylesheet" href="style.css"> | ||||
|     <script> | ||||
|         fetch('/api/version') | ||||
|             .then(response => response.json()) | ||||
|             .then(data => { | ||||
|                 const versionSpan = document.querySelector('.version'); | ||||
|                 if (versionSpan) { | ||||
|                     versionSpan.textContent = 'v' + data.version; | ||||
|                 } | ||||
|             }) | ||||
|             .catch(error => console.error('Error fetching version:', error)); | ||||
|     </script> | ||||
| </head> | ||||
| <body> | ||||
|     <div class="navbar"> | ||||
|         <div style="display: flex; align-items: center; gap: 2rem;"> | ||||
|             <img src="/logo.png" alt="FilaMan Logo" class="logo"> | ||||
|             <div class="logo-text"> | ||||
|                 <h1>FilaMan<span class="version">v1.2.61</span></h1> | ||||
|                 <h1>FilaMan<span class="version"></span></h1> | ||||
|                 <h4>Filament Management Tool</h4> | ||||
|             </div> | ||||
|         </div> | ||||
|   | ||||
| @@ -6,13 +6,24 @@ | ||||
|     <title>FilaMan - Filament Management Tool</title> | ||||
|     <link rel="icon" type="image/png" href="/favicon.ico"> | ||||
|     <link rel="stylesheet" href="style.css"> | ||||
|     <script> | ||||
|         fetch('/api/version') | ||||
|             .then(response => response.json()) | ||||
|             .then(data => { | ||||
|                 const versionSpan = document.querySelector('.version'); | ||||
|                 if (versionSpan) { | ||||
|                     versionSpan.textContent = 'v' + data.version; | ||||
|                 } | ||||
|             }) | ||||
|             .catch(error => console.error('Error fetching version:', error)); | ||||
|     </script> | ||||
| </head> | ||||
| <body> | ||||
|     <div class="navbar"> | ||||
|         <div style="display: flex; align-items: center; gap: 2rem;"> | ||||
|             <img src="/logo.png" alt="FilaMan Logo" class="logo"> | ||||
|             <div class="logo-text"> | ||||
|                 <h1>FilaMan<span class="version">v1.2.61</span></h1> | ||||
|                 <h1>FilaMan<span class="version"></span></h1> | ||||
|                 <h4>Filament Management Tool</h4> | ||||
|             </div> | ||||
|         </div> | ||||
| @@ -36,7 +47,7 @@ | ||||
|  | ||||
| <!-- head --> | ||||
|  | ||||
|     <div class="container"> | ||||
|     <div class="content"> | ||||
|         <h1>FilaMan</h1> | ||||
|         <p>Filament Management Tool</p> | ||||
|         <p>Your smart solution for <strong>Filament Management</strong> in 3D printing.</p> | ||||
| @@ -44,10 +55,11 @@ | ||||
|         <h2>About FilaMan</h2> | ||||
|         <p> | ||||
|             FilaMan is a tool designed to simplify filament spool management. It allows you to identify and weigh filament spools,  | ||||
|             automatically sync data with the self-hosted <a href="https://github.com/Donkie/Spoolman" target="_blank">Spoolman</a> platform,  | ||||
|             and ensure compatibility with <a href="https://github.com/spuder/OpenSpool" target="_blank">OpenSpool</a> for Bambu printers. | ||||
|             automatically sync data with the self-hosted <a href="https://github.com/Donkie/Spoolman" target="_blank">Spoolman</a> platform. | ||||
|         </p> | ||||
|  | ||||
|         <p>Get more information at <a href="https://www.filaman.app" target="_blank">https://www.filaman.app</a> and <a href="https://github.com/ManuelW77/Filaman" target="_blank">https://github.com/ManuelW77/Filaman</a>.</p> | ||||
|  | ||||
|         <div class="features"> | ||||
|             <div class="feature"> | ||||
|                 <h3>Spool Identification</h3> | ||||
| @@ -62,12 +74,6 @@ | ||||
|                 <p>Works with OpenSpool to recognize and activate spools on Bambu printers.</p> | ||||
|             </div> | ||||
|         </div> | ||||
|  | ||||
|         <h2>Future Plans</h2> | ||||
|         <p> | ||||
|             We are working on expanding compatibility to support smaller NFC tags like NTag213  | ||||
|             and developing custom software to enhance the OpenSpool experience. | ||||
|         </p> | ||||
|     </div> | ||||
| </body> | ||||
| </html> | ||||
|   | ||||
| @@ -6,13 +6,24 @@ | ||||
|     <title>FilaMan - Filament Management Tool</title> | ||||
|     <link rel="icon" type="image/png" href="/favicon.ico"> | ||||
|     <link rel="stylesheet" href="style.css"> | ||||
|     <script> | ||||
|         fetch('/api/version') | ||||
|             .then(response => response.json()) | ||||
|             .then(data => { | ||||
|                 const versionSpan = document.querySelector('.version'); | ||||
|                 if (versionSpan) { | ||||
|                     versionSpan.textContent = 'v' + data.version; | ||||
|                 } | ||||
|             }) | ||||
|             .catch(error => console.error('Error fetching version:', error)); | ||||
|     </script> | ||||
| </head> | ||||
| <body> | ||||
|     <div class="navbar"> | ||||
|         <div style="display: flex; align-items: center; gap: 2rem;"> | ||||
|             <img src="/logo.png" alt="FilaMan Logo" class="logo"> | ||||
|             <div class="logo-text"> | ||||
|                 <h1>FilaMan<span class="version">v1.2.61</span></h1> | ||||
|                 <h1>FilaMan<span class="version"></span></h1> | ||||
|                 <h4>Filament Management Tool</h4> | ||||
|             </div> | ||||
|         </div> | ||||
|   | ||||
| @@ -6,13 +6,24 @@ | ||||
|     <title>FilaMan - Filament Management Tool</title> | ||||
|     <link rel="icon" type="image/png" href="/favicon.ico"> | ||||
|     <link rel="stylesheet" href="style.css"> | ||||
|     <script> | ||||
|         fetch('/api/version') | ||||
|             .then(response => response.json()) | ||||
|             .then(data => { | ||||
|                 const versionSpan = document.querySelector('.version'); | ||||
|                 if (versionSpan) { | ||||
|                     versionSpan.textContent = 'v' + data.version; | ||||
|                 } | ||||
|             }) | ||||
|             .catch(error => console.error('Error fetching version:', error)); | ||||
|     </script> | ||||
| </head> | ||||
| <body> | ||||
|     <div class="navbar"> | ||||
|         <div style="display: flex; align-items: center; gap: 2rem;"> | ||||
|             <img src="/logo.png" alt="FilaMan Logo" class="logo"> | ||||
|             <div class="logo-text"> | ||||
|                 <h1>FilaMan<span class="version">v1.2.61</span></h1> | ||||
|                 <h1>FilaMan<span class="version"></span></h1> | ||||
|                 <h4>Filament Management Tool</h4> | ||||
|             </div> | ||||
|         </div> | ||||
| @@ -63,8 +74,9 @@ | ||||
|             const ip = document.getElementById('bambuIp').value; | ||||
|             const serial = document.getElementById('bambuSerial').value; | ||||
|             const code = document.getElementById('bambuCode').value; | ||||
|             const autoSend = document.getElementById('autoSend').checked; | ||||
|  | ||||
|             fetch(`/api/bambu?bambu_ip=${encodeURIComponent(ip)}&bambu_serialnr=${encodeURIComponent(serial)}&bambu_accesscode=${encodeURIComponent(code)}`) | ||||
|             fetch(`/api/bambu?bambu_ip=${encodeURIComponent(ip)}&bambu_serialnr=${encodeURIComponent(serial)}&bambu_accesscode=${encodeURIComponent(code)}&autoSend=${autoSend}`) | ||||
|                 .then(response => response.json()) | ||||
|                 .then(data => { | ||||
|                     if (data.healthy) { | ||||
| @@ -84,27 +96,42 @@ | ||||
|      | ||||
|     <div class="content"> | ||||
|         <h1>Spoolman API URL / Bambu Credentials</h1> | ||||
|         <label for="spoolmanUrl">Set URL/IP to your Spoolman-Instanz:</label> | ||||
|         <input type="text" id="spoolmanUrl" placeholder="http://ip-or-url-of-your-spoolman-instanz:port"> | ||||
|         <button onclick="checkSpoolmanInstance()">Save Spoolman URL</button> | ||||
|         <p id="statusMessage"></p> | ||||
|  | ||||
|         <h2>Bambu Lab Printer Credentials</h2> | ||||
|         <div class="bambu-settings"> | ||||
|             <div class="input-group"> | ||||
|                 <label for="bambuIp">Bambu Drucker IP-Adresse:</label> | ||||
|                 <input type="text" id="bambuIp" placeholder="192.168.1.xxx" value="{{bambuIp}}"> | ||||
|         <div class="card"> | ||||
|             <div class="card-body"> | ||||
|                 <h5 class="card-title">Set URL/IP to your Spoolman-Instanz</h5> | ||||
|                 <input type="text" id="spoolmanUrl" placeholder="http://ip-or-url-of-your-spoolman-instanz:port"> | ||||
|                 <button onclick="checkSpoolmanInstance()">Save Spoolman URL</button> | ||||
|                 <p id="statusMessage"></p> | ||||
|             </div> | ||||
|             <div class="input-group"> | ||||
|                 <label for="bambuSerial">Drucker Seriennummer:</label> | ||||
|                 <input type="text" id="bambuSerial" placeholder="BBLXXXXXXXX" value="{{bambuSerial}}"> | ||||
|         </div> | ||||
|  | ||||
|         <div class="card"> | ||||
|             <div class="card-body"> | ||||
|                 <h5 class="card-title">Bambu Lab Printer Credentials</h5> | ||||
|                 <div class="bambu-settings"> | ||||
|                     <div class="input-group"> | ||||
|                         <label for="bambuIp">Bambu Drucker IP-Adresse:</label> | ||||
|                         <input type="text" id="bambuIp" placeholder="192.168.1.xxx" value="{{bambuIp}}"> | ||||
|                     </div> | ||||
|                     <div class="input-group"> | ||||
|                         <label for="bambuSerial">Drucker Seriennummer:</label> | ||||
|                         <input type="text" id="bambuSerial" placeholder="BBLXXXXXXXX" value="{{bambuSerial}}"> | ||||
|                     </div> | ||||
|                     <div class="input-group"> | ||||
|                         <label for="bambuCode">Access Code:</label> | ||||
|                         <input type="text" id="bambuCode" placeholder="Access Code vom Drucker" value="{{bambuCode}}"> | ||||
|                     </div> | ||||
|                     <div class="input-group"> | ||||
|                         If activated, FilaMan will automatically update the next filled tray with the last scanned and weighed spool. | ||||
|                         <label for="autoSend">Auto Send to Bambu:</label> | ||||
|                         <input type="checkbox" id="autoSend" {{autoSendToBambu}}> | ||||
|                     </div> | ||||
|  | ||||
|                     <button onclick="saveBambuCredentials()">Save Bambu Credentials</button> | ||||
|                     <p id="bambuStatusMessage"></p> | ||||
|                 </div> | ||||
|             </div> | ||||
|             <div class="input-group"> | ||||
|                 <label for="bambuCode">Access Code:</label> | ||||
|                 <input type="text" id="bambuCode" placeholder="Access Code vom Drucker" value="{{bambuCode}}"> | ||||
|             </div> | ||||
|             <button onclick="saveBambuCredentials()">Save Bambu Credentials</button> | ||||
|             <p id="bambuStatusMessage"></p> | ||||
|         </div> | ||||
|     </div> | ||||
| </body> | ||||
|   | ||||
| @@ -279,9 +279,10 @@ a:hover { | ||||
|  | ||||
| /* Karten-Stil für optische Trennung */ | ||||
| .card { | ||||
|     background: #f9f9f9; | ||||
|     background: var(--primary-color); | ||||
|     width: 500px; | ||||
|     padding: 15px; | ||||
|     margin: 20px 0; | ||||
|     margin: 20px auto; | ||||
|     border-radius: 8px; | ||||
|     box-shadow: 0 2px 4px rgba(0, 0, 0, 0.1); | ||||
| } | ||||
| @@ -959,7 +960,6 @@ input[type="submit"]:disabled, | ||||
|  | ||||
| /* Bambu Settings Erweiterung */ | ||||
| .bambu-settings { | ||||
|     background: white; | ||||
|     padding: 20px; | ||||
|     border-radius: 8px; | ||||
|     box-shadow: 0 2px 4px rgba(0, 0, 0, 0.1); | ||||
| @@ -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; | ||||
| } | ||||
| @@ -6,13 +6,24 @@ | ||||
|     <title>FilaMan - Filament Management Tool</title> | ||||
|     <link rel="icon" type="image/png" href="/favicon.ico"> | ||||
|     <link rel="stylesheet" href="style.css"> | ||||
|     <script> | ||||
|         fetch('/api/version') | ||||
|             .then(response => response.json()) | ||||
|             .then(data => { | ||||
|                 const versionSpan = document.querySelector('.version'); | ||||
|                 if (versionSpan) { | ||||
|                     versionSpan.textContent = 'v' + data.version; | ||||
|                 } | ||||
|             }) | ||||
|             .catch(error => console.error('Error fetching version:', error)); | ||||
|     </script> | ||||
| </head> | ||||
| <body> | ||||
|     <div class="navbar"> | ||||
|         <div style="display: flex; align-items: center; gap: 2rem;"> | ||||
|             <img src="/logo.png" alt="FilaMan Logo" class="logo"> | ||||
|             <div class="logo-text"> | ||||
|                 <h1>FilaMan<span class="version">v1.2.61</span></h1> | ||||
|                 <h1>FilaMan<span class="version"></span></h1> | ||||
|                 <h4>Filament Management Tool</h4> | ||||
|             </div> | ||||
|         </div> | ||||
| @@ -40,18 +51,34 @@ | ||||
|         <h1>Firmware Upgrade</h1> | ||||
|  | ||||
|         <div class="warning"> | ||||
|             <strong>Warning:</strong> Please do not turn off or restart the device during the update.  | ||||
|             The device will restart automatically after the update. | ||||
|             <strong>Warning:</strong> Do not power off the device during update. | ||||
|         </div> | ||||
|  | ||||
|         <div class="update-form"> | ||||
|             <form id="updateForm" enctype='multipart/form-data'> | ||||
|                 <input type='file' name='update' accept='.bin' required> | ||||
|                 <input type='submit' value='Start Firmware Update'> | ||||
|             </form> | ||||
|         <div class="update-options"> | ||||
|             <div class="update-section"> | ||||
|                 <h2>Firmware Update</h2> | ||||
|                 <p>Upload a new firmware file (filaman_*.bin)</p> | ||||
|                 <div class="update-form"> | ||||
|                     <form id="firmwareForm" enctype='multipart/form-data' data-type="firmware"> | ||||
|                         <input type='file' name='update' accept='.bin' required> | ||||
|                         <input type='submit' value='Start Firmware Update'> | ||||
|                     </form> | ||||
|                 </div> | ||||
|             </div> | ||||
|  | ||||
|             <div class="update-section"> | ||||
|                 <h2>Webpage Update</h2> | ||||
|                 <p>Upload a new webpage file (webpage_*.bin)</p> | ||||
|                 <div class="update-form"> | ||||
|                     <form id="webpageForm" enctype='multipart/form-data' data-type="webpage"> | ||||
|                         <input type='file' name='update' accept='.bin' required> | ||||
|                         <input type='submit' value='Start Webpage Update'> | ||||
|                     </form> | ||||
|                 </div> | ||||
|             </div> | ||||
|         </div> | ||||
|  | ||||
|         <div class="progress-container"> | ||||
|         <div class="progress-container" style="display: none;"> | ||||
|             <div class="progress-bar">0%</div> | ||||
|         </div> | ||||
|         <div class="status"></div> | ||||
| @@ -64,91 +91,163 @@ | ||||
|             statusContainer.style.display = 'none'; | ||||
|         } | ||||
|  | ||||
|         document.getElementById('updateForm').addEventListener('submit', async (e) => { | ||||
|             e.preventDefault(); | ||||
|             const form = e.target; | ||||
|             const file = form.update.files[0]; | ||||
|             if (!file) { | ||||
|                 alert('Please select a firmware file.'); | ||||
|                 return; | ||||
|             } | ||||
|         const progress = document.querySelector('.progress-bar'); | ||||
|         const progressContainer = document.querySelector('.progress-container'); | ||||
|         const status = document.querySelector('.status'); | ||||
|         let updateInProgress = false; | ||||
|         let lastReceivedProgress = 0; | ||||
|  | ||||
|             const formData = new FormData(); | ||||
|             formData.append('update', file); | ||||
|         // WebSocket Handling | ||||
|         let ws = null; | ||||
|         let wsReconnectTimer = null; | ||||
|  | ||||
|             const progress = document.querySelector('.progress-bar'); | ||||
|             const progressContainer = document.querySelector('.progress-container'); | ||||
|             const status = document.querySelector('.status'); | ||||
|         function connectWebSocket() { | ||||
|             ws = new WebSocket('ws://' + window.location.host + '/ws'); | ||||
|              | ||||
|             progressContainer.style.display = 'block'; | ||||
|             status.style.display = 'none'; | ||||
|             status.className = 'status'; | ||||
|             form.querySelector('input[type=submit]').disabled = true; | ||||
|             ws.onmessage = function(event) { | ||||
|                 try { | ||||
|                     const data = JSON.parse(event.data); | ||||
|                     if (data.type === "updateProgress" && updateInProgress) { | ||||
|                         // Zeige Fortschrittsbalken | ||||
|                         progressContainer.style.display = 'block'; | ||||
|                          | ||||
|             const xhr = new XMLHttpRequest(); | ||||
|             xhr.open('POST', '/update', true); | ||||
|                         // 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; | ||||
|                         } | ||||
|                          | ||||
|             xhr.upload.onprogress = (e) => { | ||||
|                 if (e.lengthComputable) { | ||||
|                     const percentComplete = (e.loaded / e.total) * 100; | ||||
|                     progress.style.width = percentComplete + '%'; | ||||
|                     progress.textContent = Math.round(percentComplete) + '%'; | ||||
|                         // 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)'); | ||||
|                 return; | ||||
|             } | ||||
|             if (updateType === 'webpage' && !file.name.startsWith('upgrade_filaman_website_')) { | ||||
|                 alert('Please select a valid webpage file (upgrade_filaman_website_*.bin)'); | ||||
|                 return; | ||||
|             } | ||||
|              | ||||
|             // Reset UI | ||||
|             updateInProgress = true; | ||||
|             progressContainer.style.display = 'block'; | ||||
|             status.style.display = 'none'; | ||||
|             status.className = 'status'; | ||||
|             progress.style.width = '0%'; | ||||
|             progress.textContent = '0%'; | ||||
|              | ||||
|             // 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.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'); | ||||
|                 if (xhr.status !== 200 && !progress.textContent.startsWith('100')) { | ||||
|                     status.textContent = "Update failed: " + (xhr.responseText || "Unknown error"); | ||||
|                     status.className = 'status error'; | ||||
|                     status.style.display = 'block'; | ||||
|                      | ||||
|                     if (xhr.status !== 200) { | ||||
|                         form.querySelector('input[type=submit]').disabled = false; | ||||
|                     } | ||||
|                 } catch (error) { | ||||
|                     status.textContent = 'Error: ' + error.message; | ||||
|                     status.classList.add('error'); | ||||
|                     status.style.display = 'block'; | ||||
|                     form.querySelector('input[type=submit]').disabled = false; | ||||
|                     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'; | ||||
|                 form.querySelector('input[type=submit]').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(); | ||||
|             formData.append('update', file); | ||||
|             xhr.send(formData); | ||||
|         }); | ||||
|         } | ||||
|  | ||||
|         document.getElementById('firmwareForm').addEventListener('submit', handleUpdate); | ||||
|         document.getElementById('webpageForm').addEventListener('submit', handleUpdate); | ||||
|     </script> | ||||
| </body> | ||||
| </html> | ||||
| @@ -6,13 +6,24 @@ | ||||
|     <title>FilaMan - Filament Management Tool</title> | ||||
|     <link rel="icon" type="image/png" href="/favicon.ico"> | ||||
|     <link rel="stylesheet" href="style.css"> | ||||
|     <script> | ||||
|         fetch('/api/version') | ||||
|             .then(response => response.json()) | ||||
|             .then(data => { | ||||
|                 const versionSpan = document.querySelector('.version'); | ||||
|                 if (versionSpan) { | ||||
|                     versionSpan.textContent = 'v' + data.version; | ||||
|                 } | ||||
|             }) | ||||
|             .catch(error => console.error('Error fetching version:', error)); | ||||
|     </script> | ||||
| </head> | ||||
| <body> | ||||
|     <div class="navbar"> | ||||
|         <div style="display: flex; align-items: center; gap: 2rem;"> | ||||
|             <img src="/logo.png" alt="FilaMan Logo" class="logo"> | ||||
|             <div class="logo-text"> | ||||
|                 <h1>FilaMan<span class="version">v1.2.61</span></h1> | ||||
|                 <h1>FilaMan<span class="version"></span></h1> | ||||
|                 <h4>Filament Management Tool</h4> | ||||
|             </div> | ||||
|         </div> | ||||
|   | ||||
| @@ -6,13 +6,24 @@ | ||||
|     <title>FilaMan - Filament Management Tool</title> | ||||
|     <link rel="icon" type="image/png" href="/favicon.ico"> | ||||
|     <link rel="stylesheet" href="style.css"> | ||||
|     <script> | ||||
|         fetch('/api/version') | ||||
|             .then(response => response.json()) | ||||
|             .then(data => { | ||||
|                 const versionSpan = document.querySelector('.version'); | ||||
|                 if (versionSpan) { | ||||
|                     versionSpan.textContent = 'v' + data.version; | ||||
|                 } | ||||
|             }) | ||||
|             .catch(error => console.error('Error fetching version:', error)); | ||||
|     </script> | ||||
| </head> | ||||
| <body> | ||||
|     <div class="navbar"> | ||||
|         <div style="display: flex; align-items: center; gap: 2rem;"> | ||||
|             <img src="/logo.png" alt="FilaMan Logo" class="logo"> | ||||
|             <div class="logo-text"> | ||||
|                 <h1>FilaMan<span class="version">v1.2.61</span></h1> | ||||
|                 <h1>FilaMan<span class="version"></span></h1> | ||||
|                 <h4>Filament Management Tool</h4> | ||||
|             </div> | ||||
|         </div> | ||||
|   | ||||
							
								
								
									
										
											BIN
										
									
								
								img/ESP32-SPI-Pins.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								img/ESP32-SPI-Pins.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 143 KiB | 
							
								
								
									
										
											BIN
										
									
								
								img/IMG_2589.jpeg
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								img/IMG_2589.jpeg
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 136 KiB | 
							
								
								
									
										
											BIN
										
									
								
								img/IMG_2590.jpeg
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								img/IMG_2590.jpeg
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 143 KiB | 
							
								
								
									
										
											BIN
										
									
								
								img/Schaltplan.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								img/Schaltplan.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 283 KiB | 
| @@ -9,8 +9,8 @@ | ||||
| ; https://docs.platformio.org/page/projectconf.html | ||||
|  | ||||
| [common] | ||||
| version = "1.2.61" | ||||
|  | ||||
| version = "1.3.95" | ||||
| ## | ||||
| [env:esp32dev] | ||||
| platform = espressif32 | ||||
| board = esp32dev | ||||
| @@ -20,7 +20,10 @@ monitor_speed = 115200 | ||||
| lib_deps = | ||||
|     tzapu/WiFiManager @ ^2.0.17 | ||||
|     https://github.com/me-no-dev/ESPAsyncWebServer.git#master | ||||
|     me-no-dev/AsyncTCP @ ^1.1.1 | ||||
|     #me-no-dev/AsyncTCP @ ^1.1.1 | ||||
|     https://github.com/esphome/AsyncTCP.git | ||||
|     #mathieucarbou/ESPAsyncWebServer @ ^3.6.0 | ||||
|     #esp32async/AsyncTCP @ ^3.3.5 | ||||
|     bogde/HX711 @ ^0.7.5 | ||||
|     adafruit/Adafruit SSD1306 @ ^2.5.13 | ||||
|     adafruit/Adafruit GFX Library @ ^1.11.11 | ||||
| @@ -43,34 +46,26 @@ build_flags = | ||||
|     -fdata-sections | ||||
|     -DNDEBUG | ||||
|     -mtext-section-literals | ||||
|     '-D VERSION="${common.version}"' | ||||
|     -DVERSION=\"${common.version}\" | ||||
|     -DASYNCWEBSERVER_REGEX | ||||
|     -DCORE_DEBUG_LEVEL=1 | ||||
|     -DCORE_DEBUG_LEVEL=3 | ||||
|     -DCONFIG_ARDUHAL_LOG_COLORS=1 | ||||
|     -DOTA_DEBUG=1 | ||||
|     -DARDUINO_RUNNING_CORE=1 | ||||
|     -DARDUINO_EVENT_RUNNING_CORE=1 | ||||
|     -DCONFIG_OPTIMIZATION_LEVEL_DEBUG=1 | ||||
|     -DCONFIG_ESP32_PANIC_PRINT_REBOOT | ||||
|     -DCONFIG_ARDUINO_OTA_READSIZE=1024 | ||||
|     -DCONFIG_ASYNC_TCP_RUNNING_CORE=1 | ||||
|     -DCONFIG_ASYNC_TCP_USE_WDT=0 | ||||
|     -DCONFIG_LWIP_TCP_MSS=1460 | ||||
|     -DOTA_PARTITION_SUBTYPE=0x10 | ||||
|     -DPARTITION_TABLE_OFFSET=0x8000 | ||||
|     -DPARTITION_TABLE_SIZE=0x1000 | ||||
|     -DBOOT_APP_PARTITION_OTA_0=1 | ||||
|     -DCONFIG_LWIP_TCP_MSL=60000 | ||||
|     -DCONFIG_LWIP_TCP_RCV_BUF_DEFAULT=4096 | ||||
|     -DCONFIG_LWIP_MAX_ACTIVE_TCP=16 | ||||
|      | ||||
| extra_scripts =  | ||||
|     scripts/extra_script.py | ||||
|     pre:scripts/pre_build.py     ; wird zuerst ausgeführt | ||||
|     pre:scripts/pre_spiffs.py    ; wird als zweites ausgeführt | ||||
|     pre:scripts/combine_html.py  ; wird als drittes ausgeführt | ||||
|     scripts/gzip_files.py | ||||
|     ${env:buildfs.extra_scripts} | ||||
|  | ||||
| ; Remove or comment out the targets line | ||||
| ;targets = buildfs, build | ||||
| [env:buildfs] | ||||
| extra_scripts = | ||||
|     pre:scripts/combine_html.py  ; Combine header with HTML files | ||||
|     scripts/gzip_files.py       ; Compress files for SPIFFS | ||||
|  | ||||
| ; Add a custom target to build both | ||||
| [platformio] | ||||
| default_envs = esp32dev | ||||
|  | ||||
|   | ||||
| @@ -1,7 +1,39 @@ | ||||
| Import("env") | ||||
|  | ||||
| board_config = env.BoardConfig() | ||||
|  | ||||
| # Calculate SPIFFS size based on partition table | ||||
| SPIFFS_START = 0x310000  # From partitions.csv | ||||
| SPIFFS_SIZE = 0xE0000   # From partitions.csv | ||||
| SPIFFS_PAGE = 256 | ||||
| SPIFFS_BLOCK = 4096 | ||||
|  | ||||
| env.Replace( | ||||
|     MKSPIFFSTOOL="mkspiffs", | ||||
|     SPIFFSBLOCKSZ=SPIFFS_BLOCK, | ||||
|     SPIFFSBLOCKSIZE=SPIFFS_BLOCK, | ||||
|     SPIFFSSTART=SPIFFS_START, | ||||
|     SPIFFSEND=SPIFFS_START + SPIFFS_SIZE, | ||||
|     SPIFFSPAGESZ=SPIFFS_PAGE, | ||||
|     SPIFFSSIZE=SPIFFS_SIZE | ||||
| ) | ||||
|  | ||||
| # Wiederverwendung der replace_version Funktion | ||||
| exec(open("./scripts/pre_build.py").read()) | ||||
|  | ||||
| # Bind to SPIFFS build | ||||
| env.AddPreAction("buildfs", replace_version) | ||||
|  | ||||
| import os | ||||
| import shutil | ||||
| from SCons.Script import DefaultEnvironment | ||||
|  | ||||
| env = DefaultEnvironment() | ||||
|  | ||||
| # Format SPIFFS partition before uploading new files | ||||
| spiffs_dir = os.path.join(env.subst("$BUILD_DIR"), "spiffs") | ||||
| if os.path.exists(spiffs_dir): | ||||
|     shutil.rmtree(spiffs_dir) | ||||
| os.makedirs(spiffs_dir) | ||||
|  | ||||
| print("SPIFFS partition formatted.") | ||||
| @@ -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,9 +101,30 @@ 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: | ||||
|             print(f"Version {version} already exists in changelog") | ||||
|             # Version existiert bereits, aktualisiere die bestehenden Einträge | ||||
|             version_pattern = f"## \\[{version}\\] - \\d{{4}}-\\d{{2}}-\\d{{2}}" | ||||
|             next_version_pattern = "## \\[.*?\\] - \\d{4}-\\d{2}-\\d{2}" | ||||
|              | ||||
|             # Finde den Start der aktuellen Version | ||||
|             version_match = re.search(version_pattern, content) | ||||
|             if version_match: | ||||
|                 version_start = version_match.start() | ||||
|                 # Suche nach der nächsten Version | ||||
|                 next_version_match = re.search(next_version_pattern, content[version_start + 1:]) | ||||
|                  | ||||
|                 if next_version_match: | ||||
|                     # Ersetze den Inhalt zwischen aktueller und nächster Version | ||||
|                     next_version_pos = version_start + 1 + next_version_match.start() | ||||
|                     updated_content = content[:version_start] + changelog_entry + content[next_version_pos:] | ||||
|                 else: | ||||
|                     # Wenn keine nächste Version existiert, ersetze bis zum Ende | ||||
|                     updated_content = content[:version_start] + changelog_entry + "\n" | ||||
|                  | ||||
|                 with open(changelog_path, 'w') as f: | ||||
|                     f.write(updated_content) | ||||
|                 print(f"Updated entries for version {version}") | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|     update_changelog() | ||||
							
								
								
									
										111
									
								
								src/api.cpp
									
									
									
									
									
								
							
							
						
						
									
										111
									
								
								src/api.cpp
									
									
									
									
									
								
							| @@ -37,9 +37,9 @@ struct SendToApiParams { | ||||
|     } | ||||
| */ | ||||
|  | ||||
| JsonDocument fetchSpoolsForWebsite() { | ||||
| JsonDocument fetchSingleSpoolInfo(int spoolId) { | ||||
|     HTTPClient http; | ||||
|     String spoolsUrl = spoolmanUrl + apiUrl + "/spool"; | ||||
|     String spoolsUrl = spoolmanUrl + apiUrl + "/spool/" + spoolId; | ||||
|  | ||||
|     Serial.print("Rufe Spool-Daten von: "); | ||||
|     Serial.println(spoolsUrl); | ||||
| @@ -56,84 +56,45 @@ JsonDocument fetchSpoolsForWebsite() { | ||||
|             Serial.print("Fehler beim Parsen der JSON-Antwort: "); | ||||
|             Serial.println(error.c_str()); | ||||
|         } else { | ||||
|             JsonArray spools = doc.as<JsonArray>(); | ||||
|             JsonArray filteredSpools = filteredDoc.to<JsonArray>(); | ||||
|             String filamentType = doc["filament"]["material"].as<String>(); | ||||
|             String filamentBrand = doc["filament"]["vendor"]["name"].as<String>(); | ||||
|  | ||||
|             for (JsonObject spool : spools) { | ||||
|                 JsonObject filteredSpool = filteredSpools.createNestedObject(); | ||||
|                 filteredSpool["extra"]["nfc_id"] = spool["extra"]["nfc_id"]; | ||||
|             int nozzle_temp_min = 0; | ||||
|             int nozzle_temp_max = 0; | ||||
|             if (doc["filament"]["extra"]["nozzle_temperature"].is<String>()) { | ||||
|                 String tempString = doc["filament"]["extra"]["nozzle_temperature"].as<String>(); | ||||
|                 tempString.replace("[", ""); | ||||
|                 tempString.replace("]", ""); | ||||
|                 int commaIndex = tempString.indexOf(','); | ||||
|                  | ||||
|                 JsonObject filament = filteredSpool.createNestedObject("filament"); | ||||
|                 filament["sm_id"] = spool["id"]; | ||||
|                 filament["id"] = spool["filament"]["id"]; | ||||
|                 filament["name"] = spool["filament"]["name"]; | ||||
|                 filament["material"] = spool["filament"]["material"]; | ||||
|                 filament["color_hex"] = spool["filament"]["color_hex"]; | ||||
|                 filament["nozzle_temperature"] = spool["filament"]["extra"]["nozzle_temperature"]; // [190,230] | ||||
|                 filament["price_meter"] = spool["filament"]["extra"]["price_meter"]; | ||||
|                 filament["price_gramm"] = spool["filament"]["extra"]["price_gramm"]; | ||||
|  | ||||
|                 JsonObject vendor = filament.createNestedObject("vendor"); | ||||
|                 vendor["id"] = spool["filament"]["vendor"]["id"]; | ||||
|                 vendor["name"] = spool["filament"]["vendor"]["name"]; | ||||
|                 if (commaIndex != -1) { | ||||
|                     nozzle_temp_min = tempString.substring(0, commaIndex).toInt(); | ||||
|                     nozzle_temp_max = tempString.substring(commaIndex + 1).toInt(); | ||||
|                 } | ||||
|             }  | ||||
|         } | ||||
|     } else { | ||||
|         Serial.print("Fehler beim Abrufen der Spool-Daten. HTTP-Code: "); | ||||
|         Serial.println(httpCode); | ||||
|     } | ||||
|  | ||||
|     http.end(); | ||||
|     return filteredDoc; | ||||
| } | ||||
|             String filamentColor = doc["filament"]["color_hex"].as<String>(); | ||||
|             filamentColor.toUpperCase(); | ||||
|  | ||||
| JsonDocument fetchAllSpoolsInfo() { | ||||
|     HTTPClient http; | ||||
|     String spoolsUrl = spoolmanUrl + apiUrl + "/spool"; | ||||
|             String tray_info_idx = doc["filament"]["extra"]["bambu_idx"].as<String>(); | ||||
|             tray_info_idx.replace("\"", ""); | ||||
|              | ||||
|     Serial.print("Rufe Spool-Daten von: "); | ||||
|     Serial.println(spoolsUrl); | ||||
|             String cali_idx = doc["filament"]["extra"]["bambu_cali_id"].as<String>(); // "\"153\"" | ||||
|             cali_idx.replace("\"", ""); | ||||
|              | ||||
|     http.begin(spoolsUrl); | ||||
|     int httpCode = http.GET(); | ||||
|             String bambu_setting_id = doc["filament"]["extra"]["bambu_setting_id"].as<String>(); // "\"PFUSf40e9953b40d3d\"" | ||||
|             bambu_setting_id.replace("\"", ""); | ||||
|  | ||||
|     JsonDocument filteredDoc; | ||||
|     if (httpCode == HTTP_CODE_OK) { | ||||
|         String payload = http.getString(); | ||||
|         JsonDocument doc; | ||||
|         DeserializationError error = deserializeJson(doc, payload); | ||||
|         if (error) { | ||||
|             Serial.print("Fehler beim Parsen der JSON-Antwort: "); | ||||
|             Serial.println(error.c_str()); | ||||
|         } else { | ||||
|             JsonArray spools = doc.as<JsonArray>(); | ||||
|             JsonArray filteredSpools = filteredDoc.to<JsonArray>(); | ||||
|             doc.clear(); | ||||
|  | ||||
|             for (JsonObject spool : spools) { | ||||
|                 JsonObject filteredSpool = filteredSpools.createNestedObject(); | ||||
|                 filteredSpool["price"] = spool["price"]; | ||||
|                 filteredSpool["remaining_weight"] = spool["remaining_weight"]; | ||||
|                 filteredSpool["used_weight"] = spool["used_weight"]; | ||||
|                 filteredSpool["extra"]["nfc_id"] = spool["extra"]["nfc_id"]; | ||||
|  | ||||
|                 JsonObject filament = filteredSpool.createNestedObject("filament"); | ||||
|                 filament["id"] = spool["filament"]["id"]; | ||||
|                 filament["name"] = spool["filament"]["name"]; | ||||
|                 filament["material"] = spool["filament"]["material"]; | ||||
|                 filament["density"] = spool["filament"]["density"]; | ||||
|                 filament["diameter"] = spool["filament"]["diameter"]; | ||||
|                 filament["spool_weight"] = spool["filament"]["spool_weight"]; | ||||
|                 filament["color_hex"] = spool["filament"]["color_hex"]; | ||||
|  | ||||
|                 JsonObject vendor = filament.createNestedObject("vendor"); | ||||
|                 vendor["id"] = spool["filament"]["vendor"]["id"]; | ||||
|                 vendor["name"] = spool["filament"]["vendor"]["name"]; | ||||
|  | ||||
|                 JsonObject extra = filament.createNestedObject("extra"); | ||||
|                 extra["nozzle_temperature"] = spool["filament"]["extra"]["nozzle_temperature"]; | ||||
|                 extra["price_gramm"] = spool["filament"]["extra"]["price_gramm"]; | ||||
|                 extra["price_meter"] = spool["filament"]["extra"]["price_meter"]; | ||||
|             } | ||||
|             filteredDoc["color"] = filamentColor; | ||||
|             filteredDoc["type"] = filamentType; | ||||
|             filteredDoc["nozzle_temp_min"] = nozzle_temp_min; | ||||
|             filteredDoc["nozzle_temp_max"] = nozzle_temp_max; | ||||
|             filteredDoc["brand"] = filamentBrand; | ||||
|             filteredDoc["tray_info_idx"] = tray_info_idx; | ||||
|             filteredDoc["cali_idx"] = cali_idx; | ||||
|             filteredDoc["bambu_setting_id"] = bambu_setting_id; | ||||
|         } | ||||
|     } else { | ||||
|         Serial.print("Fehler beim Abrufen der Spool-Daten. HTTP-Code: "); | ||||
| @@ -186,7 +147,7 @@ bool updateSpoolTagId(String uidString, const char* payload) { | ||||
|     } | ||||
|      | ||||
|     // Überprüfe, ob die erforderlichen Felder vorhanden sind | ||||
|     if (!doc.containsKey("sm_id") || doc["sm_id"] == "") { | ||||
|     if (!doc["sm_id"].is<String>() || doc["sm_id"].as<String>() == "") { | ||||
|         Serial.println("Keine Spoolman-ID gefunden."); | ||||
|         return false; | ||||
|     } | ||||
| @@ -368,7 +329,7 @@ bool checkSpoolmanExtraFields() { | ||||
|                 for (uint8_t s = 0; s < extraLength; s++) { | ||||
|                     bool found = false; | ||||
|                     for (JsonObject field : doc.as<JsonArray>()) { | ||||
|                         if (field.containsKey("key") && field["key"] == extraFields[s]) { | ||||
|                         if (field["key"].is<String>() && field["key"] == extraFields[s]) { | ||||
|                             Serial.println("Feld gefunden: " + extraFields[s]); | ||||
|                             found = true; | ||||
|                             break; | ||||
| @@ -430,7 +391,7 @@ bool checkSpoolmanInstance(const String& url) { | ||||
|             String payload = http.getString(); | ||||
|             JsonDocument doc; | ||||
|             DeserializationError error = deserializeJson(doc, payload); | ||||
|             if (!error && doc.containsKey("status")) { | ||||
|             if (!error && doc["status"].is<String>()) { | ||||
|                 const char* status = doc["status"]; | ||||
|                 http.end(); | ||||
|  | ||||
| @@ -469,7 +430,7 @@ bool saveSpoolmanUrl(const String& url) { | ||||
|  | ||||
| String loadSpoolmanUrl() { | ||||
|     JsonDocument doc; | ||||
|     if (loadJsonValue("/spoolman_url.json", doc) && doc.containsKey("url")) { | ||||
|     if (loadJsonValue("/spoolman_url.json", doc) && doc["url"].is<String>()) { | ||||
|         return doc["url"].as<String>(); | ||||
|     } | ||||
|     Serial.println("Keine gültige Spoolman-URL gefunden."); | ||||
|   | ||||
| @@ -14,8 +14,7 @@ bool checkSpoolmanInstance(const String& url); | ||||
| bool saveSpoolmanUrl(const String& url); | ||||
| String loadSpoolmanUrl(); // Neue Funktion zum Laden der URL | ||||
| bool checkSpoolmanExtraFields(); // Neue Funktion zum Überprüfen der Extrafelder | ||||
| JsonDocument fetchSpoolsForWebsite(); // API-Funktion für die Webseite | ||||
| JsonDocument fetchAllSpoolsInfo(); | ||||
| JsonDocument fetchSingleSpoolInfo(int spoolId); // API-Funktion für die Webseite | ||||
| void sendAmsData(AsyncWebSocketClient *client); // Neue Funktion zum Senden von AMS-Daten | ||||
| bool updateSpoolTagId(String uidString, const char* payload); // Neue Funktion zum Aktualisieren eines Spools | ||||
| uint8_t updateSpoolWeight(String spoolId, uint16_t weight); // Neue Funktion zum Aktualisieren des Gewichts | ||||
|   | ||||
							
								
								
									
										170
									
								
								src/bambu.cpp
									
									
									
									
									
								
							
							
						
						
									
										170
									
								
								src/bambu.cpp
									
									
									
									
									
								
							| @@ -24,13 +24,15 @@ const char* bambu_ip = nullptr; | ||||
| const char* bambu_accesscode = nullptr; | ||||
| const char* bambu_serialnr = nullptr; | ||||
| bool bambu_connected = false; | ||||
| bool autoSendToBambu = false; | ||||
| int autoSetToBambuSpoolId = 0; | ||||
|  | ||||
| // Globale Variablen für AMS-Daten | ||||
| int ams_count = 0; | ||||
| String amsJsonData;  // Speichert das fertige JSON für WebSocket-Clients | ||||
| AMSData ams_data[MAX_AMS];  // Definition des Arrays | ||||
| AMSData ams_data[MAX_AMS];  // Definition des Arrays; | ||||
|  | ||||
| bool saveBambuCredentials(const String& ip, const String& serialnr, const String& accesscode) { | ||||
| bool saveBambuCredentials(const String& ip, const String& serialnr, const String& accesscode, bool autoSend) { | ||||
|     if (BambuMqttTask) { | ||||
|         vTaskDelete(BambuMqttTask); | ||||
|     } | ||||
| @@ -39,6 +41,7 @@ bool saveBambuCredentials(const String& ip, const String& serialnr, const String | ||||
|     doc["bambu_ip"] = ip; | ||||
|     doc["bambu_accesscode"] = accesscode; | ||||
|     doc["bambu_serialnr"] = serialnr; | ||||
|     doc["autoSendToBambu"] = autoSend; | ||||
|  | ||||
|     if (!saveJsonValue("/bambu_credentials.json", doc)) { | ||||
|         Serial.println("Fehler beim Speichern der Bambu-Credentials."); | ||||
| @@ -49,6 +52,7 @@ bool saveBambuCredentials(const String& ip, const String& serialnr, const String | ||||
|     bambu_ip = ip.c_str(); | ||||
|     bambu_accesscode = accesscode.c_str(); | ||||
|     bambu_serialnr = serialnr.c_str(); | ||||
|     autoSendToBambu = autoSend; | ||||
|  | ||||
|     vTaskDelay(100 / portTICK_PERIOD_MS); | ||||
|     if (!setupMqtt()) return false; | ||||
| @@ -58,11 +62,12 @@ bool saveBambuCredentials(const String& ip, const String& serialnr, const String | ||||
|  | ||||
| bool loadBambuCredentials() { | ||||
|     JsonDocument doc; | ||||
|     if (loadJsonValue("/bambu_credentials.json", doc) && doc.containsKey("bambu_ip")) { | ||||
|     if (loadJsonValue("/bambu_credentials.json", doc) && doc["bambu_ip"].is<String>()) { | ||||
|         // Temporäre Strings für die Werte | ||||
|         String ip = doc["bambu_ip"].as<String>(); | ||||
|         String code = doc["bambu_accesscode"].as<String>(); | ||||
|         String serial = doc["bambu_serialnr"].as<String>(); | ||||
|         autoSendToBambu = doc["autoSendToBambu"].as<bool>(); | ||||
|  | ||||
|         ip.trim(); | ||||
|         code.trim(); | ||||
| @@ -81,19 +86,23 @@ bool loadBambuCredentials() { | ||||
|     return false; | ||||
| } | ||||
|  | ||||
| String findFilamentIdx(String brand, String type) { | ||||
| struct FilamentResult { | ||||
|     String key; | ||||
|     String type; | ||||
| }; | ||||
|  | ||||
| FilamentResult findFilamentIdx(String brand, String type) { | ||||
|     // JSON-Dokument für die Filament-Daten erstellen | ||||
|     JsonDocument doc; | ||||
|      | ||||
|     // Laden der bambu_filaments.json | ||||
|     if (!loadJsonValue("/bambu_filaments.json", doc)) { | ||||
|         Serial.println("Fehler beim Laden der Filament-Daten"); | ||||
|         return "GFL99"; // Fallback auf Generic PLA | ||||
|         return {"GFL99", "PLA"}; // Fallback auf Generic PLA | ||||
|     } | ||||
|  | ||||
|     // 1. Erst versuchen wir die exakte Brand + Type Kombination zu finden | ||||
|     String searchKey; | ||||
|      | ||||
|     // 1. Suche nach Brand + Type Kombination | ||||
|     if (brand == "Bambu" || brand == "Bambulab") { | ||||
|         searchKey = "Bambu " + type; | ||||
|     } else if (brand == "PolyLite") { | ||||
| @@ -109,20 +118,43 @@ String findFilamentIdx(String brand, String type) { | ||||
|     // Durchsuche alle Einträge nach der Brand + Type Kombination | ||||
|     for (JsonPair kv : doc.as<JsonObject>()) { | ||||
|         if (kv.value().as<String>() == searchKey) { | ||||
|             return kv.key().c_str(); | ||||
|             return {kv.key().c_str(), kv.value().as<String>()}; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     // 2. Wenn nicht gefunden, suche nach Generic + Type | ||||
|     searchKey = "Generic " + type; | ||||
|     // 2. Wenn nicht gefunden, zerlege den type String in Wörter und suche nach jedem Wort | ||||
|     // Sammle alle vorhandenen Filamenttypen aus der JSON | ||||
|     std::vector<String> knownTypes; | ||||
|     for (JsonPair kv : doc.as<JsonObject>()) { | ||||
|         if (kv.value().as<String>() == searchKey) { | ||||
|             return kv.key().c_str(); | ||||
|         String value = kv.value().as<String>(); | ||||
|         // Extrahiere den Typ ohne Markennamen | ||||
|         if (value.indexOf(" ") != -1) { | ||||
|             value = value.substring(value.indexOf(" ") + 1); | ||||
|         } | ||||
|         if (!value.isEmpty()) { | ||||
|             knownTypes.push_back(value); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     // Zerlege den Input-Type in Wörter | ||||
|     String typeStr = type; | ||||
|     typeStr.trim(); | ||||
|      | ||||
|     // Durchsuche für jedes bekannte Filament, ob es im Input vorkommt | ||||
|     for (const String& knownType : knownTypes) { | ||||
|         if (typeStr.indexOf(knownType) != -1) { | ||||
|             // Suche nach diesem Typ in der Original-JSON | ||||
|             for (JsonPair kv : doc.as<JsonObject>()) { | ||||
|                 String value = kv.value().as<String>(); | ||||
|                 if (value.indexOf(knownType) != -1) { | ||||
|                     return {kv.key().c_str(), knownType}; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     // 3. Wenn immer noch nichts gefunden, gebe GFL99 zurück (Generic PLA) | ||||
|     return "GFL99"; | ||||
|     return {"GFL99", "PLA"}; | ||||
| } | ||||
|  | ||||
| bool sendMqttMessage(String payload) { | ||||
| @@ -156,15 +188,22 @@ bool setBambuSpool(String payload) { | ||||
|     int minTemp = doc["nozzle_temp_min"]; | ||||
|     int maxTemp = doc["nozzle_temp_max"]; | ||||
|     String type = doc["type"].as<String>(); | ||||
|     (type == "PLA+") ? type = "PLA" : type; | ||||
|     String brand = doc["brand"].as<String>(); | ||||
|     String tray_info_idx = (doc["tray_info_idx"].as<String>() != "-1") ? doc["tray_info_idx"].as<String>() : ""; | ||||
|     if (tray_info_idx == "") tray_info_idx = (brand != "" && type != "") ? findFilamentIdx(brand, type) : ""; | ||||
|     if (tray_info_idx == "") { | ||||
|         if (brand != "" && type != "") { | ||||
|             FilamentResult result = findFilamentIdx(brand, type); | ||||
|             tray_info_idx = result.key; | ||||
|             type = result.type;  // Aktualisiere den type mit dem gefundenen Basistyp | ||||
|         } | ||||
|     } | ||||
|     String setting_id = doc["bambu_setting_id"].as<String>(); | ||||
|     String cali_idx = doc["cali_idx"].as<String>(); | ||||
|  | ||||
|     doc.clear(); | ||||
|  | ||||
|     doc["print"]["sequence_id"] = 0; | ||||
|     doc["print"]["sequence_id"] = "0"; | ||||
|     doc["print"]["command"] = "ams_filament_setting"; | ||||
|     doc["print"]["ams_id"] = amsId < 200 ? amsId : 255; | ||||
|     doc["print"]["tray_id"] = trayId < 200 ? trayId : 254; | ||||
| @@ -172,7 +211,7 @@ bool setBambuSpool(String payload) { | ||||
|     doc["print"]["nozzle_temp_min"] = minTemp; | ||||
|     doc["print"]["nozzle_temp_max"] = maxTemp; | ||||
|     doc["print"]["tray_type"] = type; | ||||
|     doc["print"]["cali_idx"] = (cali_idx != "") ? cali_idx : ""; | ||||
|     //doc["print"]["cali_idx"] = (cali_idx != "") ? cali_idx : ""; | ||||
|     doc["print"]["tray_info_idx"] = tray_info_idx; | ||||
|     doc["print"]["setting_id"] = setting_id; | ||||
|      | ||||
| @@ -194,13 +233,13 @@ bool setBambuSpool(String payload) { | ||||
|  | ||||
|     if (cali_idx != "") { | ||||
|         yield(); | ||||
|         doc["print"]["sequence_id"] = 0; | ||||
|         doc["print"]["sequence_id"] = "0"; | ||||
|         doc["print"]["command"] = "extrusion_cali_sel"; | ||||
|         doc["print"]["filament_id"] = tray_info_idx; | ||||
|         doc["print"]["nozzle_diameter"] = "0.4"; | ||||
|         doc["print"]["cali_idx"] = cali_idx.toInt(); | ||||
|         doc["print"]["tray_id"] = trayId < 200 ? trayId : 254; | ||||
|         doc["print"]["ams_id"] = amsId < 200 ? amsId : 255; | ||||
|         //doc["print"]["ams_id"] = amsId < 200 ? amsId : 255; | ||||
|  | ||||
|         // Serialize the JSON | ||||
|         String output; | ||||
| @@ -218,44 +257,34 @@ bool setBambuSpool(String payload) { | ||||
|         doc.clear(); | ||||
|         yield(); | ||||
|     } | ||||
| /* | ||||
|     if (setting_id != "") { | ||||
|         yield(); | ||||
|         doc["print"]["sequence_id"] = 0; | ||||
|         doc["print"]["command"] = "ams_filament_setting"; | ||||
|         doc["print"]["nozzle_temp_min"] = minTemp; | ||||
|         doc["print"]["nozzle_temp_max"] = maxTemp; | ||||
|         doc["print"]["setting_id"] = setting_id; | ||||
|         doc["print"]["tray_color"] = color.length() == 8 ? color : color+"FF"; | ||||
|         doc["print"]["ams_id"] = amsId < 200 ? amsId : 255; | ||||
|         doc["print"]["tray_id"] = trayId < 200 ? trayId : 254; | ||||
|         doc["print"]["tray_info_idx"] = tray_info_idx; | ||||
|         doc["print"]["tray_type"] = type; | ||||
|  | ||||
|         // Serialize the JSON | ||||
|         String output; | ||||
|         serializeJson(doc, output); | ||||
|  | ||||
|         if (sendMqttMessage(output)) { | ||||
|             Serial.println("Filament Setting successfully set"); | ||||
|         } | ||||
|         else | ||||
|         { | ||||
|             Serial.println("Failed to set Filament setting"); | ||||
|             return false; | ||||
|         } | ||||
|  | ||||
|         doc.clear(); | ||||
|         yield(); | ||||
|     } | ||||
| */ | ||||
|  | ||||
|     return true; | ||||
| } | ||||
|  | ||||
| void autoSetSpool(int spoolId, uint8_t trayId) { | ||||
|     // wenn neue spule erkannt und autoSetToBambu > 0 | ||||
|     JsonDocument spoolInfo = fetchSingleSpoolInfo(spoolId); | ||||
|  | ||||
|     if (!spoolInfo.isNull()) | ||||
|     { | ||||
|         // AMS und TRAY id ergänzen | ||||
|         spoolInfo["amsId"] = 0; | ||||
|         spoolInfo["trayId"] = trayId; | ||||
|  | ||||
|         Serial.println("Auto set spool"); | ||||
|         Serial.println(spoolInfo.as<String>()); | ||||
|  | ||||
|         setBambuSpool(spoolInfo.as<String>()); | ||||
|     } | ||||
|  | ||||
|     // id wieder zurücksetzen damit abgeschlossen | ||||
|     autoSetToBambuSpoolId = 0; | ||||
| } | ||||
|  | ||||
| // init | ||||
| void mqtt_callback(char* topic, byte* payload, unsigned int length) { | ||||
|     String message; | ||||
|  | ||||
|     for (int i = 0; i < length; i++) { | ||||
|         message += (char)payload[i]; | ||||
|     } | ||||
| @@ -263,16 +292,27 @@ void mqtt_callback(char* topic, byte* payload, unsigned int length) { | ||||
|     // JSON-Dokument parsen | ||||
|     JsonDocument doc; | ||||
|     DeserializationError error = deserializeJson(doc, message); | ||||
|     if (error) { | ||||
|     if (error)  | ||||
|     { | ||||
|         Serial.print("Fehler beim Parsen des JSON: "); | ||||
|         Serial.println(error.c_str()); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     // Wenn bambu auto set spool aktiv und eine spule erkannt und mqtt meldung das neue spule im ams | ||||
|     if (autoSendToBambu && autoSetToBambuSpoolId > 0 &&  | ||||
|         doc["print"]["command"].as<String>() == "push_status" && doc["print"]["ams"]["tray_pre"].as<uint8_t>() | ||||
|         && !doc["print"]["ams"]["ams"].as<JsonArray>()) | ||||
|     { | ||||
|         autoSetSpool(autoSetToBambuSpoolId, doc["print"]["ams"]["tray_pre"].as<uint8_t>()); | ||||
|     } | ||||
|  | ||||
|     // Prüfen, ob "print->upgrade_state" und "print.ams.ams" existieren | ||||
|     if (doc["print"].containsKey("upgrade_state")) { | ||||
|     if (doc["print"]["upgrade_state"].is<JsonObject>())  | ||||
|     { | ||||
|         // Prüfen ob AMS-Daten vorhanden sind | ||||
|         if (!doc["print"].containsKey("ams") || !doc["print"]["ams"].containsKey("ams")) { | ||||
|         if (!doc["print"]["ams"].is<JsonObject>() || !doc["print"]["ams"]["ams"].is<JsonArray>())  | ||||
|         { | ||||
|             return; | ||||
|         } | ||||
|  | ||||
| @@ -315,7 +355,7 @@ void mqtt_callback(char* topic, byte* payload, unsigned int length) { | ||||
|         } | ||||
|  | ||||
|         // Prüfe die externe Spule | ||||
|         if (!hasChanges && doc["print"].containsKey("vt_tray")) { | ||||
|         if (!hasChanges && doc["print"]["vt_tray"].is<JsonObject>()) { | ||||
|             JsonObject vtTray = doc["print"]["vt_tray"]; | ||||
|             bool foundExternal = false; | ||||
|              | ||||
| @@ -363,7 +403,7 @@ void mqtt_callback(char* topic, byte* payload, unsigned int length) { | ||||
|         ams_count = amsArray.size(); | ||||
|  | ||||
|         // Wenn externe Spule vorhanden, füge sie hinzu | ||||
|         if (doc["print"].containsKey("vt_tray")) { | ||||
|         if (doc["print"]["vt_tray"].is<JsonObject>()) { | ||||
|             JsonObject vtTray = doc["print"]["vt_tray"]; | ||||
|             int extIdx = ams_count;  // Index für externe Spule | ||||
|             ams_data[extIdx].ams_id = 255;  // Spezielle ID für externe Spule | ||||
| @@ -374,8 +414,12 @@ void mqtt_callback(char* topic, byte* payload, unsigned int length) { | ||||
|             ams_data[extIdx].trays[0].tray_color = vtTray["tray_color"].as<String>(); | ||||
|             ams_data[extIdx].trays[0].nozzle_temp_min = vtTray["nozzle_temp_min"].as<int>(); | ||||
|             ams_data[extIdx].trays[0].nozzle_temp_max = vtTray["nozzle_temp_max"].as<int>(); | ||||
|             ams_data[extIdx].trays[0].setting_id = vtTray["setting_id"].as<String>(); | ||||
|             ams_data[extIdx].trays[0].cali_idx = vtTray["cali_idx"].as<String>(); | ||||
|  | ||||
|             if (doc["print"]["vt_tray"]["tray_type"].as<String>() != "") | ||||
|             { | ||||
|                 ams_data[extIdx].trays[0].setting_id = vtTray["setting_id"].as<String>(); | ||||
|                 ams_data[extIdx].trays[0].cali_idx = vtTray["cali_idx"].as<String>(); | ||||
|             } | ||||
|             ams_count++;  // Erhöhe ams_count für die externe Spule | ||||
|         } | ||||
|  | ||||
| @@ -387,14 +431,14 @@ void mqtt_callback(char* topic, byte* payload, unsigned int length) { | ||||
|         JsonArray wsArray = wsDoc.to<JsonArray>(); | ||||
|  | ||||
|         for (int i = 0; i < ams_count; i++) { | ||||
|             JsonObject amsObj = wsArray.createNestedObject(); | ||||
|             JsonObject amsObj = wsArray.add<JsonObject>(); | ||||
|             amsObj["ams_id"] = ams_data[i].ams_id; | ||||
|  | ||||
|             JsonArray trays = amsObj.createNestedArray("tray"); | ||||
|             JsonArray trays = amsObj["tray"].to<JsonArray>(); | ||||
|             int maxTrays = (ams_data[i].ams_id == 255) ? 1 : 4; | ||||
|              | ||||
|             for (int j = 0; j < maxTrays; j++) { | ||||
|                 JsonObject trayObj = trays.createNestedObject(); | ||||
|                 JsonObject trayObj = trays.add<JsonObject>(); | ||||
|                 trayObj["id"] = ams_data[i].trays[j].id; | ||||
|                 trayObj["tray_info_idx"] = ams_data[i].trays[j].tray_info_idx; | ||||
|                 trayObj["tray_type"] = ams_data[i].trays[j].tray_type; | ||||
| @@ -427,14 +471,14 @@ void mqtt_callback(char* topic, byte* payload, unsigned int length) { | ||||
|                 JsonArray wsArray = wsDoc.to<JsonArray>(); | ||||
|  | ||||
|                 for (int j = 0; j < ams_count; j++) { | ||||
|                     JsonObject amsObj = wsArray.createNestedObject(); | ||||
|                     JsonObject amsObj = wsArray.add<JsonObject>(); | ||||
|                     amsObj["ams_id"] = ams_data[j].ams_id; | ||||
|  | ||||
|                     JsonArray trays = amsObj.createNestedArray("tray"); | ||||
|                     JsonArray trays = amsObj["tray"].to<JsonArray>(); | ||||
|                     int maxTrays = (ams_data[j].ams_id == 255) ? 1 : 4; | ||||
|                      | ||||
|                     for (int k = 0; k < maxTrays; k++) { | ||||
|                         JsonObject trayObj = trays.createNestedObject(); | ||||
|                         JsonObject trayObj = trays.add<JsonObject>(); | ||||
|                         trayObj["id"] = ams_data[j].trays[k].id; | ||||
|                         trayObj["tray_info_idx"] = ams_data[j].trays[k].tray_info_idx; | ||||
|                         trayObj["tray_type"] = ams_data[j].trays[k].tray_type; | ||||
| @@ -462,7 +506,7 @@ void mqtt_callback(char* topic, byte* payload, unsigned int length) { | ||||
| void reconnect() { | ||||
|     // Loop until we're reconnected | ||||
|     while (!client.connected()) { | ||||
|         Serial.print("Attempting MQTT connection..."); | ||||
|         Serial.println("Attempting MQTT connection..."); | ||||
|         bambu_connected = false; | ||||
|         oledShowTopRow(); | ||||
|  | ||||
|   | ||||
| @@ -28,12 +28,15 @@ extern bool bambu_connected; | ||||
|  | ||||
| extern int ams_count; | ||||
| extern AMSData ams_data[MAX_AMS]; | ||||
| extern bool autoSendToBambu; | ||||
| extern int autoSetToBambuSpoolId; | ||||
|  | ||||
| bool loadBambuCredentials(); | ||||
| bool saveBambuCredentials(const String& bambu_ip, const String& bambu_serialnr, const String& bambu_accesscode); | ||||
| bool saveBambuCredentials(const String& bambu_ip, const String& bambu_serialnr, const String& bambu_accesscode, const bool autoSend); | ||||
| bool setupMqtt(); | ||||
| void mqtt_loop(void * parameter); | ||||
| bool setBambuSpool(String payload); | ||||
| void bambu_restart(); | ||||
|  | ||||
| extern TaskHandle_t BambuMqttTask; | ||||
| #endif | ||||
|   | ||||
| @@ -1,4 +1,5 @@ | ||||
| #include "commonFS.h" | ||||
| #include <SPIFFS.h> | ||||
|  | ||||
| bool saveJsonValue(const char* filename, const JsonDocument& doc) { | ||||
|     File file = SPIFFS.open(filename, "w"); | ||||
| @@ -35,23 +36,12 @@ bool loadJsonValue(const char* filename, JsonDocument& doc) { | ||||
|     return true; | ||||
| } | ||||
|  | ||||
| bool initializeSPIFFS() { | ||||
|     // Erster Versuch | ||||
|     if (SPIFFS.begin(true)) { | ||||
|         Serial.println("SPIFFS mounted successfully."); | ||||
|         return true; | ||||
| void initializeSPIFFS() { | ||||
|     if (!SPIFFS.begin(true, "/spiffs", 10, "spiffs")) { | ||||
|         Serial.println("SPIFFS Mount Failed"); | ||||
|         return; | ||||
|     } | ||||
|      | ||||
|     // Formatierung versuchen | ||||
|     Serial.println("Failed to mount SPIFFS. Formatting..."); | ||||
|     SPIFFS.format(); | ||||
|      | ||||
|     // Zweiter Versuch nach Formatierung | ||||
|     if (SPIFFS.begin(true)) { | ||||
|         Serial.println("SPIFFS formatted and mounted successfully."); | ||||
|         return true; | ||||
|     } | ||||
|      | ||||
|     Serial.println("SPIFFS initialization failed completely."); | ||||
|     return false; | ||||
|     Serial.printf("SPIFFS Total: %u bytes\n", SPIFFS.totalBytes()); | ||||
|     Serial.printf("SPIFFS Used: %u bytes\n", SPIFFS.usedBytes()); | ||||
|     Serial.printf("SPIFFS Free: %u bytes\n", SPIFFS.totalBytes() - SPIFFS.usedBytes()); | ||||
| } | ||||
| @@ -7,6 +7,6 @@ | ||||
|  | ||||
| bool saveJsonValue(const char* filename, const JsonDocument& doc); | ||||
| bool loadJsonValue(const char* filename, JsonDocument& doc); | ||||
| bool initializeSPIFFS(); | ||||
| void initializeSPIFFS(); | ||||
|  | ||||
| #endif | ||||
|   | ||||
| @@ -40,6 +40,10 @@ const uint8_t webserverPort = 80; | ||||
| const char* apiUrl = "/api/v1"; | ||||
| // ***** API | ||||
|  | ||||
| // ***** Bambu Auto Set Spool | ||||
| uint8_t autoSetBambuAmsCounter = 60; | ||||
| // ***** Bambu Auto Set Spool | ||||
|  | ||||
| // ***** Task Prios | ||||
| uint8_t rfidTaskCore = 1; | ||||
| uint8_t rfidTaskPrio = 1; | ||||
|   | ||||
| @@ -23,6 +23,8 @@ extern const uint8_t OLED_DATA_END; | ||||
| extern const char* apiUrl; | ||||
| extern const uint8_t webserverPort; | ||||
|  | ||||
| extern uint8_t autoSetBambuAmsCounter; | ||||
|  | ||||
| extern const unsigned char wifi_on[]; | ||||
| extern const unsigned char wifi_off[]; | ||||
| extern const unsigned char cloud_on[]; | ||||
|   | ||||
| @@ -117,7 +117,6 @@ std::vector<String> splitTextIntoLines(String text, uint8_t textSize) { | ||||
|         lines.push_back(currentLine); | ||||
|     } | ||||
|      | ||||
|     Serial.println(lines.size()); | ||||
|     return lines; | ||||
| } | ||||
|  | ||||
|   | ||||
							
								
								
									
										75
									
								
								src/main.cpp
									
									
									
									
									
								
							
							
						
						
									
										75
									
								
								src/main.cpp
									
									
									
									
									
								
							| @@ -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(); | ||||
|  | ||||
| @@ -52,7 +58,22 @@ void setup() { | ||||
|    | ||||
|   startNfc(); | ||||
|  | ||||
|   start_scale(); | ||||
|   uint8_t scaleCalibrated = start_scale(); | ||||
|   if (scaleCalibrated == 3) { | ||||
|     oledShowMessage("Scale not calibrated!"); | ||||
|     for (uint16_t i = 0; i < 50000; i++) { | ||||
|       yield(); | ||||
|       vTaskDelay(pdMS_TO_TICKS(1)); | ||||
|       esp_task_wdt_reset(); | ||||
|     } | ||||
|   } else if (scaleCalibrated == 0) { | ||||
|     oledShowMessage("HX711 not found"); | ||||
|     for (uint16_t i = 0; i < 50000; i++) { | ||||
|       yield(); | ||||
|       vTaskDelay(pdMS_TO_TICKS(1)); | ||||
|       esp_task_wdt_reset(); | ||||
|     } | ||||
|   } | ||||
|    | ||||
|   // WDT initialisieren mit 10 Sekunden Timeout | ||||
|   bool panic = true; // Wenn true, löst ein WDT-Timeout einen System-Panik aus | ||||
| @@ -69,6 +90,10 @@ void setup() { | ||||
| unsigned long lastWeightReadTime = 0; | ||||
| const unsigned long weightReadInterval = 1000; // 1 second | ||||
|  | ||||
| unsigned long lastAutoSetBambuAmsTime = 0; | ||||
| const unsigned long autoSetBambuAmsInterval = 1000; // 1 second | ||||
| uint8_t autoAmsCounter = 0; | ||||
|  | ||||
| unsigned long lastAmsSendTime = 0; | ||||
| const unsigned long amsSendInterval = 60000; // 1 minute | ||||
|  | ||||
| @@ -78,30 +103,49 @@ uint8_t wifiErrorCounter = 0; | ||||
|  | ||||
| // ##### PROGRAM START ##### | ||||
| void loop() { | ||||
|   // Überprüfe den WLAN-Status | ||||
|   if (WiFi.status() != WL_CONNECTED) { | ||||
|     wifiErrorCounter++; | ||||
|     wifiOn = false; | ||||
|   } else { | ||||
|     wifiErrorCounter = 0; | ||||
|     wifiOn = true; | ||||
|   } | ||||
|   if (wifiErrorCounter > 20) ESP.restart(); | ||||
|  | ||||
|   unsigned long currentMillis = millis(); | ||||
|  | ||||
|   // Send AMS Data min every Minute | ||||
|   if (currentMillis - lastAmsSendTime >= amsSendInterval) { | ||||
|   if (currentMillis - lastAmsSendTime >= amsSendInterval)  | ||||
|   { | ||||
|     lastAmsSendTime = currentMillis; | ||||
|     sendAmsData(nullptr); | ||||
|   } | ||||
|  | ||||
|   // Ausgabe der Waage auf Display | ||||
|   if (pauseMainTask == 0 && weight != lastWeight && hasReadRfidTag == 0) | ||||
|   // Wenn Bambu auto set Spool aktiv | ||||
|   if (autoSendToBambu && autoSetToBambuSpoolId > 0 && currentMillis - lastAutoSetBambuAmsTime >= autoSetBambuAmsInterval)  | ||||
|   { | ||||
|     (weight < 0) ? oledShowMessage("!! -1") : oledShowWeight(weight); | ||||
|     lastAutoSetBambuAmsTime = currentMillis; | ||||
|     oledShowMessage("Auto Set         " + String(autoSetBambuAmsCounter - autoAmsCounter) + "s"); | ||||
|     autoAmsCounter++; | ||||
|  | ||||
|     if (autoAmsCounter >= autoSetBambuAmsCounter)  | ||||
|     { | ||||
|       autoSetToBambuSpoolId = 0; | ||||
|       autoAmsCounter = 0; | ||||
|       oledShowWeight(weight); | ||||
|     } | ||||
|   } | ||||
|    | ||||
|  | ||||
|   // Wenn Waage nicht Kalibriert | ||||
|   if (scaleCalibrated == 3)  | ||||
|   { | ||||
|     oledShowMessage("Scale not calibrated!"); | ||||
|     vTaskDelay(5000 / portTICK_PERIOD_MS); | ||||
|     yield(); | ||||
|     esp_task_wdt_reset(); | ||||
|      | ||||
|     return; | ||||
|   }  | ||||
|  | ||||
|   // Ausgabe der Waage auf Display | ||||
|   if (pauseMainTask == 0 && weight != lastWeight && hasReadRfidTag == 0 && (!autoSendToBambu || autoSetToBambuSpoolId == 0)) | ||||
|   { | ||||
|     (weight < 2) ? ((weight < -2) ? oledShowMessage("!! -0") : oledShowWeight(0)) : oledShowWeight(weight); | ||||
|   } | ||||
|  | ||||
|  | ||||
|   // Wenn Timer abgelaufen und nicht gerade ein RFID-Tag geschrieben wird | ||||
|   if (currentMillis - lastWeightReadTime >= weightReadInterval && hasReadRfidTag < 3) | ||||
|   { | ||||
| @@ -145,6 +189,7 @@ void loop() { | ||||
|       oledShowIcon("success"); | ||||
|       vTaskDelay(2000 / portTICK_PERIOD_MS); | ||||
|       weightSend = 1; | ||||
|       autoSetToBambuSpoolId = spoolId.toInt(); | ||||
|     } | ||||
|     else | ||||
|     { | ||||
|   | ||||
| @@ -420,7 +420,7 @@ void scanRfidTask(void * parameter) { | ||||
|         //uidString = ""; | ||||
|         nfcJsonData = ""; | ||||
|         Serial.println("Tag entfernt"); | ||||
|         oledShowWeight(0); | ||||
|         if (!autoSendToBambu) oledShowWeight(weight); | ||||
|       } | ||||
|  | ||||
|       // aktualisieren der Website wenn sich der Status ändert | ||||
|   | ||||
							
								
								
									
										62
									
								
								src/ota.cpp
									
									
									
									
									
								
							
							
						
						
									
										62
									
								
								src/ota.cpp
									
									
									
									
									
								
							| @@ -1,62 +0,0 @@ | ||||
| #include <Arduino.h> | ||||
| #include "ota.h" | ||||
| #include <Update.h> | ||||
| #include <SPIFFS.h> | ||||
| #include "commonFS.h" | ||||
|  | ||||
| void handleOTAUpload(AsyncWebServerRequest *request, String filename, size_t index, uint8_t *data, size_t len, bool final) { | ||||
|     static size_t contentLength = 0; | ||||
|      | ||||
|     if (!index) { | ||||
|         contentLength = request->contentLength(); | ||||
|         Serial.printf("Update size: %u bytes\n", contentLength); | ||||
|          | ||||
|         if (contentLength == 0) { | ||||
|             request->send(400, "application/json", "{\"status\":\"error\",\"message\":\"Invalid file size\"}"); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         // Determine if this is a full image (firmware + SPIFFS) or just firmware | ||||
|         bool isFullImage = (contentLength > 0x3D0000); // SPIFFS starts at 0x3D0000 | ||||
|  | ||||
|         if (isFullImage) { | ||||
|             // For full images, we need to make sure we have enough space and properly partition it | ||||
|             if (!Update.begin(ESP.getFreeSketchSpace(), U_FLASH)) { | ||||
|                 Serial.printf("Not enough space for full image: %u bytes required\n", contentLength); | ||||
|                 request->send(400, "application/json", "{\"status\":\"error\",\"message\":\"Full image updates are not supported via OTA. Please use USB update for full images.\"}"); | ||||
|                 return; | ||||
|             } | ||||
|         } else { | ||||
|             // For firmware-only updates | ||||
|             if (!Update.begin(contentLength, U_FLASH)) { | ||||
|                 Serial.printf("Not enough space: %u required\n", contentLength); | ||||
|                 request->send(400, "application/json", "{\"status\":\"error\",\"message\":\"Not enough space available for firmware update\"}"); | ||||
|                 return; | ||||
|             } | ||||
|         } | ||||
|          | ||||
|         Serial.println(isFullImage ? "Full image update started" : "Firmware update started"); | ||||
|     } | ||||
|      | ||||
|     // Write chunk to flash | ||||
|     if (Update.write(data, len) != len) { | ||||
|         Update.printError(Serial); | ||||
|         String errorMsg = Update.errorString(); | ||||
|         request->send(400, "application/json", "{\"status\":\"error\",\"message\":\"Error writing update: " + errorMsg + "\"}"); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     if (final) { | ||||
|         if (Update.end(true)) { | ||||
|             Serial.println("Update complete"); | ||||
|             request->send(200, "application/json", "{\"status\":\"success\",\"message\":\"Update successful! Device will restart...\",\"restart\":true}"); | ||||
|             delay(1000); | ||||
|             ESP.restart(); | ||||
|         } else { | ||||
|             String errorMsg = Update.errorString(); | ||||
|             Update.printError(Serial); | ||||
|             request->send(400, "application/json", "{\"status\":\"error\",\"message\":\"Update failed: " + errorMsg + "\"}"); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| @@ -1,8 +0,0 @@ | ||||
| #ifndef OTA_H | ||||
| #define OTA_H | ||||
|  | ||||
| #include <ESPAsyncWebServer.h> | ||||
|  | ||||
| void handleOTAUpload(AsyncWebServerRequest *request, String filename, size_t index, uint8_t *data, size_t len, bool final); | ||||
|  | ||||
| #endif | ||||
| @@ -3,9 +3,9 @@ | ||||
| #include <ArduinoJson.h> | ||||
| #include "config.h" | ||||
| #include "HX711.h" | ||||
| #include <EEPROM.h> | ||||
| #include "display.h" | ||||
| #include "esp_task_wdt.h" | ||||
| #include <Preferences.h> | ||||
|  | ||||
| HX711 scale; | ||||
|  | ||||
| @@ -16,6 +16,11 @@ int16_t weight = 0; | ||||
| uint8_t weigthCouterToApi = 0; | ||||
| uint8_t scale_tare_counter = 0; | ||||
| uint8_t pauseMainTask = 0; | ||||
| uint8_t scaleCalibrated = 1; | ||||
|  | ||||
| Preferences preferences; | ||||
| const char* NVS_NAMESPACE = "scale"; | ||||
| const char* NVS_KEY_CALIBRATION = "cal_value"; | ||||
|  | ||||
| // ##### Funktionen für Waage ##### | ||||
| uint8_t tareScale() { | ||||
| @@ -46,22 +51,24 @@ void scale_loop(void * parameter) { | ||||
|   } | ||||
| } | ||||
|  | ||||
| void start_scale() { | ||||
| uint8_t start_scale() { | ||||
|   Serial.println("Prüfe Calibration Value"); | ||||
|   long calibrationValue; // calibration value (see example file "Calibration.ino") | ||||
|   //calibrationValue = 696.0; // uncomment this if you want to set the calibration value in the sketch | ||||
|   long calibrationValue; | ||||
|  | ||||
|   EEPROM.begin(512); | ||||
|   EEPROM.get(calVal_eepromAdress, calibrationValue); // uncomment this if you want to fetch the calibration value from eeprom | ||||
|  | ||||
|   //calibrationValue = EEPROM.read(calVal_eepromAdress); | ||||
|   // NVS lesen | ||||
|   preferences.begin(NVS_NAMESPACE, true); // true = readonly | ||||
|   calibrationValue = preferences.getLong(NVS_KEY_CALIBRATION, defaultScaleCalibrationValue); | ||||
|   preferences.end(); | ||||
|  | ||||
|   Serial.print("Read Scale Calibration Value "); | ||||
|   Serial.println(calibrationValue); | ||||
|  | ||||
|   scale.begin(LOADCELL_DOUT_PIN, LOADCELL_SCK_PIN); | ||||
|  | ||||
|   if (isnan(calibrationValue) || calibrationValue < 1) calibrationValue = defaultScaleCalibrationValue; | ||||
|   if (isnan(calibrationValue) || calibrationValue < 1) { | ||||
|     calibrationValue = defaultScaleCalibrationValue; | ||||
|     scaleCalibrated = 0; | ||||
|   } | ||||
|  | ||||
|   oledShowMessage("Scale Tare Please remove all"); | ||||
|   for (uint16_t i = 0; i < 2000; i++) { | ||||
| @@ -94,6 +101,8 @@ void start_scale() { | ||||
|   } else { | ||||
|       Serial.println("ScaleLoop-Task erfolgreich erstellt"); | ||||
|   } | ||||
|  | ||||
|   return (scaleCalibrated == 1) ? 1 : 3; | ||||
| } | ||||
|  | ||||
| uint8_t calibrate_scale() { | ||||
| @@ -137,18 +146,19 @@ uint8_t calibrate_scale() { | ||||
|     { | ||||
|       Serial.print("New calibration value has been set to: "); | ||||
|       Serial.println(newCalibrationValue); | ||||
|       Serial.print("Save this value to EEPROM adress "); | ||||
|       Serial.println(calVal_eepromAdress); | ||||
|  | ||||
|       //EEPROM.put(calVal_eepromAdress, newCalibrationValue); | ||||
|       EEPROM.put(calVal_eepromAdress, newCalibrationValue); | ||||
|       EEPROM.commit(); | ||||
|       // Speichern mit NVS | ||||
|       preferences.begin(NVS_NAMESPACE, false); // false = readwrite | ||||
|       preferences.putLong(NVS_KEY_CALIBRATION, newCalibrationValue); | ||||
|       preferences.end(); | ||||
|  | ||||
|       EEPROM.get(calVal_eepromAdress, newCalibrationValue); | ||||
|       //newCalibrationValue = EEPROM.read(calVal_eepromAdress); | ||||
|       // Verifizieren | ||||
|       preferences.begin(NVS_NAMESPACE, true); | ||||
|       long verifyValue = preferences.getLong(NVS_KEY_CALIBRATION, 0); | ||||
|       preferences.end(); | ||||
|  | ||||
|       Serial.print("Read Value "); | ||||
|       Serial.println(newCalibrationValue); | ||||
|       Serial.print("Verified stored value: "); | ||||
|       Serial.println(verifyValue); | ||||
|  | ||||
|       Serial.println("End calibration, revome weight"); | ||||
|  | ||||
|   | ||||
| @@ -5,7 +5,7 @@ | ||||
| #include "HX711.h" | ||||
|  | ||||
|  | ||||
| void start_scale(); | ||||
| uint8_t start_scale(); | ||||
| uint8_t calibrate_scale(); | ||||
| uint8_t tareScale(); | ||||
|  | ||||
| @@ -14,5 +14,8 @@ extern int16_t weight; | ||||
| extern uint8_t weigthCouterToApi; | ||||
| extern uint8_t scale_tare_counter; | ||||
| extern uint8_t pauseMainTask; | ||||
| extern uint8_t scaleCalibrated; | ||||
|  | ||||
| extern TaskHandle_t ScaleTask; | ||||
|  | ||||
| #endif | ||||
							
								
								
									
										272
									
								
								src/website.cpp
									
									
									
									
									
								
							
							
						
						
									
										272
									
								
								src/website.cpp
									
									
									
									
									
								
							| @@ -7,10 +7,15 @@ | ||||
| #include "nfc.h" | ||||
| #include "scale.h" | ||||
| #include "esp_task_wdt.h" | ||||
| #include "ota.h" | ||||
| #include <Update.h> | ||||
| #include "display.h" | ||||
|  | ||||
| #ifndef VERSION | ||||
|   #define VERSION "1.1.0" | ||||
| #endif | ||||
|  | ||||
| // Cache-Control Header definieren | ||||
| #define CACHE_CONTROL "max-age=31536000" // Cache für 1 Jahr | ||||
| #define CACHE_CONTROL "max-age=604800" // Cache für 1 Woche | ||||
|  | ||||
| AsyncWebServer server(webserverPort); | ||||
| AsyncWebSocket ws("/ws"); | ||||
| @@ -18,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!"); | ||||
| @@ -28,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; | ||||
| @@ -44,7 +96,7 @@ void onWsEvent(AsyncWebSocket *server, AsyncWebSocketClient *client, AwsEventTyp | ||||
|         } | ||||
|  | ||||
|         else if (doc["type"] == "writeNfcTag") { | ||||
|             if (doc.containsKey("payload")) { | ||||
|             if (doc["payload"].is<JsonObject>()) { | ||||
|                 // Versuche NFC-Daten zu schreiben | ||||
|                 String payloadString; | ||||
|                 serializeJson(doc["payload"], payloadString); | ||||
| @@ -151,11 +203,121 @@ void sendNfcData(AsyncWebSocketClient *client) { | ||||
|  | ||||
| void sendAmsData(AsyncWebSocketClient *client) { | ||||
|     if (ams_count > 0) { | ||||
|         ws.textAll("{\"type\":\"amsData\", \"payload\":" + amsJsonData + "}"); | ||||
|         ws.textAll("{\"type\":\"amsData\",\"payload\":" + amsJsonData + "}"); | ||||
|     } | ||||
| } | ||||
|  | ||||
| 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); | ||||
|  | ||||
|     // Konfiguriere Server für große Uploads | ||||
|     server.onRequestBody([](AsyncWebServerRequest *request, uint8_t *data, size_t len, size_t index, size_t total){}); | ||||
|     server.onFileUpload([](AsyncWebServerRequest *request, const String& filename, size_t index, uint8_t *data, size_t len, bool final){}); | ||||
|  | ||||
|     // Lade die Spoolman-URL beim Booten | ||||
|     spoolmanUrl = loadSpoolmanUrl(); | ||||
|     Serial.print("Geladene Spoolman-URL: "); | ||||
| @@ -189,17 +351,6 @@ void setupWebserver(AsyncWebServer &server) { | ||||
|         Serial.println("RFID-Seite gesendet"); | ||||
|     }); | ||||
|  | ||||
|     /* | ||||
|     // Neue API-Route für das Abrufen der Spool-Daten | ||||
|     server.on("/api/spools", HTTP_GET, [](AsyncWebServerRequest *request){ | ||||
|         Serial.println("API-Aufruf: /api/spools"); | ||||
|         JsonDocument spoolsData = fetchSpoolsForWebsite(); | ||||
|         String response; | ||||
|         serializeJson(spoolsData, response); | ||||
|         request->send(200, "application/json", response); | ||||
|     }); | ||||
|     */ | ||||
|  | ||||
|     server.on("/api/url", HTTP_GET, [](AsyncWebServerRequest *request){ | ||||
|         Serial.println("API-Aufruf: /api/url"); | ||||
|         String jsonResponse = "{\"spoolman_url\": \"" + String(spoolmanUrl) + "\"}"; | ||||
| @@ -222,10 +373,12 @@ void setupWebserver(AsyncWebServer &server) { | ||||
|         html.replace("{{spoolmanUrl}}", spoolmanUrl); | ||||
|  | ||||
|         JsonDocument doc; | ||||
|         if (loadJsonValue("/bambu_credentials.json", doc) && doc.containsKey("bambu_ip")) { | ||||
|         if (loadJsonValue("/bambu_credentials.json", doc) && doc["bambu_ip"].is<String>())  | ||||
|         { | ||||
|             String bambuIp = doc["bambu_ip"].as<String>(); | ||||
|             String bambuSerial = doc["bambu_serialnr"].as<String>(); | ||||
|             String bambuCode = doc["bambu_accesscode"].as<String>(); | ||||
|             autoSendToBambu = doc["autoSendToBambu"].as<bool>(); | ||||
|             bambuIp.trim(); | ||||
|             bambuSerial.trim(); | ||||
|             bambuCode.trim(); | ||||
| @@ -233,6 +386,14 @@ void setupWebserver(AsyncWebServer &server) { | ||||
|             html.replace("{{bambuIp}}", bambuIp ? bambuIp : "");             | ||||
|             html.replace("{{bambuSerial}}", bambuSerial ? bambuSerial : ""); | ||||
|             html.replace("{{bambuCode}}", bambuCode ? bambuCode : ""); | ||||
|             html.replace("{{autoSendToBambu}}", autoSendToBambu ? "checked" : ""); | ||||
|         } | ||||
|         else | ||||
|         { | ||||
|             html.replace("{{bambuIp}}", ""); | ||||
|             html.replace("{{bambuSerial}}", ""); | ||||
|             html.replace("{{bambuCode}}", ""); | ||||
|             html.replace("{{autoSendToBambu}}", ""); | ||||
|         } | ||||
|  | ||||
|         request->send(200, "text/html", html); | ||||
| @@ -264,6 +425,8 @@ void setupWebserver(AsyncWebServer &server) { | ||||
|         String bambu_ip = request->getParam("bambu_ip")->value(); | ||||
|         String bambu_serialnr = request->getParam("bambu_serialnr")->value(); | ||||
|         String bambu_accesscode = request->getParam("bambu_accesscode")->value(); | ||||
|         bool autoSend = (request->getParam("autoSend")->value() == "true") ? true : false; | ||||
|         Serial.println(autoSend); | ||||
|         bambu_ip.trim(); | ||||
|         bambu_serialnr.trim(); | ||||
|         bambu_accesscode.trim(); | ||||
| @@ -273,7 +436,7 @@ void setupWebserver(AsyncWebServer &server) { | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         bool success = saveBambuCredentials(bambu_ip, bambu_serialnr, bambu_accesscode); | ||||
|         bool success = saveBambuCredentials(bambu_ip, bambu_serialnr, bambu_accesscode, autoSend); | ||||
|  | ||||
|         request->send(200, "application/json", "{\"healthy\": " + String(success ? "true" : "false") + "}"); | ||||
|     }); | ||||
| @@ -338,30 +501,22 @@ void setupWebserver(AsyncWebServer &server) { | ||||
|         Serial.println("RFID.js gesendet"); | ||||
|     }); | ||||
|  | ||||
|     // Route for Firmware Update | ||||
|     // Vereinfachter Update-Handler | ||||
|     server.on("/upgrade", HTTP_GET, [](AsyncWebServerRequest *request) { | ||||
|         // During OTA, reduce memory usage | ||||
|         ws.enable(false);  // Temporarily disable WebSocket | ||||
|         ws.cleanupClients(); | ||||
|          | ||||
|         Serial.println("Request for /upgrade received"); | ||||
|         AsyncWebServerResponse *response = request->beginResponse(SPIFFS, "/upgrade.html.gz", "text/html"); | ||||
|         response->addHeader("Content-Encoding", "gzip"); | ||||
|         response->addHeader("Cache-Control", CACHE_CONTROL); | ||||
|         response->addHeader("Cache-Control", "no-store"); | ||||
|         request->send(response); | ||||
|     }); | ||||
|  | ||||
|     server.on("/update", HTTP_POST,  | ||||
|         [](AsyncWebServerRequest *request) { | ||||
|             // The response will be sent from handleOTAUpload when the upload is complete | ||||
|         }, | ||||
|         [](AsyncWebServerRequest *request, const String& filename, size_t index, uint8_t *data, size_t len, bool final) { | ||||
|             // Free memory before handling update | ||||
|             ws.enable(false); | ||||
|             ws.cleanupClients(); | ||||
|             handleOTAUpload(request, filename, index, data, len, final); | ||||
|         } | ||||
|     ); | ||||
|     // Update-Handler registrieren | ||||
|     handleUpdate(server); | ||||
|  | ||||
|     server.on("/api/version", HTTP_GET, [](AsyncWebServerRequest *request){ | ||||
|         String fm_version = VERSION; | ||||
|         String jsonResponse = "{\"version\": \""+ fm_version +"\"}"; | ||||
|         request->send(200, "application/json", jsonResponse); | ||||
|     }); | ||||
|  | ||||
|     // Fehlerbehandlung für nicht gefundene Seiten | ||||
|     server.onNotFound([](AsyncWebServerRequest *request){ | ||||
| @@ -379,3 +534,50 @@ void setupWebserver(AsyncWebServer &server) { | ||||
|     server.begin(); | ||||
|     Serial.println("Webserver gestartet"); | ||||
| } | ||||
|  | ||||
|  | ||||
| void backupJsonConfigs() { | ||||
|     // 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() { | ||||
|     // 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 | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -6,8 +6,8 @@ | ||||
| #include "commonFS.h" | ||||
| #include "api.h" | ||||
| #include <ArduinoJson.h> | ||||
| #include <ESPAsyncWebServer.h> | ||||
| #include <AsyncWebSocket.h> | ||||
| #include <Update.h> | ||||
| #include <AsyncTCP.h> | ||||
| #include "bambu.h" | ||||
| #include "nfc.h" | ||||
| #include "scale.h" | ||||
| @@ -17,10 +17,20 @@ extern String spoolmanUrl; | ||||
| extern AsyncWebServer server; | ||||
| extern AsyncWebSocket ws; | ||||
|  | ||||
| // Server-Initialisierung und Handler | ||||
| void initWebServer(); | ||||
| void handleUpload(AsyncWebServerRequest *request, String filename, size_t index, uint8_t *data, size_t len, bool final); | ||||
| void handleBody(AsyncWebServerRequest *request, uint8_t *data, size_t len, size_t index, size_t total); | ||||
| void setupWebserver(AsyncWebServer &server); | ||||
|  | ||||
| // WebSocket-Funktionen | ||||
| void sendAmsData(AsyncWebSocketClient *client); | ||||
| void sendNfcData(AsyncWebSocketClient *client); | ||||
| void foundNfcTag(AsyncWebSocketClient *client, uint8_t success); | ||||
| void sendWriteResult(AsyncWebSocketClient *client, uint8_t success); | ||||
|  | ||||
| // Upgrade-Funktionen | ||||
| void backupJsonConfigs(); | ||||
| void restoreJsonConfigs(); | ||||
|  | ||||
| #endif | ||||
|   | ||||
							
								
								
									
										12
									
								
								src/wlan.cpp
									
									
									
									
									
								
							
							
						
						
									
										12
									
								
								src/wlan.cpp
									
									
									
									
									
								
							| @@ -10,9 +10,19 @@ WiFiManager wm; | ||||
| bool wm_nonblocking = false; | ||||
|  | ||||
| void initWiFi() { | ||||
|     // Optimierte WiFi-Einstellungen | ||||
|     WiFi.mode(WIFI_STA); // explicitly set mode, esp defaults to STA+AP | ||||
|     WiFi.setSleep(false); // disable sleep mode | ||||
|     esp_wifi_set_ps(WIFI_PS_NONE); | ||||
|      | ||||
|     esp_wifi_set_max_tx_power(72); // Setze maximale Sendeleistung auf 20dBm | ||||
|     // Maximale Sendeleistung | ||||
|     WiFi.setTxPower(WIFI_POWER_19_5dBm); // Set maximum transmit power | ||||
|    | ||||
|     // Optimiere TCP/IP Stack | ||||
|     esp_wifi_set_protocol(WIFI_IF_STA, WIFI_PROTOCOL_11B | WIFI_PROTOCOL_11G | WIFI_PROTOCOL_11N); | ||||
|      | ||||
|     // Aktiviere WiFi-Roaming für bessere Stabilität | ||||
|     esp_wifi_set_rssi_threshold(-80); | ||||
|    | ||||
|     if(wm_nonblocking) wm.setConfigPortalBlocking(false); | ||||
|     wm.setConfigPortalTimeout(320); // Portal nach 5min schließen | ||||
|   | ||||
		Reference in New Issue
	
	Block a user