Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[Feat] Add bundler implementation to Firestore SSR Serialization feature branch #8872

Merged
merged 33 commits into from
Apr 1, 2025
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
33 commits
Select commit Hold shift + click to select a range
aee01c3
first attempt
DellaBitta Mar 14, 2025
c18ddd5
Build fixes
DellaBitta Mar 15, 2025
9ebb633
Add Firestore instance to constructor
DellaBitta Mar 17, 2025
3c2e837
lint fixes
DellaBitta Mar 17, 2025
71d24b4
first addQuery impl
DellaBitta Mar 17, 2025
5e9ceb8
reduced tools in validation utils.
DellaBitta Mar 17, 2025
5dd5ae4
remove unneeded utils.
DellaBitta Mar 18, 2025
3d45407
Ideas to support converting bundle protos to Proto3 JSON compatible f…
DellaBitta Mar 20, 2025
c361963
Format and coalesce utils.
DellaBitta Mar 20, 2025
a1e6a0b
Bundle DocumentSnaps with toJSON
DellaBitta Mar 25, 2025
5e3620d
QuerySnapshot toJSON impl, but it has bugs
DellaBitta Mar 25, 2025
ffff490
Successfully tested bundling a QuerySnapshot
DellaBitta Mar 26, 2025
3bcf31e
Remove createTime & readTime from DocumentSnapshot
DellaBitta Mar 27, 2025
ccd5205
rename documentToDocumentSnapshotBundleData
DellaBitta Mar 27, 2025
03f3ffb
Cleanup pass
DellaBitta Mar 27, 2025
06d84e0
Remove unneeded bundle_builder_validation_utils.ts
DellaBitta Mar 28, 2025
1b53351
clean up addBundleDocument readTime logic.
DellaBitta Mar 28, 2025
5f413b6
removed extra `if(queryName)` in addBundleDocument
DellaBitta Mar 28, 2025
e0f3c29
Update fromTimestamp doc.
DellaBitta Mar 28, 2025
fbfa24f
toJSON methods warn of pending writes.
DellaBitta Mar 28, 2025
070a84a
throw instead of warn.
DellaBitta Mar 28, 2025
8e0b3bb
format
DellaBitta Mar 28, 2025
65c3d8b
metadata contains hasPendingWrites
DellaBitta Mar 28, 2025
73eb8a2
Unit tests.
DellaBitta Mar 28, 2025
6251a28
Update database.test.ts
DellaBitta Mar 28, 2025
2a3a7df
Add integration test.
DellaBitta Mar 31, 2025
f009e35
Merge branch 'main' into ddb-migrate-fs-bundler
DellaBitta Mar 31, 2025
afd294a
Docs
DellaBitta Mar 31, 2025
06f71c9
debug logs
DellaBitta Apr 1, 2025
4ec62ae
Guard test against missing path.
DellaBitta Apr 1, 2025
36e4fd5
bundle test
DellaBitta Apr 1, 2025
bad6f72
Minify fix.
DellaBitta Apr 1, 2025
a5b46b7
lint fixes, console log removal.
DellaBitta Apr 1, 2025
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
4 changes: 4 additions & 0 deletions common/api-review/firestore.api.md
Original file line number Diff line number Diff line change
Expand Up @@ -178,6 +178,8 @@ export class DocumentSnapshot<AppModelType = DocumentData, DbModelType extends D
get id(): string;
readonly metadata: SnapshotMetadata;
get ref(): DocumentReference<AppModelType, DbModelType>;
// (undocumented)
toJSON(): object;
}

export { EmulatorMockTokenOptions }
Expand Down Expand Up @@ -610,6 +612,8 @@ export class QuerySnapshot<AppModelType = DocumentData, DbModelType extends Docu
readonly metadata: SnapshotMetadata;
readonly query: Query<AppModelType, DbModelType>;
get size(): number;
// (undocumented)
toJSON(): object;
}

