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

Conversation

DellaBitta
Copy link
Contributor

@DellaBitta DellaBitta commented Mar 27, 2025

Discussion

Merge the initial bundler implementation into the Firestore Result Serialization feature branch.
Implementation includes the ability to invoke toJSON() on QuerySnapshot and DocumentSnapshot instances.

Testing

Local tests to plumb the resulting bundle result into loadBundle on the client. Executed getDocFromCache to ensure that the document could be retrieved.

Also tested a new instance of Firestore by storing the bundle string in a file and executing a different node application with the string. This ensured that the cache didn't already have the original document from the result of getDoc or getDocs server provided results.

Added unit and integration tests.

API Changes

Waiting for the full feature branch to be complete before submitting an API review doc. This change includes the new methods:

  • QuerySnapshot.toJSON()
  • DocumentSnapshot.toJSON()

DellaBitta and others added 12 commits March 14, 2025 16:33
General type errors.
Removed QualifiedResourcePath.
Tested the results in loadBundle with a document consisting of all data types, and it succeeded.
name and parent fields aren't properly populated. Looks like we have extraneous "orderBy" fields, too, which might be harmless.
Also includes:

 -  format
 -  lint fixes.
 - Comments for bundler methods
 - Addition of DocumentSnapshotBundleData & QuerySnapshotBundleData types.
Copy link

changeset-bot bot commented Mar 27, 2025

⚠️ No Changeset found

Latest commit: a5b46b7

Merging this PR will not cause a version bump for any packages. If these changes should not result in a new version, you're good to go. If these changes should result in a version bump, you need to add a changeset.

This PR includes no changesets

When changesets are added to this PR, you'll see the packages that this PR includes changesets for and the associated semver types

Click here to learn what changesets are, and how to add one.

Click here if you're a maintainer who wants to add a changeset to this PR

@google-oss-bot
Copy link
Contributor

google-oss-bot commented Mar 27, 2025

Size Report 1

Affected Products

  • @firebase/firestore

    TypeBase (c8cbfff)Merge (4aa6dd5)Diff
    browser382 kB387 kB+5.67 kB (+1.5%)
    main591 kB599 kB+7.81 kB (+1.3%)
    module382 kB387 kB+5.67 kB (+1.5%)
    react-native382 kB388 kB+5.67 kB (+1.5%)
  • bundle

    TypeBase (c8cbfff)Merge (4aa6dd5)Diff
    firestore (CSI Auto Indexing Disable and Delete)272 kB274 kB+1.77 kB (+0.6%)
    firestore (CSI Auto Indexing Enable)272 kB274 kB+1.77 kB (+0.6%)
    firestore (Persistence)304 kB305 kB+1.77 kB (+0.6%)
    firestore (Query Cursors)250 kB255 kB+4.98 kB (+2.0%)
    firestore (Query)248 kB253 kB+4.98 kB (+2.0%)
    firestore (Read data once)236 kB244 kB+8.57 kB (+3.6%)
    firestore (Read Write w Persistence)328 kB332 kB+4.25 kB (+1.3%)
    firestore (Realtime updates)238 kB247 kB+9.25 kB (+3.9%)
    firestore (Transaction)215 kB221 kB+6.42 kB (+3.0%)
    firestore (Write data)214 kB216 kB+1.77 kB (+0.8%)
  • firebase

    TypeBase (c8cbfff)Merge (4aa6dd5)Diff
    firebase-compat.js794 kB799 kB+4.86 kB (+0.6%)
    firebase-firestore-compat.js340 kB345 kB+4.86 kB (+1.4%)
    firebase-firestore.js440 kB446 kB+5.68 kB (+1.3%)

Test Logs

  1. https://storage.googleapis.com/firebase-sdk-metric-reports/6YJPs0dpK4.html

@google-oss-bot
Copy link
Contributor

google-oss-bot commented Mar 27, 2025

