-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathstat_records.hpp
305 lines (233 loc) · 7.54 KB
/
stat_records.hpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
// SPDX-FileCopyrightText: 2020-2021 Ivanov Ivan
// SPDX-License-Identifier: GPL-3.0-or-later
#ifndef _STAT_RECORDS_
#define _STAT_RECORDS_
#include <Arduino.h>
//#include "FS.h"
//#include "SPIFFS.h"
#include "LITTLEFS.h"
//#include "display.hpp"
#include <TimeLib.h>
#include "i2c_max30100.h"
extern char *hrFileName;
void addDisplayStat(float val) ;
uint8_t getDisplayStat(float *buf, uint8_t bufSize) ;
struct statRecord_t{
time_t Time;
uint32_t Steps;
float HeartRate;
float SpO2;
float AmbientTempC;
float ObjectTempC;
float Pressure;
float Humidity;
float Vcc;
};
#define CACHE_DISPLAY_CNT 30
#define CACHE_RECORD_CNT 10
extern statRecord_t RTC_RECORDS[CACHE_RECORD_CNT];
extern uint8_t statRecord_cnt;
#define SPIFFS LITTLEFS
#define HOURS_IN_DAY 24
typedef enum record_idx_s{
REC_TIME_IDX = 0,
REC_VCC_IDX,
REC_STEPS_IDX,
REC_HR_IDX,
REC_AMBT_IDX,
REC_OBJT_IDX,
REC_PRESSURE_IDX,
REC_HUMIDITY_IDX,
REC_MAX_IDX
} record_idx_t;
enum skimrecord_idx_t{
// SKIMREC_TIME_IDX = TIME_IDX,
SKIMREC_VCC_IDX,
SKIMREC_STEPS_IDX,
SKIMREC_OBJT_IDX,
SKIMREC_PRESSURE_IDX,
SKIMREC_HUMIDITY_IDX,
SKIMREC_MAX_IDX
};
enum result_aggregate_t {
RESULT_AGGREGATE_AVG = 0,
RESULT_AGGREGATE_SUM,
RESULT_AGGREGATE_DIFF,
};
struct HourRecordAvg {
private:
float startValue;
float endValue;
float summ;
uint8_t cnt; // we have only 60 recors in total;
public:
HourRecordAvg();
void add(float val);
void addBack(float val);
float avg();
float diff();
float diffAsc();
float sum();
void print();
};
int buildDayFName(char *inputBuffer, int inputBuffer_size, char *dir, int16_t year, int8_t month, int8_t day);
uint32_t getArrayLogfiles(time_t lastKnownData, int32_t *resData, uint32_t resDataSize);
tmElements_t toTmElement(struct tm *tm);
class DailyRecordsAvg {
private:
HourRecordAvg val[HOURS_IN_DAY];
uint8_t cnt; // records since 0th hour
public:
DailyRecordsAvg();
int aggregateString(uint8_t _hour, float value);
int aggregateStringBack(uint8_t _hour, float value);
HourRecordAvg * hourRecord(uint8_t hourIdx);
};
struct rawForSkim_t {
DailyRecordsAvg Steps;
DailyRecordsAvg Temp;
DailyRecordsAvg Vcc;
DailyRecordsAvg Pressure;
DailyRecordsAvg Humidity;
int8_t startHourInLog;
int8_t endHourInLog;
};
struct skimData_t {
float Vcc;
uint32_t Steps;
float ObjectTempC;
float Pressure;
float Humidity;
void fill_data(float vcc , uint32_t steps , float objC, float pressure, float humidity);
bool fill_data(uint8_t version, rawForSkim_t *r4s, int _hour );
bool fill_data_v1(rawForSkim_t *r4s , int _hour);
bool fill_data_v2(rawForSkim_t *r4s , int _hour);
};
class fileRecord {
protected:
struct tm _tm;
void *_data ;
uint8_t max_cnt;
uint8_t field_cnt;
uint8_t version;
virtual void castData(char **dataPointers) ;
public:
// fileRecord();
// ~fileRecord();
bool parseLine(char *_buffer, char delimiter);
int hour() { return _tm.tm_hour; };
struct tm *tm() { return &_tm;};
};
class FileRecord: public fileRecord { // i don't want to parse and cast all data,
private:
void castData(char **dataPointers);
public:
FileRecord(uint8_t _version);
~FileRecord();
statRecord_t *data() {return (statRecord_t *) _data;};
};
class SkimFileRecord: public fileRecord { /// TODO: make parent common with FileRecord
private:
// uint8_t max_cnt = SKIMREC_MAX_IDX;
// skimData_t *_data = NULL;
void castData(char **dataPointers) ;
bool needWrite;
public:
SkimFileRecord();
~SkimFileRecord();
void setVersion(uint8_t _version);
skimData_t *get();
void set( skimData_t *s );
float getField( skimrecord_idx_t idx );
void setWritable();
bool isWritable();
bool serialize(char *str, size_t str_size);
void print();
};
struct hourStat_t {
SkimFileRecord byHours[HOURS_IN_DAY];
int8_t hoursInFile; // -1 in case not inited
int8_t hoursInTotal;
};
class SkimData {
private:
rawForSkim_t *r4s;
// char *skimDirName = "/skim";
// uint8_t logVersion;
uint8_t skimVersion;
uint8_t maxFilesCnt = 30;
uint8_t _day;
uint8_t _month;
uint16_t _year; // 64k
hourStat_t stat;
bool isCurrentHour(uint8_t _hour);
bool isToday();
bool processRaw();
void prepareFromLog(); // fill r4s // in case it happened today
bool prepareFromMemory();
bool processLine(uint8_t fVersion, char *lineString);
bool aggreggateRecord(int _hour, statRecord_t *stat_rec);
uint8_t read();
void write();
public:
SkimData(uint8_t d, uint8_t m , uint16_t y);
~SkimData();
void process();
hourStat_t *getStat();
};
typedef void (*readFileHandler)(char* fname);
#define MAX_FILES_CNT 10
#define MAX_LINE_SIZE 128
#define START_GOOD_DATE 1600000000 // some test for data Sun Sep 13 15:26:40 MSK 2020
class FileSystem {
private:
bool _canWrite = true; //false;
uint8_t logVersion;
/*
FileRecord skim_frec[HOURS_IN_DAY];
uint8_t skim_frec_cnt;
*/
File _file;
char *logDirName = "/log";
char *filenamePattern = "%04d-%02d-%02d.txt";
void printStatus();
void saveRecordsToFile();
void listDir(fs::FS &fs, const char * dirname, bool need_cat, readFileHandler r_handler );
public:
void init();
void catFile(File f);
void writeLog(statRecord_t *record);
void scanLogDir();
void rotateLogs();
void close();
int filenameDate(char *inputBuffer, int inputBuffer_size, int16_t year, int8_t month, int8_t day);
///int8_t get_values_idx(time_t _date, record_idx_t field_idx, float result_data[24] );
// int8_t get_values_idx(int32_t tm_mday, int32_t tm_month, int32_t tm_year, record_idx_t field_idx,
// result_aggregate_t result_type, Graph *graph ) ;
};
#define MAX30100_CACHE_RECORD_CNT 2
struct statPulse_t{
int32_t date;
uint16_t data[MAX30100_ARRAY_SIZE];
};
class HrLog {
private:
bool _canWrite = true; //false;
uint8_t logVersion;
File _file;
// char *logDirName = "/hr";
char *filenamePattern = "%04d-%02d-%02d.txt";
void saveRecordsToFile();
void insertHrRecord(time_t evTime, uint16_t *evData, uint32_t evDataSize);
public:
void init();
void close();
void writeLog( );
void copyFromUlpPulse( time_t date, uint32_t *raw_data, uint32_t cnt_data);
int filenameDate(char *inputBuffer, int inputBuffer_size, int16_t year, int8_t month, int8_t day);
void printPulseCashe();
void writeRecordToLog( time_t date, uint32_t *raw_data, uint32_t cnt_data) ;
uint32_t logSize();
void deleteBigFile();
};
#endif