// @public
Expand Down
12 changes: 12 additions & 0 deletions docs-devsite/firestore_.documentsnapshot.md
Original file line number Diff line number Diff line change
Expand Up @@ -41,6 +41,7 @@ export declare class DocumentSnapshot<AppModelType = DocumentData, DbModelType e
| [data(options)](./firestore_.documentsnapshot.md#documentsnapshotdata) | | Retrieves all fields in the document as an <code>Object</code>. Returns <code>undefined</code> if the document doesn't exist.<!-- -->By default, <code>serverTimestamp()</code> values that have not yet been set to their final value will be returned as <code>null</code>. You can override this by passing an options object. |
| [exists()](./firestore_.documentsnapshot.md#documentsnapshotexists) | | Returns whether or not the data exists. True if the document exists. |
| [get(fieldPath, options)](./firestore_.documentsnapshot.md#documentsnapshotget) | | Retrieves the field specified by <code>fieldPath</code>. Returns <code>undefined</code> if the document or field doesn't exist.<!-- -->By default, a <code>serverTimestamp()</code> that has not yet been set to its final value will be returned as <code>null</code>. You can override this by passing an options object. |
| [toJSON()](./firestore_.documentsnapshot.md#documentsnapshottojson) | | |

## DocumentSnapshot.(constructor)

Expand Down Expand Up @@ -144,3 +145,14 @@ any

The data at the specified field location or undefined if no such field exists in the document.

## DocumentSnapshot.toJSON()

<b>Signature:</b>

```typescript
toJSON(): object;
```
<b>Returns:</b>

object

12 changes: 12 additions & 0 deletions docs-devsite/firestore_.querysnapshot.md
Original file line number Diff line number Diff line change
Expand Up @@ -34,6 +34,7 @@ export declare class QuerySnapshot<AppModelType = DocumentData, DbModelType exte
| --- | --- | --- |
| [docChanges(options)](./firestore_.querysnapshot.md#querysnapshotdocchanges) | | Returns an array of the documents changes since the last snapshot. If this is the first snapshot, all documents will be in the list as 'added' changes. |
| [forEach(callback, thisArg)](./firestore_.querysnapshot.md#querysnapshotforeach) | | Enumerates all of the documents in the <code>QuerySnapshot</code>. |
| [toJSON()](./firestore_.querysnapshot.md#querysnapshottojson) | | |

## QuerySnapshot.docs

Expand Down Expand Up @@ -126,3 +127,14 @@ forEach(callback: (result: QueryDocumentSnapshot<AppModelType, DbModelType>) =>

void

## QuerySnapshot.toJSON()

<b>Signature:</b>

```typescript
toJSON(): object;
```
<b>Returns:</b>

object

109 changes: 109 additions & 0 deletions packages/firestore/src/api/snapshot.ts
Original file line number Diff line number Diff line change
Expand Up @@ -36,7 +36,13 @@ import { AbstractUserDataWriter } from '../lite-api/user_data_writer';
import { Document } from '../model/document';
import { DocumentKey } from '../model/document_key';
import { debugAssert, fail } from '../util/assert';
import {
BundleBuilder,
DocumentSnapshotBundleData,
QuerySnapshotBundleData
} from '../util/bundle_builder_impl';
import { Code, FirestoreError } from '../util/error';
import { AutoId } from '../util/misc';

import { Firestore } from './database';
import { SnapshotListenOptions } from './reference_impl';
Expand Down Expand Up @@ -496,6 +502,46 @@ export class DocumentSnapshot<
}
return undefined;
}

toJSON(): object {
const document = this._document;
// eslint-disable-next-line @typescript-eslint/no-explicit-any
const result: any = {};
result['bundle'] = '';
result['source'] = 'DocumentSnapshot';

if (
!document ||
!document.isValidDocument() ||
!document.isFoundDocument()
) {
return result;
}
const builder: BundleBuilder = new BundleBuilder(
this._firestore,
AutoId.newId()
);
const documentData = this._userDataWriter.convertObjectMap(
document.data.value.mapValue.fields,
'previous'
);
if (this.metadata.hasPendingWrites) {
throw new FirestoreError(
Code.FAILED_PRECONDITION,
'DocumentSnapshot.toJSON() attempted to serialize a document with pending writes. ' +
'Await waitForPendingWrites() before invoking toJSON().'
);
}
builder.addBundleDocument(
documentToDocumentSnapshotBundleData(
this.ref.path,
documentData,
document
)
);
result['bundle'] = builder.build();
return result;
}
}

/**
Expand Down Expand Up @@ -651,6 +697,52 @@ export class QuerySnapshot<

return this._cachedChanges;
}

toJSON(): object {
// eslint-disable-next-line @typescript-eslint/no-explicit-any
const result: any = {};
result['source'] = 'QuerySnapshot';
const builder: BundleBuilder = new BundleBuilder(
this._firestore,
AutoId.newId()
);
const databaseId = this._firestore._databaseId.database;
const projectId = this._firestore._databaseId.projectId;
const parent = `projects/${projectId}/databases/${databaseId}/documents`;
const docBundleDataArray: DocumentSnapshotBundleData[] = [];
const docArray = this.docs;
docArray.forEach(doc => {
if (doc._document === null) {
return;
}
const documentData = this._userDataWriter.convertObjectMap(
doc._document.data.value.mapValue.fields,
'previous'
);
if (this.metadata.hasPendingWrites) {
throw new FirestoreError(
Code.FAILED_PRECONDITION,
'QuerySnapshot.toJSON() attempted to serialize a document with pending writes. ' +
'Await waitForPendingWrites() before invoking toJSON().'
);
}
docBundleDataArray.push(
documentToDocumentSnapshotBundleData(
doc.ref.path,
documentData,
doc._document
)
);
});
const bundleData: QuerySnapshotBundleData = {
query: this.query._query,
parent,
docBundleDataArray
};
builder.addBundleQuery(bundleData);
result['bundle'] = builder.build();
return result;
}
}

/** Calculates the array of `DocumentChange`s for a given `ViewSnapshot`. */
Expand Down Expand Up @@ -790,3 +882,20 @@ export function snapshotEqual<AppModelType, DbModelType extends DocumentData>(

return false;
}

// Formats Document data for bundling a DocumentSnapshot.
function documentToDocumentSnapshotBundleData(
path: string,
documentData: DocumentData,
document: Document
): DocumentSnapshotBundleData {
return {
documentData,
documentKey: document.mutableCopy().key,
documentPath: path,
documentExists: true,
createdTime: document.createTime.toTimestamp(),
readTime: document.readTime.toTimestamp(),
versionTime: document.version.toTimestamp()
};
}
2 changes: 1 addition & 1 deletion packages/firestore/src/lite-api/user_data_reader.ts
Original file line number Diff line number Diff line change
Expand Up @@ -778,7 +778,7 @@ export function parseData(
}
}

function parseObject(
export function parseObject(
obj: Dict<unknown>,
context: ParseContextImpl
): { mapValue: ProtoMapValue } {
Expand Down
5 changes: 4 additions & 1 deletion packages/firestore/src/remote/serializer.ts
Original file line number Diff line number Diff line change
Expand Up @@ -226,7 +226,10 @@ export function toTimestamp(
}
}

function fromTimestamp(date: ProtoTimestamp): Timestamp {
/**
* Returns a Timestamp typed object given protobuf timestamp value.
*/
export function fromTimestamp(date: ProtoTimestamp): Timestamp {
const timestamp = normalizeTimestamp(date);
return new Timestamp(timestamp.seconds, timestamp.nanos);
}
Expand Down
Loading
Loading