Size Analysis Report 1

This report is too large (770,132 characters) to be displayed here in a GitHub comment. Please use the below link to see the full report on Google Cloud Storage.

Test Logs

  1. https://storage.googleapis.com/firebase-sdk-metric-reports/tViHxc9uE7.html

These were added to the API before I knew they could have been queried internally.
removed extraneous imports and private/protected changes.
Copy link
Contributor

@MarkDuckworth MarkDuckworth left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Overall, this is looking good. I have a few comments that we may need to address

if (
!originalDocument ||
(!readTime && !originalDocument.metadata.readTime) ||
(readTime && originalDocument.metadata.readTime! < readTime)
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

what happens if originalDocument.metadata.readTime is null|undefined?

Maybe this is a direct port from nodejs?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Ok, I cleaned this up quite a bit. PTAL.

if (queryName) {
const newDocument = this.documents.get(docBundleData.documentPath)!;
newDocument.metadata.queries = originalQueries || [];
if (queryName) {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think this will always be true because it's nested inside line 122

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Too right. Removed.

Comment on lines +138 to +139
* QuerySnapshot. Note we cannot accept a QuerySnapshot directly due to a circular
* dependency error.
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

nice fix for the circular dep

debugAssert(
!docBundleData.documentData.hasLocalMutations,
"Can't serialize documents with mutations."
);
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We will have to consider the impact of this assert on serializing documents with local mutations. This could be a breaking change if someone is already performing JSON.stringify(mutatedDoc). This is a use case that should not matter for SSR, but will matter because we are using toJSON for serialization.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I guess there's also a case where someone writes or modifies a document on the server side, and then runs a query to get that document (with modifications) before the server gets a write ack from the backend. This means the documentsnapshot would be with mutations...

I'm not even sure how SSR should behave in this case.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is a tricky situation for SSR. We will have to be clear that the server side code must use getDocumentsFromServer to ensure that we don't get a document with local mutation.

Also, I'm beginning to wonder if we should only override toJSON for these snapshot types when running server side. This helps avoid dealing with edge cases of serialization in the client side. Thoughts?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We discussed this offline, and resolved to throw in these situations. I've updated the branch accordingly. PTAL.

* @param value The input to validate.
* @param options Options that specify whether the string can be omitted.
*/
export function validateString(arg: string | number, value: unknown): void {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

it's not clear to me if any of the functions in this file are used or will be used?

Copy link
Contributor Author

@DellaBitta DellaBitta Mar 27, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Oh odd, I thought I had removed this source file. I must have messed up the git push. Will fix.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Fixed.

Copy link
Contributor

@MarkDuckworth MarkDuckworth left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

LGTM. One suggestion to correct the error message

throw new FirestoreError(
Code.FAILED_PRECONDITION,
'DocumentSnapshot.toJSON attempted to serialize a document with pending writes. ' +
'Await `waitForPendingWrites()` before invoking `toJSON()`.'
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think it needs to be 'Await waitForPendingWrites()` before getting documents.'

throw new FirestoreError(
Code.FAILED_PRECONDITION,
'QuerySnapshot.toJSON attempted to serialize a document with pending writes. ' +
'Await `waitForPendingWrites()` before invoking `toJSON()`.'
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

same

@DellaBitta DellaBitta changed the title [WIP] Add bundler to Firestore SSR Serialization feature branch [Feat] Add bundler implementation to Firestore SSR Serialization feature branch Apr 1, 2025
@DellaBitta DellaBitta marked this pull request as ready for review April 1, 2025 19:22
@DellaBitta DellaBitta requested review from a team as code owners April 1, 2025 19:22
@DellaBitta DellaBitta merged commit b85747b into ddb-firestore-result-serialization Apr 1, 2025
39 of 45 checks passed
@DellaBitta DellaBitta deleted the ddb-migrate-fs-bundler branch April 1, 2025 19:23
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

3 participants