Files
daily-notification-plugin/docs/android-implementation-directive.md
Matthew Raymer 6aa9140f67 docs: add comprehensive alarm/notification behavior documentation
- Add platform capability reference (Android & iOS OS-level facts)
- Add plugin behavior exploration template (executable test matrices)
- Add plugin requirements & implementation directive
- Add Android-specific implementation directive with detailed test procedures
- Add exploration findings from code inspection
- Add improvement directive for refining documentation structure
- Add Android alarm persistence directive (OS capabilities)

All documents include:
- File locations, function references, and line numbers
- Detailed test procedures with ADB commands
- Cross-platform comparisons
- Implementation checklists and code examples
2025-11-21 07:30:25 +00:00

1448 lines
41 KiB
Markdown

# Android Implementation Directive: App Launch Recovery & Missed Alarm Detection
**Author**: Matthew Raymer
**Date**: November 2025
**Status**: Active Implementation Directive - Android Only
## Purpose
This directive provides **step-by-step implementation guidance** for Android-specific gaps identified in the exploration:
1. App Launch Recovery (cold/warm/force-stop)
2. Missed Alarm Detection
3. Force Stop Detection
4. Boot Receiver Missed Alarm Handling
**Reference**: See [Exploration Findings](./exploration-findings-initial.md) for gap analysis.
---
## 1. Implementation Overview
### 1.1 What Needs to Be Implemented
| Feature | Status | Priority | Location |
| ------- | ------ | -------- | -------- |
| App Launch Recovery | ❌ Missing | **High** | `DailyNotificationPlugin.kt` - `load()` method |
| Missed Alarm Detection | ⚠️ Partial | **High** | `DailyNotificationPlugin.kt` - new method |
| Force Stop Detection | ❌ Missing | **High** | `DailyNotificationPlugin.kt` - recovery logic |
| Boot Receiver Missed Alarms | ⚠️ Partial | **Medium** | `BootReceiver.kt` - `rescheduleNotifications()` |
### 1.2 Implementation Strategy
**Phase 1**: Create `ReactivationManager` class
- Centralizes all recovery logic
- Handles cold/warm/force-stop scenarios
- Detects missed alarms
- Reschedules future alarms
**Phase 2**: Integrate into `DailyNotificationPlugin.load()`
- Call recovery manager on app launch
- Run asynchronously to avoid blocking
**Phase 3**: Enhance `BootReceiver`
- Add missed alarm detection
- Handle alarms that were scheduled before reboot
---
## 2. Implementation: ReactivationManager
### 2.1 Create New File
**File**: `android/src/main/java/com/timesafari/dailynotification/ReactivationManager.kt`
**Purpose**: Centralized recovery logic for app launch scenarios
### 2.2 Class Structure
```kotlin
package com.timesafari.dailynotification
import android.app.AlarmManager
import android.content.Context
import android.os.Build
import android.util.Log
import kotlinx.coroutines.CoroutineScope
import kotlinx.coroutines.Dispatchers
import kotlinx.coroutines.launch
/**
* Manages recovery of alarms and notifications on app launch
* Handles cold start, warm start, and force stop recovery scenarios
*
* @author Matthew Raymer
* @version 1.0.0
*/
class ReactivationManager(private val context: Context) {
companion object {
private const val TAG = "DNP-REACTIVATION"
}
/**
* Perform recovery on app launch
* Detects scenario (cold/warm/force-stop) and handles accordingly
*/
fun performRecovery() {
CoroutineScope(Dispatchers.IO).launch {
try {
Log.i(TAG, "Starting app launch recovery")
// Step 1: Detect scenario
val scenario = detectScenario()
Log.i(TAG, "Detected scenario: $scenario")
// Step 2: Handle based on scenario
when (scenario) {
RecoveryScenario.FORCE_STOP -> handleForceStopRecovery()
RecoveryScenario.COLD_START -> handleColdStartRecovery()
RecoveryScenario.WARM_START -> handleWarmStartRecovery()
}
Log.i(TAG, "App launch recovery completed")
} catch (e: Exception) {
Log.e(TAG, "Error during app launch recovery", e)
}
}
}
// ... implementation methods below ...
}
```
### 2.3 Scenario Detection
```kotlin
/**
* Detect recovery scenario based on AlarmManager state vs database
*/
private suspend fun detectScenario(): RecoveryScenario {
val db = DailyNotificationDatabase.getDatabase(context)
val dbSchedules = db.scheduleDao().getEnabled()
if (dbSchedules.isEmpty()) {
// No schedules in database - normal first launch
return RecoveryScenario.COLD_START
}
// Check AlarmManager for active alarms
val activeAlarmCount = getActiveAlarmCount()
// Force Stop Detection: DB has schedules but AlarmManager has zero
if (dbSchedules.isNotEmpty() && activeAlarmCount == 0) {
return RecoveryScenario.FORCE_STOP
}
// Check if this is warm start (app was in background)
// For now, treat as cold start - can be enhanced later
return RecoveryScenario.COLD_START
}
/**
* Get count of active alarms in AlarmManager
*/
private fun getActiveAlarmCount(): Int {
return try {
val alarmManager = context.getSystemService(Context.ALARM_SERVICE) as AlarmManager
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
// Use nextAlarmClock as indicator (not perfect, but best available)
val nextAlarm = alarmManager.nextAlarmClock
if (nextAlarm != null) {
// At least one alarm exists
// Note: This doesn't give exact count, but indicates alarms exist
return 1 // Assume at least one
} else {
return 0
}
} else {
// Pre-Lollipop: Cannot query AlarmManager directly
// Assume alarms exist if we can't check
return 1
}
} catch (e: Exception) {
Log.e(TAG, "Error checking active alarm count", e)
return 0
}
}
enum class RecoveryScenario {
COLD_START,
WARM_START,
FORCE_STOP
}
```
### 2.4 Force Stop Recovery
```kotlin
/**
* Handle force stop recovery
* All alarms were cancelled, need to restore everything
*/
private suspend fun handleForceStopRecovery() {
Log.i(TAG, "Handling force stop recovery")
val db = DailyNotificationDatabase.getDatabase(context)
val dbSchedules = db.scheduleDao().getEnabled()
val currentTime = System.currentTimeMillis()
var missedCount = 0
var rescheduledCount = 0
dbSchedules.forEach { schedule ->
try {
when (schedule.kind) {
"notify" -> {
val nextRunTime = calculateNextRunTime(schedule)
if (nextRunTime < currentTime) {
// Past alarm - was missed during force stop
missedCount++
handleMissedAlarm(schedule, nextRunTime)
// Reschedule next occurrence if repeating
if (isRepeating(schedule)) {
val nextOccurrence = calculateNextOccurrence(schedule, currentTime)
rescheduleAlarm(schedule, nextOccurrence)
rescheduledCount++
}
} else {
// Future alarm - reschedule immediately
rescheduleAlarm(schedule, nextRunTime)
rescheduledCount++
}
}
"fetch" -> {
// Reschedule fetch work
rescheduleFetch(schedule)
rescheduledCount++
}
}
} catch (e: Exception) {
Log.e(TAG, "Error recovering schedule ${schedule.id}", e)
}
}
// Record recovery in history
recordRecoveryHistory(db, "force_stop", missedCount, rescheduledCount)
Log.i(TAG, "Force stop recovery complete: $missedCount missed, $rescheduledCount rescheduled")
}
```
### 2.5 Cold Start Recovery
```kotlin
/**
* Handle cold start recovery
* Check for missed alarms and reschedule future ones
*/
private suspend fun handleColdStartRecovery() {
Log.i(TAG, "Handling cold start recovery")
val db = DailyNotificationDatabase.getDatabase(context)
val currentTime = System.currentTimeMillis()
// Step 1: Detect missed alarms
val missedNotifications = db.notificationContentDao()
.getNotificationsReadyForDelivery(currentTime)
var missedCount = 0
missedNotifications.forEach { notification ->
try {
handleMissedNotification(notification)
missedCount++
} catch (e: Exception) {
Log.e(TAG, "Error handling missed notification ${notification.id}", e)
}
}
// Step 2: Reschedule future alarms from database
val dbSchedules = db.scheduleDao().getEnabled()
var rescheduledCount = 0
dbSchedules.forEach { schedule ->
try {
val nextRunTime = calculateNextRunTime(schedule)
if (nextRunTime >= currentTime) {
// Future alarm - verify it's scheduled
if (!isAlarmScheduled(schedule, nextRunTime)) {
rescheduleAlarm(schedule, nextRunTime)
rescheduledCount++
}
}
} catch (e: Exception) {
Log.e(TAG, "Error rescheduling schedule ${schedule.id}", e)
}
}
// Record recovery in history
recordRecoveryHistory(db, "cold_start", missedCount, rescheduledCount)
Log.i(TAG, "Cold start recovery complete: $missedCount missed, $rescheduledCount rescheduled")
}
```
### 2.6 Warm Start Recovery
```kotlin
/**
* Handle warm start recovery
* Verify active alarms and check for missed ones
*/
private suspend fun handleWarmStartRecovery() {
Log.i(TAG, "Handling warm start recovery")
// Similar to cold start, but lighter weight
// Just verify alarms are still scheduled
val db = DailyNotificationDatabase.getDatabase(context)
val currentTime = System.currentTimeMillis()
// Check for missed alarms (same as cold start)
val missedNotifications = db.notificationContentDao()
.getNotificationsReadyForDelivery(currentTime)
var missedCount = 0
missedNotifications.forEach { notification ->
try {
handleMissedNotification(notification)
missedCount++
} catch (e: Exception) {
Log.e(TAG, "Error handling missed notification ${notification.id}", e)
}
}
// Verify active alarms (lighter check than cold start)
val dbSchedules = db.scheduleDao().getEnabled()
var verifiedCount = 0
dbSchedules.forEach { schedule ->
try {
val nextRunTime = calculateNextRunTime(schedule)
if (nextRunTime >= currentTime && isAlarmScheduled(schedule, nextRunTime)) {
verifiedCount++
}
} catch (e: Exception) {
Log.e(TAG, "Error verifying schedule ${schedule.id}", e)
}
}
Log.i(TAG, "Warm start recovery complete: $missedCount missed, $verifiedCount verified")
}
```
### 2.7 Helper Methods
```kotlin
/**
* Handle a missed alarm/notification
*/
private suspend fun handleMissedAlarm(schedule: Schedule, scheduledTime: Long) {
val db = DailyNotificationDatabase.getDatabase(context)
// Update delivery status
// Note: This assumes NotificationContentEntity exists for this schedule
// May need to create if it doesn't exist
// Generate missed alarm event
// Option 1: Fire callback
fireMissedAlarmCallback(schedule, scheduledTime)
// Option 2: Generate missed alarm notification
// generateMissedAlarmNotification(schedule, scheduledTime)
Log.i(TAG, "Handled missed alarm: ${schedule.id} scheduled for $scheduledTime")
}
/**
* Handle a missed notification (from NotificationContentEntity)
*/
private suspend fun handleMissedNotification(notification: NotificationContentEntity) {
val db = DailyNotificationDatabase.getDatabase(context)
// Update delivery status
notification.deliveryStatus = "missed"
db.notificationContentDao().update(notification)
// Generate missed notification event
fireMissedNotificationCallback(notification)
Log.i(TAG, "Handled missed notification: ${notification.id}")
}
/**
* Reschedule an alarm
*/
private suspend fun rescheduleAlarm(schedule: Schedule, nextRunTime: Long) {
val config = UserNotificationConfig(
enabled = schedule.enabled,
schedule = schedule.cron ?: schedule.clockTime ?: "0 9 * * *",
title = "Daily Notification",
body = "Your daily update is ready",
sound = true,
vibration = true,
priority = "normal"
)
NotifyReceiver.scheduleExactNotification(context, nextRunTime, config)
// Update schedule in database
schedule.nextRunAt = nextRunTime
val db = DailyNotificationDatabase.getDatabase(context)
db.scheduleDao().update(schedule)
Log.i(TAG, "Rescheduled alarm: ${schedule.id} for $nextRunTime")
}
/**
* Check if alarm is scheduled in AlarmManager
*/
private fun isAlarmScheduled(schedule: Schedule, triggerTime: Long): Boolean {
return NotifyReceiver.isAlarmScheduled(context, triggerTime)
}
/**
* Calculate next run time from schedule
*/
private fun calculateNextRunTime(schedule: Schedule): Long {
// Use existing logic from BootReceiver or create new
// For now, simplified version
val now = System.currentTimeMillis()
return when {
schedule.cron != null -> {
// Parse cron and calculate next run
// For now, return next day at 9 AM
now + (24 * 60 * 60 * 1000L)
}
schedule.clockTime != null -> {
// Parse HH:mm and calculate next run
// For now, return next day at specified time
now + (24 * 60 * 60 * 1000L)
}
schedule.nextRunAt != null -> {
schedule.nextRunAt
}
else -> {
// Default to next day at 9 AM
now + (24 * 60 * 60 * 1000L)
}
}
}
/**
* Check if schedule is repeating
*/
private fun isRepeating(schedule: Schedule): Boolean {
// Check cron or clockTime pattern to determine if repeating
// For now, assume daily schedules are repeating
return schedule.cron != null || schedule.clockTime != null
}
/**
* Calculate next occurrence for repeating schedule
*/
private fun calculateNextOccurrence(schedule: Schedule, fromTime: Long): Long {
// Calculate next occurrence after fromTime
// For daily: add 24 hours
// For weekly: add 7 days
// etc.
return fromTime + (24 * 60 * 60 * 1000L) // Simplified: daily
}
/**
* Reschedule fetch work
*/
private suspend fun rescheduleFetch(schedule: Schedule) {
val config = ContentFetchConfig(
enabled = schedule.enabled,
schedule = schedule.cron ?: schedule.clockTime ?: "0 9 * * *",
url = null,
timeout = 30000,
retryAttempts = 3,
retryDelay = 1000,
callbacks = CallbackConfig()
)
FetchWorker.scheduleFetch(context, config)
Log.i(TAG, "Rescheduled fetch: ${schedule.id}")
}
/**
* Record recovery in history
*/
private suspend fun recordRecoveryHistory(
db: DailyNotificationDatabase,
scenario: String,
missedCount: Int,
rescheduledCount: Int
) {
try {
db.historyDao().insert(
History(
refId = "recovery_${System.currentTimeMillis()}",
kind = "recovery",
occurredAt = System.currentTimeMillis(),
outcome = "success",
diagJson = """
{
"scenario": "$scenario",
"missed_count": $missedCount,
"rescheduled_count": $rescheduledCount
}
""".trimIndent()
)
)
} catch (e: Exception) {
Log.e(TAG, "Failed to record recovery history", e)
}
}
/**
* Fire missed alarm callback
*/
private suspend fun fireMissedAlarmCallback(schedule: Schedule, scheduledTime: Long) {
// TODO: Implement callback mechanism
// This could fire a Capacitor event or call a registered callback
Log.d(TAG, "Missed alarm callback: ${schedule.id} at $scheduledTime")
}
/**
* Fire missed notification callback
*/
private suspend fun fireMissedNotificationCallback(notification: NotificationContentEntity) {
// TODO: Implement callback mechanism
Log.d(TAG, "Missed notification callback: ${notification.id}")
}
```
---
## 3. Integration: DailyNotificationPlugin
### 3.1 Update `load()` Method
**File**: `android/src/main/java/com/timesafari/dailynotification/DailyNotificationPlugin.kt`
**Location**: Line 91
**Current Code**:
```kotlin
override fun load() {
super.load()
try {
if (context == null) {
Log.e(TAG, "Context is null, cannot initialize database")
return
}
db = DailyNotificationDatabase.getDatabase(context)
Log.i(TAG, "Daily Notification Plugin loaded successfully")
} catch (e: Exception) {
Log.e(TAG, "Failed to initialize Daily Notification Plugin", e)
}
}
```
**Updated Code**:
```kotlin
override fun load() {
super.load()
try {
if (context == null) {
Log.e(TAG, "Context is null, cannot initialize database")
return
}
db = DailyNotificationDatabase.getDatabase(context)
Log.i(TAG, "Daily Notification Plugin loaded successfully")
// Perform app launch recovery
val reactivationManager = ReactivationManager(context)
reactivationManager.performRecovery()
} catch (e: Exception) {
Log.e(TAG, "Failed to initialize Daily Notification Plugin", e)
}
}
```
---
## 4. Enhancement: BootReceiver
### 4.1 Update `rescheduleNotifications()` Method
**File**: `android/src/main/java/com/timesafari/dailynotification/BootReceiver.kt`
**Location**: Line 38
**Current Issue**: Only reschedules future alarms (line 64: `if (nextRunTime > System.currentTimeMillis())`)
**Updated Code**:
```kotlin
private suspend fun rescheduleNotifications(context: Context) {
val db = DailyNotificationDatabase.getDatabase(context)
val enabledSchedules = db.scheduleDao().getEnabled()
val currentTime = System.currentTimeMillis()
Log.i(TAG, "Found ${enabledSchedules.size} enabled schedules to reschedule")
var futureRescheduled = 0
var missedDetected = 0
enabledSchedules.forEach { schedule ->
try {
when (schedule.kind) {
"fetch" -> {
// Reschedule WorkManager fetch
val config = ContentFetchConfig(
enabled = schedule.enabled,
schedule = schedule.cron ?: schedule.clockTime ?: "0 9 * * *",
url = null,
timeout = 30000,
retryAttempts = 3,
retryDelay = 1000,
callbacks = CallbackConfig()
)
FetchWorker.scheduleFetch(context, config)
Log.i(TAG, "Rescheduled fetch for schedule: ${schedule.id}")
futureRescheduled++
}
"notify" -> {
// Reschedule AlarmManager notification
val nextRunTime = calculateNextRunTime(schedule)
if (nextRunTime > currentTime) {
// Future alarm - reschedule
val config = UserNotificationConfig(
enabled = schedule.enabled,
schedule = schedule.cron ?: schedule.clockTime ?: "0 9 * * *",
title = "Daily Notification",
body = "Your daily update is ready",
sound = true,
vibration = true,
priority = "normal"
)
NotifyReceiver.scheduleExactNotification(context, nextRunTime, config)
Log.i(TAG, "Rescheduled notification for schedule: ${schedule.id}")
futureRescheduled++
} else {
// Past alarm - was missed during reboot
missedDetected++
handleMissedAlarmOnBoot(context, schedule, nextRunTime)
// Reschedule next occurrence if repeating
if (isRepeating(schedule)) {
val nextOccurrence = calculateNextOccurrence(schedule, currentTime)
val config = UserNotificationConfig(
enabled = schedule.enabled,
schedule = schedule.cron ?: schedule.clockTime ?: "0 9 * * *",
title = "Daily Notification",
body = "Your daily update is ready",
sound = true,
vibration = true,
priority = "normal"
)
NotifyReceiver.scheduleExactNotification(context, nextOccurrence, config)
Log.i(TAG, "Rescheduled next occurrence for missed alarm: ${schedule.id}")
futureRescheduled++
}
}
}
else -> {
Log.w(TAG, "Unknown schedule kind: ${schedule.kind}")
}
}
} catch (e: Exception) {
Log.e(TAG, "Failed to reschedule ${schedule.kind} for ${schedule.id}", e)
}
}
// Record boot recovery in history
try {
db.historyDao().insert(
History(
refId = "boot_recovery_${System.currentTimeMillis()}",
kind = "boot_recovery",
occurredAt = System.currentTimeMillis(),
outcome = "success",
diagJson = """
{
"schedules_rescheduled": $futureRescheduled,
"missed_detected": $missedDetected
}
""".trimIndent()
)
)
} catch (e: Exception) {
Log.e(TAG, "Failed to record boot recovery", e)
}
Log.i(TAG, "Boot recovery complete: $futureRescheduled rescheduled, $missedDetected missed")
}
/**
* Handle missed alarm detected during boot recovery
*/
private suspend fun handleMissedAlarmOnBoot(
context: Context,
schedule: Schedule,
scheduledTime: Long
) {
// Option 1: Generate missed alarm notification
// Option 2: Fire callback
// Option 3: Update delivery status in database
Log.i(TAG, "Missed alarm detected on boot: ${schedule.id} scheduled for $scheduledTime")
// TODO: Implement missed alarm handling
// This could generate a notification or fire a callback
}
```
---
## 5. Testing Requirements
### 5.1 Test Setup
**Package Name**: `com.timesafari.dailynotification`
**Test App Location**: `test-apps/android-test-app/`
**Prerequisites**:
- Android device or emulator connected via ADB
- Test app built and installed
- ADB access enabled
- Logcat monitoring capability
**Useful ADB Commands**:
```bash
# Check if device is connected
adb devices
# Install test app
cd test-apps/android-test-app
./gradlew installDebug
# Monitor logs (filter for plugin)
adb logcat | grep -E "DNP|DailyNotification|REACTIVATION|BOOT"
# Clear logs before test
adb logcat -c
# Check scheduled alarms
adb shell dumpsys alarm | grep -i timesafari
# Check WorkManager tasks
adb shell dumpsys jobscheduler | grep -i timesafari
```
---
### 5.2 Test 1: Cold Start Recovery
**Purpose**: Verify plugin detects and handles missed alarms when app is launched after process kill.
**Test Procedure**:
#### Step 1: Schedule Alarm
**Via Test App UI**:
1. Launch test app: `adb shell am start -n com.timesafari.dailynotification/.MainActivity`
2. Click "Test Notification" button
3. Schedule alarm for 4 minutes in future (test app default)
4. Note the scheduled time
**Via ADB (Alternative)**:
```bash
# Launch app
adb shell am start -n com.timesafari.dailynotification/.MainActivity
# Wait for app to load, then use UI to schedule
# Or use monkey to click button (if button ID known)
```
**Verify Alarm Scheduled**:
```bash
# Check AlarmManager
adb shell dumpsys alarm | grep -A 5 -B 5 timesafari
# Check logs for scheduling confirmation
adb logcat -d | grep "DN|SCHEDULE\|DN|ALARM"
```
#### Step 2: Kill App Process (Not Force Stop)
**Important**: Use `am kill` NOT `am force-stop`. This simulates OS memory pressure kill.
```bash
# Kill app process (simulates OS kill)
adb shell am kill com.timesafari.dailynotification
# Verify app is killed
adb shell ps | grep timesafari
# Should return nothing (process killed)
# Verify alarm is still scheduled (should be)
adb shell dumpsys alarm | grep -i timesafari
# Should still show scheduled alarm
```
#### Step 3: Wait for Alarm Time to Pass
```bash
# Wait 5-10 minutes (longer than scheduled alarm time)
# Monitor system time
adb shell date
# Or set a timer and wait
# Alarm was scheduled for 4 minutes, wait 10 minutes total
```
**During Wait Period**:
- Alarm should NOT fire (app process is killed)
- AlarmManager still has the alarm scheduled
- No notification should appear
#### Step 4: Launch App (Cold Start)
```bash
# Launch app (triggers cold start)
adb shell am start -n com.timesafari.dailynotification/.MainActivity
# Monitor logs for recovery
adb logcat -c # Clear logs first
adb logcat | grep -E "DNP-REACTIVATION|DN|RECOVERY|missed"
```
#### Step 5: Verify Recovery
**Check Logs**:
```bash
# Look for recovery logs
adb logcat -d | grep -E "DNP-REACTIVATION|COLD_START|missed"
# Expected log entries:
# - "Starting app launch recovery"
# - "Detected scenario: COLD_START"
# - "Handling cold start recovery"
# - "Handled missed notification: <id>"
# - "Cold start recovery complete: X missed, Y rescheduled"
```
**Check Database**:
```bash
# Query database for missed notifications (requires root or debug build)
# Or check via test app UI if status display shows missed alarms
```
**Expected Results**:
- ✅ Plugin detects missed alarm
- ✅ Missed alarm event/notification generated
- ✅ Future alarms rescheduled
- ✅ Recovery logged in history
**Pass/Fail Criteria**:
- **Pass**: Logs show "missed" detection and recovery completion
- **Fail**: No recovery logs, or recovery doesn't detect missed alarm
---
### 5.3 Test 2: Force Stop Recovery
**Purpose**: Verify plugin detects force stop scenario and recovers ALL alarms (missed and future).
**Test Procedure**:
#### Step 1: Schedule Multiple Alarms
**Via Test App UI**:
1. Launch app: `adb shell am start -n com.timesafari.dailynotification/.MainActivity`
2. Schedule alarm #1 for 2 minutes in future
3. Schedule alarm #2 for 5 minutes in future
4. Schedule alarm #3 for 10 minutes in future
5. Note all scheduled times
**Verify Alarms Scheduled**:
```bash
# Check AlarmManager
adb shell dumpsys alarm | grep -A 10 timesafari
# Check logs
adb logcat -d | grep "DN|SCHEDULE"
```
#### Step 2: Force Stop App
**Important**: Use `am force-stop` to trigger hard kill.
```bash
# Force stop app (hard kill)
adb shell am force-stop com.timesafari.dailynotification
# Verify app is force-stopped
adb shell ps | grep timesafari
# Should return nothing
# Verify alarms are cancelled
adb shell dumpsys alarm | grep -i timesafari
# Should return nothing (all alarms cancelled)
```
**Expected**: All alarms immediately cancelled by OS.
#### Step 3: Wait Past Scheduled Times
```bash
# Wait 15 minutes (past all scheduled times)
# Monitor time
adb shell date
# During wait: No alarms should fire (all cancelled)
```
#### Step 4: Launch App (Force Stop Recovery)
```bash
# Launch app (triggers force stop recovery)
adb shell am start -n com.timesafari.dailynotification/.MainActivity
# Monitor logs immediately
adb logcat -c
adb logcat | grep -E "DNP-REACTIVATION|FORCE_STOP|missed|rescheduled"
```
#### Step 5: Verify Force Stop Recovery
**Check Logs**:
```bash
# Look for force stop detection
adb logcat -d | grep -E "DNP-REACTIVATION|FORCE_STOP|force_stop"
# Expected log entries:
# - "Starting app launch recovery"
# - "Detected scenario: FORCE_STOP"
# - "Handling force stop recovery"
# - "Missed alarm detected: <id>"
# - "Rescheduled alarm: <id>"
# - "Force stop recovery complete: X missed, Y rescheduled"
```
**Check AlarmManager**:
```bash
# Verify alarms are rescheduled
adb shell dumpsys alarm | grep -A 10 timesafari
# Should show rescheduled alarms
```
**Expected Results**:
- ✅ Plugin detects force stop scenario (DB has alarms, AlarmManager has zero)
- ✅ All past alarms marked as missed
- ✅ All future alarms rescheduled
- ✅ Recovery logged in history
**Pass/Fail Criteria**:
- **Pass**: Logs show "FORCE_STOP" detection and all alarms recovered
- **Fail**: No force stop detection, or alarms not recovered
---
### 5.4 Test 3: Boot Recovery Missed Alarms
**Purpose**: Verify boot receiver detects and handles missed alarms during device reboot.
**Test Procedure**:
#### Step 1: Schedule Alarm Before Reboot
**Via Test App UI**:
1. Launch app: `adb shell am start -n com.timesafari.dailynotification/.MainActivity`
2. Schedule alarm for 5 minutes in future
3. Note scheduled time
**Verify Alarm Scheduled**:
```bash
adb shell dumpsys alarm | grep -A 5 timesafari
```
#### Step 2: Reboot Device
**Emulator**:
```bash
# Reboot emulator
adb reboot
# Wait for reboot to complete (30-60 seconds)
# Monitor with:
adb wait-for-device
adb shell getprop sys.boot_completed
# Wait until returns "1"
```
**Physical Device**:
- Manually reboot device
- Wait for boot to complete
- Reconnect ADB: `adb devices`
#### Step 3: Wait for Alarm Time to Pass
```bash
# After reboot, wait 10 minutes (past scheduled alarm time)
# Do NOT open app during this time
# Monitor system time
adb shell date
```
**During Wait**:
- Boot receiver should have run (check logs after reboot)
- Alarm should NOT fire (was wiped on reboot)
- No notification should appear
#### Step 4: Check Boot Receiver Logs
```bash
# Check logs from boot
adb logcat -d | grep -E "DNP-BOOT|BOOT_COMPLETED|reschedule"
# Expected log entries:
# - "Boot completed, rescheduling notifications"
# - "Found X enabled schedules to reschedule"
# - "Rescheduled notification for schedule: <id>"
# - "Missed alarm detected on boot: <id>"
# - "Boot recovery complete: X rescheduled, Y missed"
```
#### Step 5: Launch App and Verify
```bash
# Launch app to verify state
adb shell am start -n com.timesafari.dailynotification/.MainActivity
# Check if missed alarms were handled
adb logcat -d | grep -E "missed|boot_recovery"
```
**Expected Results**:
- ✅ Boot receiver runs on reboot
- ✅ Boot receiver detects missed alarms (past scheduled time)
- ✅ Missed alarms handled (notification generated or callback fired)
- ✅ Next occurrence rescheduled if repeating
- ✅ Future alarms rescheduled
**Pass/Fail Criteria**:
- **Pass**: Boot receiver logs show missed alarm detection and handling
- **Fail**: Boot receiver doesn't run, or doesn't detect missed alarms
---
### 5.5 Test 4: Warm Start Verification
**Purpose**: Verify plugin checks for missed alarms when app returns from background.
**Test Procedure**:
#### Step 1: Schedule Alarm
**Via Test App UI**:
1. Launch app: `adb shell am start -n com.timesafari.dailynotification/.MainActivity`
2. Schedule alarm for 10 minutes in future
3. Note scheduled time
**Verify Alarm Scheduled**:
```bash
adb shell dumpsys alarm | grep -A 5 timesafari
```
#### Step 2: Put App in Background
```bash
# Press home button (puts app in background)
adb shell input keyevent KEYCODE_HOME
# Verify app is in background
adb shell dumpsys activity activities | grep -A 5 timesafari
# App should be in "stopped" or "paused" state
```
**Alternative**: Use app switcher on device/emulator to swipe away (but don't force stop).
#### Step 3: Wait and Verify Alarm Still Scheduled
```bash
# Wait 2-3 minutes (before alarm time)
# Verify alarm is still scheduled
adb shell dumpsys alarm | grep -A 5 timesafari
# Should still show scheduled alarm
# Verify app process status
adb shell ps | grep timesafari
# Process may still be running (warm start scenario)
```
#### Step 4: Bring App to Foreground
```bash
# Bring app to foreground
adb shell am start -n com.timesafari.dailynotification/.MainActivity
# Monitor logs for warm start recovery
adb logcat -c
adb logcat | grep -E "DNP-REACTIVATION|WARM_START|verified"
```
#### Step 5: Verify Warm Start Recovery
**Check Logs**:
```bash
# Look for warm start recovery
adb logcat -d | grep -E "DNP-REACTIVATION|WARM_START|verified"
# Expected log entries:
# - "Starting app launch recovery"
# - "Detected scenario: WARM_START" (or COLD_START if process was killed)
# - "Handling warm start recovery"
# - "Warm start recovery complete: X missed, Y verified"
```
**Check Alarm Status**:
```bash
# Verify alarm is still scheduled
adb shell dumpsys alarm | grep -A 5 timesafari
# Should still show scheduled alarm
```
**Expected Results**:
- ✅ Plugin checks for missed alarms on warm start
- ✅ Plugin verifies active alarms are still scheduled
- ✅ No false positives (alarm still valid, not marked as missed)
**Pass/Fail Criteria**:
- **Pass**: Logs show verification, alarm still scheduled
- **Fail**: Alarm incorrectly marked as missed, or verification fails
---
### 5.6 Test 5: Swipe from Recents (Alarm Persistence)
**Purpose**: Verify alarms persist and fire even after app is swiped from recents.
**Test Procedure**:
#### Step 1: Schedule Alarm
```bash
# Launch app and schedule alarm for 4 minutes
adb shell am start -n com.timesafari.dailynotification/.MainActivity
# Use UI to schedule alarm
```
#### Step 2: Swipe App from Recents
**Emulator**:
```bash
# Open recent apps
adb shell input keyevent KEYCODE_APP_SWITCH
# Swipe app away (requires manual interaction or automation)
# Or use:
adb shell input swipe 500 1000 500 500 # Swipe up gesture
```
**Physical Device**: Manually swipe app from recent apps list.
#### Step 3: Verify App Process Status
```bash
# Check if process is still running
adb shell ps | grep timesafari
# Process may be killed or still running (OS decision)
# Verify alarm is still scheduled (CRITICAL)
adb shell dumpsys alarm | grep -A 5 timesafari
# Should still show scheduled alarm
```
#### Step 4: Wait for Alarm Time
```bash
# Wait for scheduled alarm time
# Monitor logs
adb logcat -c
adb logcat | grep -E "DN|RECEIVE|DN|NOTIFICATION|DN|WORK"
```
#### Step 5: Verify Alarm Fires
**Check Logs**:
```bash
# Look for alarm firing
adb logcat -d | grep -E "DN|RECEIVE_START|DN|WORK_START|DN|DISPLAY"
# Expected log entries:
# - "DN|RECEIVE_START" (DailyNotificationReceiver triggered)
# - "DN|WORK_START" (WorkManager worker started)
# - "DN|DISPLAY_NOTIF_START" (Notification displayed)
```
**Check Notification**:
- Notification should appear in system notification tray
- App process should be recreated by OS to deliver notification
**Expected Results**:
- ✅ Alarm fires even though app was swiped away
- ✅ App process recreated by OS
- ✅ Notification displayed
**Pass/Fail Criteria**:
- **Pass**: Notification appears, logs show alarm fired
- **Fail**: No notification, or alarm doesn't fire
---
### 5.7 Test 6: Exact Alarm Permission (Android 12+)
**Purpose**: Verify exact alarm permission handling on Android 12+.
**Test Procedure**:
#### Step 1: Revoke Exact Alarm Permission
**Android 12+ (API 31+)**:
```bash
# Check current permission status
adb shell dumpsys package com.timesafari.dailynotification | grep -i "schedule_exact_alarm"
# Revoke permission (requires root or system app)
# Or use Settings UI:
adb shell am start -a android.settings.REQUEST_SCHEDULE_EXACT_ALARM
# Manually revoke in settings
```
**Alternative**: Use test app UI to check permission status.
#### Step 2: Attempt to Schedule Alarm
**Via Test App UI**:
1. Launch app
2. Click "Test Notification"
3. Should trigger permission request flow
**Check Logs**:
```bash
adb logcat -d | grep -E "EXACT_ALARM|permission|SCHEDULE_EXACT"
```
#### Step 3: Grant Permission
**Via Settings**:
```bash
# Open exact alarm settings
adb shell am start -a android.settings.REQUEST_SCHEDULE_EXACT_ALARM
# Or navigate manually:
adb shell am start -a android.settings.APPLICATION_DETAILS_SETTINGS -d package:com.timesafari.dailynotification
```
#### Step 4: Verify Alarm Scheduling
```bash
# Schedule alarm again
# Check if it succeeds
adb logcat -d | grep "DN|SCHEDULE"
# Verify alarm is scheduled
adb shell dumpsys alarm | grep -A 5 timesafari
```
**Expected Results**:
- ✅ Plugin detects missing permission
- ✅ Settings opened automatically
- ✅ After granting, alarm schedules successfully
---
### 5.8 Test Validation Matrix
| Test | Scenario | Expected OS Behavior | Expected Plugin Behavior | Verification Method | Pass Criteria |
| ---- | -------- | -------------------- | ------------------------ | ------------------- | ------------- |
| 1 | Cold Start | N/A | Detect missed alarms | Logs show recovery | ✅ Recovery logs present |
| 2 | Force Stop | Alarms cancelled | Detect force stop, recover all | Logs + AlarmManager check | ✅ All alarms recovered |
| 3 | Boot Recovery | Alarms wiped | Boot receiver reschedules + detects missed | Boot logs | ✅ Boot receiver handles missed |
| 4 | Warm Start | Alarms persist | Verify alarms still scheduled | Logs + AlarmManager check | ✅ Verification logs present |
| 5 | Swipe from Recents | Alarm fires | Alarm fires (OS handles) | Notification appears | ✅ Notification displayed |
| 6 | Exact Alarm Permission | Permission required | Request permission, schedule after grant | Logs + AlarmManager | ✅ Permission flow works |
---
### 5.9 Log Monitoring Commands
**Comprehensive Log Monitoring**:
```bash
# Monitor all plugin-related logs
adb logcat | grep -E "DNP|DailyNotification|REACTIVATION|BOOT|DN\|"
# Monitor recovery specifically
adb logcat | grep -E "REACTIVATION|recovery|missed|reschedule"
# Monitor alarm scheduling
adb logcat | grep -E "DN\|SCHEDULE|DN\|ALARM|setAlarmClock|setExact"
# Monitor notification delivery
adb logcat | grep -E "DN\|RECEIVE|DN\|WORK|DN\|DISPLAY|Notification"
# Save logs to file for analysis
adb logcat -d > recovery_test.log
```
**Filter by Tag**:
```bash
# DNP-REACTIVATION (recovery manager)
adb logcat -s DNP-REACTIVATION
# DNP-BOOT (boot receiver)
adb logcat -s DNP-BOOT
# DailyNotificationWorker
adb logcat -s DailyNotificationWorker
# DailyNotificationReceiver
adb logcat -s DailyNotificationReceiver
```
---
### 5.10 Troubleshooting Test Failures
**If Recovery Doesn't Run**:
```bash
# Check if plugin loaded
adb logcat -d | grep "Daily Notification Plugin loaded"
# Check for errors
adb logcat -d | grep -E "ERROR|Exception|Failed"
# Verify database initialized
adb logcat -d | grep "DailyNotificationDatabase"
```
**If Force Stop Not Detected**:
```bash
# Check AlarmManager state
adb shell dumpsys alarm | grep -c timesafari
# Count should be 0 after force stop
# Check database state
# (Requires database inspection tool or test app UI)
```
**If Missed Alarms Not Detected**:
```bash
# Check database query
adb logcat -d | grep "getNotificationsReadyForDelivery"
# Verify scheduled_time < currentTime
adb shell date +%s
# Compare with alarm scheduled times
```
---
### 5.11 Automated Test Script
**Basic Test Script** (save as `test_recovery.sh`):
```bash
#!/bin/bash
PACKAGE="com.timesafari.dailynotification"
ACTIVITY="${PACKAGE}/.MainActivity"
echo "=== Test 1: Cold Start Recovery ==="
echo "1. Schedule alarm via UI, then run:"
echo " adb shell am kill $PACKAGE"
echo "2. Wait 10 minutes"
echo "3. Launch app:"
echo " adb shell am start -n $ACTIVITY"
echo "4. Check logs:"
echo " adb logcat -d | grep DNP-REACTIVATION"
echo ""
echo "=== Test 2: Force Stop Recovery ==="
echo "1. Schedule alarms via UI"
echo "2. Force stop:"
echo " adb shell am force-stop $PACKAGE"
echo "3. Wait 15 minutes"
echo "4. Launch app:"
echo " adb shell am start -n $ACTIVITY"
echo "5. Check logs:"
echo " adb logcat -d | grep FORCE_STOP"
echo ""
echo "=== Test 3: Boot Recovery ==="
echo "1. Schedule alarm via UI"
echo "2. Reboot:"
echo " adb reboot"
echo "3. Wait for boot, then 10 minutes"
echo "4. Check boot logs:"
echo " adb logcat -d | grep DNP-BOOT"
```
**Make executable**: `chmod +x test_recovery.sh`
---
## 6. Implementation Checklist
- [ ] Create `ReactivationManager.kt` file
- [ ] Implement scenario detection
- [ ] Implement force stop recovery
- [ ] Implement cold start recovery
- [ ] Implement warm start recovery
- [ ] Implement missed alarm handling
- [ ] Implement missed notification handling
- [ ] Update `DailyNotificationPlugin.load()` to call recovery
- [ ] Update `BootReceiver.rescheduleNotifications()` to handle missed alarms
- [ ] Add helper methods (calculateNextRunTime, isRepeating, etc.)
- [ ] Add logging for debugging
- [ ] Test cold start recovery
- [ ] Test force stop recovery
- [ ] Test boot recovery missed alarms
- [ ] Test warm start verification
---
## 7. Code References
**Existing Code to Reuse**:
- `NotifyReceiver.scheduleExactNotification()` - Line 92
- `NotifyReceiver.isAlarmScheduled()` - Line 279
- `NotifyReceiver.getNextAlarmTime()` - Line 305
- `FetchWorker.scheduleFetch()` - Line 31
- `NotificationContentDao.getNotificationsReadyForDelivery()` - Line 98
- `BootReceiver.calculateNextRunTime()` - Line 103
**New Code to Create**:
- `ReactivationManager.kt` - New file
- Helper methods in `BootReceiver.kt` - Add to existing file
---
## Related Documentation
- [Exploration Findings](./exploration-findings-initial.md) - Gap analysis
- [Plugin Requirements & Implementation](./plugin-requirements-implementation.md) - Requirements
- [Platform Capability Reference](./platform-capability-reference.md) - Android OS behavior
---
## Notes
- Implementation should be done incrementally
- Test each scenario after implementation
- Log extensively for debugging
- Handle errors gracefully (don't crash on recovery failure)
- Consider performance (recovery should not block app launch)