From 5fff7af40e4a892641747bbff228be9947246e49 Mon Sep 17 00:00:00 2001 From: James Daniels Date: Fri, 13 Dec 2024 13:38:23 -0500 Subject: [PATCH 1/4] Adding more injector safety --- src/zones.ts | 59 +++++++++++++++++++++++++++++++++++++++++++++------- 1 file changed, 51 insertions(+), 8 deletions(-) diff --git a/src/zones.ts b/src/zones.ts index c32ac1dd5..53e416e6b 100644 --- a/src/zones.ts +++ b/src/zones.ts @@ -4,7 +4,9 @@ import { Injector, NgZone, PendingTasks, - inject + inject, + isDevMode, + runInInjectionContext } from '@angular/core'; import { pendingUntilEvent } from '@angular/core/rxjs-interop'; import { @@ -76,31 +78,71 @@ function getSchedulers() { return inject(ɵAngularFireSchedulers); } +var alreadyWarned = false; +function warnOutsideInjectionContext(original: any, operation: string) { + if (isDevMode()) { + console.warn(`Firebase API outside injection context (${operation})`, original); + if (!alreadyWarned) { + alreadyWarned = true; + console.error("Calling Firebase APIs outside of an Injection context may destabilize your application leading to subtle change-detection and hydration bugs. Find more at https://github.com/angular/angularfire/blob/main/docs/zones.md"); + } + } +} + function runOutsideAngular(fn: (...args: any[]) => T): T { - return inject(NgZone).runOutsideAngular(() => fn()); + let ngZone: NgZone|undefined; + try { + ngZone = inject(NgZone); + } catch(e) { + warnOutsideInjectionContext(fn, "runOutsideAngular"); + } + if (!ngZone) {return fn();} + return ngZone.runOutsideAngular(() => fn()); } function run(fn: (...args: any[]) => T): T { - return inject(NgZone).run(() => fn()); + let ngZone: NgZone|undefined; + try { + ngZone = inject(NgZone); + } catch(e) { + warnOutsideInjectionContext(fn, "run"); + } + if (!ngZone) {return fn();} + return ngZone.run(() => fn()); } export function observeOutsideAngular(obs$: Observable): Observable { - return obs$.pipe(observeOn(getSchedulers().outsideAngular)); + let schedulers: ɵAngularFireSchedulers|undefined; + try { + schedulers = getSchedulers(); + } catch(e) { + warnOutsideInjectionContext(obs$, "observeOutsideAngular"); + } + if (!schedulers) {return obs$;} + return obs$.pipe(observeOn(schedulers.outsideAngular)); } export function observeInsideAngular(obs$: Observable): Observable { - return obs$.pipe(observeOn(getSchedulers().insideAngular)); + let schedulers: ɵAngularFireSchedulers|undefined; + try { + schedulers = getSchedulers(); + } catch(e) { + warnOutsideInjectionContext(obs$, "observeInsideAngular"); + } + if (!schedulers) {return obs$;} + return obs$.pipe(observeOn(schedulers.insideAngular)); } const zoneWrapFn = ( it: (...args: any[]) => any, - taskDone: VoidFunction | undefined + taskDone: VoidFunction | undefined, + injector: Injector, ) => { return (...args: any[]) => { if (taskDone) { setTimeout(taskDone, 0); } - return run(() => it.apply(this, args)); + return runInInjectionContext(injector, () => run(() => it.apply(this, args))); }; }; @@ -117,6 +159,7 @@ export const ɵzoneWrap = (it: T, blockUntilFirst: boolean): T => { pendingTasks = inject(PendingTasks); injector = inject(Injector); } catch(e) { + warnOutsideInjectionContext(it, "ɵzoneWrap"); return (it as any).apply(this, _arguments); } // if this is a callback function, e.g, onSnapshot, we should create a pending task and complete it @@ -127,7 +170,7 @@ export const ɵzoneWrap = (it: T, blockUntilFirst: boolean): T => { taskDone ||= run(() => pendingTasks.add()); } // TODO create a microtask to track callback functions - _arguments[i] = zoneWrapFn(_arguments[i], taskDone); + _arguments[i] = zoneWrapFn(_arguments[i], taskDone, injector); } } const ret = runOutsideAngular(() => (it as any).apply(this, _arguments)); From a401cdbd191b58fdc758da97e16d78e20d5cc179 Mon Sep 17 00:00:00 2001 From: James Daniels Date: Fri, 13 Dec 2024 13:41:37 -0500 Subject: [PATCH 2/4] better warning --- src/zones.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/zones.ts b/src/zones.ts index 53e416e6b..0cee6f80e 100644 --- a/src/zones.ts +++ b/src/zones.ts @@ -81,7 +81,7 @@ function getSchedulers() { var alreadyWarned = false; function warnOutsideInjectionContext(original: any, operation: string) { if (isDevMode()) { - console.warn(`Firebase API outside injection context (${operation})`, original); + console.warn(`Firebase API called outside injection context: ${operation}(${original})`); if (!alreadyWarned) { alreadyWarned = true; console.error("Calling Firebase APIs outside of an Injection context may destabilize your application leading to subtle change-detection and hydration bugs. Find more at https://github.com/angular/angularfire/blob/main/docs/zones.md"); From a184f69f7f82b432e48e80e7db56383e1b8cf4b6 Mon Sep 17 00:00:00 2001 From: James Daniels Date: Fri, 13 Dec 2024 15:07:02 -0500 Subject: [PATCH 3/4] Pass injector scope to promises --- src/zones.ts | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/src/zones.ts b/src/zones.ts index 0cee6f80e..d8505433e 100644 --- a/src/zones.ts +++ b/src/zones.ts @@ -81,7 +81,7 @@ function getSchedulers() { var alreadyWarned = false; function warnOutsideInjectionContext(original: any, operation: string) { if (isDevMode()) { - console.warn(`Firebase API called outside injection context: ${operation}(${original})`); + console.warn(`Firebase API called outside injection context: ${operation}(${original.name})`); if (!alreadyWarned) { alreadyWarned = true; console.error("Calling Firebase APIs outside of an Injection context may destabilize your application leading to subtle change-detection and hydration bugs. Find more at https://github.com/angular/angularfire/blob/main/docs/zones.md"); @@ -196,8 +196,8 @@ export const ɵzoneWrap = (it: T, blockUntilFirst: boolean): T => { () => new Promise((resolve, reject) => { pendingTasks.run(() => ret).then( - (it) => run(() => resolve(it)), - (reason) => run(() => reject(reason)) + (it) => runInInjectionContext(injector, () => run(() => resolve(it))), + (reason) => runInInjectionContext(injector, () => run(() => reject(reason))) ); }) ); From 82756754ea33cb82a01d3a150c90bac7f4549e28 Mon Sep 17 00:00:00 2001 From: James Daniels Date: Fri, 13 Dec 2024 16:15:50 -0500 Subject: [PATCH 4/4] beforeAuthStateChanged should not block --- src/auth/firebase.ts | 2 +- tools/build.ts | 1 + 2 files changed, 2 insertions(+), 1 deletion(-) diff --git a/src/auth/firebase.ts b/src/auth/firebase.ts index d1a25919a..97c07f167 100644 --- a/src/auth/firebase.ts +++ b/src/auth/firebase.ts @@ -59,7 +59,7 @@ import { } from 'firebase/auth'; export const applyActionCode = ɵzoneWrap(_applyActionCode, true); -export const beforeAuthStateChanged = ɵzoneWrap(_beforeAuthStateChanged, true); +export const beforeAuthStateChanged = ɵzoneWrap(_beforeAuthStateChanged, false); export const checkActionCode = ɵzoneWrap(_checkActionCode, true); export const confirmPasswordReset = ɵzoneWrap(_confirmPasswordReset, true); export const connectAuthEmulator = ɵzoneWrap(_connectAuthEmulator, true); diff --git a/tools/build.ts b/tools/build.ts index 7da9a45b4..397d78906 100644 --- a/tools/build.ts +++ b/tools/build.ts @@ -75,6 +75,7 @@ ${exportedZoneWrappedFns} browserSessionPersistence: null, indexedDBLocalPersistence: null, prodErrorMap: null, + beforeAuthStateChanged: { blockUntilFirst: false }, }), reexport('database', 'rxfire', 'rxfire/database', tsKeys()), reexport('database', 'firebase', 'firebase/database', tsKeys()),