feat(ios): add Core Data DAO layer and unit tests

Implement comprehensive data access layer for Core Data entities:

- Add NotificationContentDAO, NotificationDeliveryDAO, and NotificationConfigDAO
  with full CRUD operations and query helpers
- Add DailyNotificationDataConversions utility for type conversions
  (Date ↔ Int64, Int ↔ Int32, JSON, optional strings)
- Update PersistenceController with entity verification and migration policies
- Add comprehensive unit tests for all DAO classes and data conversions
- Update Core Data model with NotificationContent, NotificationDelivery,
  and NotificationConfig entities (relationships and indexes)
- Integrate ReactivationManager into DailyNotificationPlugin.load()

DAO Features:
- Create/Insert methods with dictionary support
- Read/Query methods with predicates (by timesafariDid, notificationType,
  scheduledTime range, deliveryStatus, etc.)
- Update methods (touch, updateDeliveryStatus, recordUserInteraction)
- Delete methods (by ID, by key, delete all)
- Relationship management (NotificationContent ↔ NotificationDelivery)
- Cascade delete support

Test Coverage:
- 328 lines: DailyNotificationDataConversionsTests (time, numeric, string, JSON)
- 490 lines: NotificationContentDAOTests (CRUD, queries, updates)
- 415 lines: NotificationDeliveryDAOTests (CRUD, relationships, cascade delete)
- 412 lines: NotificationConfigDAOTests (CRUD, queries, active filtering)

All tests use in-memory Core Data stack for isolation and speed.

Completes sections 4.4, 4.5, and 6.0 of iOS implementation checklist.
This commit is contained in:
Matthew
2025-12-09 02:23:05 -08:00
parent dd8d67462f
commit a90d08c425
14 changed files with 5201 additions and 9 deletions

View File

