mirror of
https://codeberg.org/Freeyourgadget/Gadgetbridge
synced 2025-01-17 05:07:33 +01:00
Zepp OS: Add watch app logs developer option
This commit is contained in:
parent
d38afe60c2
commit
c3c4c2ce74
@ -248,6 +248,9 @@ public class DeviceSettingsPreferenceConst {
|
||||
public static final String WIFI_HOTSPOT_STOP = "wifi_hotspot_stop";
|
||||
public static final String WIFI_HOTSPOT_STATUS = "wifi_hotspot_status";
|
||||
|
||||
public static final String PREF_APP_LOGS_START = "pref_app_logs_start";
|
||||
public static final String PREF_APP_LOGS_STOP = "pref_app_logs_stop";
|
||||
|
||||
public static final String MORNING_UPDATES_ENABLED = "morning_updates_enabled";
|
||||
public static final String MORNING_UPDATES_CATEGORIES_SORTABLE = "morning_updates_categories";
|
||||
|
||||
|
@ -51,6 +51,7 @@ import nodomain.freeyourgadget.gadgetbridge.model.ActivitySummaryParser;
|
||||
import nodomain.freeyourgadget.gadgetbridge.service.devices.huami.AbstractHuami2021FWInstallHandler;
|
||||
import nodomain.freeyourgadget.gadgetbridge.service.devices.huami.zeppos.services.ZeppOsAlexaService;
|
||||
import nodomain.freeyourgadget.gadgetbridge.service.devices.huami.zeppos.services.ZeppOsContactsService;
|
||||
import nodomain.freeyourgadget.gadgetbridge.service.devices.huami.zeppos.services.ZeppOsLogsService;
|
||||
import nodomain.freeyourgadget.gadgetbridge.service.devices.huami.zeppos.services.ZeppOsShortcutCardsService;
|
||||
import nodomain.freeyourgadget.gadgetbridge.service.devices.huami.zeppos.services.ZeppOsConfigService;
|
||||
import nodomain.freeyourgadget.gadgetbridge.service.devices.huami.HuamiLanguageType;
|
||||
@ -377,6 +378,9 @@ public abstract class Huami2021Coordinator extends HuamiCoordinator {
|
||||
// Developer
|
||||
//
|
||||
settings.add(R.xml.devicesettings_header_developer);
|
||||
if (ZeppOsLogsService.isSupported(getPrefs(device))) {
|
||||
settings.add(R.xml.devicesettings_app_logs_start_stop);
|
||||
}
|
||||
if (supportsAlexa(device)) {
|
||||
settings.add(R.xml.devicesettings_huami2021_alexa);
|
||||
}
|
||||
|
@ -362,6 +362,8 @@ public class Huami2021SettingsCustomizer extends HuamiSettingsCustomizer {
|
||||
// Notify preference changed on button click, so we can react to them
|
||||
final List<Preference> wifiFtpButtons = Arrays.asList(
|
||||
handler.findPreference(DeviceSettingsPreferenceConst.PREF_BLUETOOTH_CALLS_PAIR),
|
||||
handler.findPreference(DeviceSettingsPreferenceConst.PREF_APP_LOGS_START),
|
||||
handler.findPreference(DeviceSettingsPreferenceConst.PREF_APP_LOGS_STOP),
|
||||
handler.findPreference(DeviceSettingsPreferenceConst.WIFI_HOTSPOT_START),
|
||||
handler.findPreference(DeviceSettingsPreferenceConst.WIFI_HOTSPOT_STOP),
|
||||
handler.findPreference(DeviceSettingsPreferenceConst.FTP_SERVER_START),
|
||||
|
@ -112,6 +112,7 @@ import nodomain.freeyourgadget.gadgetbridge.service.devices.huami.zeppos.service
|
||||
import nodomain.freeyourgadget.gadgetbridge.service.devices.huami.zeppos.services.ZeppOsAppsService;
|
||||
import nodomain.freeyourgadget.gadgetbridge.service.devices.huami.zeppos.services.ZeppOsCalendarService;
|
||||
import nodomain.freeyourgadget.gadgetbridge.service.devices.huami.zeppos.services.ZeppOsCannedMessagesService;
|
||||
import nodomain.freeyourgadget.gadgetbridge.service.devices.huami.zeppos.services.ZeppOsLogsService;
|
||||
import nodomain.freeyourgadget.gadgetbridge.service.devices.huami.zeppos.services.ZeppOsNotificationService;
|
||||
import nodomain.freeyourgadget.gadgetbridge.service.devices.huami.zeppos.services.ZeppOsShortcutCardsService;
|
||||
import nodomain.freeyourgadget.gadgetbridge.service.devices.huami.zeppos.services.ZeppOsConfigService;
|
||||
@ -153,6 +154,7 @@ public abstract class Huami2021Support extends HuamiSupport implements ZeppOsFil
|
||||
private final ZeppOsNotificationService notificationService = new ZeppOsNotificationService(this, fileTransferService);
|
||||
private final ZeppOsAlexaService alexaService = new ZeppOsAlexaService(this);
|
||||
private final ZeppOsAppsService appsService = new ZeppOsAppsService(this);
|
||||
private final ZeppOsLogsService logsService = new ZeppOsLogsService(this);
|
||||
|
||||
private final Map<Short, AbstractZeppOsService> mServiceMap = new LinkedHashMap<Short, AbstractZeppOsService>() {{
|
||||
put(fileTransferService.getEndpoint(), fileTransferService);
|
||||
@ -171,6 +173,7 @@ public abstract class Huami2021Support extends HuamiSupport implements ZeppOsFil
|
||||
put(notificationService.getEndpoint(), notificationService);
|
||||
put(alexaService.getEndpoint(), alexaService);
|
||||
put(appsService.getEndpoint(), appsService);
|
||||
put(logsService.getEndpoint(), logsService);
|
||||
}};
|
||||
|
||||
public Huami2021Support() {
|
||||
|
@ -0,0 +1,229 @@
|
||||
/* 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.huami.zeppos.services;
|
||||
|
||||
import android.annotation.SuppressLint;
|
||||
import android.widget.Toast;
|
||||
|
||||
import org.slf4j.Logger;
|
||||
import org.slf4j.LoggerFactory;
|
||||
|
||||
import java.nio.ByteBuffer;
|
||||
import java.nio.ByteOrder;
|
||||
import java.nio.charset.StandardCharsets;
|
||||
import java.text.SimpleDateFormat;
|
||||
import java.util.Date;
|
||||
import java.util.HashSet;
|
||||
import java.util.Set;
|
||||
|
||||
import nodomain.freeyourgadget.gadgetbridge.activities.devicesettings.DeviceSettingsPreferenceConst;
|
||||
import nodomain.freeyourgadget.gadgetbridge.deviceevents.GBDeviceEventUpdatePreferences;
|
||||
import nodomain.freeyourgadget.gadgetbridge.service.btle.TransactionBuilder;
|
||||
import nodomain.freeyourgadget.gadgetbridge.service.devices.huami.Huami2021Support;
|
||||
import nodomain.freeyourgadget.gadgetbridge.service.devices.huami.zeppos.AbstractZeppOsService;
|
||||
import nodomain.freeyourgadget.gadgetbridge.util.GB;
|
||||
import nodomain.freeyourgadget.gadgetbridge.util.Prefs;
|
||||
import nodomain.freeyourgadget.gadgetbridge.util.StringUtils;
|
||||
|
||||
public class ZeppOsLogsService extends AbstractZeppOsService {
|
||||
private static final Logger LOG = LoggerFactory.getLogger(ZeppOsLogsService.class);
|
||||
|
||||
private static final short ENDPOINT = 0x003a;
|
||||
|
||||
public static final byte CMD_CAPABILITIES_REQUEST = 0x01;
|
||||
public static final byte CMD_CAPABILITIES_RESPONSE = 0x02;
|
||||
public static final byte CMD_LOGS_START = 0x03;
|
||||
public static final byte CMD_LOGS_START_ACK = 0x04;
|
||||
public static final byte CMD_LOGS_STOP = 0x05;
|
||||
public static final byte CMD_LOGS_STOP_ACK = 0x06;
|
||||
public static final byte CMD_LOGS_DATA = 0x07;
|
||||
public static final byte CMD_UNKNOWN_8 = 0x08;
|
||||
public static final byte CMD_UNKNOWN_9 = 0x09;
|
||||
|
||||
public static final String PREF_VERSION = "zepp_os_logs_version";
|
||||
|
||||
private String logsType;
|
||||
private final Set<Byte> sessions = new HashSet<>();
|
||||
|
||||
@SuppressLint("SimpleDateFormat")
|
||||
private final SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ss");
|
||||
|
||||
public ZeppOsLogsService(final Huami2021Support support) {
|
||||
super(support);
|
||||
}
|
||||
|
||||
@Override
|
||||
public short getEndpoint() {
|
||||
return ENDPOINT;
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean isEncrypted() {
|
||||
return false;
|
||||
}
|
||||
|
||||
@Override
|
||||
public void handlePayload(final byte[] payload) {
|
||||
switch (payload[0]) {
|
||||
case CMD_CAPABILITIES_RESPONSE:
|
||||
handleCapabilitiesResponse(payload);
|
||||
break;
|
||||
case CMD_LOGS_START_ACK:
|
||||
if (payload[1] != 1) {
|
||||
LOG.warn("Failed to start logs, status = {}", payload[1]);
|
||||
GB.toast(getContext(), "Failed to start logs", Toast.LENGTH_SHORT, GB.WARN);
|
||||
return;
|
||||
}
|
||||
final byte sessionId = payload[2];
|
||||
LOG.info("Got logs start ack, sessionId = {}", sessionId);
|
||||
GB.toast(getContext(), "App logs started", Toast.LENGTH_SHORT, GB.INFO);
|
||||
sessions.add(sessionId);
|
||||
break;
|
||||
case CMD_LOGS_STOP_ACK:
|
||||
LOG.info("Got logs stop ack, status = {}", payload[1]);
|
||||
GB.toast(getContext(), "App logs stopped", Toast.LENGTH_SHORT, GB.INFO);
|
||||
break;
|
||||
case CMD_LOGS_DATA:
|
||||
handleLogsData(payload);
|
||||
break;
|
||||
case CMD_UNKNOWN_8:
|
||||
LOG.info("Got unknown 8, replying with unknown 9");
|
||||
write("reply logs unknown 9", CMD_UNKNOWN_9);
|
||||
break;
|
||||
default:
|
||||
LOG.warn("Unexpected logs payload byte {}", String.format("0x%02x", payload[0]));
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean onSendConfiguration(final String config, final Prefs prefs) {
|
||||
switch (config) {
|
||||
case DeviceSettingsPreferenceConst.PREF_APP_LOGS_START:
|
||||
start();
|
||||
return true;
|
||||
case DeviceSettingsPreferenceConst.PREF_APP_LOGS_STOP:
|
||||
stop();
|
||||
return true;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
@Override
|
||||
public void initialize(final TransactionBuilder builder) {
|
||||
requestCapabilities(builder);
|
||||
}
|
||||
|
||||
public void requestCapabilities(final TransactionBuilder builder) {
|
||||
write(builder, CMD_CAPABILITIES_REQUEST);
|
||||
}
|
||||
|
||||
public void start() {
|
||||
if (logsType == null) {
|
||||
LOG.error("logsType is null");
|
||||
return;
|
||||
}
|
||||
|
||||
LOG.info("Starting logs");
|
||||
|
||||
final byte[] logsTypeBytes = logsType.getBytes(StandardCharsets.UTF_8);
|
||||
final ByteBuffer buf = ByteBuffer.allocate(1 + logsTypeBytes.length + 1)
|
||||
.order(ByteOrder.LITTLE_ENDIAN);
|
||||
|
||||
buf.put(CMD_LOGS_START);
|
||||
buf.put(logsTypeBytes);
|
||||
buf.put((byte) 0);
|
||||
|
||||
write("start logs", buf.array());
|
||||
}
|
||||
|
||||
public void stop() {
|
||||
LOG.info("Stopping {} log sessions", sessions.size());
|
||||
|
||||
for (final Byte session : sessions) {
|
||||
stop(session);
|
||||
}
|
||||
}
|
||||
|
||||
private void stop(final byte sessionId) {
|
||||
final ByteBuffer buf = ByteBuffer.allocate(3)
|
||||
.order(ByteOrder.LITTLE_ENDIAN);
|
||||
|
||||
buf.put(CMD_LOGS_STOP);
|
||||
buf.put((byte) 0);
|
||||
buf.put(sessionId);
|
||||
|
||||
write("stop logs session " + sessionId, buf.array());
|
||||
}
|
||||
|
||||
private void handleCapabilitiesResponse(final byte[] payload) {
|
||||
final int version = payload[1] & 0xFF;
|
||||
if (version != 1) {
|
||||
LOG.warn("Unsupported logs service version {}", version);
|
||||
return;
|
||||
}
|
||||
final byte var1 = payload[2];
|
||||
if (var1 != 1) {
|
||||
LOG.warn("Unexpected value for var1 '{}'", var1);
|
||||
}
|
||||
final byte var2 = payload[3];
|
||||
if (var2 != 0) {
|
||||
LOG.warn("Unexpected value for var2 '{}'", var2);
|
||||
}
|
||||
|
||||
logsType = StringUtils.untilNullTerminator(payload, 4);
|
||||
|
||||
LOG.info("Logs version={}, var1={}, var2={}, logsType={}", version, var1, var2, logsType);
|
||||
|
||||
getSupport().evaluateGBDeviceEvent(new GBDeviceEventUpdatePreferences(PREF_VERSION, version));
|
||||
}
|
||||
|
||||
private void handleLogsData(final byte[] payload) {
|
||||
final ByteBuffer buf = ByteBuffer.wrap(payload).order(ByteOrder.LITTLE_ENDIAN);
|
||||
buf.get(); // discard first byte
|
||||
final byte index = buf.get();
|
||||
final byte sessionId = buf.get();
|
||||
if (!sessions.contains(sessionId)) {
|
||||
LOG.warn("Got log data for unknown session {}", sessionId);
|
||||
}
|
||||
final String appIdDecimal = StringUtils.untilNullTerminator(buf);
|
||||
final byte unknown1 = buf.get();
|
||||
if (unknown1 != 0) {
|
||||
LOG.warn("Unexpected value for unknown1 = {}", unknown1);
|
||||
}
|
||||
final long timestampMillis = buf.getLong();
|
||||
final byte unknown2 = buf.get();
|
||||
if (unknown2 != 2) {
|
||||
LOG.warn("Unexpected value for unknown2 = {}", unknown2);
|
||||
}
|
||||
final String message = StringUtils.untilNullTerminator(buf);
|
||||
if (buf.position() < buf.limit()) {
|
||||
LOG.warn("There are {} log data bytes still in the buffer", (buf.limit() - buf.position()));
|
||||
}
|
||||
|
||||
LOG.info(
|
||||
"Log entry - {} [{}] - {}",
|
||||
sdf.format(new Date(timestampMillis)),
|
||||
appIdDecimal,
|
||||
message
|
||||
);
|
||||
}
|
||||
|
||||
public static boolean isSupported(final Prefs devicePrefs) {
|
||||
return devicePrefs.getInt(PREF_VERSION, 0) == 1;
|
||||
}
|
||||
}
|
@ -2121,4 +2121,8 @@
|
||||
<string name="voice_service_class_title">Voice service class</string>
|
||||
<string name="voice_service_class_summary">Full service path handling voice commands</string>
|
||||
<string name="voice_service">Voice Service</string>
|
||||
<string name="pref_app_logs_title">App logs</string>
|
||||
<string name="pref_app_logs_summary">Enable logs from watch apps</string>
|
||||
<string name="pref_app_logs_start_summary">Start logging from watch apps</string>
|
||||
<string name="pref_app_logs_stop_summary">Stop logging from watch apps</string>
|
||||
</resources>
|
||||
|
26
app/src/main/res/xml/devicesettings_app_logs_start_stop.xml
Normal file
26
app/src/main/res/xml/devicesettings_app_logs_start_stop.xml
Normal file
@ -0,0 +1,26 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<androidx.preference.PreferenceScreen xmlns:android="http://schemas.android.com/apk/res/android">
|
||||
<PreferenceScreen
|
||||
android:icon="@drawable/ic_developer_mode"
|
||||
android:key="pref_screen_app_logs_start_stop"
|
||||
android:summary="@string/pref_app_logs_summary"
|
||||
android:title="@string/pref_app_logs_title">
|
||||
|
||||
<PreferenceCategory
|
||||
android:key="pref_header_app_logs_start_stop"
|
||||
android:title="@string/pref_app_logs_title">
|
||||
|
||||
<Preference
|
||||
android:icon="@drawable/ic_play"
|
||||
android:key="pref_app_logs_start"
|
||||
android:summary="@string/pref_app_logs_start_summary"
|
||||
android:title="@string/start" />
|
||||
|
||||
<Preference
|
||||
android:icon="@drawable/ic_stop"
|
||||
android:key="pref_app_logs_stop"
|
||||
android:summary="@string/pref_app_logs_stop_summary"
|
||||
android:title="@string/stop" />
|
||||
</PreferenceCategory>
|
||||
</PreferenceScreen>
|
||||
</androidx.preference.PreferenceScreen>
|
Loading…
x
Reference in New Issue
Block a user