Skip to content

Commit 036849f

Browse files
committed
Merge branch 'master' into mila/BloomFilter
2 parents eaef9da + 1455bfa commit 036849f

28 files changed

+1002
-481
lines changed

.changeset/nervous-ads-pretend.md

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
---
2+
"@firebase/database": patch
3+
---
4+
5+
Fixed issue where connectDatabaseToEmulator can be called twice during a hot reload

.changeset/quick-radios-obey.md

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
---
2+
"@firebase/firestore": patch
3+
---
4+
5+
Update canonifyFilter to compute the canonization for flat conjunctions the same as implicit AND queries.

.changeset/stupid-swans-fix.md

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
---
2+
"@firebase/firestore": patch
3+
---
4+
5+
Fix an issue that stops some performance optimization being applied.

.changeset/wild-geckos-fetch.md

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
---
2+
'@firebase/util': patch
3+
---
4+
5+
Reformat a comment that causes compile errors in some build toolchains.

.changeset/young-hornets-rescue.md

+6
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
---
2+
'@firebase/util': minor
3+
'firebase': minor
4+
---
5+
6+
Allow users to specify their environment as `node` or `browser` to override Firebase's runtime environment detection and force the SDK to act as if it were in the respective environment.

common/api-review/util.api.md

+4-2
Original file line numberDiff line numberDiff line change
@@ -193,6 +193,7 @@ export interface FirebaseDefaults {
193193
config?: Record<string, string>;
194194
// (undocumented)
195195
emulatorHosts?: Record<string, string>;
196+
forceEnvironment?: 'browser' | 'node';
196197
}
197198

198199
// Warning: (ae-missing-release-tag) "FirebaseError" is exported by the package, but it is missing a release tag (@alpha, @beta, @public, or @internal)
@@ -221,11 +222,12 @@ export const getDefaultEmulatorHost: (productName: string) => string | undefined
221222
// @public
222223
export const getDefaultEmulatorHostnameAndPort: (productName: string) => [hostname: string, port: number] | undefined;
223224

225+
// @public
226+
export const getDefaults: () => FirebaseDefaults | undefined;
227+
224228
// @public
225229
export const getExperimentalSetting: <T extends ExperimentalKey>(name: T) => FirebaseDefaults[`_${T}`];
226230

227-
// Warning: (ae-missing-release-tag) "getGlobal" is exported by the package, but it is missing a release tag (@alpha, @beta, @public, or @internal)
228-
//
229231
// @public
230232
export function getGlobal(): typeof globalThis;
231233

package.json

+2-2
Original file line numberDiff line numberDiff line change
@@ -105,7 +105,7 @@
105105
"eslint-plugin-unused-imports": "2.0.0",
106106
"express": "4.18.2",
107107
"find-free-port": "2.0.0",
108-
"firebase-tools": "11.2.2",
108+
"firebase-tools": "11.19.0",
109109
"glob": "7.2.3",
110110
"http-server": "14.1.1",
111111
"indexeddbshim": "8.0.0",
@@ -140,7 +140,7 @@
140140
"protractor": "5.4.2",
141141
"request": "2.88.2",
142142
"semver": "7.3.8",
143-
"simple-git": "3.7.1",
143+
"simple-git": "3.15.0",
144144
"sinon": "9.2.4",
145145
"sinon-chai": "3.7.0",
146146
"source-map-loader": "1.1.3",

packages/database/src/api/Database.ts