@@ -36,4 +36,92 @@
<attribute name="nextRunAt" optional="YES" attributeType="Date" usesScalarValueType="NO"/>
<attribute name="stateJson" optional="YES" attributeType="String"/>
</entity>
<entity name="NotificationContent" representedClassName="NotificationContent" syncable="YES" codeGenerationType="class">
<attribute name="id" optional="NO" attributeType="String"/>
<attribute name="pluginVersion" optional="YES" attributeType="String"/>
<attribute name="timesafariDid" optional="YES" attributeType="String"/>
<attribute name="notificationType" optional="YES" attributeType="String"/>
<attribute name="title" optional="YES" attributeType="String"/>
<attribute name="body" optional="YES" attributeType="String"/>
<attribute name="scheduledTime" optional="NO" attributeType="Date" usesScalarValueType="NO"/>
<attribute name="timezone" optional="YES" attributeType="String"/>
<attribute name="priority" optional="YES" attributeType="Integer 32" defaultValueString="0" usesScalarValueType="YES"/>
<attribute name="vibrationEnabled" optional="YES" attributeType="Boolean" defaultValueString="NO" usesScalarValueType="YES"/>
<attribute name="soundEnabled" optional="YES" attributeType="Boolean" defaultValueString="YES" usesScalarValueType="YES"/>
<attribute name="mediaUrl" optional="YES" attributeType="String"/>
<attribute name="encryptedContent" optional="YES" attributeType="String"/>
<attribute name="encryptionKeyId" optional="YES" attributeType="String"/>
<attribute name="createdAt" optional="NO" attributeType="Date" usesScalarValueType="NO"/>
<attribute name="updatedAt" optional="NO" attributeType="Date" usesScalarValueType="NO"/>
<attribute name="ttlSeconds" optional="YES" attributeType="Integer 64" defaultValueString="604800" usesScalarValueType="YES"/>
<attribute name="deliveryStatus" optional="YES" attributeType="String"/>
<attribute name="deliveryAttempts" optional="YES" attributeType="Integer 32" defaultValueString="0" usesScalarValueType="YES"/>
<attribute name="lastDeliveryAttempt" optional="YES" attributeType="Date" usesScalarValueType="NO"/>
<attribute name="userInteractionCount" optional="YES" attributeType="Integer 32" defaultValueString="0" usesScalarValueType="YES"/>
<attribute name="lastUserInteraction" optional="YES" attributeType="Date" usesScalarValueType="NO"/>
<attribute name="metadata" optional="YES" attributeType="String"/>
<relationship name="deliveries" optional="YES" toMany="YES" deletionRule="Cascade" destinationEntity="NotificationDelivery" inverseName="notificationContent" inverseEntity="NotificationDelivery"/>
<index name="index_notification_content_timesafari_did">
<indexElement value="timesafariDid"/>
</index>
<index name="index_notification_content_notification_type">
<indexElement value="notificationType"/>
</index>
<index name="index_notification_content_scheduled_time">
<indexElement value="scheduledTime"/>
</index>
</entity>
<entity name="NotificationDelivery" representedClassName="NotificationDelivery" syncable="YES" codeGenerationType="class">
<attribute name="id" optional="NO" attributeType="String"/>
<attribute name="notificationId" optional="YES" attributeType="String"/>
<attribute name="timesafariDid" optional="YES" attributeType="String"/>
<attribute name="deliveryTimestamp" optional="NO" attributeType="Date" usesScalarValueType="NO"/>
<attribute name="deliveryStatus" optional="YES" attributeType="String"/>
<attribute name="deliveryMethod" optional="YES" attributeType="String"/>
<attribute name="deliveryAttemptNumber" optional="YES" attributeType="Integer 32" defaultValueString="1" usesScalarValueType="YES"/>
<attribute name="deliveryDurationMs" optional="YES" attributeType="Integer 64" defaultValueString="0" usesScalarValueType="YES"/>
<attribute name="userInteractionType" optional="YES" attributeType="String"/>
<attribute name="userInteractionTimestamp" optional="YES" attributeType="Date" usesScalarValueType="NO"/>
<attribute name="userInteractionDurationMs" optional="YES" attributeType="Integer 64" defaultValueString="0" usesScalarValueType="YES"/>
<attribute name="errorCode" optional="YES" attributeType="String"/>
<attribute name="errorMessage" optional="YES" attributeType="String"/>
<attribute name="deviceInfo" optional="YES" attributeType="String"/>
<attribute name="networkInfo" optional="YES" attributeType="String"/>
<attribute name="batteryLevel" optional="YES" attributeType="Integer 32" defaultValueString="-1" usesScalarValueType="YES"/>
<attribute name="dozeModeActive" optional="YES" attributeType="Boolean" defaultValueString="NO" usesScalarValueType="YES"/>
<attribute name="exactAlarmPermission" optional="YES" attributeType="Boolean" defaultValueString="NO" usesScalarValueType="YES"/>
<attribute name="notificationPermission" optional="YES" attributeType="Boolean" defaultValueString="NO" usesScalarValueType="YES"/>
<attribute name="metadata" optional="YES" attributeType="String"/>
<relationship name="notificationContent" optional="YES" maxCount="1" deletionRule="Nullify" destinationEntity="NotificationContent" inverseName="deliveries" inverseEntity="NotificationContent"/>
<index name="index_notification_delivery_notification_id">
<indexElement value="notificationId"/>
</index>
<index name="index_notification_delivery_delivery_timestamp">
<indexElement value="deliveryTimestamp"/>
</index>
</entity>
<entity name="NotificationConfig" representedClassName="NotificationConfig" syncable="YES" codeGenerationType="class">
<attribute name="id" optional="NO" attributeType="String"/>
<attribute name="timesafariDid" optional="YES" attributeType="String"/>
<attribute name="configType" optional="YES" attributeType="String"/>
<attribute name="configKey" optional="YES" attributeType="String"/>
<attribute name="configValue" optional="YES" attributeType="String"/>
<attribute name="configDataType" optional="YES" attributeType="String"/>
<attribute name="isEncrypted" optional="YES" attributeType="Boolean" defaultValueString="NO" usesScalarValueType="YES"/>
<attribute name="encryptionKeyId" optional="YES" attributeType="String"/>
<attribute name="createdAt" optional="NO" attributeType="Date" usesScalarValueType="NO"/>
<attribute name="updatedAt" optional="NO" attributeType="Date" usesScalarValueType="NO"/>
<attribute name="ttlSeconds" optional="YES" attributeType="Integer 64" defaultValueString="0" usesScalarValueType="YES"/>
<attribute name="isActive" optional="YES" attributeType="Boolean" defaultValueString="YES" usesScalarValueType="YES"/>
<attribute name="metadata" optional="YES" attributeType="String"/>
<index name="index_notification_config_config_key">
<indexElement value="configKey"/>
</index>
<index name="index_notification_config_config_type">
<indexElement value="configType"/>
</index>
<index name="index_notification_config_timesafari_did">
<indexElement value="timesafariDid"/>
</index>
</entity>
</model>