Xiaomi: Implement daily activity parsing

This commit is contained in:
José Rebelo 2023-10-19 23:33:46 +01:00
parent 5dd746f2d6
commit aead518e05
11 changed files with 329 additions and 29 deletions

View File

@ -45,7 +45,7 @@ public class GBDaoGenerator {
public static void main(String[] args) throws Exception { public static void main(String[] args) throws Exception {
final Schema schema = new Schema(62, MAIN_PACKAGE + ".entities"); final Schema schema = new Schema(63, MAIN_PACKAGE + ".entities");
Entity userAttributes = addUserAttributes(schema); Entity userAttributes = addUserAttributes(schema);
Entity user = addUserInfo(schema, userAttributes); Entity user = addUserInfo(schema, userAttributes);
@ -70,6 +70,7 @@ public class GBDaoGenerator {
addHuamiHeartRateRestingSample(schema, user, device); addHuamiHeartRateRestingSample(schema, user, device);
addHuamiPaiSample(schema, user, device); addHuamiPaiSample(schema, user, device);
addHuamiSleepRespiratoryRateSample(schema, user, device); addHuamiSleepRespiratoryRateSample(schema, user, device);
addXiaomiActivitySample(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);
@ -324,6 +325,19 @@ public class GBDaoGenerator {
return sleepRespiratoryRateSample; return sleepRespiratoryRateSample;
} }
private static Entity addXiaomiActivitySample(Schema schema, Entity user, Entity device) {
Entity activitySample = addEntity(schema, "XiaomiActivitySample");
addCommonActivitySampleProperties("AbstractActivitySample", activitySample, user, device);
activitySample.implementsSerializable();
activitySample.addIntProperty(SAMPLE_RAW_INTENSITY).notNull().codeBeforeGetterAndSetter(OVERRIDE);
activitySample.addIntProperty(SAMPLE_STEPS).notNull().codeBeforeGetterAndSetter(OVERRIDE);
activitySample.addIntProperty(SAMPLE_RAW_KIND).notNull().codeBeforeGetterAndSetter(OVERRIDE);
addHeartRateProperties(activitySample);
activitySample.addIntProperty("stress");
activitySample.addIntProperty("spo2");
return activitySample;
}
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);
} }

View File

@ -0,0 +1,93 @@
/* 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;
import androidx.annotation.NonNull;
import androidx.annotation.Nullable;
import java.util.ArrayList;
import java.util.List;
import nodomain.freeyourgadget.gadgetbridge.entities.AbstractActivitySample;
import nodomain.freeyourgadget.gadgetbridge.entities.DaoSession;
import nodomain.freeyourgadget.gadgetbridge.impl.GBDevice;
import nodomain.freeyourgadget.gadgetbridge.model.TimeSample;
/**
* Wraps a {@link SampleProvider} into a {@link TimeSampleProvider}.
*/
public abstract class AbstractSampleToTimeSampleProvider<T extends TimeSample, S extends AbstractActivitySample> implements TimeSampleProvider<T> {
private final SampleProvider<S> mSampleProvider;
private final DaoSession mSession;
private final GBDevice mDevice;
protected AbstractSampleToTimeSampleProvider(final SampleProvider<S> sampleProvider, final GBDevice device, final DaoSession session) {
mSampleProvider = sampleProvider;
mDevice = device;
mSession = session;
}
protected abstract T convertSample(final S sample);
public GBDevice getDevice() {
return mDevice;
}
public DaoSession getSession() {
return mSession;
}
@NonNull
@Override
public List<T> getAllSamples(final long timestampFrom, final long timestampTo) {
final List<S> upstreamSamples = mSampleProvider.getAllActivitySamples((int) (timestampFrom / 1000L), (int) (timestampTo / 1000L));
final List<T> ret = new ArrayList<>();
for (final S sample : upstreamSamples) {
ret.add(convertSample(sample));
}
return ret;
}
@Override
public void addSample(final T timeSample) {
throw new UnsupportedOperationException("This sample provider is read-only!");
}
@Override
public void addSamples(final List<T> timeSamples) {
throw new UnsupportedOperationException("This sample provider is read-only!");
}
@Override
public T createSample() {
throw new UnsupportedOperationException("This sample provider is read-only!");
}
@Nullable
@Override
public T getLatestSample() {
final S latestSample = mSampleProvider.getLatestActivitySample();
return convertSample(latestSample);
}
@Nullable
@Override
public T getFirstSample() {
final S firstSample = mSampleProvider.getFirstActivitySample();
return convertSample(firstSample);
}
}

