More configuration options
- wifi provisioning - auto start/stop of sessions
This commit is contained in:
parent
92e7399fe4
commit
041031709f
|
@ -4,8 +4,10 @@ from collections import namedtuple
|
||||||
from datetime import datetime
|
from datetime import datetime
|
||||||
import subprocess
|
import subprocess
|
||||||
from distutils.version import StrictVersion
|
from distutils.version import StrictVersion
|
||||||
|
import subprocess
|
||||||
|
import os
|
||||||
|
|
||||||
locale.setlocale(locale.LC_ALL, 'en_US')
|
#locale.setlocale(locale.LC_ALL, 'en_US')
|
||||||
|
|
||||||
# sizeof(esp_image_header_t) + sizeof(esp_image_segment_header_t)
|
# sizeof(esp_image_header_t) + sizeof(esp_image_segment_header_t)
|
||||||
APP_DESC_OFFSET = 32
|
APP_DESC_OFFSET = 32
|
||||||
|
@ -113,9 +115,17 @@ def add_info_to_firmware(firmware_file, version):
|
||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
fimware_file = ".pio/build/esp32/firmware.bin"
|
firmware_file = ".pio/build/esp32/firmware.bin"
|
||||||
print(read_app_description(fimware_file))
|
version_file_name = "VERSION"
|
||||||
|
|
||||||
|
version = version_number_from_git()
|
||||||
|
add_info_to_firmware(firmware_file, version)
|
||||||
|
print(read_app_description(firmware_file))
|
||||||
|
with open(version_file_name, "w") as version_file:
|
||||||
|
print(version, file=version_file)
|
||||||
|
subprocess.run(["scp", firmware_file, "root@server:/volumes/swimtracker-update"])
|
||||||
|
subprocess.run(["scp", version_file_name, "root@server:/volumes/swimtracker-update"])
|
||||||
|
|
||||||
|
os.unlink(firmware_file)
|
||||||
|
os.unlink(version_file_name)
|
||||||
|
|
||||||
patch_app_description(fimware_file, version="myversion",
|
|
||||||
project_name="swimtracker.bauer.tech", time="22:16:12", date="Feb 20 2020")
|
|
||||||
print(read_app_description(fimware_file))
|
|
||||||
|
|
|
@ -4,7 +4,7 @@
|
||||||
#include <functional>
|
#include <functional>
|
||||||
#include "Dtypes.h"
|
#include "Dtypes.h"
|
||||||
|
|
||||||
constexpr int MAX_URI_HANDLERS = 20;
|
constexpr int MAX_URI_HANDLERS = 25;
|
||||||
|
|
||||||
esp_err_t rawCallback(httpd_req_t *req);
|
esp_err_t rawCallback(httpd_req_t *req);
|
||||||
|
|
||||||
|
|
|
@ -2,32 +2,23 @@
|
||||||
#include "SwimTrackerConfig.h"
|
#include "SwimTrackerConfig.h"
|
||||||
#include <cstdint>
|
#include <cstdint>
|
||||||
|
|
||||||
template <int DIVIDER = 128>
|
|
||||||
class Scale
|
class Scale
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
bool measure(uint16_t &measurementOut)
|
bool measure(uint16_t &measurementOut)
|
||||||
{
|
{
|
||||||
//if (hx711_.is_ready())
|
long value = hx711_.read_average(CONFIG_MEASUREMENT_AVG_COUNT) - offset_;
|
||||||
//{
|
if (value < 0)
|
||||||
long value = hx711_.read_average(CONFIG_MEASUREMENT_AVG_COUNT) - offset_;
|
measurementOut = (int16_t)(-(value >> valueRightShift_ ));
|
||||||
if (value < 0)
|
else
|
||||||
measurementOut = (int16_t)(-value / DIVIDER);
|
measurementOut = 0;
|
||||||
else
|
return true;
|
||||||
measurementOut = 0;
|
|
||||||
return true;
|
|
||||||
//}
|
|
||||||
//else {
|
|
||||||
// long value = hx711_.read_average(CONFIG_MEASUREMENT_AVG_COUNT) - offset_;
|
|
||||||
//
|
|
||||||
// Serial.printf("Measurement failed %ld\n", value);
|
|
||||||
// return false;
|
|
||||||
//}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void begin(uint32_t pinDOUT, uint32_t pinSCK)
|
void begin(uint32_t pinDOUT, uint32_t pinSCK, int valueRightShift)
|
||||||
{
|
{
|
||||||
hx711_.begin(pinDOUT, pinSCK);
|
hx711_.begin(pinDOUT, pinSCK);
|
||||||
|
valueRightShift_ = valueRightShift;
|
||||||
};
|
};
|
||||||
|
|
||||||
void tare(uint32_t numMeasurementsToAverage = 50)
|
void tare(uint32_t numMeasurementsToAverage = 50)
|
||||||
|
@ -38,8 +29,9 @@ public:
|
||||||
}
|
}
|
||||||
|
|
||||||
const long &offset() const { return offset_; }
|
const long &offset() const { return offset_; }
|
||||||
|
int valueRightShift() const { return valueRightShift_; }
|
||||||
private:
|
private:
|
||||||
HX711 hx711_;
|
HX711 hx711_;
|
||||||
long offset_ = 0;
|
long offset_ = 0;
|
||||||
|
int valueRightShift_;
|
||||||
};
|
};
|
||||||
|
|
|
@ -0,0 +1,79 @@
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
template <typename MeasurementT>
|
||||||
|
class AutoStart
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
void begin(MeasurementT minThreshold, MeasurementT maxThreshold, uint32_t maxNumMeasurementsBetweenPeeks)
|
||||||
|
{
|
||||||
|
minThreshold_ = minThreshold;
|
||||||
|
maxThreshold_ = maxThreshold;
|
||||||
|
maxNumMeasurementsBetweenPeeks_ = maxNumMeasurementsBetweenPeeks;
|
||||||
|
measurementsSinceLastPeek_ = maxNumMeasurementsBetweenPeeks + 1;
|
||||||
|
up_ = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool autoStart(MeasurementT measurement)
|
||||||
|
{
|
||||||
|
measurementsSinceLastPeek_ += 1;
|
||||||
|
if (!up_ && measurement > maxThreshold_)
|
||||||
|
{
|
||||||
|
up_ = true;
|
||||||
|
bool doAutoStart = measurementsSinceLastPeek_ < maxNumMeasurementsBetweenPeeks_;
|
||||||
|
measurementsSinceLastPeek_ = 0;
|
||||||
|
return doAutoStart;
|
||||||
|
}
|
||||||
|
if (up_ && measurement < minThreshold_)
|
||||||
|
up_ = false;
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
MeasurementT minThreshold_;
|
||||||
|
MeasurementT maxThreshold_;
|
||||||
|
uint32_t maxNumMeasurementsBetweenPeeks_;
|
||||||
|
|
||||||
|
// time of last peak
|
||||||
|
uint32_t measurementsSinceLastPeek_;
|
||||||
|
// if measurements have been below min threshold (false) or above max threshold (true)
|
||||||
|
bool up_;
|
||||||
|
};
|
||||||
|
|
||||||
|
// ------------------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
template <typename MeasurementT>
|
||||||
|
class AutoStop
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
void begin(MeasurementT threshold, uint32_t numMeasurementsBelowThreshold)
|
||||||
|
{
|
||||||
|
threshold_ = threshold;
|
||||||
|
numMeasurementsBelowThreshold_ = numMeasurementsBelowThreshold;
|
||||||
|
counter_ = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool autoStop(MeasurementT measurement)
|
||||||
|
{
|
||||||
|
if (measurement > threshold_)
|
||||||
|
{
|
||||||
|
counter_ = 0;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
++counter_;
|
||||||
|
if (counter_ >= numMeasurementsBelowThreshold_)
|
||||||
|
{
|
||||||
|
counter_ = 0;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
MeasurementT threshold_;
|
||||||
|
uint32_t numMeasurementsBelowThreshold_;
|
||||||
|
uint32_t counter_;
|
||||||
|
};
|
|
@ -4,18 +4,23 @@
|
||||||
#include <WiFiUdp.h>
|
#include <WiFiUdp.h>
|
||||||
#include <NTPClient.h>
|
#include <NTPClient.h>
|
||||||
|
|
||||||
|
#include "AutoStartStop.h"
|
||||||
|
|
||||||
template <typename SessionT>
|
template <typename SessionT>
|
||||||
class SessionManager
|
class SessionManager
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
using MeasurementType = typename SessionT::MeasurementType;
|
using MeasurementType = typename SessionT::MeasurementType;
|
||||||
|
|
||||||
SessionManager(int scaleDoutPin, int scaleSckPin, uint8_t tareAvgCount);
|
SessionManager();
|
||||||
|
|
||||||
void begin();
|
void begin(int scaleDoutPin, int scaleSckPin, uint8_t tareAvgCount, int valueRightShift, MeasurementType autoStartMinTh, MeasurementType autoStartMaxTh, uint32_t autoStartMeasuresBetweenPeaks,
|
||||||
|
MeasurementType autoStopTh, uint32_t autoStopNumMeasures);
|
||||||
|
|
||||||
void tare();
|
void tare();
|
||||||
long tareOffset() const { return scale_.offset(); };
|
long tareOffset() const { return scale_.offset(); };
|
||||||
|
int valueRightShift() const { return scale_.valueRightShift(); }
|
||||||
|
|
||||||
void startMeasurements();
|
void startMeasurements();
|
||||||
void stopMeasurements();
|
void stopMeasurements();
|
||||||
bool isMeasuring() const { return measuring_; }
|
bool isMeasuring() const { return measuring_; }
|
||||||
|
@ -24,10 +29,12 @@ public:
|
||||||
void iteration();
|
void iteration();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
void onMeasurementTaken(MeasurementType measurement);
|
||||||
|
|
||||||
WiFiUDP ntpUDP_;
|
WiFiUDP ntpUDP_;
|
||||||
NTPClient timeClient_;
|
NTPClient timeClient_;
|
||||||
|
|
||||||
Scale<CONFIG_VALUE_DIVIDER> scale_;
|
Scale scale_;
|
||||||
//MockScale scale;
|
//MockScale scale;
|
||||||
SessionT session_;
|
SessionT session_;
|
||||||
bool measuring_;
|
bool measuring_;
|
||||||
|
@ -36,45 +43,55 @@ private:
|
||||||
int scaleDoutPin_;
|
int scaleDoutPin_;
|
||||||
int scaleSckPin_;
|
int scaleSckPin_;
|
||||||
uint8_t tareAvgCount_;
|
uint8_t tareAvgCount_;
|
||||||
|
int valueRightShift_;
|
||||||
|
|
||||||
|
AutoStart<MeasurementT> autoStart_;
|
||||||
|
AutoStop<MeasurementT> autoStop_;
|
||||||
};
|
};
|
||||||
|
|
||||||
// ------------------------------------------------------------------------------------------------
|
// ------------------------------------------------------------------------------------------------
|
||||||
|
|
||||||
template <typename SessionT>
|
template <typename SessionT>
|
||||||
SessionManager<SessionT>::SessionManager(int scaleDoutPin, int scaleSckPin, uint8_t tareAvgCount)
|
SessionManager<SessionT>::SessionManager()
|
||||||
: timeClient_(ntpUDP_, "pool.ntp.org"),
|
: timeClient_(ntpUDP_, "pool.ntp.org"),
|
||||||
measuring_(false),
|
measuring_(false),
|
||||||
lastCallTime_(0),
|
lastCallTime_(0)
|
||||||
scaleDoutPin_(scaleDoutPin),
|
|
||||||
scaleSckPin_(scaleSckPin),
|
|
||||||
tareAvgCount_(tareAvgCount)
|
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename SessionT>
|
template <typename SessionT>
|
||||||
void SessionManager<SessionT>::tare()
|
void SessionManager<SessionT>::tare()
|
||||||
{
|
{
|
||||||
if(measuring_)
|
if (measuring_)
|
||||||
stopMeasurements();
|
stopMeasurements();
|
||||||
Serial.println("Beginning tare");
|
Serial.println("Beginning tare");
|
||||||
scale_.begin(scaleDoutPin_, scaleSckPin_);
|
scale_.begin(scaleDoutPin_, scaleSckPin_, valueRightShift_);
|
||||||
scale_.tare(CONFIG_TARE_AVG_COUNT);
|
scale_.tare(CONFIG_TARE_AVG_COUNT);
|
||||||
Serial.println("Finished tare");
|
Serial.println("Finished tare");
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename SessionT>
|
template <typename SessionT>
|
||||||
void SessionManager<SessionT>::begin()
|
void SessionManager<SessionT>::begin(int scaleDoutPin, int scaleSckPin, uint8_t tareAvgCount, int valueRightShift,
|
||||||
|
MeasurementType autoStartMinTh, MeasurementType autoStartMaxTh, uint32_t autoStartMeasuresBetweenPeaks,
|
||||||
|
MeasurementType autoStopTh, uint32_t autoStopNumMeasures)
|
||||||
{
|
{
|
||||||
|
scaleDoutPin_ = scaleDoutPin;
|
||||||
|
scaleSckPin_ = scaleSckPin;
|
||||||
|
tareAvgCount_ = tareAvgCount;
|
||||||
|
|
||||||
timeClient_.begin();
|
timeClient_.begin();
|
||||||
timeClient_.update();
|
timeClient_.update();
|
||||||
tare();
|
tare();
|
||||||
session_.init(timeClient_.getEpochTime());
|
session_.init(timeClient_.getEpochTime());
|
||||||
|
|
||||||
|
autoStart_.begin(autoStartMinTh, autoStartMaxTh, autoStartMeasuresBetweenPeaks);
|
||||||
|
autoStop_.begin(autoStopTh, autoStopNumMeasures);
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename SessionT>
|
template <typename SessionT>
|
||||||
void SessionManager<SessionT>::startMeasurements()
|
void SessionManager<SessionT>::startMeasurements()
|
||||||
{
|
{
|
||||||
if(measuring_ == true)
|
if (measuring_ == true)
|
||||||
return;
|
return;
|
||||||
measuring_ = true;
|
measuring_ = true;
|
||||||
lastCallTime_ = 0;
|
lastCallTime_ = 0;
|
||||||
|
@ -84,23 +101,25 @@ void SessionManager<SessionT>::startMeasurements()
|
||||||
template <typename SessionT>
|
template <typename SessionT>
|
||||||
void SessionManager<SessionT>::stopMeasurements()
|
void SessionManager<SessionT>::stopMeasurements()
|
||||||
{
|
{
|
||||||
if(measuring_ == false)
|
if (measuring_ == false)
|
||||||
return;
|
return;
|
||||||
session_.finalize();
|
session_.finalize();
|
||||||
measuring_ = false;
|
measuring_ = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
template <typename SessionT>
|
template <typename SessionT>
|
||||||
void SessionManager<SessionT>::iteration()
|
void SessionManager<SessionT>::iteration()
|
||||||
{
|
{
|
||||||
if (!measuring_) {
|
if (!measuring_)
|
||||||
|
{
|
||||||
delay(1);
|
delay(1);
|
||||||
return; // give control to HTTP server thread
|
return; // give control to HTTP server thread
|
||||||
}
|
}
|
||||||
|
|
||||||
uint16_t measurement = -1;
|
MeasurementType measurement = -1;
|
||||||
bool measurementDone = false;
|
bool measurementDone = false;
|
||||||
while(!measurementDone)
|
while (!measurementDone)
|
||||||
measurementDone = scale_.measure(measurement);
|
measurementDone = scale_.measure(measurement);
|
||||||
bool addPointSuccessful = session_.addPoint(measurement);
|
bool addPointSuccessful = session_.addPoint(measurement);
|
||||||
//Serial.printf("Measured: %d\n", measurement);
|
//Serial.printf("Measured: %d\n", measurement);
|
||||||
|
@ -131,3 +150,67 @@ void SessionManager<SessionT>::iteration()
|
||||||
}
|
}
|
||||||
lastCallTime_ = millis();
|
lastCallTime_ = millis();
|
||||||
}
|
}
|
||||||
|
*/
|
||||||
|
|
||||||
|
template <typename SessionT>
|
||||||
|
void SessionManager<SessionT>::iteration()
|
||||||
|
{
|
||||||
|
MeasurementType measurement = -1;
|
||||||
|
bool measurementDone = false;
|
||||||
|
while (!measurementDone)
|
||||||
|
measurementDone = scale_.measure(measurement);
|
||||||
|
|
||||||
|
onMeasurementTaken(measurement);
|
||||||
|
|
||||||
|
if (lastCallTime_ != 0)
|
||||||
|
{
|
||||||
|
const long cycleDuration = millis() - lastCallTime_;
|
||||||
|
if (cycleDuration <= CONFIG_MEASURE_DELAY)
|
||||||
|
{
|
||||||
|
delay(CONFIG_MEASURE_DELAY - cycleDuration);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
const long skipped = (cycleDuration / CONFIG_MEASURE_DELAY);
|
||||||
|
Serial.printf("Measurements skipped: %ld, cycleDuration %ld\n", skipped, cycleDuration);
|
||||||
|
|
||||||
|
for (int i = 0; i < skipped; ++i)
|
||||||
|
onMeasurementTaken(measurement);
|
||||||
|
|
||||||
|
delay(CONFIG_MEASURE_DELAY * (skipped + 1) - cycleDuration);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
lastCallTime_ = millis();
|
||||||
|
}
|
||||||
|
|
||||||
|
template <typename SessionT>
|
||||||
|
void SessionManager<SessionT>::onMeasurementTaken(MeasurementType measurement)
|
||||||
|
{
|
||||||
|
if (measuring_)
|
||||||
|
{
|
||||||
|
bool autoStop = autoStop_.autoStop(measurement);
|
||||||
|
if (autoStop)
|
||||||
|
{
|
||||||
|
Serial.println("Auto stop");
|
||||||
|
stopMeasurements();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool addPointSuccessful = session_.addPoint(measurement);
|
||||||
|
if (!addPointSuccessful)
|
||||||
|
{
|
||||||
|
Serial.println("Maximum time of session reached - stopping");
|
||||||
|
stopMeasurements();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if (autoStart_.autoStart(measurement))
|
||||||
|
{
|
||||||
|
Serial.println("Auto start");
|
||||||
|
startMeasurements();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -0,0 +1,105 @@
|
||||||
|
#include "WifiManager.h"
|
||||||
|
|
||||||
|
void WifiManager::begin(const String &hostname)
|
||||||
|
{
|
||||||
|
hostname_ = hostname;
|
||||||
|
prefs_.begin("st_wifi_manager");
|
||||||
|
startWifi();
|
||||||
|
}
|
||||||
|
|
||||||
|
void WifiManager::startWifi()
|
||||||
|
{
|
||||||
|
WiFi.disconnect();
|
||||||
|
|
||||||
|
const String apPassword = prefs_.getString("apPassword", "");
|
||||||
|
const String staSSID = prefs_.getString("staSSID", "");
|
||||||
|
const String staPassword = prefs_.getString("staPassword", "");
|
||||||
|
|
||||||
|
if (staSSID.length() > 0 && staPassword.length() > 0)
|
||||||
|
{
|
||||||
|
// station mode
|
||||||
|
WiFi.mode(WIFI_STA);
|
||||||
|
WiFi.begin(staSSID.c_str(), staPassword.c_str());
|
||||||
|
WiFi.setHostname(hostname_.c_str());
|
||||||
|
int connectCounter = 0;
|
||||||
|
bool successful = true;
|
||||||
|
while (WiFi.status() != WL_CONNECTED)
|
||||||
|
{
|
||||||
|
delay(2000);
|
||||||
|
WiFi.begin(staSSID.c_str(), staPassword.c_str());
|
||||||
|
connectCounter += 1;
|
||||||
|
if (connectCounter >= 60) // for two minutes no connection
|
||||||
|
{
|
||||||
|
successful = false;
|
||||||
|
break; // fallback to AP mode
|
||||||
|
}
|
||||||
|
}
|
||||||
|
state_ = STA;
|
||||||
|
if (successful)
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
WiFi.mode(WIFI_AP);
|
||||||
|
WiFi.softAPConfig(IPAddress(192, 168, 42, 1), IPAddress(192, 168, 42, 1), IPAddress(255, 255, 255, 0));
|
||||||
|
WiFi.softAPsetHostname(hostname_.c_str());
|
||||||
|
if (apPassword.length() > 0)
|
||||||
|
{
|
||||||
|
WiFi.softAP(hostname_.c_str(), apPassword.c_str());
|
||||||
|
state_ = AP_SECURE;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
WiFi.softAP(hostname_.c_str());
|
||||||
|
state_ = AP_PROVISIONING;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void WifiManager::setStaCredentials(const char *wifiName, const char *password)
|
||||||
|
{
|
||||||
|
assert(state_ != INVALID);
|
||||||
|
|
||||||
|
prefs_.putString("staSSID", wifiName);
|
||||||
|
prefs_.putString("staPassword", password);
|
||||||
|
}
|
||||||
|
|
||||||
|
void WifiManager::setApCredentials(const char *password)
|
||||||
|
{
|
||||||
|
assert(state_ != INVALID);
|
||||||
|
|
||||||
|
prefs_.remove("staSSID");
|
||||||
|
prefs_.remove("staPassword");
|
||||||
|
prefs_.putString("apPassword", password);
|
||||||
|
}
|
||||||
|
|
||||||
|
void WifiManager::resetToApProvisioning()
|
||||||
|
{
|
||||||
|
prefs_.remove("apPassword");
|
||||||
|
prefs_.remove("staSSID");
|
||||||
|
prefs_.remove("staPassword");
|
||||||
|
}
|
||||||
|
|
||||||
|
void WifiManager::wifiWatchdog()
|
||||||
|
{
|
||||||
|
if (state_ == STA && WiFi.status() != WL_CONNECTED) {
|
||||||
|
startWifi();
|
||||||
|
Serial.println("Connection lost - Restarting WIFI");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
const char * WifiManager::stateToString(WifiManager::State state)
|
||||||
|
{
|
||||||
|
switch (state)
|
||||||
|
{
|
||||||
|
case INVALID:
|
||||||
|
return "INVALID";
|
||||||
|
case STA:
|
||||||
|
return "STATION_MODE";
|
||||||
|
case AP_PROVISIONING:
|
||||||
|
return "AP_PROVISIONING";
|
||||||
|
case AP_SECURE:
|
||||||
|
return "AP_SECURE";
|
||||||
|
default:
|
||||||
|
return "";
|
||||||
|
};
|
||||||
|
}
|
|
@ -0,0 +1,51 @@
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "Preferences.h"
|
||||||
|
#include <WiFi.h>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Manages connection and provisioning of WiFI
|
||||||
|
*
|
||||||
|
* Starts in state AP_PROVISIONING, where device is an access point without password.
|
||||||
|
* Device should not work normally in this mode, only API is served to set password.
|
||||||
|
* Two options:
|
||||||
|
* - connect to existing WiFi (station mode), via setStaCredentials
|
||||||
|
* - create secured access point, via setApCredentials
|
||||||
|
*
|
||||||
|
* When operating in access point mode, the device IP is 192.168.42.1
|
||||||
|
*
|
||||||
|
* call wifiWatchdog regularly to reconnect in station mode if connection was lost
|
||||||
|
*/
|
||||||
|
class WifiManager
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
enum State
|
||||||
|
{
|
||||||
|
INVALID,
|
||||||
|
STA,
|
||||||
|
AP_PROVISIONING,
|
||||||
|
AP_SECURE,
|
||||||
|
};
|
||||||
|
WifiManager() : state_(INVALID) {}
|
||||||
|
|
||||||
|
void begin(const String &hostname);
|
||||||
|
|
||||||
|
void setStaCredentials(const char *wifiName, const char *password);
|
||||||
|
void setApCredentials(const char *password);
|
||||||
|
void resetToApProvisioning();
|
||||||
|
|
||||||
|
void wifiWatchdog();
|
||||||
|
|
||||||
|
bool inProvisioningMode() const { return state_ == INVALID || state_ == AP_PROVISIONING; }
|
||||||
|
|
||||||
|
State state() const { return state_; }
|
||||||
|
const char *stateStr() const { return stateToString(state_); }
|
||||||
|
|
||||||
|
static const char *stateToString(State state);
|
||||||
|
|
||||||
|
private:
|
||||||
|
void startWifi();
|
||||||
|
Preferences prefs_;
|
||||||
|
State state_;
|
||||||
|
String hostname_;
|
||||||
|
};
|
|
@ -27,6 +27,7 @@ monitor_speed = 115200
|
||||||
lib_deps =
|
lib_deps =
|
||||||
NTPClient
|
NTPClient
|
||||||
HX711@0.7.4
|
HX711@0.7.4
|
||||||
|
ArduinoJson
|
||||||
https://github.com/gilmaimon/ArduinoWebsockets.git
|
https://github.com/gilmaimon/ArduinoWebsockets.git
|
||||||
src_filter = +<*> -<native_main.cpp>
|
src_filter = +<*> -<native_main.cpp>
|
||||||
board_build.partitions = partitions_custom.csv
|
board_build.partitions = partitions_custom.csv
|
||||||
|
|
|
@ -7,18 +7,11 @@
|
||||||
//#define _HW_V_20
|
//#define _HW_V_20
|
||||||
#define NEW_HEAVY_LOAD_CELL
|
#define NEW_HEAVY_LOAD_CELL
|
||||||
|
|
||||||
// ------------------------------------------ WiFi ---------------------------------------------------------------------------------
|
|
||||||
|
|
||||||
const char *CONFIG_WIFI_SSID = "WLAN";
|
|
||||||
//const char *CONFIG_WIFI_SSID = "MartinHandy";
|
|
||||||
const char *CONFIG_WIFI_PASSWORD = "Bau3rWLAN";
|
|
||||||
//const char *CONFIG_WIFI_PASSWORD = "35cbbd203afe";
|
|
||||||
const char *CONFIG_HOSTNAME = "swimtracker";
|
const char *CONFIG_HOSTNAME = "swimtracker";
|
||||||
|
|
||||||
|
|
||||||
// ------------------------------------- Hardware & Measurement Settings ------------------------------------------------------------
|
// ------------------------------------- Hardware & Measurement Settings ------------------------------------------------------------
|
||||||
|
|
||||||
|
|
||||||
const uint8_t CONFIG_MEASUREMENT_AVG_COUNT = 1; // number of measurements in normal phase
|
const uint8_t CONFIG_MEASUREMENT_AVG_COUNT = 1; // number of measurements in normal phase
|
||||||
const uint8_t CONFIG_TARE_AVG_COUNT = 20; // number of measurements in tare-phase (to find 0 )
|
const uint8_t CONFIG_TARE_AVG_COUNT = 20; // number of measurements in tare-phase (to find 0 )
|
||||||
const int CONFIG_MEASURE_DELAY = 100; // interval in ms between measurements
|
const int CONFIG_MEASURE_DELAY = 100; // interval in ms between measurements
|
||||||
|
@ -26,10 +19,24 @@ const uint32_t CONFIG_SESSION_MAX_LENGTH_HOURS = 3; // maximum length of one ses
|
||||||
const char *CONFIG_DATA_PATH = "/dat"; // folder in SPIFFS file system to store measurement data
|
const char *CONFIG_DATA_PATH = "/dat"; // folder in SPIFFS file system to store measurement data
|
||||||
using MeasurementT = uint16_t; // data type for one measurement
|
using MeasurementT = uint16_t; // data type for one measurement
|
||||||
#ifdef NEW_HEAVY_LOAD_CELL
|
#ifdef NEW_HEAVY_LOAD_CELL
|
||||||
const int CONFIG_VALUE_DIVIDER = 8; // uint32 measurements are divided by this factor, before stored in uint16_t
|
const int CONFIG_VALUE_RIGHT_SHIFT = 3; // uint32 measurements are divided by this power, before stored in uint16_t
|
||||||
#else
|
#else
|
||||||
const int CONFIG_VALUE_DIVIDER = 128; // uint32 measurements are divided by this factor, before stored in uint16_t
|
const int CONFIG_VALUE_RIGHT_SHIFT = 7;
|
||||||
#endif
|
#endif
|
||||||
|
const MeasurementT CONFIG_KG_FACTOR_INV = 701; // after shifting - how many "measurement units" are one kg
|
||||||
|
|
||||||
|
|
||||||
|
const char * UPDATE_URL = "https://swimtracker-update.bauer.tech/firmware.bin";
|
||||||
|
|
||||||
|
|
||||||
|
// auto start/stop
|
||||||
|
MeasurementT CONFIG_AUTO_START_MIN_THRESHOLD = CONFIG_KG_FACTOR_INV * 1;
|
||||||
|
MeasurementT CONFIG_AUTO_START_MAX_THRESHOLD = CONFIG_KG_FACTOR_INV * 3;
|
||||||
|
uint32_t CONFIG_AUTO_START_MAX_MEASUREMENTS_BETWEEN_PEAKS = (1000 / CONFIG_MEASURE_DELAY) * 6;
|
||||||
|
|
||||||
|
MeasurementT CONFIG_AUTO_STOP_THRESHOLD = CONFIG_KG_FACTOR_INV * 1;
|
||||||
|
//uint32_t CONFIG_AUTO_STOP_NUM_MEASUREMENTS = (1000 / CONFIG_MEASURE_DELAY) * 60 * 15;
|
||||||
|
uint32_t CONFIG_AUTO_STOP_NUM_MEASUREMENTS = (1000 / CONFIG_MEASURE_DELAY) * 30;
|
||||||
|
|
||||||
|
|
||||||
// ------------------------------------- Derived Settings -----------------------------------------------------------------------------
|
// ------------------------------------- Derived Settings -----------------------------------------------------------------------------
|
||||||
|
@ -37,7 +44,6 @@ const int CONFIG_VALUE_DIVIDER = 128; // uint32 measurements are d
|
||||||
const uint32_t CONFIG_SESSION_MAX_SIZE = CONFIG_SESSION_MAX_LENGTH_HOURS * 3600 * (1000 / CONFIG_MEASURE_DELAY) * sizeof(uint16_t);
|
const uint32_t CONFIG_SESSION_MAX_SIZE = CONFIG_SESSION_MAX_LENGTH_HOURS * 3600 * (1000 / CONFIG_MEASURE_DELAY) * sizeof(uint16_t);
|
||||||
static_assert(CONFIG_SESSION_MAX_SIZE < 1024 * 1024, "Measurement data takes more than 1MiB space");
|
static_assert(CONFIG_SESSION_MAX_SIZE < 1024 * 1024, "Measurement data takes more than 1MiB space");
|
||||||
|
|
||||||
// HX711 load cell
|
|
||||||
#ifdef _HW_V_20
|
#ifdef _HW_V_20
|
||||||
const int CONFIG_SCALE_DOUT_PIN = 23;
|
const int CONFIG_SCALE_DOUT_PIN = 23;
|
||||||
const int CONFIG_SCALE_SCK_PIN = 22;
|
const int CONFIG_SCALE_SCK_PIN = 22;
|
||||||
|
|
|
@ -3,10 +3,14 @@
|
||||||
#include "Dtypes.h"
|
#include "Dtypes.h"
|
||||||
#include "SwimTrackerConfig.h"
|
#include "SwimTrackerConfig.h"
|
||||||
|
|
||||||
#include <WiFi.h>
|
#include <ArduinoJson.h>
|
||||||
#include <HTTPClient.h>
|
#include <HTTPClient.h>
|
||||||
|
|
||||||
|
#include "esp_https_ota.h"
|
||||||
|
#include "ESPmDNS.h"
|
||||||
|
|
||||||
// Own libs
|
// Own libs
|
||||||
|
#include "WifiManager.h"
|
||||||
#include "MockScale.h"
|
#include "MockScale.h"
|
||||||
#include "Scale.h"
|
#include "Scale.h"
|
||||||
#include "MeasurementSession.h"
|
#include "MeasurementSession.h"
|
||||||
|
@ -17,15 +21,14 @@
|
||||||
#include "WebDAV.h"
|
#include "WebDAV.h"
|
||||||
#include "WebsocketServer.h"
|
#include "WebsocketServer.h"
|
||||||
|
|
||||||
#include "esp_https_ota.h"
|
|
||||||
#include "ESPmDNS.h"
|
|
||||||
|
|
||||||
using Session_T = SimpleMeasurementSession<MeasurementT, CONFIG_SESSION_MAX_SIZE>;
|
using Session_T = SimpleMeasurementSession<MeasurementT, CONFIG_SESSION_MAX_SIZE>;
|
||||||
SessionManager<Session_T> sessionManager(CONFIG_SCALE_DOUT_PIN, CONFIG_SCALE_SCK_PIN, CONFIG_TARE_AVG_COUNT);
|
SessionManager<Session_T> sessionManager;
|
||||||
|
|
||||||
EspHttp espHttpServer;
|
EspHttp espHttpServer;
|
||||||
WebsocketServer<Session_T> webSocketServer(sessionManager, 81);
|
WebsocketServer<Session_T> webSocketServer(sessionManager, 81);
|
||||||
|
|
||||||
|
WifiManager wifiManager;
|
||||||
|
|
||||||
extern const uint8_t certificate_pem[] asm("_binary_certificate_pem_start");
|
extern const uint8_t certificate_pem[] asm("_binary_certificate_pem_start");
|
||||||
|
|
||||||
bool firmwareUpdate()
|
bool firmwareUpdate()
|
||||||
|
@ -33,7 +36,7 @@ bool firmwareUpdate()
|
||||||
esp_http_client_config_t config;
|
esp_http_client_config_t config;
|
||||||
Serial.println((char *)certificate_pem);
|
Serial.println((char *)certificate_pem);
|
||||||
memset(&config, 0, sizeof(esp_http_client_config_t));
|
memset(&config, 0, sizeof(esp_http_client_config_t));
|
||||||
config.url = "https://swimtracker-update.bauer.tech/firmware.bin";
|
config.url = UPDATE_URL;
|
||||||
config.cert_pem = (char *)certificate_pem;
|
config.cert_pem = (char *)certificate_pem;
|
||||||
Serial.println("Starting firmware upgrade");
|
Serial.println("Starting firmware upgrade");
|
||||||
esp_err_t ret = esp_https_ota(&config);
|
esp_err_t ret = esp_https_ota(&config);
|
||||||
|
@ -51,7 +54,7 @@ bool firmwareUpdate()
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename SessionT>
|
template <typename SessionT>
|
||||||
void httpSetup(SessionManager<SessionT> *sessionManager)
|
void httpSetup(SessionManager<SessionT> *sessionManager, WifiManager *wifiManager)
|
||||||
{
|
{
|
||||||
auto cbStartSession = [sessionManager](httpd_req_t *req) {
|
auto cbStartSession = [sessionManager](httpd_req_t *req) {
|
||||||
httpd_resp_set_hdr(req, "Access-Control-Allow-Origin", "*");
|
httpd_resp_set_hdr(req, "Access-Control-Allow-Origin", "*");
|
||||||
|
@ -79,63 +82,58 @@ void httpSetup(SessionManager<SessionT> *sessionManager)
|
||||||
firmwareUpdate();
|
firmwareUpdate();
|
||||||
};
|
};
|
||||||
auto cbStatus = [sessionManager](httpd_req_t *req) {
|
auto cbStatus = [sessionManager](httpd_req_t *req) {
|
||||||
String result;
|
httpd_resp_set_hdr(req, "Access-Control-Allow-Origin", "*");
|
||||||
result.reserve(512);
|
|
||||||
httpd_resp_set_hdr(req, "Content-Type", "application/json");
|
httpd_resp_set_hdr(req, "Content-Type", "application/json");
|
||||||
|
|
||||||
|
StaticJsonDocument<1024> json;
|
||||||
// session
|
// session
|
||||||
{
|
{
|
||||||
result += "{ \"session\": ";
|
JsonObject sessionObj = json.createNestedObject("session");
|
||||||
if (sessionManager->isMeasuring())
|
if (sessionManager->isMeasuring())
|
||||||
{
|
{
|
||||||
const auto &session = sessionManager->session();
|
const auto &session = sessionManager->session();
|
||||||
result += "{ \"started\":" + String(session.getStartTime()) + ", ";
|
sessionObj["started"] = session.getStartTime();
|
||||||
result += "\"num_measurements\":" + String(session.numMeasurements()) + "},\n";
|
sessionObj["num_measurements"] = session.numMeasurements();
|
||||||
}
|
}
|
||||||
else
|
|
||||||
result += "{},\n";
|
|
||||||
}
|
}
|
||||||
// scale
|
// scale
|
||||||
{
|
{
|
||||||
const String tareOffset(sessionManager->tareOffset());
|
JsonObject scaleObj = json.createNestedObject("scale");
|
||||||
const String divider(CONFIG_VALUE_DIVIDER);
|
scaleObj["tare_offset"] = sessionManager->tareOffset();
|
||||||
result += "\"scale\": { \"tare_offset\": " + tareOffset + ", \"divider\":" + divider + "},\n";
|
scaleObj["value_right_shift"] = sessionManager->valueRightShift();
|
||||||
}
|
}
|
||||||
// flash
|
// flash
|
||||||
{
|
{
|
||||||
const String usedBytes(portablefs::usedBytes());
|
JsonObject fsObj = json.createNestedObject("flash");
|
||||||
const String freeBytes(portablefs::totalBytes() - portablefs::usedBytes());
|
fsObj["used"] = portablefs::usedBytes();
|
||||||
result += "\"file_system\": { \"used\": " + usedBytes + ", \"free\":" + freeBytes + "},\n";
|
fsObj["free"] = portablefs::totalBytes() - portablefs::usedBytes();
|
||||||
}
|
}
|
||||||
// RAM
|
// RAM
|
||||||
{
|
{
|
||||||
const String freeBytes(ESP.getFreeHeap());
|
JsonObject ramObj = json.createNestedObject("ram");
|
||||||
const String usedBytes(ESP.getHeapSize() - ESP.getFreeHeap());
|
ramObj["used"] = ESP.getHeapSize() - ESP.getFreeHeap();
|
||||||
result += "\"ram\": { \"used\": " + usedBytes + ", \"free\":" + freeBytes + "},\n";
|
ramObj["free"] = ESP.getFreeHeap();
|
||||||
}
|
}
|
||||||
// PSRAM
|
// PSRAM
|
||||||
{
|
{
|
||||||
const String freeBytes(ESP.getFreePsram());
|
JsonObject psramObj = json.createNestedObject("psram");
|
||||||
const String usedBytes(ESP.getPsramSize() - ESP.getFreePsram());
|
psramObj["used"] = ESP.getPsramSize() - ESP.getFreePsram();
|
||||||
result += "\"psram\": { \"used\": " + usedBytes + ", \"free\":" + freeBytes + "},\n";
|
psramObj["free"] = ESP.getFreePsram();
|
||||||
}
|
}
|
||||||
// firmware
|
// firmware
|
||||||
{
|
{
|
||||||
auto descr = esp_ota_get_app_description();
|
auto descr = esp_ota_get_app_description();
|
||||||
const String projectName(descr->project_name);
|
|
||||||
const String versionStr(descr->version);
|
JsonObject firmware = json.createNestedObject("firmware");
|
||||||
const String idfVersion(descr->idf_ver);
|
firmware["name"] = descr->project_name;
|
||||||
const String compileDate(descr->date);
|
firmware["version"] = descr->version;
|
||||||
const String compileTime(descr->time);
|
firmware["idf_version"] = descr->idf_ver;
|
||||||
result += "\"firmware\": { \"name\" : \"" +
|
firmware["compile_date"] = descr->date;
|
||||||
projectName + "\", \"version\": \"" +
|
firmware["compile_time"] = descr->time;
|
||||||
versionStr + "\", \"idfVersion\": \"" +
|
|
||||||
idfVersion + "\", \"compile_date\": \"" +
|
|
||||||
compileDate + +"\", \"compile_time\": \"" +
|
|
||||||
compileTime + "\" }\n";
|
|
||||||
}
|
}
|
||||||
result += "}";
|
char jsonText[512];
|
||||||
httpd_resp_send(req, result.c_str(), result.length());
|
auto bytesWritten = serializeJson(json, jsonText);
|
||||||
|
httpd_resp_send(req, jsonText, bytesWritten);
|
||||||
};
|
};
|
||||||
auto cbGetData = [sessionManager](httpd_req_t *req) {
|
auto cbGetData = [sessionManager](httpd_req_t *req) {
|
||||||
auto sessionId = sessionManager->session().getStartTime();
|
auto sessionId = sessionManager->session().getStartTime();
|
||||||
|
@ -166,36 +164,88 @@ void httpSetup(SessionManager<SessionT> *sessionManager)
|
||||||
httpd_resp_send(req, buf, totalSize);
|
httpd_resp_send(req, buf, totalSize);
|
||||||
free(buf);
|
free(buf);
|
||||||
};
|
};
|
||||||
|
auto cbWifiGet = [wifiManager](httpd_req_t *req) {
|
||||||
|
httpd_resp_set_hdr(req, "Access-Control-Allow-Origin", "*");
|
||||||
|
StaticJsonDocument<128> json;
|
||||||
|
json["state"] = wifiManager->stateStr();
|
||||||
|
char jsonText[128];
|
||||||
|
auto bytesWritten = serializeJson(json, jsonText);
|
||||||
|
httpd_resp_send(req, jsonText, bytesWritten);
|
||||||
|
};
|
||||||
|
auto cbWifiPost = [wifiManager](httpd_req_t *req) {
|
||||||
|
httpd_resp_set_hdr(req, "Access-Control-Allow-Origin", "*");
|
||||||
|
StaticJsonDocument<1024> json;
|
||||||
|
char content[512];
|
||||||
|
size_t recvSize = min(req->content_len, sizeof(content)); //truncate if too long
|
||||||
|
int ret = httpd_req_recv(req, content, recvSize);
|
||||||
|
if (ret <= 0)
|
||||||
|
{
|
||||||
|
if (ret == HTTPD_SOCK_ERR_TIMEOUT)
|
||||||
|
httpd_resp_send_408(req);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
DeserializationError err = deserializeJson(json, content);
|
||||||
|
if (err)
|
||||||
|
{
|
||||||
|
httpd_resp_set_status(req, "400 Bad Request");
|
||||||
|
httpd_resp_send(req, "JSON parse error", -1);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (json.containsKey("reset_to_provisioning") && json["reset_to_provisioning"].as<bool>())
|
||||||
|
{
|
||||||
|
wifiManager->resetToApProvisioning();
|
||||||
|
httpd_resp_send(req, "OK", -1);
|
||||||
|
ESP.restart();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (json.containsKey("ap_password"))
|
||||||
|
{
|
||||||
|
httpd_resp_send(req, "OK", -1);
|
||||||
|
wifiManager->setApCredentials(json["ap_password"].as<const char *>());
|
||||||
|
ESP.restart();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (json.containsKey("sta_ssid") && json.containsKey("sta_password"))
|
||||||
|
{
|
||||||
|
wifiManager->setStaCredentials(json["sta_ssid"].as<const char *>(), //
|
||||||
|
json["sta_password"].as<const char *>());
|
||||||
|
httpd_resp_send(req, "OK", -1);
|
||||||
|
ESP.restart();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
httpd_resp_set_status(req, "400 Bad Request");
|
||||||
|
httpd_resp_send(req, "Invalid keys in JSON", -1);
|
||||||
|
};
|
||||||
|
|
||||||
espHttpServer.start();
|
espHttpServer.start();
|
||||||
espHttpServer.on("/api/session/start", HTTP_GET, cbStartSession);
|
|
||||||
espHttpServer.on("/api/session/start", HTTP_POST, cbStartSession);
|
|
||||||
|
|
||||||
espHttpServer.on("/api/session/stop", HTTP_GET, cbStopSession);
|
espHttpServer.on("/api/wifi", HTTP_GET, cbWifiGet);
|
||||||
espHttpServer.on("/api/session/stop", HTTP_POST, cbStopSession);
|
espHttpServer.on("/api/wifi", HTTP_POST, cbWifiPost);
|
||||||
|
|
||||||
espHttpServer.on("/api/session/data", HTTP_GET, cbGetData);
|
|
||||||
espHttpServer.on("/api/status", HTTP_GET, cbStatus);
|
|
||||||
espHttpServer.on("/api/tare", HTTP_GET, cbTare);
|
|
||||||
espHttpServer.on("/api/restart", HTTP_GET, cbRestart);
|
espHttpServer.on("/api/restart", HTTP_GET, cbRestart);
|
||||||
espHttpServer.on("/api/firmwareupdate", HTTP_GET, cbFirmwareUpdate);
|
|
||||||
|
|
||||||
auto webdav = webdavHandler("/webdav/", "/dat");
|
if (!wifiManager->inProvisioningMode())
|
||||||
espHttpServer.on("/webdav/*?", HTTP_GET, webdav);
|
|
||||||
espHttpServer.on("/webdav/*?", HTTP_PROPFIND, webdav);
|
|
||||||
espHttpServer.on("/webdav/*?", HTTP_DELETE, webdav);
|
|
||||||
espHttpServer.on("/webdav/*?", HTTP_OPTIONS, webdav);
|
|
||||||
|
|
||||||
Serial.println("HTTP setup done");
|
|
||||||
}
|
|
||||||
|
|
||||||
void checkWifi()
|
|
||||||
{
|
|
||||||
while (WiFi.status() != WL_CONNECTED)
|
|
||||||
{
|
{
|
||||||
Serial.println("WiFi disconnected. Try to reconnect");
|
espHttpServer.on("/api/session/start", HTTP_GET, cbStartSession);
|
||||||
WiFi.reconnect();
|
espHttpServer.on("/api/session/start", HTTP_POST, cbStartSession);
|
||||||
delay(2000);
|
|
||||||
|
espHttpServer.on("/api/session/stop", HTTP_GET, cbStopSession);
|
||||||
|
espHttpServer.on("/api/session/stop", HTTP_POST, cbStopSession);
|
||||||
|
|
||||||
|
espHttpServer.on("/api/session/data", HTTP_GET, cbGetData);
|
||||||
|
espHttpServer.on("/api/status", HTTP_GET, cbStatus);
|
||||||
|
espHttpServer.on("/api/tare", HTTP_GET, cbTare);
|
||||||
|
espHttpServer.on("/api/firmwareupdate", HTTP_GET, cbFirmwareUpdate);
|
||||||
|
|
||||||
|
auto webdav = webdavHandler("/webdav/", "/dat");
|
||||||
|
espHttpServer.on("/webdav/*?", HTTP_GET, webdav);
|
||||||
|
espHttpServer.on("/webdav/*?", HTTP_PROPFIND, webdav);
|
||||||
|
espHttpServer.on("/webdav/*?", HTTP_DELETE, webdav);
|
||||||
|
espHttpServer.on("/webdav/*?", HTTP_OPTIONS, webdav);
|
||||||
|
Serial.println("HTTP setup done");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
Serial.println("HTTP setup with limited API in provisioning mode");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -213,25 +263,13 @@ void mdnsSetup(const String &fullHostname)
|
||||||
if (!MDNS.begin(fullHostname.c_str()))
|
if (!MDNS.begin(fullHostname.c_str()))
|
||||||
{
|
{
|
||||||
Serial.println("Error setting up MDNS responder!");
|
Serial.println("Error setting up MDNS responder!");
|
||||||
while (1)
|
while (true)
|
||||||
{
|
{
|
||||||
delay(1000);
|
delay(1000);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Serial.printf("mDNS started %s\n", fullHostname.c_str());
|
Serial.printf("mDNS started %s\n", fullHostname.c_str());
|
||||||
MDNS.addService("_swimtracker", "tcp", 80);
|
MDNS.addService("swimtracker", "tcp", 81);
|
||||||
/*
|
|
||||||
// mDNS
|
|
||||||
esp_err_t err = mdns_init();
|
|
||||||
if (err)
|
|
||||||
Serial.printf("MDNS Init failed: %d\n", err);
|
|
||||||
else {
|
|
||||||
Serial.printf("Setting up zeroconf hostname %s\n", fullHostname.c_str());
|
|
||||||
mdns_hostname_set(fullHostname.c_str());
|
|
||||||
mdns_service_add(NULL, "_swimtracker", "_tcp", 81, NULL, 0);
|
|
||||||
mdns_instance_name_set("SwimTracker by bauer.tech");
|
|
||||||
}
|
|
||||||
*/
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void setup()
|
void setup()
|
||||||
|
@ -241,7 +279,6 @@ void setup()
|
||||||
while (!Serial)
|
while (!Serial)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
Serial.printf("Starting SwimTracker Firmware - connecting to %s\n", CONFIG_WIFI_SSID);
|
|
||||||
|
|
||||||
// File system
|
// File system
|
||||||
bool spiffsResult = SPIFFS.begin(true);
|
bool spiffsResult = SPIFFS.begin(true);
|
||||||
|
@ -251,41 +288,40 @@ void setup()
|
||||||
ESP_ERROR_CHECK(esp_event_loop_create_default());
|
ESP_ERROR_CHECK(esp_event_loop_create_default());
|
||||||
|
|
||||||
// WiFi
|
// WiFi
|
||||||
WiFi.disconnect();
|
|
||||||
WiFi.mode(WIFI_STA);
|
|
||||||
WiFi.begin(CONFIG_WIFI_SSID, CONFIG_WIFI_PASSWORD);
|
|
||||||
String fullHostname = String(CONFIG_HOSTNAME) + getIdSuffix();
|
String fullHostname = String(CONFIG_HOSTNAME) + getIdSuffix();
|
||||||
WiFi.setHostname(fullHostname.c_str());
|
wifiManager.begin(fullHostname);
|
||||||
Serial.println(F("Waiting for WIFI connection..."));
|
|
||||||
int connectCounter = 0;
|
|
||||||
while (WiFi.status() != WL_CONNECTED)
|
|
||||||
{
|
|
||||||
delay(1000);
|
|
||||||
connectCounter += 1;
|
|
||||||
if (connectCounter > 5)
|
|
||||||
{
|
|
||||||
Serial.println("Couldn't obtain WIFI - retrying..");
|
|
||||||
WiFi.begin(CONFIG_WIFI_SSID, CONFIG_WIFI_PASSWORD);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Serial.print("Connected to WiFi. IP:");
|
Serial.print("Connected to WiFi. IP:");
|
||||||
Serial.println(WiFi.localIP());
|
Serial.println(WiFi.localIP());
|
||||||
|
Serial.printf("WIFI state: %s\n", wifiManager.stateStr());
|
||||||
|
|
||||||
mdnsSetup(fullHostname);
|
mdnsSetup(fullHostname);
|
||||||
|
|
||||||
|
Preferences scalePrefs;
|
||||||
|
scalePrefs.begin("st_prefs");
|
||||||
|
int valueRightShift = scalePrefs.getInt("valueRightShift", CONFIG_VALUE_RIGHT_SHIFT);
|
||||||
|
uint8_t tareAvgCount = scalePrefs.getUInt("tareAvgCount", CONFIG_TARE_AVG_COUNT);
|
||||||
|
MeasurementT autoStartMinThreshold = scalePrefs.getUInt("autoStartMinThreshold", CONFIG_AUTO_START_MIN_THRESHOLD);
|
||||||
|
MeasurementT autoStartMaxThreshold = scalePrefs.getUInt("autoStartMinThreshold", CONFIG_AUTO_START_MAX_THRESHOLD);
|
||||||
|
uint32_t autoStartMaxMeasurementsBetweenPeaks = scalePrefs.getUInt("autoStartMaxMeasurementsBetweenPeaks", CONFIG_AUTO_START_MAX_MEASUREMENTS_BETWEEN_PEAKS);
|
||||||
|
MeasurementT autoStopThreshold = scalePrefs.getUInt("autoStopThreshold", CONFIG_AUTO_STOP_THRESHOLD);
|
||||||
|
uint32_t autoStopNumMeasurements = scalePrefs.getUInt("autoStopNumMeasurements", CONFIG_AUTO_STOP_NUM_MEASUREMENTS);
|
||||||
|
|
||||||
|
|
||||||
// Session
|
// Session
|
||||||
sessionManager.begin();
|
sessionManager.begin(CONFIG_SCALE_DOUT_PIN, CONFIG_SCALE_SCK_PIN,
|
||||||
|
tareAvgCount, valueRightShift,
|
||||||
|
autoStartMinThreshold, autoStartMaxThreshold, autoStartMaxMeasurementsBetweenPeaks,
|
||||||
|
autoStopThreshold, autoStopNumMeasurements);
|
||||||
|
|
||||||
// HTTP & Websocket server
|
// HTTP & Websocket server
|
||||||
httpSetup(&sessionManager);
|
httpSetup(&sessionManager, &wifiManager);
|
||||||
webSocketServer.begin();
|
if (!wifiManager.inProvisioningMode())
|
||||||
|
webSocketServer.begin();
|
||||||
}
|
}
|
||||||
|
|
||||||
int measurementsSent = 0;
|
|
||||||
|
|
||||||
void loop()
|
void loop()
|
||||||
{
|
{
|
||||||
sessionManager.iteration();
|
sessionManager.iteration();
|
||||||
webSocketServer.iteration();
|
webSocketServer.iteration();
|
||||||
checkWifi();
|
wifiManager.wifiWatchdog();
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,2 +0,0 @@
|
||||||
|
|
||||||
https://github.com/adafruit/Adafruit-HUZZAH32-ESP32-Feather-PCB.git
|
|
|
@ -0,0 +1,29 @@
|
||||||
|
|
||||||
|
|
||||||
|
Components
|
||||||
|
==========
|
||||||
|
https://lcsc.com/
|
||||||
|
|
||||||
|
|
||||||
|
- ESP32-WROVER-E (has PSRAM compared to wroom, the IE version doesn't have pcb antenna)
|
||||||
|
https://www.mouser.de/ProductDetail/Espressif-Systems/ESP32-WROVER-EM213EH2864PH3Q0?qs=sGAEpiMZZMu3sxpa5v1qrgqRbH4gaXhhyqOoBsYWKak%3D
|
||||||
|
- HX711 ADC for weight scale
|
||||||
|
mouser doesn't have it?
|
||||||
|
|
||||||
|
|
||||||
|
Tools
|
||||||
|
=====
|
||||||
|
- easyeda has lcsc library, to import in kicad use:
|
||||||
|
https://wokwi.com/easyeda2kicad
|
||||||
|
|
||||||
|
|
||||||
|
Related open source PCBs
|
||||||
|
========================
|
||||||
|
|
||||||
|
https://github.com/adafruit/Adafruit-HUZZAH32-ESP32-Feather-PCB.git
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
Requirements HX711
|
||||||
|
==================
|
||||||
|
-
|
Loading…
Reference in New Issue