- SnapshotModels.swift: CachedQuote (slug-unique, themes/regions
als CSV), SnapshotMeta (singleton mit lastSyncedAt + totalCount),
SnapshotContainer.make() mit App-Group-Store-URL (Fallback auf
App-Container für Dev ohne Apple-Dev-Portal-Setup)
- SnapshotSync (actor) mit injectable Loader für Tests: refresh /
refreshIfStale / tryRefresh (fail-soft). Re-konsolidiert beim Pull
(Update + Insert + Delete entzogene Slugs). 24h-Staleness-Default.
- DailyQuoteWidget: Hash-of-Day-Picker aus SwiftData, drei Sizes,
Mitternacht-Refresh-Policy, Placeholder bei leerem Store. Widget-
Target zieht SnapshotModels.swift mit (project.yml).
- ZitareNativeApp triggert SnapshotSync.tryRefresh() bei Launch +
WidgetCenter.reloadAllTimelines() danach.
- AppConfig.snapshotURL = webBaseURL/index-min.json (Web-Endpoint
noch nicht live, fail-soft).
- DeepLinkRouter Substring-Guard fix (`/t` statt `/t/` im
Prefix-Array, sonst greift hasPrefix("/t//") nicht).
- 22 Tests grün (6 AppConfig + 11 DeepLinkRouter + 3 SnapshotSync +
1 UI + 1 Widget-Compile-Smoke), swiftlint 0 violations in 22 Files
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
44 lines
2 KiB
Swift
44 lines
2 KiB
Swift
import Foundation
|
|
import ManaCore
|
|
|
|
/// App-spezifische Konfiguration für Zitare. Implementiert
|
|
/// `ManaAppConfig` aus ManaCore und ergänzt die Zitare-eigene
|
|
/// `apiBaseURL` (zitare-api, getrennt von mana-auth) sowie
|
|
/// `webBaseURL` (zitare.com, für WKWebView und Universal-Links)
|
|
/// und `appBaseURL` (zitare.mana.how, für eingeloggte Pfade).
|
|
enum AppConfig {
|
|
static let manaAppConfig: ManaAppConfig = DefaultManaAppConfig(
|
|
authBaseURL: URL(string: "https://auth.mana.how")!,
|
|
keychainService: "ev.mana.zitare",
|
|
keychainAccessGroup: nil
|
|
)
|
|
|
|
/// `zitare-api.mana.how` — API-Backend (Hono+Bun).
|
|
static let apiBaseURL = URL(string: "https://zitare-api.mana.how")!
|
|
|
|
/// `zitare.com` — geplante öffentliche Domain (CC-BY-SA-Korpus,
|
|
/// statisch). Universal-Link-Domain für AASA. **Heute DNS noch
|
|
/// nicht live** (Cloudflare-Zone-Onboarding offen, siehe
|
|
/// `zitare/STATUS.md`); bis dahin nutzt der WebView `appBaseURL`
|
|
/// (`zitare.mana.how`) — der Container liefert beide Surfaces.
|
|
static let publicWebURL = URL(string: "https://zitare.com")!
|
|
|
|
/// `zitare.mana.how` — SPA-Surface, eingeloggte Pfade. Heute auch
|
|
/// der Default für Lese-Surfaces, bis `zitare.com` live ist.
|
|
static let appBaseURL = URL(string: "https://zitare.mana.how")!
|
|
|
|
/// Effektive Default-URL für den WebView. Zeigt vorerst auf
|
|
/// `appBaseURL` (`zitare.mana.how`); nach Cloudflare-Zone-Cut
|
|
/// kommt das zurück auf `publicWebURL`.
|
|
static let webBaseURL = appBaseURL
|
|
|
|
/// App-Group für Daten-Sharing zwischen App ↔ Widget ↔ ShareExt.
|
|
static let appGroup = "group.ev.mana.zitare"
|
|
|
|
/// Endpoint für den Korpus-Snapshot (Phase ζ-2). Heute noch nicht
|
|
/// als statische HTTP-Datei publiziert — Aufgabe im Web-Repo:
|
|
/// `apps/zitare/static/index-min.json` aus dem Snapshot-Job
|
|
/// zusätzlich rauskopieren. Bis dahin schlägt der Pull mit 404
|
|
/// fehl und `SnapshotSync.tryRefresh()` macht fail-soft no-op.
|
|
static let snapshotURL = webBaseURL.appendingPathComponent("index-min.json")
|
|
}
|