mirror of
https://codeberg.org/Freeyourgadget/Gadgetbridge.git
synced 2025-01-26 16:41:43 +01:00
Huami: Persist PAI samples
This commit is contained in:
parent
24f78655c2
commit
a1e07b5d1b
@ -66,6 +66,7 @@ public class GBDaoGenerator {
|
|||||||
addHuamiHeartRateManualSample(schema, user, device);
|
addHuamiHeartRateManualSample(schema, user, device);
|
||||||
addHuamiHeartRateMaxSample(schema, user, device);
|
addHuamiHeartRateMaxSample(schema, user, device);
|
||||||
addHuamiHeartRateRestingSample(schema, user, device);
|
addHuamiHeartRateRestingSample(schema, user, device);
|
||||||
|
addHuamiPaiSample(schema, user, device);
|
||||||
addPebbleHealthActivitySample(schema, user, device);
|
addPebbleHealthActivitySample(schema, user, device);
|
||||||
addPebbleHealthActivityKindOverlay(schema, user, device);
|
addPebbleHealthActivityKindOverlay(schema, user, device);
|
||||||
addPebbleMisfitActivitySample(schema, user, device);
|
addPebbleMisfitActivitySample(schema, user, device);
|
||||||
@ -284,6 +285,21 @@ public class GBDaoGenerator {
|
|||||||
return hrRestingSample;
|
return hrRestingSample;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static Entity addHuamiPaiSample(Schema schema, Entity user, Entity device) {
|
||||||
|
Entity paiSample = addEntity(schema, "HuamiPaiSample");
|
||||||
|
addCommonTimeSampleProperties("AbstractPaiSample", paiSample, user, device);
|
||||||
|
paiSample.addIntProperty("utcOffset").notNull();
|
||||||
|
paiSample.addFloatProperty("paiLow").notNull().codeBeforeGetter(OVERRIDE);
|
||||||
|
paiSample.addFloatProperty("paiModerate").notNull().codeBeforeGetter(OVERRIDE);
|
||||||
|
paiSample.addFloatProperty("paiHigh").notNull().codeBeforeGetter(OVERRIDE);
|
||||||
|
paiSample.addIntProperty("timeLow").notNull().codeBeforeGetter(OVERRIDE);
|
||||||
|
paiSample.addIntProperty("timeModerate").notNull().codeBeforeGetter(OVERRIDE);
|
||||||
|
paiSample.addIntProperty("timeHigh").notNull().codeBeforeGetter(OVERRIDE);
|
||||||
|
paiSample.addFloatProperty("paiToday").notNull().codeBeforeGetter(OVERRIDE);
|
||||||
|
paiSample.addFloatProperty("paiTotal").notNull().codeBeforeGetter(OVERRIDE);
|
||||||
|
return paiSample;
|
||||||
|
}
|
||||||
|
|
||||||
private static void addHeartRateProperties(Entity activitySample) {
|
private static void addHeartRateProperties(Entity activitySample) {
|
||||||
activitySample.addIntProperty(SAMPLE_HEART_RATE).notNull().codeBeforeGetterAndSetter(OVERRIDE);
|
activitySample.addIntProperty(SAMPLE_HEART_RATE).notNull().codeBeforeGetterAndSetter(OVERRIDE);
|
||||||
}
|
}
|
||||||
|
@ -57,6 +57,7 @@ import nodomain.freeyourgadget.gadgetbridge.impl.GBDeviceCandidate;
|
|||||||
import nodomain.freeyourgadget.gadgetbridge.model.ActivitySummaryParser;
|
import nodomain.freeyourgadget.gadgetbridge.model.ActivitySummaryParser;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.model.BatteryConfig;
|
import nodomain.freeyourgadget.gadgetbridge.model.BatteryConfig;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.model.HeartRateSample;
|
import nodomain.freeyourgadget.gadgetbridge.model.HeartRateSample;
|
||||||
|
import nodomain.freeyourgadget.gadgetbridge.model.PaiSample;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.model.Spo2Sample;
|
import nodomain.freeyourgadget.gadgetbridge.model.Spo2Sample;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.model.StressSample;
|
import nodomain.freeyourgadget.gadgetbridge.model.StressSample;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.util.Prefs;
|
import nodomain.freeyourgadget.gadgetbridge.util.Prefs;
|
||||||
@ -179,6 +180,11 @@ public abstract class AbstractDeviceCoordinator implements DeviceCoordinator {
|
|||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public TimeSampleProvider<? extends PaiSample> getPaiSampleProvider(GBDevice device, DaoSession session) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@Nullable
|
@Nullable
|
||||||
public ActivitySummaryParser getActivitySummaryParser(final GBDevice device) {
|
public ActivitySummaryParser getActivitySummaryParser(final GBDevice device) {
|
||||||
@ -266,6 +272,11 @@ public abstract class AbstractDeviceCoordinator implements DeviceCoordinator {
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean supportsPai() {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean supportsAlarmSnoozing() {
|
public boolean supportsAlarmSnoozing() {
|
||||||
return false;
|
return false;
|
||||||
|
@ -42,6 +42,7 @@ import nodomain.freeyourgadget.gadgetbridge.model.ActivitySummaryParser;
|
|||||||
import nodomain.freeyourgadget.gadgetbridge.model.BatteryConfig;
|
import nodomain.freeyourgadget.gadgetbridge.model.BatteryConfig;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.model.DeviceType;
|
import nodomain.freeyourgadget.gadgetbridge.model.DeviceType;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.model.HeartRateSample;
|
import nodomain.freeyourgadget.gadgetbridge.model.HeartRateSample;
|
||||||
|
import nodomain.freeyourgadget.gadgetbridge.model.PaiSample;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.model.Spo2Sample;
|
import nodomain.freeyourgadget.gadgetbridge.model.Spo2Sample;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.model.StressSample;
|
import nodomain.freeyourgadget.gadgetbridge.model.StressSample;
|
||||||
|
|
||||||
@ -218,6 +219,12 @@ public interface DeviceCoordinator {
|
|||||||
*/
|
*/
|
||||||
boolean supportsHeartRateStats();
|
boolean supportsHeartRateStats();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns true if PAI (Personal Activity Intelligence) measurement and fetching is supported by
|
||||||
|
* the device (with this coordinator).
|
||||||
|
*/
|
||||||
|
boolean supportsPai();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns true if activity data fetching is supported AND possible at this
|
* Returns true if activity data fetching is supported AND possible at this
|
||||||
* very moment. This will consider the device state (being connected/disconnected/busy...)
|
* very moment. This will consider the device state (being connected/disconnected/busy...)
|
||||||
@ -260,6 +267,11 @@ public interface DeviceCoordinator {
|
|||||||
*/
|
*/
|
||||||
TimeSampleProvider<? extends HeartRateSample> getHeartRateManualSampleProvider(GBDevice device, DaoSession session);
|
TimeSampleProvider<? extends HeartRateSample> getHeartRateManualSampleProvider(GBDevice device, DaoSession session);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the sample provider for PAI data, for the device being supported.
|
||||||
|
*/
|
||||||
|
TimeSampleProvider<? extends PaiSample> getPaiSampleProvider(GBDevice device, DaoSession session);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns the {@link ActivitySummaryParser} for the device being supported.
|
* Returns the {@link ActivitySummaryParser} for the device being supported.
|
||||||
*
|
*
|
||||||
|
@ -127,6 +127,11 @@ public abstract class Huami2021Coordinator extends HuamiCoordinator {
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean supportsPai() {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean supportsMusicInfo() {
|
public boolean supportsMusicInfo() {
|
||||||
return true;
|
return true;
|
||||||
|
@ -165,6 +165,11 @@ public abstract class HuamiCoordinator extends AbstractBLEDeviceCoordinator {
|
|||||||
return new HuamiHeartRateManualSampleProvider(device, session);
|
return new HuamiHeartRateManualSampleProvider(device, session);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public HuamiPaiSampleProvider getPaiSampleProvider(GBDevice device, DaoSession session) {
|
||||||
|
return new HuamiPaiSampleProvider(device, session);
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public ActivitySummaryParser getActivitySummaryParser(final GBDevice device) {
|
public ActivitySummaryParser getActivitySummaryParser(final GBDevice device) {
|
||||||
return new HuamiActivitySummaryParser();
|
return new HuamiActivitySummaryParser();
|
||||||
|
@ -0,0 +1,56 @@
|
|||||||
|
/* Copyright (C) 2023 José Rebelo
|
||||||
|
|
||||||
|
This file is part of Gadgetbridge.
|
||||||
|
|
||||||
|
Gadgetbridge is free software: you can redistribute it and/or modify
|
||||||
|
it under the terms of the GNU Affero General Public License as published
|
||||||
|
by the Free Software Foundation, either version 3 of the License, or
|
||||||
|
(at your option) any later version.
|
||||||
|
|
||||||
|
Gadgetbridge is distributed in the hope that it will be useful,
|
||||||
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
GNU Affero General Public License for more details.
|
||||||
|
|
||||||
|
You should have received a copy of the GNU Affero General Public License
|
||||||
|
along with this program. If not, see <http://www.gnu.org/licenses/>. */
|
||||||
|
package nodomain.freeyourgadget.gadgetbridge.devices.huami;
|
||||||
|
|
||||||
|
import androidx.annotation.NonNull;
|
||||||
|
|
||||||
|
import de.greenrobot.dao.AbstractDao;
|
||||||
|
import de.greenrobot.dao.Property;
|
||||||
|
import nodomain.freeyourgadget.gadgetbridge.devices.AbstractTimeSampleProvider;
|
||||||
|
import nodomain.freeyourgadget.gadgetbridge.entities.DaoSession;
|
||||||
|
import nodomain.freeyourgadget.gadgetbridge.entities.HuamiPaiSample;
|
||||||
|
import nodomain.freeyourgadget.gadgetbridge.entities.HuamiPaiSampleDao;
|
||||||
|
import nodomain.freeyourgadget.gadgetbridge.impl.GBDevice;
|
||||||
|
|
||||||
|
public class HuamiPaiSampleProvider extends AbstractTimeSampleProvider<HuamiPaiSample> {
|
||||||
|
public HuamiPaiSampleProvider(final GBDevice device, final DaoSession session) {
|
||||||
|
super(device, session);
|
||||||
|
}
|
||||||
|
|
||||||
|
@NonNull
|
||||||
|
@Override
|
||||||
|
public AbstractDao<HuamiPaiSample, ?> getSampleDao() {
|
||||||
|
return getSession().getHuamiPaiSampleDao();
|
||||||
|
}
|
||||||
|
|
||||||
|
@NonNull
|
||||||
|
@Override
|
||||||
|
protected Property getTimestampSampleProperty() {
|
||||||
|
return HuamiPaiSampleDao.Properties.Timestamp;
|
||||||
|
}
|
||||||
|
|
||||||
|
@NonNull
|
||||||
|
@Override
|
||||||
|
protected Property getDeviceIdentifierSampleProperty() {
|
||||||
|
return HuamiPaiSampleDao.Properties.DeviceId;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public HuamiPaiSample createSample() {
|
||||||
|
return new HuamiPaiSample();
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,42 @@
|
|||||||
|
/* Copyright (C) 2023 José Rebelo
|
||||||
|
|
||||||
|
This file is part of Gadgetbridge.
|
||||||
|
|
||||||
|
Gadgetbridge is free software: you can redistribute it and/or modify
|
||||||
|
it under the terms of the GNU Affero General Public License as published
|
||||||
|
by the Free Software Foundation, either version 3 of the License, or
|
||||||
|
(at your option) any later version.
|
||||||
|
|
||||||
|
Gadgetbridge is distributed in the hope that it will be useful,
|
||||||
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
GNU Affero General Public License for more details.
|
||||||
|
|
||||||
|
You should have received a copy of the GNU Affero General Public License
|
||||||
|
along with this program. If not, see <http://www.gnu.org/licenses/>. */
|
||||||
|
package nodomain.freeyourgadget.gadgetbridge.entities;
|
||||||
|
|
||||||
|
import androidx.annotation.NonNull;
|
||||||
|
|
||||||
|
import nodomain.freeyourgadget.gadgetbridge.model.PaiSample;
|
||||||
|
import nodomain.freeyourgadget.gadgetbridge.util.DateTimeUtils;
|
||||||
|
|
||||||
|
public abstract class AbstractPaiSample extends AbstractTimeSample implements PaiSample {
|
||||||
|
@NonNull
|
||||||
|
@Override
|
||||||
|
public String toString() {
|
||||||
|
return getClass().getSimpleName() + "{" +
|
||||||
|
"timestamp=" + DateTimeUtils.formatDateTime(DateTimeUtils.parseTimestampMillis(getTimestamp())) +
|
||||||
|
", paiLow=" + getPaiLow() +
|
||||||
|
", paiModerate=" + getPaiModerate() +
|
||||||
|
", paiHigh=" + getPaiHigh() +
|
||||||
|
", timeLow=" + getTimeLow() +
|
||||||
|
", timeModerate=" + getTimeModerate() +
|
||||||
|
", timeHigh=" + getTimeHigh() +
|
||||||
|
", paiToday=" + getPaiToday() +
|
||||||
|
", paiTotal=" + getPaiTotal() +
|
||||||
|
", userId=" + getUserId() +
|
||||||
|
", deviceId=" + getDeviceId() +
|
||||||
|
"}";
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,35 @@
|
|||||||
|
/* Copyright (C) 2023 José Rebelo
|
||||||
|
|
||||||
|
This file is part of Gadgetbridge.
|
||||||
|
|
||||||
|
Gadgetbridge is free software: you can redistribute it and/or modify
|
||||||
|
it under the terms of the GNU Affero General Public License as published
|
||||||
|
by the Free Software Foundation, either version 3 of the License, or
|
||||||
|
(at your option) any later version.
|
||||||
|
|
||||||
|
Gadgetbridge is distributed in the hope that it will be useful,
|
||||||
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
GNU Affero General Public License for more details.
|
||||||
|
|
||||||
|
You should have received a copy of the GNU Affero General Public License
|
||||||
|
along with this program. If not, see <http://www.gnu.org/licenses/>. */
|
||||||
|
package nodomain.freeyourgadget.gadgetbridge.model;
|
||||||
|
|
||||||
|
public interface PaiSample extends TimeSample {
|
||||||
|
float getPaiLow();
|
||||||
|
|
||||||
|
float getPaiModerate();
|
||||||
|
|
||||||
|
float getPaiHigh();
|
||||||
|
|
||||||
|
int getTimeLow();
|
||||||
|
|
||||||
|
int getTimeModerate();
|
||||||
|
|
||||||
|
int getTimeHigh();
|
||||||
|
|
||||||
|
float getPaiToday();
|
||||||
|
|
||||||
|
float getPaiTotal();
|
||||||
|
}
|
@ -26,6 +26,7 @@ public class RecordedDataTypes {
|
|||||||
public static final int TYPE_SPO2 = 0x00000020;
|
public static final int TYPE_SPO2 = 0x00000020;
|
||||||
public static final int TYPE_STRESS = 0x00000040;
|
public static final int TYPE_STRESS = 0x00000040;
|
||||||
public static final int TYPE_HEART_RATE = 0x00000080;
|
public static final int TYPE_HEART_RATE = 0x00000080;
|
||||||
|
public static final int TYPE_PAI = 0x00000100;
|
||||||
|
|
||||||
public static final int TYPE_ALL = (int)0xffffffff;
|
public static final int TYPE_ALL = (int)0xffffffff;
|
||||||
}
|
}
|
||||||
|
@ -121,6 +121,7 @@ import nodomain.freeyourgadget.gadgetbridge.service.devices.huami.operations.Abs
|
|||||||
import nodomain.freeyourgadget.gadgetbridge.service.devices.huami.operations.FetchHeartRateManualOperation;
|
import nodomain.freeyourgadget.gadgetbridge.service.devices.huami.operations.FetchHeartRateManualOperation;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.service.devices.huami.operations.FetchHeartRateMaxOperation;
|
import nodomain.freeyourgadget.gadgetbridge.service.devices.huami.operations.FetchHeartRateMaxOperation;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.service.devices.huami.operations.FetchHeartRateRestingOperation;
|
import nodomain.freeyourgadget.gadgetbridge.service.devices.huami.operations.FetchHeartRateRestingOperation;
|
||||||
|
import nodomain.freeyourgadget.gadgetbridge.service.devices.huami.operations.FetchPaiOperation;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.service.devices.huami.operations.FetchSpo2NormalOperation;
|
import nodomain.freeyourgadget.gadgetbridge.service.devices.huami.operations.FetchSpo2NormalOperation;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.service.devices.huami.operations.FetchSportsSummaryOperation;
|
import nodomain.freeyourgadget.gadgetbridge.service.devices.huami.operations.FetchSportsSummaryOperation;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.service.devices.huami.operations.FetchStressAutoOperation;
|
import nodomain.freeyourgadget.gadgetbridge.service.devices.huami.operations.FetchStressAutoOperation;
|
||||||
@ -1681,6 +1682,10 @@ public abstract class HuamiSupport extends AbstractBTLEDeviceSupport implements
|
|||||||
this.fetchOperationQueue.add(new FetchHeartRateRestingOperation(this));
|
this.fetchOperationQueue.add(new FetchHeartRateRestingOperation(this));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if ((dataTypes & RecordedDataTypes.TYPE_PAI) != 0 && coordinator.supportsPai()) {
|
||||||
|
this.fetchOperationQueue.add(new FetchPaiOperation(this));
|
||||||
|
}
|
||||||
|
|
||||||
final AbstractFetchOperation nextOperation = this.fetchOperationQueue.poll();
|
final AbstractFetchOperation nextOperation = this.fetchOperationQueue.poll();
|
||||||
if (nextOperation != null) {
|
if (nextOperation != null) {
|
||||||
try {
|
try {
|
||||||
|
@ -16,15 +16,29 @@
|
|||||||
along with this program. If not, see <http://www.gnu.org/licenses/>. */
|
along with this program. If not, see <http://www.gnu.org/licenses/>. */
|
||||||
package nodomain.freeyourgadget.gadgetbridge.service.devices.huami.operations;
|
package nodomain.freeyourgadget.gadgetbridge.service.devices.huami.operations;
|
||||||
|
|
||||||
|
import android.widget.Toast;
|
||||||
|
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
|
|
||||||
import java.nio.ByteBuffer;
|
import java.nio.ByteBuffer;
|
||||||
import java.nio.ByteOrder;
|
import java.nio.ByteOrder;
|
||||||
|
import java.util.ArrayList;
|
||||||
import java.util.GregorianCalendar;
|
import java.util.GregorianCalendar;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
import nodomain.freeyourgadget.gadgetbridge.GBApplication;
|
||||||
|
import nodomain.freeyourgadget.gadgetbridge.database.DBHandler;
|
||||||
|
import nodomain.freeyourgadget.gadgetbridge.database.DBHelper;
|
||||||
|
import nodomain.freeyourgadget.gadgetbridge.devices.huami.HuamiCoordinator;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.devices.huami.HuamiService;
|
import nodomain.freeyourgadget.gadgetbridge.devices.huami.HuamiService;
|
||||||
|
import nodomain.freeyourgadget.gadgetbridge.devices.huami.HuamiPaiSampleProvider;
|
||||||
|
import nodomain.freeyourgadget.gadgetbridge.entities.DaoSession;
|
||||||
|
import nodomain.freeyourgadget.gadgetbridge.entities.Device;
|
||||||
|
import nodomain.freeyourgadget.gadgetbridge.entities.HuamiPaiSample;
|
||||||
|
import nodomain.freeyourgadget.gadgetbridge.entities.User;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.service.devices.huami.HuamiSupport;
|
import nodomain.freeyourgadget.gadgetbridge.service.devices.huami.HuamiSupport;
|
||||||
|
import nodomain.freeyourgadget.gadgetbridge.util.DeviceHelper;
|
||||||
import nodomain.freeyourgadget.gadgetbridge.util.GB;
|
import nodomain.freeyourgadget.gadgetbridge.util.GB;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -39,6 +53,8 @@ public class FetchPaiOperation extends AbstractRepeatingFetchOperation {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected boolean handleActivityData(final GregorianCalendar timestamp, final byte[] bytes) {
|
protected boolean handleActivityData(final GregorianCalendar timestamp, final byte[] bytes) {
|
||||||
|
final List<HuamiPaiSample> samples = new ArrayList<>();
|
||||||
|
|
||||||
final ByteBuffer buf = ByteBuffer.wrap(bytes).order(ByteOrder.LITTLE_ENDIAN);
|
final ByteBuffer buf = ByteBuffer.wrap(bytes).order(ByteOrder.LITTLE_ENDIAN);
|
||||||
|
|
||||||
while (buf.position() < bytes.length) {
|
while (buf.position() < bytes.length) {
|
||||||
@ -68,7 +84,7 @@ public class FetchPaiOperation extends AbstractRepeatingFetchOperation {
|
|||||||
byte[] unknown2 = new byte[39];
|
byte[] unknown2 = new byte[39];
|
||||||
buf.get(unknown2);
|
buf.get(unknown2);
|
||||||
|
|
||||||
LOG.debug(
|
LOG.trace(
|
||||||
"PAI at {} + {}: paiLow={} paiModerate={} paiHigh={} timeLow={} timeMid={} timeHigh={} paiToday={} paiTotal={} unknown1={} unknown2={}",
|
"PAI at {} + {}: paiLow={} paiModerate={} paiHigh={} timeLow={} timeMid={} timeHigh={} paiToday={} paiTotal={} unknown1={} unknown2={}",
|
||||||
timestamp.getTime(), utcOffsetInQuarterHours,
|
timestamp.getTime(), utcOffsetInQuarterHours,
|
||||||
paiLow, paiModerate, paiHigh,
|
paiLow, paiModerate, paiHigh,
|
||||||
@ -78,7 +94,43 @@ public class FetchPaiOperation extends AbstractRepeatingFetchOperation {
|
|||||||
GB.hexdump(unknown2)
|
GB.hexdump(unknown2)
|
||||||
);
|
);
|
||||||
|
|
||||||
// TODO save
|
final HuamiPaiSample sample = new HuamiPaiSample();
|
||||||
|
sample.setTimestamp(timestamp.getTimeInMillis());
|
||||||
|
sample.setUtcOffset(utcOffsetInQuarterHours * 900000);
|
||||||
|
sample.setPaiLow(paiLow);
|
||||||
|
sample.setPaiModerate(paiModerate);
|
||||||
|
sample.setPaiHigh(paiHigh);
|
||||||
|
sample.setTimeLow(timeLow);
|
||||||
|
sample.setTimeModerate(timeModerate);
|
||||||
|
sample.setTimeHigh(timeHigh);
|
||||||
|
sample.setPaiToday(paiToday);
|
||||||
|
sample.setPaiTotal(paiTotal);
|
||||||
|
samples.add(sample);
|
||||||
|
}
|
||||||
|
|
||||||
|
return persistSamples(samples);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected boolean persistSamples(final List<HuamiPaiSample> samples) {
|
||||||
|
try (DBHandler handler = GBApplication.acquireDB()) {
|
||||||
|
final DaoSession session = handler.getDaoSession();
|
||||||
|
|
||||||
|
final Device device = DBHelper.getDevice(getDevice(), session);
|
||||||
|
final User user = DBHelper.getUser(session);
|
||||||
|
|
||||||
|
final HuamiCoordinator coordinator = (HuamiCoordinator) DeviceHelper.getInstance().getCoordinator(getDevice());
|
||||||
|
final HuamiPaiSampleProvider sampleProvider = coordinator.getPaiSampleProvider(getDevice(), session);
|
||||||
|
|
||||||
|
for (final HuamiPaiSample sample : samples) {
|
||||||
|
sample.setDevice(device);
|
||||||
|
sample.setUser(user);
|
||||||
|
}
|
||||||
|
|
||||||
|
LOG.debug("Will persist {} pai samples", samples.size());
|
||||||
|
sampleProvider.addSamples(samples);
|
||||||
|
} catch (final Exception e) {
|
||||||
|
GB.toast(getContext(), "Error saving pai samples", Toast.LENGTH_LONG, GB.ERROR, e);
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
|
Loading…
Reference in New Issue
Block a user