From 75f723dfa4a476ad4f0eaae100be44a98bae81e1 Mon Sep 17 00:00:00 2001 From: Stephen Rosa Date: Mon, 17 Mar 2025 13:45:32 -0700 Subject: [PATCH 01/10] feat(react/fdc) add for parsing arguments to variadic generated SDK query hook function signatures --- .../js/default-connector/README.md | 110 +++++++++++------- .../js/default-connector/index.cjs.js | 5 - .../js/default-connector/index.d.ts | 5 - package.json | 8 +- packages/react/src/data-connect/index.ts | 13 ++- .../data-connect/validateReactArgs.test.tsx | 0 .../src/data-connect/validateReactArgs.ts | 66 +++++++++++ pnpm-lock.yaml | 10 +- 8 files changed, 154 insertions(+), 63 deletions(-) create mode 100644 packages/react/src/data-connect/validateReactArgs.test.tsx create mode 100644 packages/react/src/data-connect/validateReactArgs.ts diff --git a/dataconnect-sdk/js/default-connector/README.md b/dataconnect-sdk/js/default-connector/README.md index 6ad5971f..19addde3 100644 --- a/dataconnect-sdk/js/default-connector/README.md +++ b/dataconnect-sdk/js/default-connector/README.md @@ -1,23 +1,34 @@ -# Generated TypeScript README +# Table of Contents +- [**Overview**](#generated-typescript-readme) +- [**Accessing the connector**](#accessing-the-connector) + - [*Connecting to the local Emulator*](#connecting-to-the-local-emulator) +- [**Queries**](#queries) + - [*ListMovies*](#listmovies) + - [*GetMovieById*](#getmoviebyid) +- [**Mutations**](#mutations) + - [*CreateMovie*](#createmovie) + - [*UpsertMovie*](#upsertmovie) + - [*DeleteMovie*](#deletemovie) + +# Generated TypeScript README This README will guide you through the process of using the generated TypeScript SDK package for the connector `default`. It will also provide examples on how to use your generated SDK to call your Data Connect queries and mutations. ***NOTE:** This README is generated alongside the generated SDK. If you make changes to this file, they will be overwritten when the SDK is regenerated.* You can use this generated SDK by importing from the package `@dataconnect/default-connector` as shown below. Both CommonJS and ESM imports are supported. + You can also follow the instructions from the [Data Connect documentation](https://firebase.google.com/docs/data-connect/web-sdk#set-client). # Accessing the connector -A connector is a collection of queries and mutations. One SDK is generated for each connector - this SDK is generated for the connector `default`. +A connector is a collection of Queries and Mutations. One SDK is generated for each connector - this SDK is generated for the connector `default`. You can find more information about connectors in the [Data Connect documentation](https://firebase.google.com/docs/data-connect#how-does). -In order to call Data Connect queries and mutations, you need to create an instance of the connector in your application code. - ```javascript import { getDataConnect, DataConnect } from 'firebase/data-connect'; import { connectorConfig } from '@dataconnect/default-connector'; -const connector: DataConnect = getDataConnect(connectorConfig); +const dataConnect = getDataConnect(connectorConfig); ``` ## Connecting to the local Emulator @@ -27,17 +38,17 @@ To connect to the emulator, you can use the following code. You can also follow the emulator instructions from the [Data Connect documentation](https://firebase.google.com/docs/data-connect/web-sdk#instrument-clients). ```javascript -// add connectDataConnectEmulator to your imports import { connectDataConnectEmulator, getDataConnect, DataConnect } from 'firebase/data-connect'; import { connectorConfig } from '@dataconnect/default-connector'; -const connector: DataConnect = getDataConnect(connectorConfig); -connectDataConnectEmulator(connector, 'localhost', 9399); +const dataConnect = getDataConnect(connectorConfig); +connectDataConnectEmulator(dataConnect, 'localhost', 9399); ``` -After it's initialized, you can call your Data Connect [queries](#queries) and [mutations](#mutations) from your generated SDK. +After it's initialized, you can call your Data Connect [queries](#queries) and [mutations](#mutations) from your generated SDK. # Queries + There are two ways to execute a Data Connect Query using the generated Web SDK: - Using a Query Reference function, which returns a `QueryRef` - The `QueryRef` can be used as an argument to `executeQuery()`, which will execute the Query and return a `QueryPromise` @@ -47,7 +58,7 @@ There are two ways to execute a Data Connect Query using the generated Web SDK: The following is true for both the action shortcut function and the `QueryRef` function: - The `QueryPromise` returned will resolve to the result of the Query once it has finished executing - If the Query accepts arguments, both the action shortcut function and the `QueryRef` function accept a single argument: an object that contains all the required variables (and the optional variables) for the Query -- Both functions can be called with or without passing in a `DataConnect` instance as an argument +- Both functions can be called with or without passing in a `DataConnect` instance as an argument. If no `DataConnect` argument is passed in, then the generated SDK will call `getDataConnect(connectorConfig)` behind the scenes for you. Below are examples of how to use the `default` connector's generated functions to execute each query. You can also follow the examples from the [Data Connect documentation](https://firebase.google.com/docs/data-connect/web-sdk#using-queries). @@ -68,7 +79,7 @@ listMoviesRef(dc: DataConnect): QueryRef; ### Variables The `ListMovies` query has no variables. ### Return Type -Recall that executing the `ListMovies` query returns a `QueryPromise` that resolves to an object with a `data` property. +Recall that executing the `ListMovies` query returns a `QueryPromise` that resolves to an object with a `data` property. The `data` property is an object of type `ListMoviesData`, which is defined in [default-connector/index.d.ts](./index.d.ts). It has the following fields: ```javascript @@ -87,13 +98,14 @@ export interface ListMoviesData { import { getDataConnect, DataConnect } from 'firebase/data-connect'; import { connectorConfig, listMovies } from '@dataconnect/default-connector'; + // Call the `listMovies()` function to execute the query. // You can use the `await` keyword to wait for the promise to resolve. const { data } = await listMovies(); // You can also pass in a `DataConnect` instance to the action shortcut function. -const connector: DataConnect = getDataConnect(connectorConfig); -const { data } = await listMovies(connector); +const dataConnect = getDataConnect(connectorConfig); +const { data } = await listMovies(dataConnect); console.log(data.movies); @@ -110,12 +122,13 @@ listMovies().then((response) => { import { getDataConnect, DataConnect, executeQuery } from 'firebase/data-connect'; import { connectorConfig, listMoviesRef } from '@dataconnect/default-connector'; + // Call the `listMoviesRef()` function to get a reference to the query. const ref = listMoviesRef(); // You can also pass in a `DataConnect` instance to the `QueryRef` function. -const connector: DataConnect = getDataConnect(connectorConfig); -const ref = listMoviesRef(connector); +const dataConnect = getDataConnect(connectorConfig); +const ref = listMoviesRef(dataConnect); // Call `executeQuery()` on the reference to execute the query. // You can use the `await` keyword to wait for the promise to resolve. @@ -153,7 +166,7 @@ export interface GetMovieByIdVariables { } ``` ### Return Type -Recall that executing the `GetMovieById` query returns a `QueryPromise` that resolves to an object with a `data` property. +Recall that executing the `GetMovieById` query returns a `QueryPromise` that resolves to an object with a `data` property. The `data` property is an object of type `GetMovieByIdData`, which is defined in [default-connector/index.d.ts](./index.d.ts). It has the following fields: ```javascript @@ -171,10 +184,11 @@ export interface GetMovieByIdData { ```javascript import { getDataConnect, DataConnect } from 'firebase/data-connect'; import { connectorConfig, getMovieById, GetMovieByIdVariables } from '@dataconnect/default-connector'; + // The `GetMovieById` query requires an argument of type `GetMovieByIdVariables`: const getMovieByIdVars: GetMovieByIdVariables = { id: ..., -} +}; // Call the `getMovieById()` function to execute the query. // You can use the `await` keyword to wait for the promise to resolve. @@ -183,8 +197,8 @@ const { data } = await getMovieById(getMovieByIdVars); const { data } = await getMovieById({ id: ..., }); // You can also pass in a `DataConnect` instance to the action shortcut function. -const connector: DataConnect = getDataConnect(connectorConfig); -const { data } = await getMovieById(connector, getMovieByIdVars); +const dataConnect = getDataConnect(connectorConfig); +const { data } = await getMovieById(dataConnect, getMovieByIdVars); console.log(data.movie); @@ -200,10 +214,11 @@ getMovieById(getMovieByIdVars).then((response) => { ```javascript import { getDataConnect, DataConnect, executeQuery } from 'firebase/data-connect'; import { connectorConfig, getMovieByIdRef, GetMovieByIdVariables } from '@dataconnect/default-connector'; + // The `GetMovieById` query requires an argument of type `GetMovieByIdVariables`: const getMovieByIdVars: GetMovieByIdVariables = { id: ..., -} +}; // Call the `getMovieByIdRef()` function to get a reference to the query. const ref = getMovieByIdRef(getMovieByIdVars); @@ -211,8 +226,8 @@ const ref = getMovieByIdRef(getMovieByIdVars); const ref = getMovieByIdRef({ id: ..., }); // You can also pass in a `DataConnect` instance to the `QueryRef` function. -const connector: DataConnect = getDataConnect(connectorConfig); -const ref = getMovieByIdRef(connector, getMovieByIdVars); +const dataConnect = getDataConnect(connectorConfig); +const ref = getMovieByIdRef(dataConnect, getMovieByIdVars); // Call `executeQuery()` on the reference to execute the query. // You can use the `await` keyword to wait for the promise to resolve. @@ -228,6 +243,7 @@ executeQuery(ref).then((response) => { ``` # Mutations + There are two ways to execute a Data Connect Mutation using the generated Web SDK: - Using a Mutation Reference function, which returns a `MutationRef` - The `MutationRef` can be used as an argument to `executeMutation()`, which will execute the Mutation and return a `MutationPromise` @@ -237,7 +253,7 @@ There are two ways to execute a Data Connect Mutation using the generated Web SD The following is true for both the action shortcut function and the `MutationRef` function: - The `MutationPromise` returned will resolve to the result of the Mutation once it has finished executing - If the Mutation accepts arguments, both the action shortcut function and the `MutationRef` function accept a single argument: an object that contains all the required variables (and the optional variables) for the Mutation -- Both functions can be called with or without passing in a `DataConnect` instance as an argument +- Both functions can be called with or without passing in a `DataConnect` instance as an argument. If no `DataConnect` argument is passed in, then the generated SDK will call `getDataConnect(connectorConfig)` behind the scenes for you. Below are examples of how to use the `default` connector's generated functions to execute each mutation. You can also follow the examples from the [Data Connect documentation](https://firebase.google.com/docs/data-connect/web-sdk#using-mutations). @@ -266,7 +282,7 @@ export interface CreateMovieVariables { } ``` ### Return Type -Recall that executing the `CreateMovie` mutation returns a `MutationPromise` that resolves to an object with a `data` property. +Recall that executing the `CreateMovie` mutation returns a `MutationPromise` that resolves to an object with a `data` property. The `data` property is an object of type `CreateMovieData`, which is defined in [default-connector/index.d.ts](./index.d.ts). It has the following fields: ```javascript @@ -279,12 +295,13 @@ export interface CreateMovieData { ```javascript import { getDataConnect, DataConnect } from 'firebase/data-connect'; import { connectorConfig, createMovie, CreateMovieVariables } from '@dataconnect/default-connector'; + // The `CreateMovie` mutation requires an argument of type `CreateMovieVariables`: const createMovieVars: CreateMovieVariables = { title: ..., genre: ..., imageUrl: ..., -} +}; // Call the `createMovie()` function to execute the mutation. // You can use the `await` keyword to wait for the promise to resolve. @@ -293,8 +310,8 @@ const { data } = await createMovie(createMovieVars); const { data } = await createMovie({ title: ..., genre: ..., imageUrl: ..., }); // You can also pass in a `DataConnect` instance to the action shortcut function. -const connector: DataConnect = getDataConnect(connectorConfig); -const { data } = await createMovie(connector, createMovieVars); +const dataConnect = getDataConnect(connectorConfig); +const { data } = await createMovie(dataConnect, createMovieVars); console.log(data.movie_insert); @@ -310,12 +327,13 @@ createMovie(createMovieVars).then((response) => { ```javascript import { getDataConnect, DataConnect, executeMutation } from 'firebase/data-connect'; import { connectorConfig, createMovieRef, CreateMovieVariables } from '@dataconnect/default-connector'; + // The `CreateMovie` mutation requires an argument of type `CreateMovieVariables`: const createMovieVars: CreateMovieVariables = { title: ..., genre: ..., imageUrl: ..., -} +}; // Call the `createMovieRef()` function to get a reference to the mutation. const ref = createMovieRef(createMovieVars); @@ -323,8 +341,8 @@ const ref = createMovieRef(createMovieVars); const ref = createMovieRef({ title: ..., genre: ..., imageUrl: ..., }); // You can also pass in a `DataConnect` instance to the `MutationRef` function. -const connector: DataConnect = getDataConnect(connectorConfig); -const ref = createMovieRef(connector, createMovieVars); +const dataConnect = getDataConnect(connectorConfig); +const ref = createMovieRef(dataConnect, createMovieVars); // Call `executeMutation()` on the reference to execute the mutation. // You can use the `await` keyword to wait for the promise to resolve. @@ -364,7 +382,7 @@ export interface UpsertMovieVariables { } ``` ### Return Type -Recall that executing the `UpsertMovie` mutation returns a `MutationPromise` that resolves to an object with a `data` property. +Recall that executing the `UpsertMovie` mutation returns a `MutationPromise` that resolves to an object with a `data` property. The `data` property is an object of type `UpsertMovieData`, which is defined in [default-connector/index.d.ts](./index.d.ts). It has the following fields: ```javascript @@ -377,12 +395,13 @@ export interface UpsertMovieData { ```javascript import { getDataConnect, DataConnect } from 'firebase/data-connect'; import { connectorConfig, upsertMovie, UpsertMovieVariables } from '@dataconnect/default-connector'; + // The `UpsertMovie` mutation requires an argument of type `UpsertMovieVariables`: const upsertMovieVars: UpsertMovieVariables = { id: ..., title: ..., imageUrl: ..., -} +}; // Call the `upsertMovie()` function to execute the mutation. // You can use the `await` keyword to wait for the promise to resolve. @@ -391,8 +410,8 @@ const { data } = await upsertMovie(upsertMovieVars); const { data } = await upsertMovie({ id: ..., title: ..., imageUrl: ..., }); // You can also pass in a `DataConnect` instance to the action shortcut function. -const connector: DataConnect = getDataConnect(connectorConfig); -const { data } = await upsertMovie(connector, upsertMovieVars); +const dataConnect = getDataConnect(connectorConfig); +const { data } = await upsertMovie(dataConnect, upsertMovieVars); console.log(data.movie_upsert); @@ -408,12 +427,13 @@ upsertMovie(upsertMovieVars).then((response) => { ```javascript import { getDataConnect, DataConnect, executeMutation } from 'firebase/data-connect'; import { connectorConfig, upsertMovieRef, UpsertMovieVariables } from '@dataconnect/default-connector'; + // The `UpsertMovie` mutation requires an argument of type `UpsertMovieVariables`: const upsertMovieVars: UpsertMovieVariables = { id: ..., title: ..., imageUrl: ..., -} +}; // Call the `upsertMovieRef()` function to get a reference to the mutation. const ref = upsertMovieRef(upsertMovieVars); @@ -421,8 +441,8 @@ const ref = upsertMovieRef(upsertMovieVars); const ref = upsertMovieRef({ id: ..., title: ..., imageUrl: ..., }); // You can also pass in a `DataConnect` instance to the `MutationRef` function. -const connector: DataConnect = getDataConnect(connectorConfig); -const ref = upsertMovieRef(connector, upsertMovieVars); +const dataConnect = getDataConnect(connectorConfig); +const ref = upsertMovieRef(dataConnect, upsertMovieVars); // Call `executeMutation()` on the reference to execute the mutation. // You can use the `await` keyword to wait for the promise to resolve. @@ -460,7 +480,7 @@ export interface DeleteMovieVariables { } ``` ### Return Type -Recall that executing the `DeleteMovie` mutation returns a `MutationPromise` that resolves to an object with a `data` property. +Recall that executing the `DeleteMovie` mutation returns a `MutationPromise` that resolves to an object with a `data` property. The `data` property is an object of type `DeleteMovieData`, which is defined in [default-connector/index.d.ts](./index.d.ts). It has the following fields: ```javascript @@ -473,10 +493,11 @@ export interface DeleteMovieData { ```javascript import { getDataConnect, DataConnect } from 'firebase/data-connect'; import { connectorConfig, deleteMovie, DeleteMovieVariables } from '@dataconnect/default-connector'; + // The `DeleteMovie` mutation requires an argument of type `DeleteMovieVariables`: const deleteMovieVars: DeleteMovieVariables = { id: ..., -} +}; // Call the `deleteMovie()` function to execute the mutation. // You can use the `await` keyword to wait for the promise to resolve. @@ -485,8 +506,8 @@ const { data } = await deleteMovie(deleteMovieVars); const { data } = await deleteMovie({ id: ..., }); // You can also pass in a `DataConnect` instance to the action shortcut function. -const connector: DataConnect = getDataConnect(connectorConfig); -const { data } = await deleteMovie(connector, deleteMovieVars); +const dataConnect = getDataConnect(connectorConfig); +const { data } = await deleteMovie(dataConnect, deleteMovieVars); console.log(data.movie_delete); @@ -502,10 +523,11 @@ deleteMovie(deleteMovieVars).then((response) => { ```javascript import { getDataConnect, DataConnect, executeMutation } from 'firebase/data-connect'; import { connectorConfig, deleteMovieRef, DeleteMovieVariables } from '@dataconnect/default-connector'; + // The `DeleteMovie` mutation requires an argument of type `DeleteMovieVariables`: const deleteMovieVars: DeleteMovieVariables = { id: ..., -} +}; // Call the `deleteMovieRef()` function to get a reference to the mutation. const ref = deleteMovieRef(deleteMovieVars); @@ -513,8 +535,8 @@ const ref = deleteMovieRef(deleteMovieVars); const ref = deleteMovieRef({ id: ..., }); // You can also pass in a `DataConnect` instance to the `MutationRef` function. -const connector: DataConnect = getDataConnect(connectorConfig); -const ref = deleteMovieRef(connector, deleteMovieVars); +const dataConnect = getDataConnect(connectorConfig); +const ref = deleteMovieRef(dataConnect, deleteMovieVars); // Call `executeMutation()` on the reference to execute the mutation. // You can use the `await` keyword to wait for the promise to resolve. diff --git a/dataconnect-sdk/js/default-connector/index.cjs.js b/dataconnect-sdk/js/default-connector/index.cjs.js index 88a43c97..d162c425 100644 --- a/dataconnect-sdk/js/default-connector/index.cjs.js +++ b/dataconnect-sdk/js/default-connector/index.cjs.js @@ -12,7 +12,6 @@ exports.createMovieRef = function createMovieRef(dcOrVars, vars) { dcInstance._useGeneratedSdk(); return mutationRef(dcInstance, 'CreateMovie', inputVars); } - exports.createMovie = function createMovie(dcOrVars, vars) { return executeMutation(createMovieRef(dcOrVars, vars)); }; @@ -21,7 +20,6 @@ exports.upsertMovieRef = function upsertMovieRef(dcOrVars, vars) { dcInstance._useGeneratedSdk(); return mutationRef(dcInstance, 'UpsertMovie', inputVars); } - exports.upsertMovie = function upsertMovie(dcOrVars, vars) { return executeMutation(upsertMovieRef(dcOrVars, vars)); }; @@ -30,7 +28,6 @@ exports.deleteMovieRef = function deleteMovieRef(dcOrVars, vars) { dcInstance._useGeneratedSdk(); return mutationRef(dcInstance, 'DeleteMovie', inputVars); } - exports.deleteMovie = function deleteMovie(dcOrVars, vars) { return executeMutation(deleteMovieRef(dcOrVars, vars)); }; @@ -39,7 +36,6 @@ exports.listMoviesRef = function listMoviesRef(dc) { dcInstance._useGeneratedSdk(); return queryRef(dcInstance, 'ListMovies'); } - exports.listMovies = function listMovies(dc) { return executeQuery(listMoviesRef(dc)); }; @@ -48,7 +44,6 @@ exports.getMovieByIdRef = function getMovieByIdRef(dcOrVars, vars) { dcInstance._useGeneratedSdk(); return queryRef(dcInstance, 'GetMovieById', inputVars); } - exports.getMovieById = function getMovieById(dcOrVars, vars) { return executeQuery(getMovieByIdRef(dcOrVars, vars)); }; diff --git a/dataconnect-sdk/js/default-connector/index.d.ts b/dataconnect-sdk/js/default-connector/index.d.ts index 937b6ee2..5fb465e7 100644 --- a/dataconnect-sdk/js/default-connector/index.d.ts +++ b/dataconnect-sdk/js/default-connector/index.d.ts @@ -68,7 +68,6 @@ export interface UpsertMovieVariables { imageUrl: string; } - /* Allow users to create refs without passing in DataConnect */ export function createMovieRef(vars: CreateMovieVariables): MutationRef; /* Allow users to pass in custom DataConnect instances */ @@ -77,7 +76,6 @@ export function createMovieRef(dc: DataConnect, vars: CreateMovieVariables): Mut export function createMovie(vars: CreateMovieVariables): MutationPromise; export function createMovie(dc: DataConnect, vars: CreateMovieVariables): MutationPromise; - /* Allow users to create refs without passing in DataConnect */ export function upsertMovieRef(vars: UpsertMovieVariables): MutationRef; /* Allow users to pass in custom DataConnect instances */ @@ -86,7 +84,6 @@ export function upsertMovieRef(dc: DataConnect, vars: UpsertMovieVariables): Mut export function upsertMovie(vars: UpsertMovieVariables): MutationPromise; export function upsertMovie(dc: DataConnect, vars: UpsertMovieVariables): MutationPromise; - /* Allow users to create refs without passing in DataConnect */ export function deleteMovieRef(vars: DeleteMovieVariables): MutationRef; /* Allow users to pass in custom DataConnect instances */ @@ -95,7 +92,6 @@ export function deleteMovieRef(dc: DataConnect, vars: DeleteMovieVariables): Mut export function deleteMovie(vars: DeleteMovieVariables): MutationPromise; export function deleteMovie(dc: DataConnect, vars: DeleteMovieVariables): MutationPromise; - /* Allow users to create refs without passing in DataConnect */ export function listMoviesRef(): QueryRef; /* Allow users to pass in custom DataConnect instances */ @@ -104,7 +100,6 @@ export function listMoviesRef(dc: DataConnect): QueryRef; export function listMovies(dc: DataConnect): QueryPromise; - /* Allow users to create refs without passing in DataConnect */ export function getMovieByIdRef(vars: GetMovieByIdVariables): QueryRef; /* Allow users to pass in custom DataConnect instances */ diff --git a/package.json b/package.json index 87adc173..d2c9a1d4 100644 --- a/package.json +++ b/package.json @@ -24,6 +24,12 @@ "vitest": "^2.0.5" }, "dependencies": { - "@angular/fire": "^19.0.0" + "@angular/fire": "^19.0.0", + "default-connector": "link:../../../../../../Library/pnpm/global/5/node_modules/dataconnect-sdk/js/default-connector" + }, + "pnpm": { + "overrides": { + "default-connector": "link:../../../../../../Library/pnpm/global/5/node_modules/dataconnect-sdk/js/default-connector" + } } } diff --git a/packages/react/src/data-connect/index.ts b/packages/react/src/data-connect/index.ts index d833515d..6e40e0bd 100644 --- a/packages/react/src/data-connect/index.ts +++ b/packages/react/src/data-connect/index.ts @@ -1,13 +1,14 @@ export { - DataConnectQueryClient, - type DataConnectQueryOptions, + DataConnectQueryClient, + type DataConnectQueryOptions, } from "./query-client"; export { - useDataConnectQuery, - type useDataConnectQueryOptions, + useDataConnectQuery, + type useDataConnectQueryOptions, } from "./useDataConnectQuery"; export { - useDataConnectMutation, - type useDataConnectMutationOptions, + useDataConnectMutation, + type useDataConnectMutationOptions, } from "./useDataConnectMutation"; +export { validateReactArgs } from "./validateReactArgs"; export type { FlattenedQueryResult, FlattenedMutationResult } from "./types"; diff --git a/packages/react/src/data-connect/validateReactArgs.test.tsx b/packages/react/src/data-connect/validateReactArgs.test.tsx new file mode 100644 index 00000000..e69de29b diff --git a/packages/react/src/data-connect/validateReactArgs.ts b/packages/react/src/data-connect/validateReactArgs.ts new file mode 100644 index 00000000..bc20069d --- /dev/null +++ b/packages/react/src/data-connect/validateReactArgs.ts @@ -0,0 +1,66 @@ +import { + ConnectorConfig, + DataConnect, + getDataConnect, +} from "firebase/data-connect"; + +interface ParsedReactArgs { + dc: DataConnect; + vars: Variables; + options: Options; +} + +/** + * The generated React SDK will allow the user to pass in variables, a Data Connect instance, or operation options. + * The only required argument is the variables, which are only required when the operation has at least one required + * variable. Otherwise, all arguments are optional. This function validates the variables and returns back the DataConnect + * instance, variables, and options based on the arguments passed in. + * @param connectorConfig DataConnect connector config + * @param dcOrVarsOrOptions the first argument provided to a generated react function + * @param varsOrOptions the second argument provided to a generated react function + * @param options the third argument provided to a generated react function + * @param hasVars boolean parameter indicating whether the operation has variables + * @param validateVars boolean parameter indicating whether we should expect to find a value for realVars + * @returns parsed DataConnect, Variables, and Options for the operation + * @internal + */ +export function validateReactArgs< + Variables extends object, + Options extends object +>( + connectorConfig: ConnectorConfig, + dcOrVarsOrOptions?: DataConnect | Variables | Options, + varsOrOptions?: Variables | Options, + options?: Options, + hasVars?: boolean, + validateVars?: boolean +): ParsedReactArgs { + let dcInstance: DataConnect; + let realVars: Variables; + let realOptions: Options; + + if (dcOrVarsOrOptions && "enableEmulator" in dcOrVarsOrOptions) { + dcInstance = dcOrVarsOrOptions as DataConnect; + if (hasVars) { + realVars = varsOrOptions as Variables; + realOptions = options as Options; + } else { + realVars = undefined as unknown as Variables; + realOptions = varsOrOptions as Options; + } + } else { + dcInstance = getDataConnect(connectorConfig); + if (hasVars) { + realVars = dcOrVarsOrOptions as Variables; + realOptions = varsOrOptions as Options; + } else { + realVars = undefined as unknown as Variables; + realOptions = dcOrVarsOrOptions as Options; + } + } + + if (!dcInstance || (!realVars && validateVars)) { + throw new Error("invalid-argument: Variables required."); // copied from firebase error codes + } + return { dc: dcInstance, vars: realVars, options: realOptions }; +} diff --git a/pnpm-lock.yaml b/pnpm-lock.yaml index 471e2ea5..2b3d9a8b 100644 --- a/pnpm-lock.yaml +++ b/pnpm-lock.yaml @@ -4,6 +4,9 @@ settings: autoInstallPeers: true excludeLinksFromLockfile: false +overrides: + default-connector: link:../../../../../../Library/pnpm/global/5/node_modules/dataconnect-sdk/js/default-connector + importers: .: @@ -11,6 +14,9 @@ importers: '@angular/fire': specifier: ^19.0.0 version: 19.0.0(@angular/common@19.1.8(@angular/core@19.1.8(rxjs@7.8.1)(zone.js@0.15.0))(rxjs@7.8.1))(@angular/core@19.1.8(rxjs@7.8.1)(zone.js@0.15.0))(@angular/platform-browser-dynamic@19.1.8(@angular/common@19.1.8(@angular/core@19.1.8(rxjs@7.8.1)(zone.js@0.15.0))(rxjs@7.8.1))(@angular/compiler@19.1.8(@angular/core@19.1.8(rxjs@7.8.1)(zone.js@0.15.0)))(@angular/core@19.1.8(rxjs@7.8.1)(zone.js@0.15.0))(@angular/platform-browser@19.1.8(@angular/animations@19.1.8(@angular/core@19.1.8(rxjs@7.8.1)(zone.js@0.15.0)))(@angular/common@19.1.8(@angular/core@19.1.8(rxjs@7.8.1)(zone.js@0.15.0))(rxjs@7.8.1))(@angular/core@19.1.8(rxjs@7.8.1)(zone.js@0.15.0))))(@angular/platform-browser@19.1.8(@angular/animations@19.1.8(@angular/core@19.1.8(rxjs@7.8.1)(zone.js@0.15.0)))(@angular/common@19.1.8(@angular/core@19.1.8(rxjs@7.8.1)(zone.js@0.15.0))(rxjs@7.8.1))(@angular/core@19.1.8(rxjs@7.8.1)(zone.js@0.15.0)))(chokidar@4.0.3)(rxjs@7.8.1) + default-connector: + specifier: link:../../../../../../Library/pnpm/global/5/node_modules/dataconnect-sdk/js/default-connector + version: link:../../../../../../Library/pnpm/global/5/node_modules/dataconnect-sdk/js/default-connector devDependencies: '@angular/core': specifier: ^19.1.8 @@ -5174,7 +5180,7 @@ snapshots: transitivePeerDependencies: - supports-color - eslint-module-utils@2.12.0(@typescript-eslint/parser@8.25.0(eslint@9.21.0(jiti@1.21.7))(typescript@5.7.3))(eslint-import-resolver-node@0.3.9)(eslint-import-resolver-typescript@3.8.3(eslint-plugin-import@2.31.0)(eslint@9.21.0(jiti@1.21.7)))(eslint@9.21.0(jiti@1.21.7)): + eslint-module-utils@2.12.0(@typescript-eslint/parser@8.25.0(eslint@9.21.0(jiti@1.21.7))(typescript@5.7.3))(eslint-import-resolver-node@0.3.9)(eslint-import-resolver-typescript@3.8.3)(eslint@9.21.0(jiti@1.21.7)): dependencies: debug: 3.2.7 optionalDependencies: @@ -5196,7 +5202,7 @@ snapshots: doctrine: 2.1.0 eslint: 9.21.0(jiti@1.21.7) eslint-import-resolver-node: 0.3.9 - eslint-module-utils: 2.12.0(@typescript-eslint/parser@8.25.0(eslint@9.21.0(jiti@1.21.7))(typescript@5.7.3))(eslint-import-resolver-node@0.3.9)(eslint-import-resolver-typescript@3.8.3(eslint-plugin-import@2.31.0)(eslint@9.21.0(jiti@1.21.7)))(eslint@9.21.0(jiti@1.21.7)) + eslint-module-utils: 2.12.0(@typescript-eslint/parser@8.25.0(eslint@9.21.0(jiti@1.21.7))(typescript@5.7.3))(eslint-import-resolver-node@0.3.9)(eslint-import-resolver-typescript@3.8.3)(eslint@9.21.0(jiti@1.21.7)) hasown: 2.0.2 is-core-module: 2.16.1 is-glob: 4.0.3 From 744afefa4b3ed1ac2fc50874c1fe534bf11fb699 Mon Sep 17 00:00:00 2001 From: Stephen Rosa Date: Mon, 17 Mar 2025 15:06:47 -0700 Subject: [PATCH 02/10] add unit tests for validateReactArgs --- .../data-connect/validateReactArgs.test.tsx | 288 ++++++++++++++++++ 1 file changed, 288 insertions(+) diff --git a/packages/react/src/data-connect/validateReactArgs.test.tsx b/packages/react/src/data-connect/validateReactArgs.test.tsx index e69de29b..62a7e2de 100644 --- a/packages/react/src/data-connect/validateReactArgs.test.tsx +++ b/packages/react/src/data-connect/validateReactArgs.test.tsx @@ -0,0 +1,288 @@ +import { describe, expect, test } from "vitest"; +import { validateReactArgs } from "./validateReactArgs"; +import { connectorConfig } from "@/dataconnect/default-connector"; +import { getDataConnect } from "firebase/data-connect"; +import { firebaseApp } from "~/testing-utils"; + +// initialize firebase app +firebaseApp; + +describe("validateReactArgs", () => { + const dataConnect = getDataConnect(connectorConfig); + + const emptyObjectVars = {}; + const strictlyRequiredArgs = { limit: 5 }; + const strictlyOptionalVars = { title: "a" }; + const allArgs = { limit: 5, title: "a" }; + + const options = { meta: { hasOptions: true } }; + + test.each([ + { + argsDescription: "no args are provided", + dcOrOptions: undefined, + options: undefined, + expectedInputVars: undefined, + expectedInputOpts: undefined, + }, + { + argsDescription: "only dataconnect is provided", + dcOrOptions: dataConnect, + options: undefined, + expectedInputVars: undefined, + expectedInputOpts: undefined, + }, + { + argsDescription: "only options are provided", + dcOrOptions: options, + options: undefined, + expectedInputVars: undefined, + expectedInputOpts: options, + }, + { + argsDescription: "dataconnect and options are provided", + dcOrOptions: dataConnect, + options: options, + expectedInputVars: undefined, + expectedInputOpts: options, + }, + ])( + "parses args correctly when $argsDescription for an operation with no variables", + ({ dcOrOptions, options, expectedInputVars, expectedInputOpts }) => { + const { + dc: dcInstance, + vars: inputVars, + options: inputOpts, + } = validateReactArgs( + connectorConfig, + dcOrOptions, + options + // hasVars = undefined (false-y) + // validateArgs = undefined (false-y) + ); + + expect(dcInstance).toBe(dataConnect); + + if (expectedInputVars) { + expect(inputVars).toBe(expectedInputVars); + } else { + expect(inputVars).toBeUndefined(); + } + + if (expectedInputOpts) { + expect(inputOpts).toBe(expectedInputOpts); + } else { + expect(inputOpts).toBeUndefined(); + } + } + ); + + test.each([ + { + argsDescription: "no args are provided", + dcOrVarsOrOptions: undefined, + varsOrOptions: undefined, + options: undefined, + expectedInputVars: undefined, + expectedInputOpts: undefined, + }, + { + argsDescription: "only dataconnect is provided", + dcOrVarsOrOptions: dataConnect, + varsOrOptions: undefined, + options: undefined, + expectedInputVars: undefined, + expectedInputOpts: undefined, + }, + { + argsDescription: "only an empty vars object is provided", + dcOrVarsOrOptions: emptyObjectVars, + varsOrOptions: undefined, + options: undefined, + expectedInputVars: emptyObjectVars, + expectedInputOpts: undefined, + }, + { + argsDescription: "only vars are provided", + dcOrVarsOrOptions: strictlyOptionalVars, + varsOrOptions: undefined, + options: undefined, + expectedInputVars: strictlyOptionalVars, + expectedInputOpts: undefined, + }, + { + argsDescription: "only options are provided", + dcOrVarsOrOptions: undefined, + varsOrOptions: options, + options: undefined, + expectedInputVars: undefined, + expectedInputOpts: options, + }, + { + argsDescription: "dataconnect and vars are provided", + dcOrVarsOrOptions: dataConnect, + varsOrOptions: strictlyOptionalVars, + options: undefined, + expectedInputVars: strictlyOptionalVars, + expectedInputOpts: undefined, + }, + { + argsDescription: "dataconnect and options are provided", + dcOrVarsOrOptions: dataConnect, + varsOrOptions: undefined, + options: options, + expectedInputVars: undefined, + expectedInputOpts: options, + }, + { + argsDescription: "dataconnect and vars and options are provided", + dcOrVarsOrOptions: dataConnect, + varsOrOptions: strictlyOptionalVars, + options: options, + expectedInputVars: strictlyOptionalVars, + expectedInputOpts: options, + }, + ])( + "parses args correctly when $argsDescription for an operation with all optional variables", + ({ + dcOrVarsOrOptions, + varsOrOptions, + options, + expectedInputVars, + expectedInputOpts, + }) => { + const { + dc: dcInstance, + vars: inputVars, + options: inputOpts, + } = validateReactArgs( + connectorConfig, + dcOrVarsOrOptions, + varsOrOptions, + options, + true, // hasVars = true + false // validateArgs = false + ); + + expect(dcInstance).toBe(dataConnect); + + if (expectedInputVars) { + expect(inputVars).toBe(expectedInputVars); + } else { + expect(inputVars).toBeUndefined(); + } + + if (expectedInputOpts) { + expect(inputOpts).toBe(expectedInputOpts); + } else { + expect(inputOpts).toBeUndefined(); + } + } + ); + + test.each([ + { + argsDescription: "only vars are provided", + dcOrVarsOrOptions: strictlyRequiredArgs, + varsOrOptions: undefined, + options: undefined, + expectedInputVars: strictlyRequiredArgs, + expectedInputOpts: undefined, + }, + { + argsDescription: "dataconnect and vars are provided", + dcOrVarsOrOptions: dataConnect, + varsOrOptions: strictlyRequiredArgs, + options: undefined, + expectedInputVars: strictlyRequiredArgs, + expectedInputOpts: undefined, + }, + { + argsDescription: "vars and options are provided", + dcOrVarsOrOptions: strictlyRequiredArgs, + varsOrOptions: options, + options: undefined, + expectedInputVars: strictlyRequiredArgs, + expectedInputOpts: options, + }, + { + argsDescription: "dataconnect and vars and options are provided", + dcOrVarsOrOptions: dataConnect, + varsOrOptions: strictlyRequiredArgs, + options: options, + expectedInputVars: strictlyRequiredArgs, + expectedInputOpts: options, + }, + ])( + "parses args correctly when $argsDescription for an operation with any required variables", + ({ + dcOrVarsOrOptions, + varsOrOptions, + options, + expectedInputVars, + expectedInputOpts, + }) => { + const { + dc: dcInstance, + vars: inputVars, + options: inputOpts, + } = validateReactArgs( + connectorConfig, + dcOrVarsOrOptions, + varsOrOptions, + options, + true, // hasVars = true + true // validateArgs = true + ); + + expect(dcInstance).toBe(dataConnect); + + if (expectedInputVars) { + expect(inputVars).toBe(expectedInputVars); + } else { + expect(inputVars).toBeUndefined(); + } + + if (expectedInputOpts) { + expect(inputOpts).toBe(expectedInputOpts); + } else { + expect(inputOpts).toBeUndefined(); + } + } + ); + + test.each([ + { + argsDescription: "only dataconnect is provided", + dcOrVarsOrOptions: dataConnect, + varsOrOptions: undefined, + options: undefined, + }, + { + argsDescription: "only options are provided", + dcOrVarsOrOptions: undefined, + varsOrOptions: options, + options: undefined, + }, + { + argsDescription: "only dataconnect and options are provided", + dcOrVarsOrOptions: dataConnect, + varsOrOptions: undefined, + options: options, + }, + ])( + "throws error when $argsDescription for an operation with any required variables", + ({ dcOrVarsOrOptions, varsOrOptions, options }) => { + expect(() => { + validateReactArgs( + connectorConfig, + dcOrVarsOrOptions, + varsOrOptions, + options, + true, // hasVars = true + true // validateArgs = true + ); + }).toThrowError("invalid-argument: Variables required."); + } + ); +}); From 7b57607bd57f5f0554967da04b2eb40874bfa8ed Mon Sep 17 00:00:00 2001 From: Stephen Rosa Date: Mon, 17 Mar 2025 15:12:03 -0700 Subject: [PATCH 03/10] update unit tests for validateReactArgs --- .../data-connect/validateReactArgs.test.tsx | 32 +++++++++---------- 1 file changed, 15 insertions(+), 17 deletions(-) diff --git a/packages/react/src/data-connect/validateReactArgs.test.tsx b/packages/react/src/data-connect/validateReactArgs.test.tsx index 62a7e2de..8c499bcc 100644 --- a/packages/react/src/data-connect/validateReactArgs.test.tsx +++ b/packages/react/src/data-connect/validateReactArgs.test.tsx @@ -11,9 +11,7 @@ describe("validateReactArgs", () => { const dataConnect = getDataConnect(connectorConfig); const emptyObjectVars = {}; - const strictlyRequiredArgs = { limit: 5 }; - const strictlyOptionalVars = { title: "a" }; - const allArgs = { limit: 5, title: "a" }; + const nonEmptyVars = { limit: 5 }; const options = { meta: { hasOptions: true } }; @@ -104,10 +102,10 @@ describe("validateReactArgs", () => { }, { argsDescription: "only vars are provided", - dcOrVarsOrOptions: strictlyOptionalVars, + dcOrVarsOrOptions: nonEmptyVars, varsOrOptions: undefined, options: undefined, - expectedInputVars: strictlyOptionalVars, + expectedInputVars: nonEmptyVars, expectedInputOpts: undefined, }, { @@ -121,9 +119,9 @@ describe("validateReactArgs", () => { { argsDescription: "dataconnect and vars are provided", dcOrVarsOrOptions: dataConnect, - varsOrOptions: strictlyOptionalVars, + varsOrOptions: nonEmptyVars, options: undefined, - expectedInputVars: strictlyOptionalVars, + expectedInputVars: nonEmptyVars, expectedInputOpts: undefined, }, { @@ -137,9 +135,9 @@ describe("validateReactArgs", () => { { argsDescription: "dataconnect and vars and options are provided", dcOrVarsOrOptions: dataConnect, - varsOrOptions: strictlyOptionalVars, + varsOrOptions: nonEmptyVars, options: options, - expectedInputVars: strictlyOptionalVars, + expectedInputVars: nonEmptyVars, expectedInputOpts: options, }, ])( @@ -183,34 +181,34 @@ describe("validateReactArgs", () => { test.each([ { argsDescription: "only vars are provided", - dcOrVarsOrOptions: strictlyRequiredArgs, + dcOrVarsOrOptions: nonEmptyVars, varsOrOptions: undefined, options: undefined, - expectedInputVars: strictlyRequiredArgs, + expectedInputVars: nonEmptyVars, expectedInputOpts: undefined, }, { argsDescription: "dataconnect and vars are provided", dcOrVarsOrOptions: dataConnect, - varsOrOptions: strictlyRequiredArgs, + varsOrOptions: nonEmptyVars, options: undefined, - expectedInputVars: strictlyRequiredArgs, + expectedInputVars: nonEmptyVars, expectedInputOpts: undefined, }, { argsDescription: "vars and options are provided", - dcOrVarsOrOptions: strictlyRequiredArgs, + dcOrVarsOrOptions: nonEmptyVars, varsOrOptions: options, options: undefined, - expectedInputVars: strictlyRequiredArgs, + expectedInputVars: nonEmptyVars, expectedInputOpts: options, }, { argsDescription: "dataconnect and vars and options are provided", dcOrVarsOrOptions: dataConnect, - varsOrOptions: strictlyRequiredArgs, + varsOrOptions: nonEmptyVars, options: options, - expectedInputVars: strictlyRequiredArgs, + expectedInputVars: nonEmptyVars, expectedInputOpts: options, }, ])( From 9b232297ebf3770b42017d0f59f19c108700571b Mon Sep 17 00:00:00 2001 From: Stephen Rosa Date: Mon, 17 Mar 2025 15:52:04 -0700 Subject: [PATCH 04/10] remove some changes unintentionally added to this PR, fix formatting --- dataconnect-sdk/js/default-connector/index.cjs.js | 5 +++++ package.json | 8 +------- packages/react/src/data-connect/index.ts | 15 ++++++--------- 3 files changed, 12 insertions(+), 16 deletions(-) diff --git a/dataconnect-sdk/js/default-connector/index.cjs.js b/dataconnect-sdk/js/default-connector/index.cjs.js index d162c425..464a1b21 100644 --- a/dataconnect-sdk/js/default-connector/index.cjs.js +++ b/dataconnect-sdk/js/default-connector/index.cjs.js @@ -15,6 +15,7 @@ exports.createMovieRef = function createMovieRef(dcOrVars, vars) { exports.createMovie = function createMovie(dcOrVars, vars) { return executeMutation(createMovieRef(dcOrVars, vars)); }; + exports.upsertMovieRef = function upsertMovieRef(dcOrVars, vars) { const { dc: dcInstance, vars: inputVars} = validateArgs(connectorConfig, dcOrVars, vars, true); dcInstance._useGeneratedSdk(); @@ -23,6 +24,7 @@ exports.upsertMovieRef = function upsertMovieRef(dcOrVars, vars) { exports.upsertMovie = function upsertMovie(dcOrVars, vars) { return executeMutation(upsertMovieRef(dcOrVars, vars)); }; + exports.deleteMovieRef = function deleteMovieRef(dcOrVars, vars) { const { dc: dcInstance, vars: inputVars} = validateArgs(connectorConfig, dcOrVars, vars, true); dcInstance._useGeneratedSdk(); @@ -31,6 +33,7 @@ exports.deleteMovieRef = function deleteMovieRef(dcOrVars, vars) { exports.deleteMovie = function deleteMovie(dcOrVars, vars) { return executeMutation(deleteMovieRef(dcOrVars, vars)); }; + exports.listMoviesRef = function listMoviesRef(dc) { const { dc: dcInstance} = validateArgs(connectorConfig, dc, undefined); dcInstance._useGeneratedSdk(); @@ -39,6 +42,7 @@ exports.listMoviesRef = function listMoviesRef(dc) { exports.listMovies = function listMovies(dc) { return executeQuery(listMoviesRef(dc)); }; + exports.getMovieByIdRef = function getMovieByIdRef(dcOrVars, vars) { const { dc: dcInstance, vars: inputVars} = validateArgs(connectorConfig, dcOrVars, vars, true); dcInstance._useGeneratedSdk(); @@ -47,3 +51,4 @@ exports.getMovieByIdRef = function getMovieByIdRef(dcOrVars, vars) { exports.getMovieById = function getMovieById(dcOrVars, vars) { return executeQuery(getMovieByIdRef(dcOrVars, vars)); }; + diff --git a/package.json b/package.json index d2c9a1d4..87adc173 100644 --- a/package.json +++ b/package.json @@ -24,12 +24,6 @@ "vitest": "^2.0.5" }, "dependencies": { - "@angular/fire": "^19.0.0", - "default-connector": "link:../../../../../../Library/pnpm/global/5/node_modules/dataconnect-sdk/js/default-connector" - }, - "pnpm": { - "overrides": { - "default-connector": "link:../../../../../../Library/pnpm/global/5/node_modules/dataconnect-sdk/js/default-connector" - } + "@angular/fire": "^19.0.0" } } diff --git a/packages/react/src/data-connect/index.ts b/packages/react/src/data-connect/index.ts index 6e40e0bd..cf72f5da 100644 --- a/packages/react/src/data-connect/index.ts +++ b/packages/react/src/data-connect/index.ts @@ -1,14 +1,11 @@ +export { DataConnectQueryClient, type DataConnectQueryOptions } from "./query-client"; export { - DataConnectQueryClient, - type DataConnectQueryOptions, -} from "./query-client"; -export { - useDataConnectQuery, - type useDataConnectQueryOptions, + useDataConnectQuery, + type useDataConnectQueryOptions, } from "./useDataConnectQuery"; export { - useDataConnectMutation, - type useDataConnectMutationOptions, + useDataConnectMutation, + type useDataConnectMutationOptions, } from "./useDataConnectMutation"; -export { validateReactArgs } from "./validateReactArgs"; export type { FlattenedQueryResult, FlattenedMutationResult } from "./types"; +export { validateReactArgs } from "./validateReactArgs"; From 7381716f9af88698748d3ad4872757fe50101d19 Mon Sep 17 00:00:00 2001 From: Stephen Rosa Date: Mon, 17 Mar 2025 15:52:58 -0700 Subject: [PATCH 05/10] remove some changes unintentionally added to this PR --- packages/react/src/data-connect/index.ts | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/packages/react/src/data-connect/index.ts b/packages/react/src/data-connect/index.ts index cf72f5da..d6171943 100644 --- a/packages/react/src/data-connect/index.ts +++ b/packages/react/src/data-connect/index.ts @@ -1,4 +1,7 @@ -export { DataConnectQueryClient, type DataConnectQueryOptions } from "./query-client"; +export { + DataConnectQueryClient, + type DataConnectQueryOptions, +} from "./query-client"; export { useDataConnectQuery, type useDataConnectQueryOptions, From 1b579a3ca0097816b70c86e210748d033067221f Mon Sep 17 00:00:00 2001 From: Stephen Rosa Date: Mon, 17 Mar 2025 15:54:35 -0700 Subject: [PATCH 06/10] remove some changes unintentionally added to this PR --- dataconnect-sdk/js/default-connector/index.cjs.js | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/dataconnect-sdk/js/default-connector/index.cjs.js b/dataconnect-sdk/js/default-connector/index.cjs.js index 464a1b21..bb3906fd 100644 --- a/dataconnect-sdk/js/default-connector/index.cjs.js +++ b/dataconnect-sdk/js/default-connector/index.cjs.js @@ -12,6 +12,7 @@ exports.createMovieRef = function createMovieRef(dcOrVars, vars) { dcInstance._useGeneratedSdk(); return mutationRef(dcInstance, 'CreateMovie', inputVars); } + exports.createMovie = function createMovie(dcOrVars, vars) { return executeMutation(createMovieRef(dcOrVars, vars)); }; @@ -21,6 +22,7 @@ exports.upsertMovieRef = function upsertMovieRef(dcOrVars, vars) { dcInstance._useGeneratedSdk(); return mutationRef(dcInstance, 'UpsertMovie', inputVars); } + exports.upsertMovie = function upsertMovie(dcOrVars, vars) { return executeMutation(upsertMovieRef(dcOrVars, vars)); }; @@ -30,6 +32,7 @@ exports.deleteMovieRef = function deleteMovieRef(dcOrVars, vars) { dcInstance._useGeneratedSdk(); return mutationRef(dcInstance, 'DeleteMovie', inputVars); } + exports.deleteMovie = function deleteMovie(dcOrVars, vars) { return executeMutation(deleteMovieRef(dcOrVars, vars)); }; @@ -39,6 +42,7 @@ exports.listMoviesRef = function listMoviesRef(dc) { dcInstance._useGeneratedSdk(); return queryRef(dcInstance, 'ListMovies'); } + exports.listMovies = function listMovies(dc) { return executeQuery(listMoviesRef(dc)); }; @@ -48,7 +52,7 @@ exports.getMovieByIdRef = function getMovieByIdRef(dcOrVars, vars) { dcInstance._useGeneratedSdk(); return queryRef(dcInstance, 'GetMovieById', inputVars); } + exports.getMovieById = function getMovieById(dcOrVars, vars) { return executeQuery(getMovieByIdRef(dcOrVars, vars)); }; - From 414a21907cf1e3ca402fe45d34b2d67e48d17e4d Mon Sep 17 00:00:00 2001 From: Stephen Rosa Date: Mon, 17 Mar 2025 16:02:57 -0700 Subject: [PATCH 07/10] remove changes to pnpm-lock.yaml unintentionally added to this PR --- pnpm-lock.yaml | 12 +++--------- 1 file changed, 3 insertions(+), 9 deletions(-) diff --git a/pnpm-lock.yaml b/pnpm-lock.yaml index 2b3d9a8b..eaac801f 100644 --- a/pnpm-lock.yaml +++ b/pnpm-lock.yaml @@ -4,9 +4,6 @@ settings: autoInstallPeers: true excludeLinksFromLockfile: false -overrides: - default-connector: link:../../../../../../Library/pnpm/global/5/node_modules/dataconnect-sdk/js/default-connector - importers: .: @@ -14,9 +11,6 @@ importers: '@angular/fire': specifier: ^19.0.0 version: 19.0.0(@angular/common@19.1.8(@angular/core@19.1.8(rxjs@7.8.1)(zone.js@0.15.0))(rxjs@7.8.1))(@angular/core@19.1.8(rxjs@7.8.1)(zone.js@0.15.0))(@angular/platform-browser-dynamic@19.1.8(@angular/common@19.1.8(@angular/core@19.1.8(rxjs@7.8.1)(zone.js@0.15.0))(rxjs@7.8.1))(@angular/compiler@19.1.8(@angular/core@19.1.8(rxjs@7.8.1)(zone.js@0.15.0)))(@angular/core@19.1.8(rxjs@7.8.1)(zone.js@0.15.0))(@angular/platform-browser@19.1.8(@angular/animations@19.1.8(@angular/core@19.1.8(rxjs@7.8.1)(zone.js@0.15.0)))(@angular/common@19.1.8(@angular/core@19.1.8(rxjs@7.8.1)(zone.js@0.15.0))(rxjs@7.8.1))(@angular/core@19.1.8(rxjs@7.8.1)(zone.js@0.15.0))))(@angular/platform-browser@19.1.8(@angular/animations@19.1.8(@angular/core@19.1.8(rxjs@7.8.1)(zone.js@0.15.0)))(@angular/common@19.1.8(@angular/core@19.1.8(rxjs@7.8.1)(zone.js@0.15.0))(rxjs@7.8.1))(@angular/core@19.1.8(rxjs@7.8.1)(zone.js@0.15.0)))(chokidar@4.0.3)(rxjs@7.8.1) - default-connector: - specifier: link:../../../../../../Library/pnpm/global/5/node_modules/dataconnect-sdk/js/default-connector - version: link:../../../../../../Library/pnpm/global/5/node_modules/dataconnect-sdk/js/default-connector devDependencies: '@angular/core': specifier: ^19.1.8 @@ -5180,7 +5174,7 @@ snapshots: transitivePeerDependencies: - supports-color - eslint-module-utils@2.12.0(@typescript-eslint/parser@8.25.0(eslint@9.21.0(jiti@1.21.7))(typescript@5.7.3))(eslint-import-resolver-node@0.3.9)(eslint-import-resolver-typescript@3.8.3)(eslint@9.21.0(jiti@1.21.7)): + eslint-module-utils@2.12.0(@typescript-eslint/parser@8.25.0(eslint@9.21.0(jiti@1.21.7))(typescript@5.7.3))(eslint-import-resolver-node@0.3.9)(eslint-import-resolver-typescript@3.8.3(eslint-plugin-import@2.31.0)(eslint@9.21.0(jiti@1.21.7)))(eslint@9.21.0(jiti@1.21.7)): dependencies: debug: 3.2.7 optionalDependencies: @@ -5202,7 +5196,7 @@ snapshots: doctrine: 2.1.0 eslint: 9.21.0(jiti@1.21.7) eslint-import-resolver-node: 0.3.9 - eslint-module-utils: 2.12.0(@typescript-eslint/parser@8.25.0(eslint@9.21.0(jiti@1.21.7))(typescript@5.7.3))(eslint-import-resolver-node@0.3.9)(eslint-import-resolver-typescript@3.8.3)(eslint@9.21.0(jiti@1.21.7)) + eslint-module-utils: 2.12.0(@typescript-eslint/parser@8.25.0(eslint@9.21.0(jiti@1.21.7))(typescript@5.7.3))(eslint-import-resolver-node@0.3.9)(eslint-import-resolver-typescript@3.8.3(eslint-plugin-import@2.31.0)(eslint@9.21.0(jiti@1.21.7)))(eslint@9.21.0(jiti@1.21.7)) hasown: 2.0.2 is-core-module: 2.16.1 is-glob: 4.0.3 @@ -6863,4 +6857,4 @@ snapshots: yocto-queue@0.1.0: {} - zone.js@0.15.0: {} + zone.js@0.15.0: {} \ No newline at end of file From 80649161fce8e7ad0c0ca7bd1d2ddb586ed03c8e Mon Sep 17 00:00:00 2001 From: Stephen Rosa Date: Mon, 17 Mar 2025 16:04:36 -0700 Subject: [PATCH 08/10] remove changes to pnpm-lock.yaml unintentionally added to this PR --- pnpm-lock.yaml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/pnpm-lock.yaml b/pnpm-lock.yaml index eaac801f..471e2ea5 100644 --- a/pnpm-lock.yaml +++ b/pnpm-lock.yaml @@ -6857,4 +6857,4 @@ snapshots: yocto-queue@0.1.0: {} - zone.js@0.15.0: {} \ No newline at end of file + zone.js@0.15.0: {} From 4e89404af4bd906a775bdcea3359f91a70a5c5dc Mon Sep 17 00:00:00 2001 From: Stephen Rosa Date: Wed, 19 Mar 2025 16:07:02 -0700 Subject: [PATCH 09/10] address review comments - collapse expect checks in testing, create new explicit DataConnectOptions type --- .../data-connect/validateReactArgs.test.tsx | 42 +++---------------- .../src/data-connect/validateReactArgs.ts | 34 ++++++++------- 2 files changed, 24 insertions(+), 52 deletions(-) diff --git a/packages/react/src/data-connect/validateReactArgs.test.tsx b/packages/react/src/data-connect/validateReactArgs.test.tsx index 8c499bcc..b58f3cef 100644 --- a/packages/react/src/data-connect/validateReactArgs.test.tsx +++ b/packages/react/src/data-connect/validateReactArgs.test.tsx @@ -60,18 +60,8 @@ describe("validateReactArgs", () => { ); expect(dcInstance).toBe(dataConnect); - - if (expectedInputVars) { - expect(inputVars).toBe(expectedInputVars); - } else { - expect(inputVars).toBeUndefined(); - } - - if (expectedInputOpts) { - expect(inputOpts).toBe(expectedInputOpts); - } else { - expect(inputOpts).toBeUndefined(); - } + expect(inputVars).toBe(expectedInputVars); + expect(inputOpts).toBe(expectedInputOpts) } ); @@ -163,18 +153,8 @@ describe("validateReactArgs", () => { ); expect(dcInstance).toBe(dataConnect); - - if (expectedInputVars) { - expect(inputVars).toBe(expectedInputVars); - } else { - expect(inputVars).toBeUndefined(); - } - - if (expectedInputOpts) { - expect(inputOpts).toBe(expectedInputOpts); - } else { - expect(inputOpts).toBeUndefined(); - } + expect(inputVars).toBe(expectedInputVars); + expect(inputOpts).toBe(expectedInputOpts) } ); @@ -234,18 +214,8 @@ describe("validateReactArgs", () => { ); expect(dcInstance).toBe(dataConnect); - - if (expectedInputVars) { - expect(inputVars).toBe(expectedInputVars); - } else { - expect(inputVars).toBeUndefined(); - } - - if (expectedInputOpts) { - expect(inputOpts).toBe(expectedInputOpts); - } else { - expect(inputOpts).toBeUndefined(); - } + expect(inputVars).toBe(expectedInputVars); + expect(inputOpts).toBe(expectedInputOpts) } ); diff --git a/packages/react/src/data-connect/validateReactArgs.ts b/packages/react/src/data-connect/validateReactArgs.ts index bc20069d..7905d8f4 100644 --- a/packages/react/src/data-connect/validateReactArgs.ts +++ b/packages/react/src/data-connect/validateReactArgs.ts @@ -3,11 +3,16 @@ import { DataConnect, getDataConnect, } from "firebase/data-connect"; +import { useDataConnectQueryOptions } from "./useDataConnectQuery"; -interface ParsedReactArgs { +type DataConnectOptions = + | useDataConnectQueryOptions + | useDataConnectQueryOptions; + +interface ParsedReactArgs { dc: DataConnect; vars: Variables; - options: Options; + options: DataConnectOptions; } /** @@ -24,39 +29,36 @@ interface ParsedReactArgs { * @returns parsed DataConnect, Variables, and Options for the operation * @internal */ -export function validateReactArgs< - Variables extends object, - Options extends object ->( +export function validateReactArgs( connectorConfig: ConnectorConfig, - dcOrVarsOrOptions?: DataConnect | Variables | Options, - varsOrOptions?: Variables | Options, - options?: Options, + dcOrVarsOrOptions?: DataConnect | Variables | DataConnectOptions, + varsOrOptions?: Variables | DataConnectOptions, + options?: DataConnectOptions, hasVars?: boolean, validateVars?: boolean -): ParsedReactArgs { +): ParsedReactArgs { let dcInstance: DataConnect; let realVars: Variables; - let realOptions: Options; + let realOptions: DataConnectOptions; if (dcOrVarsOrOptions && "enableEmulator" in dcOrVarsOrOptions) { dcInstance = dcOrVarsOrOptions as DataConnect; if (hasVars) { realVars = varsOrOptions as Variables; - realOptions = options as Options; + realOptions = options as DataConnectOptions; } else { realVars = undefined as unknown as Variables; - realOptions = varsOrOptions as Options; + realOptions = varsOrOptions as DataConnectOptions; } } else { dcInstance = getDataConnect(connectorConfig); if (hasVars) { realVars = dcOrVarsOrOptions as Variables; - realOptions = varsOrOptions as Options; + realOptions = varsOrOptions as DataConnectOptions; } else { realVars = undefined as unknown as Variables; - realOptions = dcOrVarsOrOptions as Options; - } + realOptions = dcOrVarsOrOptions as DataConnectOptions; + } } if (!dcInstance || (!realVars && validateVars)) { From f5e567099f96df41d84aabefb8ab31bea7329b7a Mon Sep 17 00:00:00 2001 From: Stephen Rosa Date: Wed, 19 Mar 2025 16:12:27 -0700 Subject: [PATCH 10/10] run VSCode file formatter --- packages/react/src/data-connect/validateReactArgs.test.tsx | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/packages/react/src/data-connect/validateReactArgs.test.tsx b/packages/react/src/data-connect/validateReactArgs.test.tsx index b58f3cef..478bdaa7 100644 --- a/packages/react/src/data-connect/validateReactArgs.test.tsx +++ b/packages/react/src/data-connect/validateReactArgs.test.tsx @@ -61,7 +61,7 @@ describe("validateReactArgs", () => { expect(dcInstance).toBe(dataConnect); expect(inputVars).toBe(expectedInputVars); - expect(inputOpts).toBe(expectedInputOpts) + expect(inputOpts).toBe(expectedInputOpts); } ); @@ -154,7 +154,7 @@ describe("validateReactArgs", () => { expect(dcInstance).toBe(dataConnect); expect(inputVars).toBe(expectedInputVars); - expect(inputOpts).toBe(expectedInputOpts) + expect(inputOpts).toBe(expectedInputOpts); } ); @@ -215,7 +215,7 @@ describe("validateReactArgs", () => { expect(dcInstance).toBe(dataConnect); expect(inputVars).toBe(expectedInputVars); - expect(inputOpts).toBe(expectedInputOpts) + expect(inputOpts).toBe(expectedInputOpts); } );