mirror of
https://github.com/jomjol/AI-on-the-edge-device.git
synced 2025-12-14 07:26:53 +03:00
Compare commits
28 Commits
16.0.0-RC7
...
update-to-
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
4a76bc3eb3 | ||
|
|
51416b1358 | ||
|
|
c320e4c921 | ||
|
|
43b29f3408 | ||
|
|
b1c65c0a71 | ||
|
|
00091fc3f9 | ||
|
|
69a43fb068 | ||
|
|
82f28cb5bc | ||
|
|
34818c0dc1 | ||
|
|
0b3a6e1057 | ||
|
|
f06ef7b80e | ||
|
|
962a674058 | ||
|
|
c57cd83948 | ||
|
|
6991c41060 | ||
|
|
8bb274cd84 | ||
|
|
168ec5b485 | ||
|
|
7a0a34e32e | ||
|
|
64bf79b288 | ||
|
|
61085d3861 | ||
|
|
8494f36069 | ||
|
|
3e67aeec0d | ||
|
|
e85e92762e | ||
|
|
eb9bf3c7c1 | ||
|
|
f542d842cf | ||
|
|
5bbc2f3da5 | ||
|
|
2831478e02 | ||
|
|
1e0cdfaba1 | ||
|
|
bfebcd5d15 |
32
.github/workflows/build.yaml
vendored
32
.github/workflows/build.yaml
vendored
@@ -1,6 +1,10 @@
|
|||||||
name: Build and Pack
|
name: Build and Pack
|
||||||
|
|
||||||
on: [push, pull_request]
|
on:
|
||||||
|
push:
|
||||||
|
pull_request:
|
||||||
|
release:
|
||||||
|
types: [released] # Only trigger on published releases (not drafts or pre-released)
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
#########################################################################################
|
#########################################################################################
|
||||||
@@ -84,7 +88,8 @@ jobs:
|
|||||||
cd ../..
|
cd ../..
|
||||||
|
|
||||||
cp -r ./sd-card/html/* ./html/
|
cp -r ./sd-card/html/* ./html/
|
||||||
|
rm -f ./html/edit_config_template.html # Remove the config page template, it is no longer needed
|
||||||
|
|
||||||
echo "Replacing variables..."
|
echo "Replacing variables..."
|
||||||
cd html
|
cd html
|
||||||
find . -type f -exec sed -i 's/$COMMIT_HASH/${{ steps.vars.outputs.sha_short }}/g' {} \;
|
find . -type f -exec sed -i 's/$COMMIT_HASH/${{ steps.vars.outputs.sha_short }}/g' {} \;
|
||||||
@@ -285,8 +290,9 @@ jobs:
|
|||||||
rm -rf ./sd-card/html
|
rm -rf ./sd-card/html
|
||||||
rm -rf ./sd-card/demo
|
rm -rf ./sd-card/demo
|
||||||
cp -r ./html ./sd-card/ # Overwrite the Web UI with the preprocessed files
|
cp -r ./html ./sd-card/ # Overwrite the Web UI with the preprocessed files
|
||||||
|
rm -f ./sd-card/Readme.md
|
||||||
cp -r ./demo ./sd-card/
|
cp -r ./demo ./sd-card/
|
||||||
cd sd-card; zip -r ../manual_setup/sd-card.zip *; cd ..
|
cd sd-card; rm -rf html/param-tooltips; zip -r ../manual_setup/sd-card.zip *; cd ..
|
||||||
cd ./manual_setup
|
cd ./manual_setup
|
||||||
|
|
||||||
- name: Upload manual_setup.zip artifact (Firmware + Bootloader + Partitions + Web UI)
|
- name: Upload manual_setup.zip artifact (Firmware + Bootloader + Partitions + Web UI)
|
||||||
@@ -302,7 +308,7 @@ jobs:
|
|||||||
prepare-release:
|
prepare-release:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
needs: [pack-for-update, pack-for-manual_setup, pack-for-remote_setup]
|
needs: [pack-for-update, pack-for-manual_setup, pack-for-remote_setup]
|
||||||
if: startsWith(github.ref, 'refs/tags/')
|
if: github.event_name == 'release' # Only run when the trigger is a release
|
||||||
|
|
||||||
# Sets permissions of the GITHUB_TOKEN to allow updating the branches
|
# Sets permissions of the GITHUB_TOKEN to allow updating the branches
|
||||||
permissions:
|
permissions:
|
||||||
@@ -404,7 +410,7 @@ jobs:
|
|||||||
#########################################################################################
|
#########################################################################################
|
||||||
# Make sure to also update update-webinstaller.yml!
|
# Make sure to also update update-webinstaller.yml!
|
||||||
update-web-installer:
|
update-web-installer:
|
||||||
if: github.event_name == 'release' && github.event.action == 'published' # Only run on release but not on prerelease
|
if: github.event_name == 'release' # Only run when the trigger is a release
|
||||||
needs: [prepare-release]
|
needs: [prepare-release]
|
||||||
environment:
|
environment:
|
||||||
name: github-pages
|
name: github-pages
|
||||||
@@ -433,22 +439,22 @@ jobs:
|
|||||||
- name: Add binary to Web Installer and update manifest
|
- name: Add binary to Web Installer and update manifest
|
||||||
run: |
|
run: |
|
||||||
echo "Updating Web installer to use firmware from ${{ steps.last_release.outputs.tag_name }}..."
|
echo "Updating Web installer to use firmware from ${{ steps.last_release.outputs.tag_name }}..."
|
||||||
rm -f docs/binary/firmware.bin
|
rm -f webinstaller/binary/firmware.bin
|
||||||
wget ${{ github.server_url }}/${{ github.repository }}/releases/download/${{ steps.last_release.outputs.tag_name }}/AI-on-the-edge-device__update__${{ steps.last_release.outputs.tag_name }}.zip
|
wget ${{ github.server_url }}/${{ github.repository }}/releases/download/${{ steps.last_release.outputs.tag_name }}/AI-on-the-edge-device__update__${{ steps.last_release.outputs.tag_name }}.zip
|
||||||
unzip AI-on-the-edge-device__update__${{ steps.last_release.outputs.tag_name }}.zip
|
unzip AI-on-the-edge-device__update__${{ steps.last_release.outputs.tag_name }}.zip
|
||||||
cp -f firmware.bin docs/binary/firmware.bin
|
cp -f firmware.bin webinstaller/binary/firmware.bin
|
||||||
echo "Updating index and manifest file..."
|
echo "Updating index and manifest file..."
|
||||||
sed -i 's/$VERSION/${{ steps.last_release.outputs.tag_name }}/g' docs/index.html
|
sed -i 's/$VERSION/${{ steps.last_release.outputs.tag_name }}/g' webinstaller/index.html
|
||||||
sed -i 's/$VERSION/${{ steps.last_release.outputs.tag_name }}/g' docs/manifest.json
|
sed -i 's/$VERSION/${{ steps.last_release.outputs.tag_name }}/g' webinstaller/manifest.json
|
||||||
|
|
||||||
- name: Setup Pages
|
- name: Setup Pages
|
||||||
uses: actions/configure-pages@v4
|
uses: actions/configure-pages@v5
|
||||||
|
|
||||||
- name: Upload artifact
|
- name: Upload artifact
|
||||||
uses: actions/upload-pages-artifact@v2
|
uses: actions/upload-pages-artifact@v3
|
||||||
with:
|
with:
|
||||||
path: 'docs'
|
path: 'webinstaller'
|
||||||
|
|
||||||
- name: Deploy to GitHub Pages
|
- name: Deploy to GitHub Pages
|
||||||
id: deployment
|
id: deployment
|
||||||
uses: actions/deploy-pages@v3 # Note: v4 does not work!
|
uses: actions/deploy-pages@v4.0.5 # Note: v4 does not work!
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
# This updates the Web Installer with the files from the docs folder and the binary of the latest release
|
# This updates the Web Installer with the files from the webinstaller folder and the binary of the latest release
|
||||||
# it only gets run on:
|
# it only gets run on:
|
||||||
# - Manually triggered
|
# - Manually triggered
|
||||||
# Make sure to also update the lower part of build.yml!
|
# Make sure to also update the lower part of build.yml!
|
||||||
@@ -11,7 +11,7 @@ on:
|
|||||||
# branches:
|
# branches:
|
||||||
# - rolling
|
# - rolling
|
||||||
# paths:
|
# paths:
|
||||||
# - docs # The path filter somehow does not work, so lets run it on every change to rolling
|
# - webinstaller # The path filter somehow does not work, so lets run it on every change to rolling
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
manually-update-web-installer:
|
manually-update-web-installer:
|
||||||
@@ -42,13 +42,13 @@ jobs:
|
|||||||
- name: Add binary to Web Installer and update manifest
|
- name: Add binary to Web Installer and update manifest
|
||||||
run: |
|
run: |
|
||||||
echo "Updating Web installer to use firmware from ${{ steps.last_release.outputs.tag_name }}..."
|
echo "Updating Web installer to use firmware from ${{ steps.last_release.outputs.tag_name }}..."
|
||||||
rm -f docs/binary/firmware.bin
|
rm -f webinstaller/binary/firmware.bin
|
||||||
wget https://github.com/jomjol/AI-on-the-edge-device/releases/download/${{ steps.last_release.outputs.tag_name }}/AI-on-the-edge-device__update__${{ steps.last_release.outputs.tag_name }}.zip
|
wget ${{ github.server_url }}/${{ github.repository }}/releases/download/${{ steps.last_release.outputs.tag_name }}/AI-on-the-edge-device__update__${{ steps.last_release.outputs.tag_name }}.zip
|
||||||
unzip AI-on-the-edge-device__update__${{ steps.last_release.outputs.tag_name }}.zip
|
unzip AI-on-the-edge-device__update__${{ steps.last_release.outputs.tag_name }}.zip
|
||||||
cp -f firmware.bin docs/binary/firmware.bin
|
cp -f firmware.bin webinstaller/binary/firmware.bin
|
||||||
echo "Updating index and manifest file..."
|
echo "Updating index and manifest file..."
|
||||||
sed -i 's/$VERSION/${{ steps.last_release.outputs.tag_name }}/g' docs/index.html
|
sed -i 's/$VERSION/${{ steps.last_release.outputs.tag_name }}/g' webinstaller/index.html
|
||||||
sed -i 's/$VERSION/${{ steps.last_release.outputs.tag_name }}/g' docs/manifest.json
|
sed -i 's/$VERSION/${{ steps.last_release.outputs.tag_name }}/g' webinstaller/manifest.json
|
||||||
|
|
||||||
- name: Setup Pages
|
- name: Setup Pages
|
||||||
uses: actions/configure-pages@v5
|
uses: actions/configure-pages@v5
|
||||||
@@ -56,7 +56,7 @@ jobs:
|
|||||||
- name: Upload artifact
|
- name: Upload artifact
|
||||||
uses: actions/upload-pages-artifact@v3
|
uses: actions/upload-pages-artifact@v3
|
||||||
with:
|
with:
|
||||||
path: 'docs'
|
path: 'webinstaller'
|
||||||
|
|
||||||
- name: Deploy to GitHub Pages
|
- name: Deploy to GitHub Pages
|
||||||
id: deployment
|
id: deployment
|
||||||
|
|||||||
2
.github/workflows/reply-bot.yaml
vendored
2
.github/workflows/reply-bot.yaml
vendored
@@ -18,7 +18,7 @@ permissions:
|
|||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
comment:
|
comment:
|
||||||
runs-on: ubuntu-20.04
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v4
|
||||||
|
|
||||||
|
|||||||
425
Changelog.md
425
Changelog.md
File diff suppressed because it is too large
Load Diff
@@ -4,7 +4,7 @@
|
|||||||
```
|
```
|
||||||
git clone https://github.com/jomjol/AI-on-the-edge-device.git
|
git clone https://github.com/jomjol/AI-on-the-edge-device.git
|
||||||
cd AI-on-the-edge-device
|
cd AI-on-the-edge-device
|
||||||
git checkout rolling
|
git checkout main
|
||||||
git submodule update --init
|
git submodule update --init
|
||||||
```
|
```
|
||||||
|
|
||||||
@@ -12,10 +12,10 @@ git submodule update --init
|
|||||||
```
|
```
|
||||||
cd /components/submodule-name (e.g. tflite-micro-example)
|
cd /components/submodule-name (e.g. tflite-micro-example)
|
||||||
git checkout VERSION (e.g. HASH of latest tflite-micro-example build)
|
git checkout VERSION (e.g. HASH of latest tflite-micro-example build)
|
||||||
cd ../../ (auf Ebene von code)
|
cd ../../ (at the code level)
|
||||||
git submodule update --init
|
git submodule update --init
|
||||||
```
|
```
|
||||||
Evt. muss man vorher noch einige Verzeichnisse in compenents von Hand löschen, da sie beim checkout nicht gelöscht wurden (vor update -- init)
|
You may need to manually delete some directories in the 'components' folder beforehand, as they were not deleted during checkout (before update -- init)
|
||||||
|
|
||||||
## Build and Flash within terminal
|
## Build and Flash within terminal
|
||||||
See further down to build it within an IDE.
|
See further down to build it within an IDE.
|
||||||
@@ -51,7 +51,7 @@ pio device monitor -p /dev/ttyUSB0
|
|||||||
```
|
```
|
||||||
git clone https://github.com/jomjol/AI-on-the-edge-device.git
|
git clone https://github.com/jomjol/AI-on-the-edge-device.git
|
||||||
cd AI-on-the-edge-device
|
cd AI-on-the-edge-device
|
||||||
git checkout rolling
|
git checkout main
|
||||||
git submodule update --init
|
git submodule update --init
|
||||||
```
|
```
|
||||||
|
|
||||||
|
|||||||
@@ -898,7 +898,7 @@ bool ClassFlowPostProcessing::doFlow(string zwtime) {
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
if (PreValueUse && NUMBERS[j]->PreValueOkay) {
|
if (PreValueUse && NUMBERS[j]->PreValueOkay) {
|
||||||
if (NUMBERS[j]->Nachkomma > 0) {
|
if ((NUMBERS[j]->Nachkomma > 0) && (NUMBERS[j]->ChangeRateThreshold > 0)) {
|
||||||
double _difference1 = (NUMBERS[j]->PreValue - (NUMBERS[j]->ChangeRateThreshold / pow(10, NUMBERS[j]->Nachkomma)));
|
double _difference1 = (NUMBERS[j]->PreValue - (NUMBERS[j]->ChangeRateThreshold / pow(10, NUMBERS[j]->Nachkomma)));
|
||||||
double _difference2 = (NUMBERS[j]->PreValue + (NUMBERS[j]->ChangeRateThreshold / pow(10, NUMBERS[j]->Nachkomma)));
|
double _difference2 = (NUMBERS[j]->PreValue + (NUMBERS[j]->ChangeRateThreshold / pow(10, NUMBERS[j]->Nachkomma)));
|
||||||
|
|
||||||
|
|||||||
@@ -31,7 +31,12 @@ extern "C"
|
|||||||
#include "ClassLogFile.h"
|
#include "ClassLogFile.h"
|
||||||
|
|
||||||
#include "esp_vfs_fat.h"
|
#include "esp_vfs_fat.h"
|
||||||
|
|
||||||
|
#if (ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 4, 0))
|
||||||
|
#include "esp_private/sdmmc_common.h"
|
||||||
|
#else
|
||||||
#include "../sdmmc_common.h"
|
#include "../sdmmc_common.h"
|
||||||
|
#endif
|
||||||
|
|
||||||
static const char *TAG = "HELPER";
|
static const char *TAG = "HELPER";
|
||||||
|
|
||||||
@@ -806,11 +811,21 @@ struct SDCard_Manufacturer_database sd_database[] = {
|
|||||||
.id = 0x03,
|
.id = 0x03,
|
||||||
.manufacturer = "SanDisk",
|
.manufacturer = "SanDisk",
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
.type = "sd",
|
||||||
|
.id = 0x05,
|
||||||
|
.manufacturer = "Lenovo",
|
||||||
|
},
|
||||||
{
|
{
|
||||||
.type = "sd",
|
.type = "sd",
|
||||||
.id = 0x08,
|
.id = 0x08,
|
||||||
.manufacturer = "Silicon Power",
|
.manufacturer = "Silicon Power",
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
.type = "sd",
|
||||||
|
.id = 0x09,
|
||||||
|
.manufacturer = "ATP",
|
||||||
|
},
|
||||||
{
|
{
|
||||||
.type = "sd",
|
.type = "sd",
|
||||||
.id = 0x18,
|
.id = 0x18,
|
||||||
@@ -894,7 +909,27 @@ struct SDCard_Manufacturer_database sd_database[] = {
|
|||||||
{
|
{
|
||||||
.type = "sd",
|
.type = "sd",
|
||||||
.id = 0x89,
|
.id = 0x89,
|
||||||
.manufacturer = "Unknown",
|
.manufacturer = "Netac",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
.type = "sd",
|
||||||
|
.id = 0x9f,
|
||||||
|
.manufacturer = "Kingston/Kodak/Silicon Power",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
.type = "sd",
|
||||||
|
.id = 0xad,
|
||||||
|
.manufacturer = "Amazon Basics/Lexar/OV",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
.type = "sd",
|
||||||
|
.id = 0xdf,
|
||||||
|
.manufacturer = "Lenovo",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
.type = "sd",
|
||||||
|
.id = 0xfe,
|
||||||
|
.manufacturer = "Bekit/Cloudisk/HP/Reletech",
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
@@ -85,7 +85,7 @@ bool MQTTPublish(std::string _key, std::string _content, int qos, bool retained_
|
|||||||
_content.append("..");
|
_content.append("..");
|
||||||
}
|
}
|
||||||
|
|
||||||
LogFile.WriteToFile(ESP_LOG_DEBUG, TAG, "Published topic: " + _key + ", content: " + _content + " (msg_id=" + std::to_string(msg_id) + ")");
|
LogFile.WriteToFile(ESP_LOG_DEBUG, TAG, "Published topic: " + _key + ", content: " + _content);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
@@ -465,7 +465,7 @@ void MQTTconnected(){
|
|||||||
if (subscribeFunktionMap != NULL) {
|
if (subscribeFunktionMap != NULL) {
|
||||||
for(std::map<std::string, std::function<bool(std::string, char*, int)>>::iterator it = subscribeFunktionMap->begin(); it != subscribeFunktionMap->end(); ++it) {
|
for(std::map<std::string, std::function<bool(std::string, char*, int)>>::iterator it = subscribeFunktionMap->begin(); it != subscribeFunktionMap->end(); ++it) {
|
||||||
int msg_id = esp_mqtt_client_subscribe(client, it->first.c_str(), 0);
|
int msg_id = esp_mqtt_client_subscribe(client, it->first.c_str(), 0);
|
||||||
LogFile.WriteToFile(ESP_LOG_DEBUG, TAG, "topic " + it->first + " subscribe successful, msg_id=" + std::to_string(msg_id));
|
LogFile.WriteToFile(ESP_LOG_DEBUG, TAG, "topic " + it->first + " subscribe successful");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -174,14 +174,14 @@ bool MQTThomeassistantDiscovery(int qos) {
|
|||||||
int aFreeInternalHeapSizeBefore = heap_caps_get_free_size(MALLOC_CAP_8BIT | MALLOC_CAP_INTERNAL);
|
int aFreeInternalHeapSizeBefore = heap_caps_get_free_size(MALLOC_CAP_8BIT | MALLOC_CAP_INTERNAL);
|
||||||
|
|
||||||
// Group | Field | User Friendly Name | Icon | Unit | Device Class | State Class | Entity Category
|
// Group | Field | User Friendly Name | Icon | Unit | Device Class | State Class | Entity Category
|
||||||
allSendsSuccessed |= sendHomeAssistantDiscoveryTopic("", "uptime", "Uptime", "clock-time-eight-outline", "s", "", "", "diagnostic", qos);
|
allSendsSuccessed |= sendHomeAssistantDiscoveryTopic("", "uptime", "Uptime", "progress-clock", "s", "duration", "measurement", "diagnostic", qos);
|
||||||
allSendsSuccessed |= sendHomeAssistantDiscoveryTopic("", "MAC", "MAC Address", "network-outline", "", "", "", "diagnostic", qos);
|
allSendsSuccessed |= sendHomeAssistantDiscoveryTopic("", "MAC", "MAC Address", "network-outline", "", "", "", "diagnostic", qos);
|
||||||
allSendsSuccessed |= sendHomeAssistantDiscoveryTopic("", "fwVersion", "Firmware Version", "application-outline", "", "", "", "diagnostic", qos);
|
allSendsSuccessed |= sendHomeAssistantDiscoveryTopic("", "fwVersion", "Firmware Version", "application-outline", "", "", "", "diagnostic", qos);
|
||||||
allSendsSuccessed |= sendHomeAssistantDiscoveryTopic("", "hostname", "Hostname", "network-outline", "", "", "", "diagnostic", qos);
|
allSendsSuccessed |= sendHomeAssistantDiscoveryTopic("", "hostname", "Hostname", "network-outline", "", "", "", "diagnostic", qos);
|
||||||
allSendsSuccessed |= sendHomeAssistantDiscoveryTopic("", "freeMem", "Free Memory", "memory", "B", "", "measurement", "diagnostic", qos);
|
allSendsSuccessed |= sendHomeAssistantDiscoveryTopic("", "freeMem", "Free Memory", "memory", "B", "", "measurement", "diagnostic", qos);
|
||||||
allSendsSuccessed |= sendHomeAssistantDiscoveryTopic("", "wifiRSSI", "Wi-Fi RSSI", "wifi", "dBm", "signal_strength", "", "diagnostic", qos);
|
allSendsSuccessed |= sendHomeAssistantDiscoveryTopic("", "wifiRSSI", "Wi-Fi RSSI", "wifi", "dBm", "signal_strength", "", "diagnostic", qos);
|
||||||
allSendsSuccessed |= sendHomeAssistantDiscoveryTopic("", "CPUtemp", "CPU Temperature", "thermometer", "°C", "temperature", "measurement", "diagnostic", qos);
|
allSendsSuccessed |= sendHomeAssistantDiscoveryTopic("", "CPUtemp", "CPU Temperature", "thermometer", "°C", "temperature", "measurement", "diagnostic", qos);
|
||||||
allSendsSuccessed |= sendHomeAssistantDiscoveryTopic("", "interval", "Interval", "clock-time-eight-outline", "min", "" , "measurement", "diagnostic", qos);
|
allSendsSuccessed |= sendHomeAssistantDiscoveryTopic("", "interval", "Interval", "clock-time-eight-outline", "min", "", "measurement", "diagnostic", qos);
|
||||||
allSendsSuccessed |= sendHomeAssistantDiscoveryTopic("", "IP", "IP", "network-outline", "", "", "", "diagnostic", qos);
|
allSendsSuccessed |= sendHomeAssistantDiscoveryTopic("", "IP", "IP", "network-outline", "", "", "", "diagnostic", qos);
|
||||||
allSendsSuccessed |= sendHomeAssistantDiscoveryTopic("", "status", "Status", "list-status", "", "", "", "diagnostic", qos);
|
allSendsSuccessed |= sendHomeAssistantDiscoveryTopic("", "status", "Status", "list-status", "", "", "", "diagnostic", qos);
|
||||||
allSendsSuccessed |= sendHomeAssistantDiscoveryTopic("", "flowstart", "Manual Flow Start", "timer-play-outline", "", "", "", "", qos);
|
allSendsSuccessed |= sendHomeAssistantDiscoveryTopic("", "flowstart", "Manual Flow Start", "timer-play-outline", "", "", "", "", qos);
|
||||||
|
|||||||
@@ -20,7 +20,7 @@
|
|||||||
[common:esp32-idf]
|
[common:esp32-idf]
|
||||||
extends = common:idf
|
extends = common:idf
|
||||||
; PlatformIO releases, see https://github.com/platformio/platform-espressif32/releases
|
; PlatformIO releases, see https://github.com/platformio/platform-espressif32/releases
|
||||||
platform = platformio/espressif32 @ 6.9.0
|
platform = platformio/espressif32 @ 6.11.0
|
||||||
framework = espidf
|
framework = espidf
|
||||||
lib_deps =
|
lib_deps =
|
||||||
${common:idf.lib_deps}
|
${common:idf.lib_deps}
|
||||||
|
|||||||
@@ -4,4 +4,4 @@ Default Value: `undefined`
|
|||||||
Dedicated definition of the field for InfluxDB use for saving in the Influx database (e.g.: "watermeter/value").
|
Dedicated definition of the field for InfluxDB use for saving in the Influx database (e.g.: "watermeter/value").
|
||||||
|
|
||||||
!!! Note
|
!!! Note
|
||||||
This parameter must be prefixed with `<NUMBER>` followed by a dot (eg. `main.Field`). `<NUMBER>` is the name of the number sequence defined in the ROI's.
|
If you edit the config file manually, you must prefix this parameter with `<NUMBER>` followed by a dot (eg. `main.Field`). The reason is that this parameter is specific for each `<NUMBER>` (`<NUMBER>` is the name of the number sequence defined in the ROI's).
|
||||||
|
|||||||
@@ -4,4 +4,4 @@ Default Value: `undefined`
|
|||||||
Field for InfluxDB v2 to use for saving.
|
Field for InfluxDB v2 to use for saving.
|
||||||
|
|
||||||
!!! Note
|
!!! Note
|
||||||
This parameter must be prefixed with `<NUMBER>` followed by a dot (eg. `main.Field`). `<NUMBER>` is the name of the number sequence defined in the ROI's.
|
If you edit the config file manually, you must prefix this parameter with `<NUMBER>` followed by a dot (eg. `main.Field`). The reason is that this parameter is specific for each `<NUMBER>` (`<NUMBER>` is the name of the number sequence defined in the ROI's).
|
||||||
|
|||||||
@@ -10,11 +10,11 @@ Alternatively you can set the parameter `DecimalShift` to `3` so the value is co
|
|||||||
List of supported options:
|
List of supported options:
|
||||||
|
|
||||||
- `other`
|
- `other`
|
||||||
- `water_m3` (uses `m^3/min` as rate)
|
- `water_m3` (uses `m^3/h` as rate)
|
||||||
- `water_l` (uses `l/h` as rate, not officially supported by Homeassistant!)
|
- `water_l` (uses `l/h` as rate, not officially supported by Homeassistant!)
|
||||||
- `water_gal` (uses `gal/h` as rate, not officially supported by Homeassistant!)
|
- `water_gal` (uses `gal/h` as rate, not officially supported by Homeassistant!)
|
||||||
- `water_ft3` (uses `ft^3/min` as rate)
|
- `water_ft3` (uses `ft^3/min` as rate)
|
||||||
- `gas_m3` (uses `m^3/min` as rate)
|
- `gas_m3` (uses `m^3/h` as rate)
|
||||||
- `gas_ft3` (uses `ft^3/min` as rate)
|
- `gas_ft3` (uses `ft^3/min` as rate)
|
||||||
- `energy_wh` (uses `W` as rate)
|
- `energy_wh` (uses `W` as rate)
|
||||||
- `energy_kwh` (uses `KW` as rate)
|
- `energy_kwh` (uses `KW` as rate)
|
||||||
|
|||||||
@@ -4,4 +4,4 @@ Default Value: `0`
|
|||||||
The Idx number for the counter device. Can be obtained from the devices setup page on the Domoticz system.
|
The Idx number for the counter device. Can be obtained from the devices setup page on the Domoticz system.
|
||||||
|
|
||||||
!!! Note
|
!!! Note
|
||||||
This parameter must be prefixed with `<NUMBER>` followed by a dot (eg. `main.DomoticzIDX`). `<NUMBER>` is the name of the number sequence defined in the ROI's.
|
If you edit the config file manually, you must prefix this parameter with `<NUMBER>` followed by a dot (eg. `main.DomoticzIDX`). The reason is that this parameter is specific for each `<NUMBER>` (`<NUMBER>` is the name of the number sequence defined in the ROI's).
|
||||||
|
|||||||
@@ -7,4 +7,4 @@ Allow a meter to count backwards (decreasing values).
|
|||||||
This is unusual (it means there is a negative rate) and not wanted in most cases!
|
This is unusual (it means there is a negative rate) and not wanted in most cases!
|
||||||
|
|
||||||
!!! Note
|
!!! Note
|
||||||
This parameter must be prefixed with `<NUMBER>` followed by a dot (eg. `main.AllowNegativeRates`). `<NUMBER>` is the name of the number sequence defined in the ROI's.
|
If you edit the config file manually, you must prefix this parameter with `<NUMBER>` followed by a dot (eg. `main.AllowNegativeRates`). The reason is that this parameter is specific for each `<NUMBER>` (`<NUMBER>` is the name of the number sequence defined in the ROI's).
|
||||||
|
|||||||
@@ -9,4 +9,4 @@ See [here](../Watermeter-specific-analog---digit-transition) for details.
|
|||||||
Range: `6.0` .. `9.9`.
|
Range: `6.0` .. `9.9`.
|
||||||
|
|
||||||
!!! Note
|
!!! Note
|
||||||
This parameter must be prefixed with `<NUMBER>` followed by a dot (eg. `main.AnalogToDigitTransitionStart`). `<NUMBER>` is the name of the number sequence defined in the ROI's.
|
If you edit the config file manually, you must prefix this parameter with `<NUMBER>` followed by a dot (eg. `main.AnalogToDigitTransitionStart`). The reason is that this parameter is specific for each `<NUMBER>` (`<NUMBER>` is the name of the number sequence defined in the ROI's).
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
# Parameter `ChangeRateThreshold`
|
# Parameter `ChangeRateThreshold`
|
||||||
Default Value: `2`
|
Default Value: `2`
|
||||||
|
|
||||||
Range: `1` .. `9`.
|
Range: `0` .. `9`.
|
||||||
|
|
||||||
Threshold parameter for change rate detection.<br>
|
Threshold parameter for change rate detection.<br>
|
||||||
This parameter is intended to compensate for small reading fluctuations that occur when the meter does not change its value for a long time (e.g. at night) or slightly turns backwards. This can eg. happen on watermeters.
|
This parameter is intended to compensate for small reading fluctuations that occur when the meter does not change its value for a long time (e.g. at night) or slightly turns backwards. This can eg. happen on watermeters.
|
||||||
@@ -10,18 +10,18 @@ It is only applied to the last digit of the read value (See example below).
|
|||||||
If the read value is within PreValue +/- Threshold, no further calculation is carried out and the Value/Prevalue remains at the old value.
|
If the read value is within PreValue +/- Threshold, no further calculation is carried out and the Value/Prevalue remains at the old value.
|
||||||
|
|
||||||
!!! Note
|
!!! Note
|
||||||
This parameter must be prefixed with `<NUMBER>` followed by a dot (eg. `main.ChangeRateThreshold`). `<NUMBER>` is the name of the number sequence defined in the ROI's.
|
If you edit the config file manually, you must prefix this parameter with `<NUMBER>` followed by a dot (eg. `main.ChangeRateThreshold`). The reason is that this parameter is specific for each `<NUMBER>` (`<NUMBER>` is the name of the number sequence defined in the ROI's).
|
||||||
|
|
||||||
## Example
|
## Example
|
||||||
|
|
||||||
- Smallest ROI provides value for `0.000'x` (Eg. a water meter with 4 pointers behind the decimal point)
|
- Smallest ROI provides value for `0.000'x` (Eg. a water meter with 4 pointers behind the decimal point)
|
||||||
- ChangeRateThreshold = 2
|
- ChangeRateThreshold = 2
|
||||||
|
|
||||||
#### With `Extended Resolution` **disabled**
|
#### With `ExtendedResolution` **disabled**
|
||||||
PreValue: `123.456'7` -> Threshold = `+/-0.000'2`.<br>
|
PreValue: `123.456'7` -> Threshold = `+/-0.000'2`.<br>
|
||||||
All changes between `123.456'5` and `123.456'9` get ignored
|
All changes between `123.456'5` and `123.456'9` get ignored
|
||||||
|
|
||||||
#### With `Extended Resolution` **enabled**
|
#### With `ExtendedResolution` **enabled**
|
||||||
PreValue: `123.456'78` -> Threshold = `+/-0.000'02`.<br>
|
PreValue: `123.456'78` -> Threshold = `+/-0.000'02`.<br>
|
||||||
All changes between `123.456'76` and `123.456'80` get ignored.
|
All changes between `123.456'76` and `123.456'80` get ignored.
|
||||||
|
|
||||||
|
|||||||
@@ -8,4 +8,4 @@ An additional consistency check.
|
|||||||
It especially improves the zero crossing check between digits.
|
It especially improves the zero crossing check between digits.
|
||||||
|
|
||||||
!!! Note
|
!!! Note
|
||||||
This parameter must be prefixed with `<NUMBER>` followed by a dot (eg. `main.CheckDigitIncreaseConsistency`). `<NUMBER>` is the name of the number sequence defined in the ROI's.
|
If you edit the config file manually, you must prefix this parameter with `<NUMBER>` followed by a dot (eg. `main.CheckDigitIncreaseConsistency`). The reason is that this parameter is specific for each `<NUMBER>` (`<NUMBER>` is the name of the number sequence defined in the ROI's).
|
||||||
|
|||||||
@@ -5,4 +5,4 @@ Shift the decimal separator (positiv or negativ).
|
|||||||
Eg. to move from `m³` to `liter` (`1 m³` equals `1000 liters`), you need to set it to `+3`.
|
Eg. to move from `m³` to `liter` (`1 m³` equals `1000 liters`), you need to set it to `+3`.
|
||||||
|
|
||||||
!!! Note
|
!!! Note
|
||||||
This parameter must be prefixed with `<NUMBER>` followed by a dot (eg. `main.DecimalShift`). `<NUMBER>` is the name of the number sequence defined in the ROI's.
|
If you edit the config file manually, you must prefix this parameter with `<NUMBER>` followed by a dot (eg. `main.DecimalShift`). The reason is that this parameter is specific for each `<NUMBER>` (`<NUMBER>` is the name of the number sequence defined in the ROI's).
|
||||||
|
|||||||
@@ -7,4 +7,4 @@ Use the decimal place of the last analog counter for increased accuracy.
|
|||||||
This parameter is only supported on the `*-class*` and `*-const` models! See [Choosing-the-Model](../Choosing-the-Model) for details.
|
This parameter is only supported on the `*-class*` and `*-const` models! See [Choosing-the-Model](../Choosing-the-Model) for details.
|
||||||
|
|
||||||
!!! Note
|
!!! Note
|
||||||
This parameter must be prefixed with `<NUMBER>` followed by a dot (eg. `main.ExtendedResolution`). `<NUMBER>` is the name of the number sequence defined in the ROI's.
|
If you edit the config file manually, you must prefix this parameter with `<NUMBER>` followed by a dot (eg. `main.ExtendedResolution`). The reason is that this parameter is specific for each `<NUMBER>` (`<NUMBER>` is the name of the number sequence defined in the ROI's).
|
||||||
|
|||||||
@@ -6,4 +6,4 @@ This is only relevant for models which use `N`!
|
|||||||
See [here](../Choosing-the-Model) for details.
|
See [here](../Choosing-the-Model) for details.
|
||||||
|
|
||||||
!!! Note
|
!!! Note
|
||||||
This parameter must be prefixed with `<NUMBER>` followed by a dot (eg. `main.IgnoreLeadingNaN`). `<NUMBER>` is the name of the number sequence defined in the ROI's.
|
If you edit the config file manually, you must prefix this parameter with `<NUMBER>` followed by a dot (eg. `main.IgnoreLeadingNaN`). The reason is that this parameter is specific for each `<NUMBER>` (`<NUMBER>` is the name of the number sequence defined in the ROI's).
|
||||||
|
|||||||
@@ -5,4 +5,4 @@ Defines if the **Change Rate** is calculated as the difference between the last
|
|||||||
as the difference normalized to the interval (`RateChange` = difference per minute).
|
as the difference normalized to the interval (`RateChange` = difference per minute).
|
||||||
|
|
||||||
!!! Note
|
!!! Note
|
||||||
This parameter must be prefixed with `<NUMBER>` followed by a dot (eg. `main.MaxRateType`). `<NUMBER>` is the name of the number sequence defined in the ROI's.
|
If you edit the config file manually, you must prefix this parameter with `<NUMBER>` followed by a dot (eg. `main.MaxRateType`). The reason is that this parameter is specific for each `<NUMBER>` (`<NUMBER>` is the name of the number sequence defined in the ROI's).
|
||||||
|
|||||||
@@ -6,4 +6,4 @@ Maximum allowed change between two readings, if exceeded the last reading will b
|
|||||||
If negative rate is disallowed and no maximum rate value is set, one false high reading will lead to a period of missing measurements until the measurement reaches the previous false high reading. E.g. if the counter is at `600,00` and it's read incorrectly as` 610,00`, all measurements will be skipped until the counter reaches `610,00`. Setting the MaxRateValue to `0,05` leads to a rejection of all readings with a difference `> 0,05`, in this case `610,00`. The rejection also applies to correct readings with a difference `> 0,05`!
|
If negative rate is disallowed and no maximum rate value is set, one false high reading will lead to a period of missing measurements until the measurement reaches the previous false high reading. E.g. if the counter is at `600,00` and it's read incorrectly as` 610,00`, all measurements will be skipped until the counter reaches `610,00`. Setting the MaxRateValue to `0,05` leads to a rejection of all readings with a difference `> 0,05`, in this case `610,00`. The rejection also applies to correct readings with a difference `> 0,05`!
|
||||||
|
|
||||||
!!! Note
|
!!! Note
|
||||||
This parameter must be prefixed with `<NUMBER>` followed by a dot (eg. `main.MaxRateValue`). `<NUMBER>` is the name of the number sequence defined in the ROI's.
|
If you edit the config file manually, you must prefix this parameter with `<NUMBER>` followed by a dot (eg. `main.MaxRateValue`). The reason is that this parameter is specific for each `<NUMBER>` (`<NUMBER>` is the name of the number sequence defined in the ROI's).
|
||||||
|
|||||||
5
sd-card/Readme.md
Normal file
5
sd-card/Readme.md
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
# SD Card content
|
||||||
|
This folder contains the files which are required to setup the SD card.
|
||||||
|
|
||||||
|
❗ Do not directly copy this folder onto your SD card, **it will not work!** Instead, you can use any of the artifacts generaded in any of the Pipeline runs of the [Build-Pipeline](https://github.com/jomjol/AI-on-the-edge-device/actions/workflows/build.yaml).
|
||||||
|
The files in the `html` folder here only serve as templates. The real `html` folder get generated using the Github actions.
|
||||||
BIN
sd-card/config/ana-cont_1500_s2_q.tflite
Normal file
BIN
sd-card/config/ana-cont_1500_s2_q.tflite
Normal file
Binary file not shown.
@@ -43,7 +43,7 @@ AlignmentAlgo = default
|
|||||||
/config/ref1.jpg 442 142
|
/config/ref1.jpg 442 142
|
||||||
|
|
||||||
[Digits]
|
[Digits]
|
||||||
Model = /config/dig-cont_0712_s3_q.tflite
|
Model = /config/dig-cont_0900_s3_q.tflite
|
||||||
CNNGoodThreshold = 0.5
|
CNNGoodThreshold = 0.5
|
||||||
;ROIImagesLocation = /log/digit
|
;ROIImagesLocation = /log/digit
|
||||||
;ROIImagesRetention = 3
|
;ROIImagesRetention = 3
|
||||||
@@ -52,7 +52,7 @@ main.dig2 343 126 30 54 false
|
|||||||
main.dig3 391 126 30 54 false
|
main.dig3 391 126 30 54 false
|
||||||
|
|
||||||
[Analog]
|
[Analog]
|
||||||
Model = /config/ana-cont_1300_s2.tflite
|
Model = /config/ana-cont_1500_s2_q.tflite
|
||||||
CNNGoodThreshold = 0.5
|
CNNGoodThreshold = 0.5
|
||||||
;ROIImagesLocation = /log/analog
|
;ROIImagesLocation = /log/analog
|
||||||
;ROIImagesRetention = 3
|
;ROIImagesRetention = 3
|
||||||
@@ -73,7 +73,7 @@ main.MaxRateValue = 0.05
|
|||||||
main.ExtendedResolution = false
|
main.ExtendedResolution = false
|
||||||
main.IgnoreLeadingNaN = false
|
main.IgnoreLeadingNaN = false
|
||||||
ErrorMessage = true
|
ErrorMessage = true
|
||||||
CheckDigitIncreaseConsistency = false
|
main.CheckDigitIncreaseConsistency = false
|
||||||
|
|
||||||
;[MQTT]
|
;[MQTT]
|
||||||
;Uri = mqtt://IP-ADRESS:1883
|
;Uri = mqtt://IP-ADRESS:1883
|
||||||
|
|||||||
BIN
sd-card/config/dig-class100-0180-s2-q.tflite
Normal file
BIN
sd-card/config/dig-class100-0180-s2-q.tflite
Normal file
Binary file not shown.
BIN
sd-card/config/dig-cont_0900_s3_q.tflite
Normal file
BIN
sd-card/config/dig-cont_0900_s3_q.tflite
Normal file
Binary file not shown.
@@ -43,14 +43,14 @@ AlignmentAlgo = default
|
|||||||
/config/ref1.jpg 536 113
|
/config/ref1.jpg 536 113
|
||||||
|
|
||||||
[Digits]
|
[Digits]
|
||||||
Model = /config/dig-cont_0710_s3_q.tflite
|
Model = /config/dig-cont_0810_s3_q.tflite
|
||||||
CNNGoodThreshold = 0.5
|
CNNGoodThreshold = 0.5
|
||||||
;ROIImagesLocation = /log/digit
|
;ROIImagesLocation = /log/digit
|
||||||
;ROIImagesRetention = 3
|
;ROIImagesRetention = 3
|
||||||
main.dig1 438 62 49 71 false
|
main.dig1 438 62 49 71 false
|
||||||
|
|
||||||
[Analog]
|
[Analog]
|
||||||
Model = /config/ana-cont_1300_s2.tflite
|
Model = /config/ana-cont_1400_s2_q.tflite
|
||||||
;ROIImagesLocation = /log/analog
|
;ROIImagesLocation = /log/analog
|
||||||
;ROIImagesRetention = 3
|
;ROIImagesRetention = 3
|
||||||
main.ana1 452 199 120 120 false
|
main.ana1 452 199 120 120 false
|
||||||
@@ -67,7 +67,7 @@ main.AllowNegativeRates = true
|
|||||||
main.ExtendedResolution = true
|
main.ExtendedResolution = true
|
||||||
main.IgnoreLeadingNaN = false
|
main.IgnoreLeadingNaN = false
|
||||||
ErrorMessage = true
|
ErrorMessage = true
|
||||||
CheckDigitIncreaseConsistency = false
|
main.CheckDigitIncreaseConsistency = false
|
||||||
|
|
||||||
;[MQTT]
|
;[MQTT]
|
||||||
;Uri = mqtt://IP-ADRESS:1883
|
;Uri = mqtt://IP-ADRESS:1883
|
||||||
@@ -76,11 +76,14 @@ CheckDigitIncreaseConsistency = false
|
|||||||
;user = USERNAME
|
;user = USERNAME
|
||||||
;password = PASSWORD
|
;password = PASSWORD
|
||||||
RetainMessages = false
|
RetainMessages = false
|
||||||
|
;DomoticzTopicIn = undefined
|
||||||
|
;main.DomoticzIDX = undefined
|
||||||
HomeassistantDiscovery = false
|
HomeassistantDiscovery = false
|
||||||
;MeterType = other
|
;MeterType = other
|
||||||
;CACert = /config/certs/RootCA.pem
|
;CACert = /config/certs/RootCA.pem
|
||||||
;ClientCert = /config/certs/client.pem.crt
|
;ClientCert = /config/certs/client.pem.crt
|
||||||
;ClientKey = /config/certs/client.pem.key
|
;ClientKey = /config/certs/client.pem.key
|
||||||
|
;ValidateServerCert = true
|
||||||
|
|
||||||
;[InfluxDB]
|
;[InfluxDB]
|
||||||
;Uri = undefined
|
;Uri = undefined
|
||||||
@@ -133,4 +136,3 @@ TimeZone = CET-1CEST,M3.5.0,M10.5.0/3
|
|||||||
RSSIThreshold = -75
|
RSSIThreshold = -75
|
||||||
CPUFrequency = 160
|
CPUFrequency = 160
|
||||||
SetupMode = false
|
SetupMode = false
|
||||||
|
|
||||||
|
|||||||
@@ -968,8 +968,8 @@
|
|||||||
<label for=PostProcessing_ChangeRateThreshold_enabled><class id="PostProcessing_ChangeRateThreshold_text" style="color:black;">Change Rate Threshold</class></label>
|
<label for=PostProcessing_ChangeRateThreshold_enabled><class id="PostProcessing_ChangeRateThreshold_text" style="color:black;">Change Rate Threshold</class></label>
|
||||||
</td>
|
</td>
|
||||||
<td>
|
<td>
|
||||||
<input required type="number" id="PostProcessing_ChangeRateThreshold_value1" step="1" min="1" max="9" value="2"
|
<input required type="number" id="PostProcessing_ChangeRateThreshold_value1" step="1" min="0" max="9" value="2"
|
||||||
oninput="(!validity.rangeUnderflow||(value=1)) && (!validity.rangeOverflow||(value=9)) &&
|
oninput="(!validity.rangeUnderflow||(value=0)) && (!validity.rangeOverflow||(value=9)) &&
|
||||||
(!validity.stepMismatch||(value=parseInt(this.value)));">
|
(!validity.stepMismatch||(value=parseInt(this.value)));">
|
||||||
</td>
|
</td>
|
||||||
<td>$TOOLTIP_PostProcessing_NUMBER.ChangeRateThreshold</td>
|
<td>$TOOLTIP_PostProcessing_NUMBER.ChangeRateThreshold</td>
|
||||||
|
|||||||
3
webinstaller/Readme.md
Normal file
3
webinstaller/Readme.md
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
# Webinstaller
|
||||||
|
This folder is used to provide the required files to generate the [Web-Installer](https://jomjol.github.io/AI-on-the-edge-device/).
|
||||||
|
The Webinstaller gets automatically updated on a release using the Github actions.
|
||||||
2
webinstaller/binary/Readme.md
Normal file
2
webinstaller/binary/Readme.md
Normal file
@@ -0,0 +1,2 @@
|
|||||||
|
# Binary folder of the webinstaller
|
||||||
|
The firmware itself (`firmware.bin`) gets copied to this folder through the Github action.
|
||||||
@@ -67,21 +67,19 @@
|
|||||||
|
|
||||||
<div class="footer">
|
<div class="footer">
|
||||||
<div class="footer-section">
|
<div class="footer-section">
|
||||||
<span>Support & Contact Us</span>
|
<span>Support & Contact</span>
|
||||||
<a href="https://github.com/jomjol/AI-on-the-edge-device" target="_blank" title="GitHub">
|
<a href="https://github.com/jomjol/AI-on-the-edge-device" target="_blank" title="GitHub">
|
||||||
<img src="https://github.com/jomjol/AI-on-the-edge-device/images/github-logo.png" alt="GitHub">
|
<img src="https://raw.githubusercontent.com/jomjol/AI-on-the-edge-device/refs/heads/main/images/github-logo.png" alt="GitHub">
|
||||||
</a>
|
|
||||||
<img src="https://github.com/jomjol/AI-on-the-edge-device/images/gmail-logo.png" alt="Email">
|
|
||||||
</a>
|
</a>
|
||||||
<a href="https://github.com/jomjol/AI-on-the-edge-device/discussions" target="_blank" title="GitHub">
|
<a href="https://github.com/jomjol/AI-on-the-edge-device/discussions" target="_blank" title="GitHub">
|
||||||
<img src="https://github.com/jomjol/AI-on-the-edge-device/images/discussion-logo" alt="GitHub">
|
<img src="https://raw.githubusercontent.com/jomjol/AI-on-the-edge-device/refs/heads/main/images/discussion-logo.png" alt="GitHub">
|
||||||
</a>
|
</a>
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
<div class="footer-section">
|
<div class="footer-section">
|
||||||
<span>Donations</span>
|
<span>Donations</span>
|
||||||
<a href="https://www.paypal.com/donate?hosted_button_id=8TRSVYNYKDSWL" target="_blank" title="Donate via PayPal">
|
<a href="https://www.paypal.com/donate?hosted_button_id=8TRSVYNYKDSWL" target="_blank" title="Donate via PayPal">
|
||||||
<img src="https://github.com/jomjol/AI-on-the-edge-device/images/paypal.png" alt="PayPal" style="width: 60px; height: auto;">
|
<img src="https://raw.githubusercontent.com/jomjol/AI-on-the-edge-device/refs/heads/main/images/paypal.png" alt="PayPal" style="width: 60px; height: auto;">
|
||||||
</a>
|
</a>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
Reference in New Issue
Block a user