2020-06-21 11:10:45 +02:00
|
|
|
#include "Dtypes.h"
|
|
|
|
#include "SessionChunk.h"
|
|
|
|
#include "FilesystemAbstraction.h"
|
|
|
|
|
|
|
|
template <typename Measurement_T, uint32_t MAX_SIZE>
|
|
|
|
class SimpleMeasurementSession
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
using ChunkT = SessionChunk<Measurement_T, MAX_SIZE>;
|
|
|
|
|
|
|
|
// save interval in number of measurements (by default every minute)
|
|
|
|
SimpleMeasurementSession(uint32_t saveInterval = 10 * 60)
|
|
|
|
: chunk(nullptr), saveInterval_(saveInterval)
|
|
|
|
{
|
|
|
|
}
|
2020-06-21 16:03:00 +02:00
|
|
|
|
2020-06-21 11:10:45 +02:00
|
|
|
~SimpleMeasurementSession()
|
|
|
|
{
|
|
|
|
if (chunk != nullptr)
|
|
|
|
free(chunk);
|
|
|
|
}
|
|
|
|
|
|
|
|
void init(uint32_t epochStartTime)
|
|
|
|
{
|
2020-06-21 16:03:00 +02:00
|
|
|
if (chunk == nullptr)
|
2020-06-21 11:10:45 +02:00
|
|
|
{
|
|
|
|
// psram allocation doesn't seem to work in constructor
|
|
|
|
chunk = (ChunkT *)heap_caps_malloc(sizeof(ChunkT), MALLOC_CAP_SPIRAM);
|
|
|
|
new (chunk) ChunkT(); // placement new to init chunk
|
|
|
|
}
|
|
|
|
chunk->init(epochStartTime, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool addPoint(Measurement_T measurement)
|
|
|
|
{
|
|
|
|
bool success = chunk->addPoint(measurement);
|
|
|
|
if (success && (chunk->numMeasurements() % saveInterval_) == 0)
|
|
|
|
saveToFileSystem();
|
2020-06-21 16:03:00 +02:00
|
|
|
if (!success)
|
2020-06-21 11:10:45 +02:00
|
|
|
Serial.println("Failed to add point");
|
|
|
|
return success;
|
|
|
|
}
|
|
|
|
|
|
|
|
void finalize()
|
|
|
|
{
|
2020-06-21 16:03:00 +02:00
|
|
|
if (numMeasurements() > 0)
|
|
|
|
saveToFileSystem();
|
2020-06-21 11:10:45 +02:00
|
|
|
chunk->init(0, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t getStartTime() const
|
|
|
|
{
|
|
|
|
return chunk->getStartTime();
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t numMeasurements() const
|
|
|
|
{
|
|
|
|
return chunk->numMeasurements();
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename Encoder_T>
|
|
|
|
void serialize(Encoder_T &encoder, uint32_t startIdx) const
|
|
|
|
{
|
|
|
|
ChunkT::sendHeader(encoder, chunk->getStartTime(), startIdx);
|
|
|
|
auto numElementsToSend = chunk->numMeasurements() - startIdx;
|
|
|
|
encoder.sendArray(chunk->getDataPointer() + startIdx, numElementsToSend);
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
void saveToFileSystem()
|
|
|
|
{
|
|
|
|
// todo: check this! free doesn't mean that the file writing actually works ok
|
|
|
|
// use error codes of write instead? anyway: test it!
|
|
|
|
deleteUntilBytesFree(CONFIG_SESSION_MAX_SIZE);
|
|
|
|
|
|
|
|
String filename = String(CONFIG_DATA_PATH) + "/" + String(chunk->getStartTime());
|
|
|
|
if (portablefs::exists(filename.c_str()))
|
|
|
|
{
|
|
|
|
auto file = portablefs::open(filename.c_str(), "a");
|
2020-06-23 21:35:28 +02:00
|
|
|
file.seek(0, SeekSet);
|
|
|
|
StreamingMsgPackEncoder<portablefs::File> encoder(&file);
|
|
|
|
chunk->sendHeader(encoder, chunk->getStartTime(), 0);
|
|
|
|
|
2020-06-21 11:10:45 +02:00
|
|
|
file.seek(0, SeekEnd);
|
|
|
|
size_t existingMeasurements = (file.size() - ChunkT::valueOffset()) / sizeof(Measurement_T);
|
|
|
|
Serial.printf("Incremental save, existing %d\n", existingMeasurements);
|
|
|
|
size_t measurementsToWrite = chunk->numMeasurements() - existingMeasurements;
|
|
|
|
Measurement_T *startPtr = chunk->getDataPointer() + existingMeasurements;
|
|
|
|
file.write((uint8_t *)(startPtr), measurementsToWrite * sizeof(Measurement_T));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
Serial.println("First save");
|
|
|
|
auto file = portablefs::open(filename.c_str(), "w");
|
|
|
|
StreamingMsgPackEncoder<portablefs::File> encoder(&file);
|
|
|
|
chunk->serialize(encoder);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void deleteUntilBytesFree(size_t requiredSpace)
|
|
|
|
{
|
|
|
|
auto freeBytes = portablefs::totalBytes() - portablefs::usedBytes();
|
|
|
|
while (freeBytes < requiredSpace)
|
|
|
|
{
|
|
|
|
uint32_t nextSessionToDelete = 0;
|
|
|
|
|
|
|
|
auto dir = portablefs::openDir(CONFIG_DATA_PATH);
|
|
|
|
String filenameToDelete;
|
|
|
|
while (dir.next())
|
|
|
|
{
|
|
|
|
if (dir.isFile())
|
|
|
|
{
|
|
|
|
const auto fileName = dir.fileName();
|
|
|
|
const auto fileNameWithoutDir = fileName.substring(fileName.lastIndexOf("/") + 1);
|
|
|
|
auto sessionId = fileNameWithoutDir.toInt();
|
|
|
|
if (sessionId < nextSessionToDelete)
|
|
|
|
{
|
|
|
|
nextSessionToDelete = sessionId;
|
|
|
|
filenameToDelete = dir.fileName();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
assert(nextSessionToDelete > 0);
|
|
|
|
Serial.printf("Removing old session %s to make space\n", filenameToDelete.c_str());
|
|
|
|
portablefs::remove(filenameToDelete.c_str());
|
|
|
|
auto newFreeBytes = portablefs::totalBytes() - portablefs::usedBytes();
|
|
|
|
assert(newFreeBytes > freeBytes);
|
|
|
|
freeBytes = newFreeBytes;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ChunkT *chunk;
|
|
|
|
uint32_t saveInterval_;
|
|
|
|
};
|