View File

@ -70,8 +70,7 @@ public abstract class XiaomiCoordinator extends AbstractBLEDeviceCoordinator {
@Override @Override
public TimeSampleProvider<? extends StressSample> getStressSampleProvider(final GBDevice device, final DaoSession session) { public TimeSampleProvider<? extends StressSample> getStressSampleProvider(final GBDevice device, final DaoSession session) {
// TODO XiaomiStressSampleProvider return new XiaomiStressSampleProvider(device, session);
return super.getStressSampleProvider(device, session);
} }
@Override @Override
@ -182,7 +181,7 @@ public abstract class XiaomiCoordinator extends AbstractBLEDeviceCoordinator {
@Override @Override
public boolean supportsPai() { public boolean supportsPai() {
// TODO does it? // TODO does it?
return true; return false;
} }
@Override @Override

View File

@ -23,37 +23,36 @@ import de.greenrobot.dao.AbstractDao;
import de.greenrobot.dao.Property; import de.greenrobot.dao.Property;
import nodomain.freeyourgadget.gadgetbridge.devices.AbstractSampleProvider; import nodomain.freeyourgadget.gadgetbridge.devices.AbstractSampleProvider;
import nodomain.freeyourgadget.gadgetbridge.entities.DaoSession; import nodomain.freeyourgadget.gadgetbridge.entities.DaoSession;
import nodomain.freeyourgadget.gadgetbridge.entities.HuamiExtendedActivitySample; import nodomain.freeyourgadget.gadgetbridge.entities.XiaomiActivitySample;
import nodomain.freeyourgadget.gadgetbridge.entities.HuamiExtendedActivitySampleDao; import nodomain.freeyourgadget.gadgetbridge.entities.XiaomiActivitySampleDao;
import nodomain.freeyourgadget.gadgetbridge.impl.GBDevice; import nodomain.freeyourgadget.gadgetbridge.impl.GBDevice;
// TODO s/HuamiExtendedActivitySample/XiaomiActivitySample/g public class XiaomiSampleProvider extends AbstractSampleProvider<XiaomiActivitySample> {
public class XiaomiSampleProvider extends AbstractSampleProvider<HuamiExtendedActivitySample> {
public XiaomiSampleProvider(final GBDevice device, final DaoSession session) { public XiaomiSampleProvider(final GBDevice device, final DaoSession session) {
super(device, session); super(device, session);
} }
@Override @Override
public AbstractDao<HuamiExtendedActivitySample, ?> getSampleDao() { public AbstractDao<XiaomiActivitySample, ?> getSampleDao() {
return getSession().getHuamiExtendedActivitySampleDao(); return getSession().getXiaomiActivitySampleDao();
} }
@Nullable @Nullable
@Override @Override
protected Property getRawKindSampleProperty() { protected Property getRawKindSampleProperty() {
return HuamiExtendedActivitySampleDao.Properties.RawKind; return XiaomiActivitySampleDao.Properties.RawKind;
} }
@NonNull @NonNull
@Override @Override
protected Property getTimestampSampleProperty() { protected Property getTimestampSampleProperty() {
return HuamiExtendedActivitySampleDao.Properties.Timestamp; return XiaomiActivitySampleDao.Properties.Timestamp;
} }
@NonNull @NonNull
@Override @Override
protected Property getDeviceIdentifierSampleProperty() { protected Property getDeviceIdentifierSampleProperty() {
return HuamiExtendedActivitySampleDao.Properties.DeviceId; return XiaomiActivitySampleDao.Properties.DeviceId;
} }
@Override @Override
@ -64,16 +63,18 @@ public class XiaomiSampleProvider extends AbstractSampleProvider<HuamiExtendedAc
@Override @Override
public int toRawActivityKind(final int activityKind) { public int toRawActivityKind(final int activityKind) {
// TODO
return activityKind; return activityKind;
} }
@Override @Override
public float normalizeIntensity(final int rawIntensity) { public float normalizeIntensity(final int rawIntensity) {
// TODO
return rawIntensity; return rawIntensity;
} }
@Override @Override
public HuamiExtendedActivitySample createActivitySample() { public XiaomiActivitySample createActivitySample() {
return new HuamiExtendedActivitySample(); return new XiaomiActivitySample();
} }
} }

View File

@ -0,0 +1,62 @@
/* 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.xiaomi;
import nodomain.freeyourgadget.gadgetbridge.devices.AbstractSampleToTimeSampleProvider;
import nodomain.freeyourgadget.gadgetbridge.entities.DaoSession;
import nodomain.freeyourgadget.gadgetbridge.entities.XiaomiActivitySample;
import nodomain.freeyourgadget.gadgetbridge.impl.GBDevice;
import nodomain.freeyourgadget.gadgetbridge.model.StressSample;
public class XiaomiStressSampleProvider extends AbstractSampleToTimeSampleProvider<StressSample, XiaomiActivitySample> {
public XiaomiStressSampleProvider(final GBDevice device, final DaoSession session) {
super(new XiaomiSampleProvider(device, session), device, session);
}
@Override
protected StressSample convertSample(final XiaomiActivitySample sample) {
return new XiaomiStressSample(
sample.getTimestamp() * 1000L,
sample.getStress()
);
}
protected static class XiaomiStressSample implements StressSample {
private final long timestamp;
private final int stress;
public XiaomiStressSample(final long timestamp, final int stress) {
this.timestamp = timestamp;
this.stress = stress;
}
@Override
public long getTimestamp() {
return timestamp;
}
@Override
public Type getType() {
return Type.UNKNOWN;
}
@Override
public int getStress() {
return stress;
}
}
}

View File

@ -20,6 +20,7 @@ public interface StressSample extends TimeSample {
enum Type { enum Type {
MANUAL(0), MANUAL(0),
AUTOMATIC(1), AUTOMATIC(1),
UNKNOWN(2),
; ;
private final int num; private final int num;

View File

@ -117,7 +117,7 @@ public class XiaomiActivityFileFetcher {
final XiaomiActivityParser activityParser = XiaomiActivityParser.create(fileId); final XiaomiActivityParser activityParser = XiaomiActivityParser.create(fileId);
if (activityParser == null) { if (activityParser == null) {
LOG.warn("Failed to find activity parser for {}", fileId); LOG.warn("Failed to find parser for {}", fileId);
triggerNextFetch(); triggerNextFetch();
return; return;
} }

View File

@ -114,9 +114,9 @@ public class XiaomiActivityFileId {
return getClass().getSimpleName() + "{" + return getClass().getSimpleName() + "{" +
"timestamp=" + DateTimeUtils.formatIso8601(timestamp) + "timestamp=" + DateTimeUtils.formatIso8601(timestamp) +
", timezone=" + timezone + ", timezone=" + timezone +
", type=" + (typeName != Type.UNKNOWN ? typeName : "UNKNOWN(" + type + ")") + ", type=" + (typeName + "(" + type + ")") +
", subtype=" + (subtypeName != Subtype.UNKNOWN ? subtypeName : "UNKNOWN(" + subtype + ")") + ", subtype=" + (subtypeName + "(" + subtype + ")") +
", detailType=" + (detailTypeName != DetailType.UNKNOWN ? detailTypeName : "UNKNOWN(" + detailType + ")") + ", detailType=" + (detailTypeName + "(" + detailType + ")") +
", version=" + version + ", version=" + version +
"}"; "}";
} }

View File

@ -22,6 +22,7 @@ import org.slf4j.Logger;
import org.slf4j.LoggerFactory; import org.slf4j.LoggerFactory;
import nodomain.freeyourgadget.gadgetbridge.service.devices.xiaomi.XiaomiSupport; import nodomain.freeyourgadget.gadgetbridge.service.devices.xiaomi.XiaomiSupport;
import nodomain.freeyourgadget.gadgetbridge.service.devices.xiaomi.activity.impl.DailyDetailsParser;
public abstract class XiaomiActivityParser { public abstract class XiaomiActivityParser {
private static final Logger LOG = LoggerFactory.getLogger(XiaomiActivityParser.class); private static final Logger LOG = LoggerFactory.getLogger(XiaomiActivityParser.class);
@ -46,26 +47,22 @@ public abstract class XiaomiActivityParser {
switch (fileId.getSubtype()) { switch (fileId.getSubtype()) {
case ACTIVITY_DAILY: case ACTIVITY_DAILY:
switch (fileId.getDetailType()) { if (fileId.getDetailType() == XiaomiActivityFileId.DetailType.DETAILS) {
case DETAILS: return new DailyDetailsParser();
return null;
case SUMMARY:
return null;
} }
break;
case ACTIVITY_SLEEP:
// TODO
break; break;
} }
LOG.warn("No parser for activity subtype in {}", fileId);
return null; return null;
} }
private static XiaomiActivityParser createForSports(final XiaomiActivityFileId fileId) { private static XiaomiActivityParser createForSports(final XiaomiActivityFileId fileId) {
assert fileId.getType() == XiaomiActivityFileId.Type.SPORTS; assert fileId.getType() == XiaomiActivityFileId.Type.SPORTS;
LOG.warn("No parser for sports subtype in {}", fileId);
return null; return null;
} }
} }

View File

@ -0,0 +1,132 @@
/* 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.service.devices.xiaomi.activity.impl;
import android.widget.Toast;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.nio.ByteBuffer;
import java.nio.ByteOrder;
import java.util.ArrayList;
import java.util.Calendar;
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.DeviceCoordinator;
import nodomain.freeyourgadget.gadgetbridge.devices.SampleProvider;
import nodomain.freeyourgadget.gadgetbridge.entities.DaoSession;
import nodomain.freeyourgadget.gadgetbridge.entities.Device;
import nodomain.freeyourgadget.gadgetbridge.entities.User;
import nodomain.freeyourgadget.gadgetbridge.entities.XiaomiActivitySample;
import nodomain.freeyourgadget.gadgetbridge.impl.GBDevice;
import nodomain.freeyourgadget.gadgetbridge.service.devices.xiaomi.XiaomiSupport;
import nodomain.freeyourgadget.gadgetbridge.service.devices.xiaomi.activity.XiaomiActivityFileId;
import nodomain.freeyourgadget.gadgetbridge.service.devices.xiaomi.activity.XiaomiActivityParser;
import nodomain.freeyourgadget.gadgetbridge.util.GB;
public class DailyDetailsParser extends XiaomiActivityParser {
private static final Logger LOG = LoggerFactory.getLogger(DailyDetailsParser.class);
@Override
public boolean parse(final XiaomiSupport support, final XiaomiActivityFileId fileId, final byte[] bytes) {
final int version = fileId.getVersion();
final int headerSize;
final int recordSize;
switch (version) {
case 1:
case 2:
headerSize = 4;
recordSize = 10;
break;
case 3:
headerSize = 5;
recordSize = 12;
break;
default:
LOG.warn("Unable to parse daily details version {}", fileId.getVersion());
return false;
}
final ByteBuffer buf = ByteBuffer.wrap(bytes).order(ByteOrder.LITTLE_ENDIAN);
final byte[] header = new byte[headerSize];
buf.get(header);
if ((buf.limit() - buf.position()) % recordSize != 0) {
LOG.warn("Remaining data in the buffer is not a multiple of {}", recordSize);
return false;
}
final List<XiaomiActivitySample> samples = new ArrayList<>();
while (buf.position() < buf.limit()) {
final XiaomiActivitySample sample = new XiaomiActivitySample();
sample.setSteps(buf.getShort());
final byte[] unknown1 = new byte[4];
buf.get(unknown1); // TODO intensity and kind?
sample.setHeartRate(buf.get() & 0xff);
final byte[] unknown2 = new byte[3];
buf.get(unknown2); // TODO intensity and kind?
if (version == 3) {
sample.setSpo2(buf.get() & 0xff);
sample.setStress(buf.get() & 0xff);
}
samples.add(sample);
}
// save all the samples that we got
final Calendar timestamp = Calendar.getInstance();
timestamp.setTime(fileId.getTimestamp());
try (DBHandler handler = GBApplication.acquireDB()) {
final DaoSession session = handler.getDaoSession();
final GBDevice gbDevice = support.getDevice();
final DeviceCoordinator coordinator = gbDevice.getDeviceCoordinator();
final SampleProvider<XiaomiActivitySample> sampleProvider = (SampleProvider<XiaomiActivitySample>) coordinator.getSampleProvider(gbDevice, session);
final Device device = DBHelper.getDevice(gbDevice, session);
final User user = DBHelper.getUser(session);
for (final XiaomiActivitySample sample : samples) {
sample.setDevice(device);
sample.setUser(user);
sample.setTimestamp((int) (timestamp.getTimeInMillis() / 1000));
sample.setProvider(sampleProvider);
timestamp.add(Calendar.MINUTE, 1);
}
sampleProvider.addGBActivitySamples(samples.toArray(new XiaomiActivitySample[0]));
timestamp.add(Calendar.MINUTE, -1);
return true;
} catch (final Exception e) {
GB.toast(support.getContext(), "Error saving activity samples", Toast.LENGTH_LONG, GB.ERROR);
LOG.error("Error saving activity samples", e);
return false;
}
}
}

View File

@ -44,6 +44,7 @@ import nodomain.freeyourgadget.gadgetbridge.entities.DaoSession;
import nodomain.freeyourgadget.gadgetbridge.entities.Device; import nodomain.freeyourgadget.gadgetbridge.entities.Device;
import nodomain.freeyourgadget.gadgetbridge.entities.HuamiExtendedActivitySample; import nodomain.freeyourgadget.gadgetbridge.entities.HuamiExtendedActivitySample;
import nodomain.freeyourgadget.gadgetbridge.entities.User; import nodomain.freeyourgadget.gadgetbridge.entities.User;
import nodomain.freeyourgadget.gadgetbridge.entities.XiaomiActivitySample;
import nodomain.freeyourgadget.gadgetbridge.impl.GBDevice; import nodomain.freeyourgadget.gadgetbridge.impl.GBDevice;
import nodomain.freeyourgadget.gadgetbridge.model.ActivityKind; import nodomain.freeyourgadget.gadgetbridge.model.ActivityKind;
import nodomain.freeyourgadget.gadgetbridge.model.ActivitySample; import nodomain.freeyourgadget.gadgetbridge.model.ActivitySample;
@ -543,7 +544,7 @@ public class XiaomiHealthService extends AbstractXiaomiService {
previousSteps = realTimeStats.getSteps(); previousSteps = realTimeStats.getSteps();
} }
final HuamiExtendedActivitySample sample; final XiaomiActivitySample sample;
try (final DBHandler dbHandler = GBApplication.acquireDB()) { try (final DBHandler dbHandler = GBApplication.acquireDB()) {
final DaoSession session = dbHandler.getDaoSession(); final DaoSession session = dbHandler.getDaoSession();