Expose reactive Stream for organization list updates
epic-organization-selection-and-onboarding-data-layer-task-004 — Wire up a Supabase Realtime subscription within OrganizationRepository to emit updated organization lists whenever the underlying table changes. Expose a Stream<List<Organization>> that higher-level Riverpod providers can watch. The stream must replay the latest cached value immediately upon subscription and merge real-time updates with the in-memory cache to avoid redundant network fetches.
Acceptance Criteria
Technical Requirements
Execution Context
Tier 3 - 413 tasks
Can start after Tier 2 completes
Implementation Notes
Use `rxdart`'s `BehaviorSubject>` as the internal sink — it naturally replays the latest value to new subscribers. In the constructor, after the initial fetch populates the cache, add the list to the subject. Subscribe to `supabase.channel('organizations').onPostgresChanges(...)` and on each event, fetch the updated row (or re-query the full list if simpler), update the cache, and add to the subject. Keep a ref-count or use `shareValueSeeded` to manage the Realtime channel lifecycle.
Expose the stream as `organizationsStream` via `subject.stream`. In Riverpod, wire via `StreamProvider.autoDispose` so the channel is torn down when no widgets are listening. Ensure the Realtime filter uses `schema: 'public', table: 'organizations'` with `event: '*'`.
Testing Requirements
Write unit tests in flutter_test with a fake/mock RealtimeChannel: (1) verify stream emits cached value immediately on subscription, (2) simulate an INSERT payload and assert stream emits updated list, (3) simulate sign-out and assert stream emits empty list or closes, (4) assert subscribing twice reuses the same channel (no duplicate channels). Write one integration test (or widget test with fakeAsync) confirming a Riverpod StreamProvider watching organizationsStream rebuilds on stream emission.
TenantSessionStore must write to both SecureStorageAdapter and Supabase session synchronously. If the Supabase write succeeds but the local secure storage write fails (or vice versa), the system ends up in an inconsistent state: local app thinks org A is selected but Supabase queries are scoped to org B (or unscoped), causing silent data leakage or empty result sets.
Mitigation & Contingency
Mitigation: Implement the dual-write with compensating rollback: if the second write fails, undo the first write and surface a typed DualWriteFailureError to callers. Add a startup integrity check in restoreSession() that validates local storage and Supabase session agree on the current org_id.
Contingency: If integrity check fails on startup, clear both stores and redirect the user to the org selection screen rather than proceeding with potentially mismatched state.
An organization could be deactivated in Supabase between the time the org list is cached and the time the user taps to select it. If the repository serves stale cached data the org-selection-service will attempt to seed a context for an inactive org, potentially causing RLS scope issues or confusing error states.
Mitigation & Contingency
Mitigation: The OrganizationRepository.getOrganizationById() path used during selection validation always performs a fresh network fetch (bypassing cache) to confirm the org is still active before the TenantSessionStore writes anything.
Contingency: If a freshness check finds the org is inactive, surface a localized error message on the selection screen ('This organization is no longer available') and refresh the org list to show only currently active partners.