+5-3
Original file line numberDiff line numberDiff line change
@@ -320,9 +320,11 @@ export function getDatabase(
320320
const db = _getProvider(app, 'database').getImmediate({
321321
identifier: url
322322
}) as Database;
323-
const emulator = getDefaultEmulatorHostnameAndPort('database');
324-
if (emulator) {
325-
connectDatabaseEmulator(db, ...emulator);
323+
if (!db._instanceStarted) {
324+
const emulator = getDefaultEmulatorHostnameAndPort('database');
325+
if (emulator) {
326+
connectDatabaseEmulator(db, ...emulator);
327+
}
326328
}
327329
return db;
328330
}

packages/database/test/exp/integration.test.ts

+18
Original file line numberDiff line numberDiff line change
@@ -77,6 +77,24 @@ describe('Database@exp Tests', () => {
7777
const db = getDatabase(defaultApp);
7878
expect(db).to.be.ok;
7979
});
80+
it("doesn't try to connect to emulator after database has already started", async () => {
81+
const db = getDatabase(defaultApp);
82+
const r = ref(db, '.info/connected');
83+
const deferred = new Deferred();
84+
onValue(r, snapshot => {
85+
if (snapshot.val()) {
86+
deferred.resolve();
87+
}
88+
});
89+
await deferred.promise;
90+
process.env.__FIREBASE_DEFAULTS__ = JSON.stringify({
91+
emulatorHosts: {
92+
database: 'localhost:9000'
93+
}
94+
});
95+
expect(() => getDatabase(defaultApp)).to.not.throw();
96+
delete process.env.__FIREBASE_DEFAULTS__;
97+
});
8098

8199
it('Can get database with custom URL', () => {
82100
const db = getDatabase(defaultApp, 'http://foo.bar.com');

packages/firestore/src/core/filter.ts

+8
Original file line numberDiff line numberDiff line change
@@ -325,6 +325,14 @@ export function canonifyFilter(filter: Filter): string {
325325
filter.op.toString() +
326326
canonicalId(filter.value)
327327
);
328+
} else if (compositeFilterIsFlatConjunction(filter)) {
329+
// Older SDK versions use an implicit AND operation between their filters.
330+
// In the new SDK versions, the developer may use an explicit AND filter.
331+
// To stay consistent with the old usages, we add a special case to ensure
332+
// the canonical ID for these two are the same. For example:
333+
// `col.whereEquals("a", 1).whereEquals("b", 2)` should have the same
334+
// canonical ID as `col.where(and(equals("a",1), equals("b",2)))`.
335+
return filter.filters.map(filter => canonifyFilter(filter)).join(',');
328336
} else {
329337
// filter instanceof CompositeFilter
330338
const canonicalIdsString = filter.filters

packages/firestore/src/lite-api/query.ts

+14-12
Original file line numberDiff line numberDiff line change
@@ -364,13 +364,14 @@ export type QueryFilterConstraint =
364364
| QueryCompositeFilterConstraint;
365365

366366
/**
367-
* Creates a {@link QueryCompositeFilterConstraint} that performs a logical OR
368-
* of all the provided {@link QueryFilterConstraint}s.
367+
* Creates a new {@link QueryCompositeFilterConstraint} that is a disjunction of
368+
* the given filter constraints. A disjunction filter includes a document if it
369+
* satisfies any of the given filters.
369370
*
370-
* @param queryConstraints - Optional. The {@link QueryFilterConstraint}s
371-
* for OR operation. These must be created with calls to {@link where},
372-
* {@link or}, or {@link and}.
373-
* @returns The created {@link QueryCompositeFilterConstraint}.
371+
* @param queryConstraints - Optional. The list of
372+
* {@link QueryFilterConstraint}s to perform a disjunction for. These must be
373+
* created with calls to {@link where}, {@link or}, or {@link and}.
374+
* @returns The newly created {@link QueryCompositeFilterConstraint}.
374375
* @internal TODO remove this internal tag with OR Query support in the server
375376
*/
376377
export function or(
@@ -388,13 +389,14 @@ export function or(
388389
}
389390

390391
/**
391-
* Creates a {@link QueryCompositeFilterConstraint} that performs a logical AND
392-
* of all the provided {@link QueryFilterConstraint}s.
392+
* Creates a new {@link QueryCompositeFilterConstraint} that is a conjunction of
393+
* the given filter constraints. A conjunction filter includes a document if it
394+
* satisfies all of the given filters.
393395
*
394-
* @param queryConstraints - Optional. The {@link QueryFilterConstraint}s
395-
* for AND operation. These must be created with calls to {@link where},
396-
* {@link or}, or {@link and}.
397-
* @returns The created {@link QueryCompositeFilterConstraint}.
396+
* @param queryConstraints - Optional. The list of
397+
* {@link QueryFilterConstraint}s to perform a conjunction for. These must be
398+
* created with calls to {@link where}, {@link or}, or {@link and}.
399+
* @returns The newly created {@link QueryCompositeFilterConstraint}.
398400
* @internal TODO remove this internal tag with OR Query support in the server
399401
*/
400402
export function and(

packages/firestore/src/local/local_documents_view.ts

+4
Original file line numberDiff line numberDiff line change
@@ -242,6 +242,10 @@ export class LocalDocumentsView {
242242
overlay.mutation.getFieldMask(),
243243
Timestamp.now()
244244
);
245+
} else {
246+
// no overlay exists
247+
// Using EMPTY to indicate there is no overlay for the document.
248+
mutatedFields.set(doc.key, FieldMask.empty());
245249
}
246250
});
247251

packages/firestore/src/local/overlayed_document.ts

+4-2
Original file line numberDiff line numberDiff line change
@@ -27,8 +27,10 @@ export class OverlayedDocument {
2727
readonly overlayedDocument: Document,
2828

2929
/**
30-
* The fields that are locally mutated by patch mutations. If the overlayed
31-
* document is from set or delete mutations, this returns null.
30+
* The fields that are locally mutated by patch mutations.
31+
*
32+
* If the overlayed document is from set or delete mutations, this is `null`.
33+
* If there is no overlay (mutation) for the document, this is an empty `FieldMask`.
3234
*/
3335
readonly mutatedFields: FieldMask | null
3436
) {}

packages/firestore/src/model/mutation_batch.ts

+1
Original file line numberDiff line numberDiff line change
@@ -166,6 +166,7 @@ export class MutationBatch {
166166
if (overlay !== null) {
167167
overlays.set(m.key, overlay);
168168
}
169+
169170
if (!mutableDocument.isValidDocument()) {
170171
mutableDocument.convertToNoDocument(SnapshotVersion.min());
171172
}

packages/firestore/test/unit/core/filter.test.ts

+13-1
Original file line numberDiff line numberDiff line change
@@ -25,7 +25,9 @@ import {
2525
FieldFilter,
2626
Operator
2727
} from '../../../src/core/filter';
28-
import { andFilter, filter, orFilter } from '../../util/helpers';
28+
import { queryToTarget } from '../../../src/core/query';
29+
import { canonifyTarget } from '../../../src/core/target';
30+
import { andFilter, filter, orFilter, query } from '../../util/helpers';
2931

3032
describe('FieldFilter', () => {
3133
it('exposes field filter members', () => {
@@ -93,4 +95,14 @@ describe('CompositeFilter', () => {
9395
expect(compositeFilterIsFlat(orFilter2)).false;
9496
expect(compositeFilterIsFlatConjunction(orFilter2)).false;
9597
});
98+
99+
it('computes canonical id of flat conjunctions', () => {
100+
const target1 = query('col', a, b, c);
101+
102+
const target2 = query('col', andFilter(a, b, c));
103+
104+
expect(canonifyTarget(queryToTarget(target1))).to.equal(
105+
canonifyTarget(queryToTarget(target2))
106+
);
107+
});
96108
});

0 commit comments

Comments
 (0)