Skip to content

Commit 4ab5a9c

Browse files
authoredFeb 17, 2021
Implement and type auth.emulatorConfig. (#4481)
* Add emulator methods to auth-types. * Implement and type auth.emulatorConfig. * Create itchy-grapes-drum.md
1 parent b3f2eef commit 4ab5a9c

File tree

11 files changed

+106
-7
lines changed

11 files changed

+106
-7
lines changed
 

‎.changeset/itchy-grapes-drum.md

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
---
2+
"@firebase/auth-types": patch
3+
---
4+
5+
Add emulator methods to auth-types.

‎packages-exp/auth-compat-exp/src/auth.ts

+4
Original file line numberDiff line numberDiff line change
@@ -74,6 +74,10 @@ export class Auth
7474
);
7575
}
7676

77+
get emulatorConfig(): compat.EmulatorConfig | null {
78+
return this.auth.emulatorConfig;
79+
}
80+
7781
get currentUser(): compat.User | null {
7882
if (!this.auth.currentUser) {
7983
return null;

‎packages-exp/auth-exp/src/api/index.test.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -383,7 +383,7 @@ describe('api/_performApiRequest', () => {
383383

384384
it('works properly with an emulated environment', () => {
385385
(auth.config as ConfigInternal).emulator = {
386-
url: 'http://localhost:5000'
386+
url: 'http://localhost:5000/'
387387
};
388388
expect(_getFinalTarget(auth, 'host', '/path', 'query=test')).to.eq(
389389
'http://localhost:5000/host/path?query=test'

‎packages-exp/auth-exp/src/core/auth/auth_impl.ts

+1
Original file line numberDiff line numberDiff line change
@@ -59,6 +59,7 @@ export const enum DefaultConfig {
5959

6060
export class AuthImpl implements Auth, _FirebaseService {
6161
currentUser: externs.User | null = null;
62+
emulatorConfig: externs.EmulatorConfig | null = null;
6263
private operations = Promise.resolve();
6364
private persistenceManager?: PersistenceUserManager;
6465
private redirectPersistenceManager?: PersistenceUserManager;

‎packages-exp/auth-exp/src/core/auth/emulator.test.ts

+37
Original file line numberDiff line numberDiff line change
@@ -83,6 +83,13 @@ describe('core/auth/emulator', () => {
8383
expect(emulatorEndpoint.calls.length).to.eq(1);
8484
});
8585

86+
it('updates the endpoint appropriately with trailing slash', async () => {
87+
useAuthEmulator(auth, 'http://localhost:2020/');
88+
await user.delete();
89+
expect(normalEndpoint.calls.length).to.eq(0);
90+
expect(emulatorEndpoint.calls.length).to.eq(1);
91+
});
92+
8693
it('checks the scheme properly', () => {
8794
expect(() => useAuthEmulator(auth, 'http://localhost:2020')).not.to.throw;
8895
delete auth.config.emulator;
@@ -134,6 +141,36 @@ describe('core/auth/emulator', () => {
134141
expect(document.querySelector('.firebase-emulator-warning')).to.be.null;
135142
}
136143
});
144+
145+
it('sets emulatorConfig on the Auth object', async () => {
146+
useAuthEmulator(auth, 'http://localhost:2020');
147+
expect(auth.emulatorConfig).to.eql({
148+
protocol: 'http',
149+
host: 'localhost',
150+
port: 2020,
151+
options: { disableWarnings: false }
152+
});
153+
});
154+
155+
it('sets disableWarnings in emulatorConfig accordingly', async () => {
156+
useAuthEmulator(auth, 'https://127.0.0.1', { disableWarnings: true });
157+
expect(auth.emulatorConfig).to.eql({
158+
protocol: 'https',
159+
host: '127.0.0.1',
160+
port: null,
161+
options: { disableWarnings: true }
162+
});
163+
});
164+
165+
it('quotes IPv6 address in emulatorConfig', async () => {
166+
useAuthEmulator(auth, 'http://[::1]:2020/');
167+
expect(auth.emulatorConfig).to.eql({
168+
protocol: 'http',
169+
host: '[::1]',
170+
port: 2020,
171+
options: { disableWarnings: false }
172+
});
173+
});
137174
});
138175

139176
context('toJSON', () => {

‎packages-exp/auth-exp/src/core/auth/emulator.ts

+13-2
Original file line numberDiff line numberDiff line change
@@ -58,9 +58,20 @@ export function useAuthEmulator(
5858
AuthErrorCode.INVALID_EMULATOR_SCHEME
5959
);
6060

61-
authInternal.config.emulator = { url };
61+
const parsedUrl = new URL(url);
62+
const disableWarnings = !!options?.disableWarnings;
63+
64+
// Store the normalized URL whose path is always nonempty (i.e. containing at least a single '/').
65+
authInternal.config.emulator = { url: parsedUrl.toString() };
6266
authInternal.settings.appVerificationDisabledForTesting = true;
63-
emitEmulatorWarning(!!options?.disableWarnings);
67+
authInternal.emulatorConfig = Object.freeze({
68+
host: parsedUrl.hostname,
69+
port: parsedUrl.port ? Number(parsedUrl.port) : null,
70+
protocol: parsedUrl.protocol.replace(':', ''),
71+
options: Object.freeze({ disableWarnings })
72+
});
73+
74+
emitEmulatorWarning(disableWarnings);
6475
}
6576

6677
function emitEmulatorWarning(disableBanner: boolean): void {

‎packages-exp/auth-exp/src/core/util/emulator.test.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -23,7 +23,7 @@ import { _emulatorUrl } from './emulator';
2323
describe('core/util/emulator', () => {
2424
const config: ConfigInternal = {
2525
emulator: {
26-
url: 'http://localhost:4000'
26+
url: 'http://localhost:4000/'
2727
}
2828
} as ConfigInternal;
2929

‎packages-exp/auth-exp/src/core/util/emulator.ts

+2-3
Original file line numberDiff line numberDiff line change
@@ -21,11 +21,10 @@ import { debugAssert } from './assert';
2121
export function _emulatorUrl(config: ConfigInternal, path?: string): string {
2222
debugAssert(config.emulator, 'Emulator should always be set here');
2323
const { url } = config.emulator;
24-
const emulatorHost = new URL(url).toString();
2524

2625
if (!path) {
27-
return emulatorHost;
26+
return url;
2827
}
2928

30-
return `${emulatorHost}${path.startsWith('/') ? path.slice(1) : path}`;
29+
return `${url}${path.startsWith('/') ? path.slice(1) : path}`;
3130
}

‎packages-exp/auth-exp/src/model/auth.ts

+1
Original file line numberDiff line numberDiff line change
@@ -37,6 +37,7 @@ export interface ConfigInternal extends externs.Config {
3737

3838
export interface Auth extends externs.Auth {
3939
currentUser: externs.User | null;
40+
emulatorConfig: externs.EmulatorConfig | null;
4041
_canInitEmulator: boolean;
4142
_isInitialized: boolean;
4243
_initializationPromise: Promise<void> | null;

‎packages-exp/auth-types-exp/index.d.ts

+30
Original file line numberDiff line numberDiff line change
@@ -288,6 +288,8 @@ export interface Auth {
288288
): Unsubscribe;
289289
/** The currently signed-in user (or null). */
290290
readonly currentUser: User | null;
291+
/** The current emulator configuration (or null). */
292+
readonly emulatorConfig: EmulatorConfig | null;
291293
/**
292294
* Asynchronously sets the provided user as {@link Auth.currentUser} on the {@link Auth} instance.
293295
*
@@ -1511,6 +1513,34 @@ declare module '@firebase/component' {
15111513
}
15121514
}
15131515

1516+
/**
1517+
* Configuration of Firebase Authentication Emulator.
1518+
*/
1519+
export interface EmulatorConfig {
1520+
/**
1521+
* The protocol used to communicate with the emulator ("http"/"https").
1522+
*/
1523+
readonly protocol: string;
1524+
/**
1525+
* The hostname of the emulator, which may be a domain ("localhost"), IPv4 address ("127.0.0.1")
1526+
* or quoted IPv6 address ("[::1]").
1527+
*/
1528+
readonly host: string;
1529+
/**
1530+
* The port of the emulator, or null if port isn't specified (i.e. protocol default).
1531+
*/
1532+
readonly port: number | null;
1533+
/**
1534+
* The emulator-specific options.
1535+
*/
1536+
readonly options: {
1537+
/**
1538+
* Whether the warning banner attached to the DOM was disabled.
1539+
*/
1540+
readonly disableWarnings: boolean;
1541+
};
1542+
}
1543+
15141544
/**
15151545
* A mapping of error codes to error messages.
15161546
*

‎packages/auth-types/index.d.ts

+11
Original file line numberDiff line numberDiff line change
@@ -389,6 +389,15 @@ export class PhoneMultiFactorGenerator {
389389
): PhoneMultiFactorAssertion;
390390
}
391391

392+
export interface EmulatorConfig {
393+
readonly protocol: string;
394+
readonly host: string;
395+
readonly port: number | null;
396+
readonly options: {
397+
readonly disableWarnings: boolean;
398+
};
399+
}
400+
392401
export class FirebaseAuth {
393402
private constructor();
394403

@@ -407,6 +416,7 @@ export class FirebaseAuth {
407416
password: string
408417
): Promise<UserCredential>;
409418
currentUser: User | null;
419+
readonly emulatorConfig: EmulatorConfig | null;
410420
fetchSignInMethodsForEmail(email: string): Promise<Array<string>>;
411421
isSignInWithEmailLink(emailLink: string): boolean;
412422
getRedirectResult(): Promise<UserCredential>;
@@ -455,6 +465,7 @@ export class FirebaseAuth {
455465
tenantId: string | null;
456466
updateCurrentUser(user: User | null): Promise<void>;
457467
useDeviceLanguage(): void;
468+
useEmulator(url: string, options?: { disableWarnings?: boolean }): void;
458469
verifyPasswordResetCode(code: string): Promise<string>;
459470
}
460471

0 commit comments

Comments
 (0)
Please sign in to comment.