swimtracker-app/data_processing/DataProcessing.js

139 lines
5.8 KiB
JavaScript
Raw Normal View History

2020-06-02 17:19:09 +02:00
import DeviceHttpDataSource from './DeviceDataSource';
import { List } from 'immutable';
import { reportDeviceData } from '../state/ActionCreators';
import { PeakDetectorSimple} from './PeakDetection';
// todo: put in settings?
const NUM_MEASUREMENTS_PER_SECOND = 10;
const WINDOW_SIZE_SECS = 5;
// This is the main data processing entry point, coupling between device and redux store
// - periodically fetch data
// - feeds them to analysis, (manages analysis classes)
// - adds them to redux store
class DataProcessing {
constructor(reduxStore) {
this.store = reduxStore;
this.store.subscribe(this.onStateChange);
this.state = this.store.getState();
this.dataSource = null;
//console.log("state", this.state);
console.assert(this.state.session.running === false, "Created DataProcessing with running=True");
this.onDataSourceChanged(this.state.settings.deviceURL);
this.rawMeasurements = List();
this.sessionStartTime = 0;
this.peakDetectorSimple = new PeakDetectorSimple(this.state.settings.peakDetectorSimpleThreshold);
this.peaks = List();
}
onStateChange = () => {
const newState = this.store.getState();
if (newState.settings.deviceURL !== this.state.settings.deviceURL)
this.onDataSourceChanged(newState.settings.deviceURL);
2020-06-02 22:43:48 +02:00
if (newState.session.running !== this.state.session.running) {
this.onRunningChanged(newState.session.running, newState.settings.deviceURL);
2020-06-02 17:19:09 +02:00
};
if(newState.settings.peakDetectorSimpleThreshold !== this.state.settings.peakDetectorSimpleThreshold) {
this.peakDetectorSimple = new PeakDetectorSimple(newState.settings.peakDetectorSimpleThreshold, this.onNewPeak);
this.peaks = List(this.peakDetectorSimple.addVector(this.rawMeasurements));
};
this.state = newState;
}
onDataSourceChanged = (newDeviceURL) => {
if (this.dataSource !== null) {
this.dataSource.stop();
this.dataSource = null;
}
2020-06-02 22:43:48 +02:00
this.dataSource = new DeviceHttpDataSource(newDeviceURL + "/api/session/data", this.onNewData);
2020-06-02 17:19:09 +02:00
}
onRunningChanged = (running, deviceURL) => {
2020-06-02 22:43:48 +02:00
let req = new XMLHttpRequest();
2020-06-02 17:19:09 +02:00
if (running) {
2020-06-02 22:43:48 +02:00
console.log("Starting session", deviceURL + "/api/session/start" );
2020-06-02 17:19:09 +02:00
req.open("GET", deviceURL + "/api/session/start");
2020-06-02 22:43:48 +02:00
req.send();
2020-06-02 17:19:09 +02:00
this.dataSource.startIndex = 0;
this.dataSource.start();
} else {
2020-06-02 22:43:48 +02:00
console.log("Stopping session");
2020-06-02 17:19:09 +02:00
req.open("GET", deviceURL + "/api/session/stop");
2020-06-02 22:43:48 +02:00
req.send();
2020-06-02 17:19:09 +02:00
this.dataSource.stop();
this.dataSource.startIndex = 0;
}
}
onNewData = (data) => {
data.values;
data.sessionStartTime;
data.startIndex;
let success = false;
2020-06-02 22:43:48 +02:00
if (data.sessionStartTime == this.sessionStartTime && data.startIndex == this.rawMeasurements.size) {
2020-06-02 17:19:09 +02:00
// normal case, add received data to measurement array
2020-06-02 22:43:48 +02:00
console.log("success: normal case");
this.rawMeasurements = this.rawMeasurements.concat(List(data.values));
2020-06-02 17:19:09 +02:00
this.analyzeNewMeasurements(data.startIndex);
success = true;
}
else if (data.startIndex === 0) {
// new start
this.sessionStartTime = data.sessionStartTime;
this.rawMeasurements = List(data.values);
success = true;
2020-06-02 22:43:48 +02:00
console.log("New start", this.sessionStartTime, this.rawMeasurements.toArray());
2020-06-02 17:19:09 +02:00
} else {
2020-06-02 22:43:48 +02:00
console.log("Requery :(");
console.log("this.sessionStartTime", this.sessionStartTime);
console.log("this.rawMeasurements", this.rawMeasurements.toArray());
console.log("data", data);
2020-06-02 17:19:09 +02:00
// missed some data -> re-query
this.dataSource.startIndex = 0;
this.sessionStartTime = 0;
//console.log("Problem: got non-consequtive data. Received:", data,
// "Own state ", { startTime: this.sessionStartTime, values: this.rawMeasurements });
}
if (success) {
const analysis = this.analyzeNewMeasurements(data.startIndex);
2020-06-02 22:43:48 +02:00
const report = reportDeviceData(this.sessionStartTime, data.startIndex, this.rawMeasurements, analysis);
console.log("reporting device data", report);
this.store.dispatch(report);
2020-06-02 17:19:09 +02:00
}
}
analyzeNewMeasurements = (newDataStartIdx) => {
2020-06-02 22:43:48 +02:00
//TODO is ".toArray()" really necessary here?
const newPeaks = this.peakDetectorSimple.addVector(this.rawMeasurements.slice(newDataStartIdx).toArray());
2020-06-02 17:19:09 +02:00
this.peaks = this.peaks.concat(List(newPeaks));
2020-06-02 22:43:48 +02:00
console.log("new peaks", newPeaks, "total peaks", this.peaks.toArray());
2020-06-02 17:19:09 +02:00
const totalMomentum = this.rawMeasurements.reduce((sum, x) => sum + x, 0);
2020-06-02 22:43:48 +02:00
const peakMax = this.rawMeasurements.reduce((running, x) => Math.max(x, running), 0);
2020-06-02 17:19:09 +02:00
// windowed quantities
const windowSizeMeasurements = WINDOW_SIZE_SECS * NUM_MEASUREMENTS_PER_SECOND;
const windowedSeq = this.rawMeasurements.slice(-windowSizeMeasurements);
2020-06-02 22:43:48 +02:00
const peakMaxWindow = windowedSeq.reduce((running, x) => Math.max(x, running), 0);
2020-06-02 17:19:09 +02:00
const momentumWindow = windowedSeq.reduce((sum, x) => sum + x, 0);
return {
peaks: this.peaks,
totalTime: this.rawMeasurements.length / NUM_MEASUREMENTS_PER_SECOND,
activeTime: 0,
totalMomentum: totalMomentum,
peakFrequency: 0,
peakMax: peakMax,
// windowed quantities
momentumWindow: momentumWindow,
frequencyWindow: 0,
peakMaxWindow: peakMaxWindow,
}
};
};
export default DataProcessing;