diff --git a/AnkiDroid/src/main/java/com/ichi2/anki/NotificationDatastore.kt b/AnkiDroid/src/main/java/com/ichi2/anki/NotificationDatastore.kt new file mode 100644 index 000000000000..d4404b738af7 --- /dev/null +++ b/AnkiDroid/src/main/java/com/ichi2/anki/NotificationDatastore.kt @@ -0,0 +1,270 @@ +/* + * Copyright (c) 2022 Prateek Singh + * + * This program is free software; you can redistribute it and/or modify it under + * the terms of the GNU General Public License as published by the Free Software + * Foundation; either version 3 of the License, or (at your option) any later + * version. + * + * This program 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 General Public License for more details. + * + * You should have received a copy of the GNU General Public License along with + * this program. If not, see . + */ + +package com.ichi2.anki + +import android.content.Context +import androidx.datastore.core.DataStore +import androidx.datastore.preferences.core.Preferences +import androidx.datastore.preferences.core.edit +import androidx.datastore.preferences.core.intPreferencesKey +import androidx.datastore.preferences.core.stringPreferencesKey +import androidx.datastore.preferences.preferencesDataStore +import com.fasterxml.jackson.core.JacksonException +import com.fasterxml.jackson.databind.ObjectMapper +import com.ichi2.anki.NotificationDatastore.Companion.getInstance +import com.ichi2.anki.model.DeckNotification +import com.ichi2.libanki.DeckId +import kotlinx.coroutines.CoroutineScope +import kotlinx.coroutines.Dispatchers +import kotlinx.coroutines.flow.firstOrNull +import kotlinx.coroutines.launch +import org.json.JSONObject +import timber.log.Timber +import java.util.* + +/** + * Time at which deck notification will trigger. Stores time in millisecond in EPOCH format. + */ +/* +* We use TimeOfNotification as key of json object. So it must be string. +*/ +typealias TimeOfNotification = String + +/** + * DeckIds which is going to trigger at particular time. + */ +typealias SimultaneouslyTriggeredDeckIds = HashSet + +/** + * Indicates all notifications that we must eventually trigger. If a notification time is in the past, we must do it as soon as possible, otherwise we must trigger the notification later. + * Implemented as map from time to set of deck ids triggered at this time + */ +typealias NotificationTodo = TreeMap + +/** + * Notification which is going to trigger next. + * */ +fun NotificationTodo.earliestNotifications(): MutableMap.MutableEntry? = + firstEntry() + +/** + * Compare the [TimeOfNotification]. It avoid the failure of comparing 2 [TimeOfNotification] when timestamp gets an extra digit because we'll get new timestamps + * starting with a 1 that will be a number greater than previous numbers but alphabetically shorter + * */ +fun TimeOfNotification.compare(timeOfNotification: TimeOfNotification) = + this.toLong().compareTo(timeOfNotification.toLong()) + +/** + * Default object for [NotificationTodo]. This object adds comparator for string values. + * */ +fun NotificationTodoObject() = NotificationTodo { timeOfNotification1, timeOfNotification2 -> + timeOfNotification1.compare(timeOfNotification2) +} + +/** + * Stores the scheduled notification details + * This is a singleton class, use [getInstance] + * */ +class NotificationDatastore private constructor(val context: Context) { + + /** + * 1. Stores the String in Notification Datastore. + * 2. It stores the data asynchronously. + * 3. Calling this function guarantees to store value in database. + * @param key The Key of value. Used in fetching the data. + * @param value Value that needs to be stored (VALUE MUST BE STRING). + * */ + suspend fun putStringAsync(key: String, value: String) { + val dataStoreKey = stringPreferencesKey(key) + context.notificationDatastore.edit { metaData -> + metaData[dataStoreKey] = value + } + } + + /** + * Stores the String in Notification Datastore + * It stores the data synchronously. It will create Coroutine [Dispatchers.IO] Scope Internally. + * @param key The Key of value. Used in fetching the data. + * @param value Value that needs to be stored (VALUE MUST BE STRING). + * */ + fun putStringSync(key: String, value: String) { + CoroutineScope(Dispatchers.IO).launch { + putStringAsync(key, value) + } + } + + /** + * Fetches the String value from Datastore. + * @prams The Key of deck whose data you want to fetch. + * @return Value associated to `key` by the last call to [putStringSync], [putStringAsync], or [default] if none + * */ + suspend fun getString(key: String, default: String): String { + val dataStoreKey = stringPreferencesKey(key) + return context.notificationDatastore.data.firstOrNull()?.let { + it[dataStoreKey] + } ?: default + } + + /** + * Stores the Integer in Notification Datastore + * It stores the data asynchronously. + * Calling this function guarantees to store value in database. + * @param key The Key of value. Created while storing the data. + * @param value Value that needs to be stored (VALUE MUST BE INTEGER). + * */ + suspend fun putIntAsync(key: String, value: Int) { + val dataStoreKey = intPreferencesKey(key) + context.notificationDatastore.edit { metaDataEditor -> + metaDataEditor[dataStoreKey] = value + } + } + + /** + * Stores the Integer in Notification Datastore + * It stores the data synchronously. It will create Coroutine [Dispatchers.IO] Scope Internally. + * @param key The Key of value. Created while storing the data. + * @param value Value that needs to be stored (VALUE MUST BE INTEGER). + * */ + fun putIntSync(key: String, value: Int) { + CoroutineScope(Dispatchers.IO).launch { + putIntAsync(key, value) + } + } + + /** + * Fetches the Integer value from Datastore. + * @prams The Key of deck whose data you want to fetch. + * @return Value associated to `key` by the last call to [putIntSync], [putIntAsync], or [default] if none + * */ + suspend fun getInt(key: String, default: Int): Int { + val dataStoreKey = intPreferencesKey(key) + return context.notificationDatastore.data.firstOrNull()?.let { + it[dataStoreKey] + } ?: default + } + + /** + * Stores the Map of time and list of deck ids to Datastore + * It stores the data asynchronously. + * */ + suspend fun setTimeDeckData(data: Map>) { + val dataStoreKey = stringPreferencesKey("TIME_DECK_DATA") + val jsonObj = JSONObject(data) + context.notificationDatastore.edit { metaData -> + metaData[dataStoreKey] = jsonObj.toString() + } + } + + /** + * Fetches the Map of time and list of deck ids from Datastore. + * @return The current AllTimeAndDecksMap + * */ + /* + * We actually are not blocking the thread. This method throws an exception. It will not create problem for us. + * */ + @Suppress("UNCHECKED_CAST", "BlockingMethodInNonBlockingContext") + suspend fun getTimeDeckData(): NotificationTodo? { + val datastoreKey = stringPreferencesKey("TIME_DECK_DATA") + return context.notificationDatastore.data.firstOrNull()?.let { + try { + objectMapper.readValue( + it[datastoreKey], + TreeMap::class.java + ) as NotificationTodo + } catch (ex: JacksonException) { + Timber.d(ex.cause) + null + } + } + } + + /** + * Stores the details of the [notification] scheduling of deck [did] + * @return operation successful of not. + * */ + suspend fun setDeckSchedData(did: DeckId, notification: DeckNotification): Boolean { + val dataStoreKey = stringPreferencesKey(did.toString()) + return runCatching { + val json = objectMapper.writeValueAsString(notification) + context.notificationDatastore.edit { metaData -> + metaData[dataStoreKey] = json + } + }.isSuccess + } + + /** + * Fetches the details of particular deck scheduling. + * @return Deck Notification model for particular deck. + * */ + /* + * We actually are not blocking the thread. This method throws an exception. It will not create problem for us. + * TODO: unit test that : + * * if there is no preference at all, we return null + * * if there is a preference without entry for this key we return null + * * if there is a preference whose entry for this key can't be cast to DeckNotification, throw + * * if there is a preference with entry for this key that can be cast, we get expected notification + */ + @Suppress("BlockingMethodInNonBlockingContext") + suspend fun getDeckSchedData(did: DeckId): DeckNotification? { + val datastoreKey = stringPreferencesKey(did.toString()) + return context.notificationDatastore.data.firstOrNull()?.let { + try { + if (!it.contains(datastoreKey)) { + // Deck data not stored. + return null + } + objectMapper.readValue( + it[datastoreKey], + DeckNotification::class.java + ) + } catch (ex: Exception) { + // Let the exception throw + CrashReportService.sendExceptionReport( + ex, + "Notification Datastore-getDeckSchedData", + "Exception Occurred during fetching of data." + ) + throw Exception("Unable to find schedule data of given deck id: $did", ex) + } + } + } + + companion object { + private val objectMapper = ObjectMapper() + private lateinit var INSTANCE: NotificationDatastore + private val Context.notificationDatastore: DataStore by preferencesDataStore("NotificationDatastore") + private fun instanceInitializedOr( + context: Context, + block: (context: Context) -> NotificationDatastore + ) = if (this::INSTANCE.isInitialized) INSTANCE else block(context) + + /** + * Thread safe. We're using this pattern because we need a Context to get an instance. + * @return The singleton NotificationDatastore + */ + fun getInstance(context: Context) = instanceInitializedOr(context) { + synchronized(this) { + // Check again whether [INSTANCE] is initialized because it could have been initialized while waiting for synchronization. + instanceInitializedOr(context) { + NotificationDatastore(context).also { + INSTANCE = it + } + } + } + } + } +} diff --git a/AnkiDroid/src/main/java/com/ichi2/anki/model/DeckNotification.kt b/AnkiDroid/src/main/java/com/ichi2/anki/model/DeckNotification.kt new file mode 100644 index 000000000000..88d08fbb00ca --- /dev/null +++ b/AnkiDroid/src/main/java/com/ichi2/anki/model/DeckNotification.kt @@ -0,0 +1,81 @@ +/* + * Copyright (c) 2022 Prateek Singh + * + * This program is free software; you can redistribute it and/or modify it under + * the terms of the GNU General Public License as published by the Free Software + * Foundation; either version 3 of the License, or (at your option) any later + * version. + * + * This program 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 General Public License for more details. + * + * You should have received a copy of the GNU General Public License along with + * this program. If not, see . + */ + +package com.ichi2.anki.model + +import com.ichi2.libanki.DeckId +import com.ichi2.libanki.Decks + +/** + * Notification details of particular deck. + * */ +data class DeckNotification( + val enabled: Boolean, + val did: DeckId, + val notificationTime: NotificationTime, + val deckPreference: DeckPreference, + val includeSubdecks: Boolean +) { + + // Empty constructor. Required for jackson to serialize. + constructor() : this( + false, + Decks.NOT_FOUND_DECK_ID, + NotificationTime(), + DeckPreference(), + false + ) + + /** + * Stores the time at which the deck notification is going to trigger. + * */ + data class NotificationTime( + val hour: Int, + val minutes: Int + ) { + + // Empty constructor. Required for jackson to serialize. + constructor() : this( + 0, + 0 + ) + } + + /** + * Store the preference of deck which is checked while triggering the Deck Notification. + * */ + data class DeckPreference( + val number: Int, + val valueType: PreferenceType, + ) { + + // Empty constructor. Required for jackson to serialize. + constructor() : this( + 10, + PreferenceType.CARDS + ) + + /** + * Defines the type of value stored in [number] in [DeckPreference]. + * if [valueType] is [CARDS] then [number] represents the card studied and + * if [valueType] is [MINUTES] then [number] represents the minutes studied. + * */ + enum class PreferenceType(value: String) { + CARDS("CARDS"), + MINUTES("MINUTES") + } + } +}