Deprecated: Sync health data manually (Deprecated)
Sync Health Connect data by yourself.
This section is deprecated, latest synchronization functions are documented here
Sync health data manually
There are 2 types of health data Summaries and Events.
Health Data | Timezone | Oldest date of retrieval | Latest date of retrieval | Class |
---|---|---|---|---|
Summary | UTC | 29 days | Yesterday | RookSummaryManager |
Event | UTC | 29 days | Today | RookEventManager |
Create the required instances:
val rookSummaryManager = RookSummaryManager(rookConfigurationManager)
val rookEventManager = RookEventManager(rookConfigurationManager)
The rookConfigurationManager
instance should be same that you created and initialized in the previous steps.
Sync summaries
To sync any type of summary, you need to provide a date. This date cannot be the current day and cannot be older than 29 days. See the examples below:
Current date | Provided date | Is valid? |
---|---|---|
2023-01-08 | 2023-01-08 | No, the date is from today |
2023-01-08 | 2023-01-07 | Yes, the date is from yesterday |
2023-01-08 | 2022-11-01 | No, the date is older than 29 days |
2023-01-08 | 2023-01-01 | Yes, the date is 7 days old |
To get health data, call sync_xxx_summary
and provide a LocalDate instance of the day you want to retrieve the data
from.
For example, if you want to sync yesterday's sleep summary, call syncSleepSummary
.
val yesterday = LocalDate.now().minusDays(1)
val result = rookSummaryManager.syncSleepSummary(yesterday)
result.fold(
{
// Sleep summary synced successfully
},
{
// Error syncing Sleep summary
val error = when (it) {
is SDKNotInitializedException -> "SDKNotInitializedException: ${it.message}"
is UserNotInitializedException -> "UserNotInitializedException: ${it.message}"
is HealthConnectNotInstalledException -> "HealthConnectNotInstalledException: ${it.message}"
is DeviceNotSupportedException -> "DeviceNotSupportedException: ${it.message}"
is MissingPermissionsException -> "MissingPermissionsException: ${it.message}"
is TimeoutException -> "TimeoutException: ${it.message}"
is HttpRequestException -> "HttpRequestException: code: ${it.code} message: ${it.message}"
else -> "${it.message}"
}
}
)
Sync pending summaries
When you call sync_xxx_summary
, what happens it's that:
- Health data is extracted from Health Connect
- The extracted data is enqueued
- The enqueued data is uploaded to ROOK servers.
- If success the queued is cleared. Otherwise, the summary is stored.
To retry sending those stored summaries call syncPendingSummaries
val result = rookSummaryManager.syncPendingSummaries()
result.fold(
{
// Pending summaries synced successfully
},
{
// Error syncing pending summaries
val error = when (it) {
is SDKNotInitializedException -> "SDKNotInitializedException: ${it.message}"
is UserNotInitializedException -> "UserNotInitializedException: ${it.message}"
is TimeoutException -> "TimeoutException: ${it.message}"
is HttpRequestException -> "HttpRequestException: code: ${it.code} message: ${it.message}"
else -> "${it.message}"
}
}
)
Recommendations
Summaries are collections of health data from a past day, so you should not sync them more than once a day.
/**
* This function assumes that you have already checked availability and permissions.
*/
suspend fun syncSummaries() {
val yesterday = LocalDate.now().minusDays(1)
rookSummaryManager.syncSleepSummary(yesterday).fold(
{
// Sleep summary synced successfully
},
{
// Error syncing Sleep summary
}
)
// Sync other types of summaries...
// Finally you can call syncPendingSummaries to retry all failed uploads
// rookSummaryManager.syncPendingSummaries() (optional)
}
- Be cautions of syncing summaries an excessive amount of times, Health Connect has a daily usage limit and your app could be blocked for some hours or a full day.
Sync events
To sync any type of event, you need to provide a date. This date cannot be older than 29 days. See the examples below:
Current date | Provided date | Is valid? |
---|---|---|
2023-01-08 | 2023-01-08 | Yes, the date is from today |
2023-01-08 | 2023-01-07 | Yes, the date is from yesterday |
2023-01-08 | 2022-11-01 | No, the date is older than 29 days |
2023-01-08 | 2023-01-01 | Yes, the date is 7 days old |
To get health data, call sync_xxx_events
and provide a LocalDate instance of the day you want to retrieve the data
from.
For example, if you want to sync today's physical events, call syncPhysicalEvents
.
val result = rookEventManager.syncPhysicalEvents(today)
result.fold(
{
// Physical events synced successfully
},
{
// Error syncing Physical events
val error = when (it) {
is SDKNotInitializedException -> "SDKNotInitializedException: ${it.message}"
is UserNotInitializedException -> "UserNotInitializedException: ${it.message}"
is HealthConnectNotInstalledException -> "HealthConnectNotInstalledException: ${it.message}"
is DeviceNotSupportedException -> "DeviceNotSupportedException: ${it.message}"
is MissingPermissionsException -> "MissingPermissionsException: ${it.message}"
is TimeoutException -> "TimeoutException: ${it.message}"
is HttpRequestException -> "HttpRequestException: code: ${it.code} message: ${it.message}"
else -> "${it.message}"
}
}
)
Sync pending events
When you call sync_xxx_events
, what happens it's that:
- Health data is extracted from Health Connect
- The extracted data is enqueued
- The enqueued data is uploaded to ROOK servers.
- If success the queued is cleared. Otherwise, the events is stored.
To retry sending those stored events call syncPendingEvents
val result = rookEventManager.syncPendingEvents()
result.fold(
{
// Pending events synced successfully
},
{
// Error syncing pending events
val error = when (it) {
is SDKNotInitializedException -> "SDKNotInitializedException: ${it.message}"
is UserNotInitializedException -> "UserNotInitializedException: ${it.message}"
is TimeoutException -> "TimeoutException: ${it.message}"
is HttpRequestException -> "HttpRequestException: code: ${it.code} message: ${it.message}"
else -> "${it.message}"
}
}
)
Recommendations
Events are collections of health data divided in intervals of 1 hour, so you can/should sync them frequently.
/**
* This function assumes that you have already checked availability and permissions.
*/
suspend fun syncEvents() {
val today = LocalDate.now()
rookEventManager.syncPhysicalEvents(today).fold(
{
// Physical events synced successfully
},
{
// Error syncing pending events
}
)
// Sync other types of events...
// Finally you can call syncPendingEvents to retry all failed uploads
// rookEventManager.syncPendingEvents() (optional)
}
- Be cautions of syncing events an excessive amount of times, Health Connect has a daily usage limit and your app could be blocked for some hours or a full day.
Sync current day events
Current day events allow to send small amounts of health data, but they also return the data that was sent, you can use this data to update your UI without waiting for the webhook event.
Steps events
Retrieve and upload current day steps count of Health Connect.
rookEventManager.syncTodayHealthConnectStepsCount().fold(
{
when (it) {
SyncStatusWithData.RecordsNotFound -> {
// No steps events found
}
is SyncStatusWithData.Synced -> {
val steps: Int? = it.data
// Update the UI with steps count
}
}
},
{
// Handle error
}
)
Warning: This function contributes to the Health Connect rate limit, don't call it too frequently.
Calories events
Retrieve and upload current day calories count of Health Connect.
rookEventManager.getTodayCaloriesCount().fold(
{
when (it) {
SyncStatusWithData.RecordsNotFound -> {
// No calories events found
}
is SyncStatusWithData.Synced -> {
val dailyCalories: DailyCalories = it.data
// Update the UI with calories count
}
}
},
{
// Handle error
}
)
Warning: This function contributes to the Health Connect rate limit, don't call it too frequently.
Javadoc
See the complete specification of RookEventManager and RookSummaryManager in the Javadoc