From 4dea59a39e0fefa945a351218f11e460409734c8 Mon Sep 17 00:00:00 2001 From: Mehdi Salemi Date: Mon, 18 Nov 2024 16:06:03 -0500 Subject: [PATCH] adds functions api client to fetch schema for dev dash --- bin/get-graphql-schemas.js | 8 +- graphql.config.ts | 3 +- packages/app/project.json | 26 +++++- .../functions/api_schema_definition.ts | 6 -- .../schema-definition-by-api-type.ts | 74 +++++++++++++++ .../generated/schema-definition-by-target.ts | 84 +++++++++++++++++ .../graphql/functions/generated/types.d.ts | 24 +++++ .../schema-definition-by-api-type.graphql | 7 ++ .../schema-definition-by-target.graphql | 9 ++ .../functions/target_schema_definition.ts | 6 -- .../app/src/cli/commands/app/function/run.ts | 4 +- .../src/cli/commands/app/function/schema.ts | 3 +- .../app/src/cli/models/app/app.test-data.ts | 10 +- .../src/cli/services/function/common.test.ts | 4 +- .../app/src/cli/services/function/common.ts | 9 +- .../src/cli/services/generate-schema.test.ts | 91 ++++++++++++------- .../app/src/cli/services/generate-schema.ts | 45 ++++++--- .../utilities/developer-platform-client.ts | 19 +++- .../app-management-client.ts | 54 +++++++++-- .../partners-client.ts | 32 +++++-- .../cli-kit/src/public/node/api/functions.ts | 59 ++++++++++++ .../rules/no-inline-graphql.js | 6 +- 22 files changed, 488 insertions(+), 95 deletions(-) create mode 100644 packages/app/src/cli/api/graphql/functions/generated/schema-definition-by-api-type.ts create mode 100644 packages/app/src/cli/api/graphql/functions/generated/schema-definition-by-target.ts create mode 100644 packages/app/src/cli/api/graphql/functions/generated/types.d.ts create mode 100644 packages/app/src/cli/api/graphql/functions/queries/schema-definition-by-api-type.graphql create mode 100644 packages/app/src/cli/api/graphql/functions/queries/schema-definition-by-target.graphql create mode 100644 packages/cli-kit/src/public/node/api/functions.ts diff --git a/bin/get-graphql-schemas.js b/bin/get-graphql-schemas.js index 886ec93509..b251d997e9 100755 --- a/bin/get-graphql-schemas.js +++ b/bin/get-graphql-schemas.js @@ -38,7 +38,13 @@ const schemas = [ repo: 'shopify', pathToFile: 'areas/core/shopify/db/graphql/admin_schema_unstable_public.graphql', localPath: './packages/cli-kit/src/cli/api/graphql/admin/admin_schema.graphql', - } + }, + { + repo: 'shopify', + pathToFile: 'areas/core/shopify/db/graphql/functions_cli_api_schema_unstable_public.graphql', + localPath: './packages/app/src/cli/api/graphql/functions/functions_cli_schema.graphql', + branch: 'ms.proto3-with-cli', + }, ] function runCommand(command, args) { diff --git a/graphql.config.ts b/graphql.config.ts index b1a5c631f2..18494c3b64 100644 --- a/graphql.config.ts +++ b/graphql.config.ts @@ -1,7 +1,7 @@ function projectFactory(name: string, schemaName: string, project: string = 'app') { return { schema: `./packages/${project}/src/cli/api/graphql/${name}/${schemaName}`, - documents: [`./packages/${project}/src/cli/api/graphql/${name}/queries/**/*.graphql`,`./packages/${project}/src/cli/api/graphql/${name}/mutations/**/*.graphql`], + documents: [`./packages/${project}/src/cli/api/graphql/${name}/queries/**/*.graphql`,`./packages/${project}/src/cli/api/graphql/${name}/mutations/**/*.graphql`,], extensions: { codegen: { generates: { @@ -78,5 +78,6 @@ export default { appDev: projectFactory('app-dev', 'app_dev_schema.graphql'), appManagement: projectFactory('app-management', 'app_management_schema.graphql'), admin: projectFactory('admin', 'admin_schema.graphql', 'cli-kit'), + functions: projectFactory('functions', 'functions_cli_schema.graphql', 'app'), }, } diff --git a/packages/app/project.json b/packages/app/project.json index 9bf718c9f1..c3e88fc512 100644 --- a/packages/app/project.json +++ b/packages/app/project.json @@ -80,7 +80,8 @@ "{projectRoot}/src/cli/api/graphql/business-platform-destinations/generated/**/*.ts", "{projectRoot}/src/cli/api/graphql/business-platform-organizations/generated/**/*.ts", "{projectRoot}/src/cli/api/graphql/app-dev/generated/**/*.ts", - "{projectRoot}/src/cli/api/graphql/app-management/generated/**/*.ts" + "{projectRoot}/src/cli/api/graphql/app-management/generated/**/*.ts", + "{projectRoot}/src/cli/api/graphql/functions/generated/**/*.ts" ], "options": { "commands": [ @@ -88,7 +89,8 @@ "pnpm eslint 'src/cli/api/graphql/business-platform-destinations/generated/**/*.{ts,tsx}' --fix", "pnpm eslint 'src/cli/api/graphql/business-platform-organizations/generated/**/*.{ts,tsx}' --fix", "pnpm eslint 'src/cli/api/graphql/app-dev/generated/**/*.{ts,tsx}' --fix", - "pnpm eslint 'src/cli/api/graphql/app-management/generated/**/*.{ts,tsx}' --fix" + "pnpm eslint 'src/cli/api/graphql/app-management/generated/**/*.{ts,tsx}' --fix", + "pnpm eslint 'src/cli/api/graphql/functions/generated/**/*.{ts,tsx}' --fix" ], "cwd": "packages/app" } @@ -148,6 +150,17 @@ "cwd": "{workspaceRoot}" } }, + "graphql-codegen:generate:functions": { + "executor": "nx:run-commands", + "inputs": ["{workspaceRoot}/graphql.config.ts", "{projectRoot}/src/cli/api/graphql/functions/**/*.graphql"], + "outputs": ["{projectRoot}/src/cli/api/graphql/functions/generated/**/*.ts"], + "options": { + "commands": [ + "pnpm exec graphql-codegen --project=functions" + ], + "cwd": "{workspaceRoot}" + } + }, "graphql-codegen:postfix": { "executor": "nx:run-commands", "dependsOn": [ @@ -155,7 +168,8 @@ "graphql-codegen:generate:business-platform-destinations", "graphql-codegen:generate:business-platform-organizations", "graphql-codegen:generate:app-dev", - "graphql-codegen:generate:app-management" + "graphql-codegen:generate:app-management", + "graphql-codegen:generate:functions" ], "inputs": [{ "dependentTasksOutputFiles": "**/*.ts" }], "outputs": [ @@ -163,7 +177,8 @@ "{projectRoot}/src/cli/api/graphql/business-platform-destinations/generated/**/*.ts", "{projectRoot}/src/cli/api/graphql/business-platform-organizations/generated/**/*.ts", "{projectRoot}/src/cli/api/graphql/app-dev/generated/**/*.ts", - "{projectRoot}/src/cli/api/graphql/app-management/generated/**/*.ts" + "{projectRoot}/src/cli/api/graphql/app-management/generated/**/*.ts", + "{projectRoot}/src/cli/api/graphql/functions/generated/**/*.ts" ], "options": { "commands": [ @@ -171,7 +186,8 @@ "find ./packages/app/src/cli/api/graphql/business-platform-destinations/generated/ -type f -name '*.ts' -exec sh -c 'sed -i \"\" \"s|import \\* as Types from '\\''./types'\\'';|import \\* as Types from '\\''./types.js'\\'';|g; s|export const \\([A-Za-z0-9_]*\\)Document =|export const \\1 =|g\" \"$0\"' {} \\;", "find ./packages/app/src/cli/api/graphql/business-platform-organizations/generated/ -type f -name '*.ts' -exec sh -c 'sed -i \"\" \"s|import \\* as Types from '\\''./types'\\'';|import \\* as Types from '\\''./types.js'\\'';|g; s|export const \\([A-Za-z0-9_]*\\)Document =|export const \\1 =|g\" \"$0\"' {} \\;", "find ./packages/app/src/cli/api/graphql/app-dev/generated/ -type f -name '*.ts' -exec sh -c 'sed -i \"\" \"s|import \\* as Types from '\\''./types'\\'';|import \\* as Types from '\\''./types.js'\\'';|g; s|export const \\([A-Za-z0-9_]*\\)Document =|export const \\1 =|g\" \"$0\"' {} \\;", - "find ./packages/app/src/cli/api/graphql/app-management/generated/ -type f -name '*.ts' -exec sh -c 'sed -i \"\" \"s|import \\* as Types from '\\''./types'\\'';|import \\* as Types from '\\''./types.js'\\'';|g; s|export const \\([A-Za-z0-9_]*\\)Document =|export const \\1 =|g\" \"$0\"' {} \\;" + "find ./packages/app/src/cli/api/graphql/app-management/generated/ -type f -name '*.ts' -exec sh -c 'sed -i \"\" \"s|import \\* as Types from '\\''./types'\\'';|import \\* as Types from '\\''./types.js'\\'';|g; s|export const \\([A-Za-z0-9_]*\\)Document =|export const \\1 =|g\" \"$0\"' {} \\;", + "find ./packages/app/src/cli/api/graphql/functions/generated/ -type f -name '*.ts' -exec sh -c 'sed -i \"\" \"s|import \\* as Types from '\\''./types'\\'';|import \\* as Types from '\\''./types.js'\\'';|g; s|export const \\([A-Za-z0-9_]*\\)Document =|export const \\1 =|g\" \"$0\"' {} \\;" ], "cwd": "{workspaceRoot}" } diff --git a/packages/app/src/cli/api/graphql/functions/api_schema_definition.ts b/packages/app/src/cli/api/graphql/functions/api_schema_definition.ts index 09a3806b3d..bf6be02a5d 100644 --- a/packages/app/src/cli/api/graphql/functions/api_schema_definition.ts +++ b/packages/app/src/cli/api/graphql/functions/api_schema_definition.ts @@ -9,9 +9,3 @@ export const ApiSchemaDefinitionQuery = gql` export interface ApiSchemaDefinitionQuerySchema { definition: string | null } - -export interface ApiSchemaDefinitionQueryVariables { - apiKey: string - version: string - type: string -} diff --git a/packages/app/src/cli/api/graphql/functions/generated/schema-definition-by-api-type.ts b/packages/app/src/cli/api/graphql/functions/generated/schema-definition-by-api-type.ts new file mode 100644 index 0000000000..59d66f3461 --- /dev/null +++ b/packages/app/src/cli/api/graphql/functions/generated/schema-definition-by-api-type.ts @@ -0,0 +1,74 @@ +/* eslint-disable @typescript-eslint/consistent-type-definitions */ +import * as Types from './types.js' + +import {TypedDocumentNode as DocumentNode} from '@graphql-typed-document-node/core' + +export type SchemaDefinitionByApiTypeQueryVariables = Types.Exact<{ + type: Types.Scalars['String']['input'] + version: Types.Scalars['String']['input'] +}> + +export type SchemaDefinitionByApiTypeQuery = {api?: {schema?: {definition: string} | null} | null} + +export const SchemaDefinitionByApiType = { + kind: 'Document', + definitions: [ + { + kind: 'OperationDefinition', + operation: 'query', + name: {kind: 'Name', value: 'SchemaDefinitionByApiType'}, + variableDefinitions: [ + { + kind: 'VariableDefinition', + variable: {kind: 'Variable', name: {kind: 'Name', value: 'type'}}, + type: {kind: 'NonNullType', type: {kind: 'NamedType', name: {kind: 'Name', value: 'String'}}}, + }, + { + kind: 'VariableDefinition', + variable: {kind: 'Variable', name: {kind: 'Name', value: 'version'}}, + type: {kind: 'NonNullType', type: {kind: 'NamedType', name: {kind: 'Name', value: 'String'}}}, + }, + ], + selectionSet: { + kind: 'SelectionSet', + selections: [ + { + kind: 'Field', + name: {kind: 'Name', value: 'api'}, + arguments: [ + { + kind: 'Argument', + name: {kind: 'Name', value: 'type'}, + value: {kind: 'Variable', name: {kind: 'Name', value: 'type'}}, + }, + ], + selectionSet: { + kind: 'SelectionSet', + selections: [ + { + kind: 'Field', + name: {kind: 'Name', value: 'schema'}, + arguments: [ + { + kind: 'Argument', + name: {kind: 'Name', value: 'version'}, + value: {kind: 'Variable', name: {kind: 'Name', value: 'version'}}, + }, + ], + selectionSet: { + kind: 'SelectionSet', + selections: [ + {kind: 'Field', name: {kind: 'Name', value: 'definition'}}, + {kind: 'Field', name: {kind: 'Name', value: '__typename'}}, + ], + }, + }, + {kind: 'Field', name: {kind: 'Name', value: '__typename'}}, + ], + }, + }, + ], + }, + }, + ], +} as unknown as DocumentNode diff --git a/packages/app/src/cli/api/graphql/functions/generated/schema-definition-by-target.ts b/packages/app/src/cli/api/graphql/functions/generated/schema-definition-by-target.ts new file mode 100644 index 0000000000..f7dadf4593 --- /dev/null +++ b/packages/app/src/cli/api/graphql/functions/generated/schema-definition-by-target.ts @@ -0,0 +1,84 @@ +/* eslint-disable @typescript-eslint/consistent-type-definitions */ +import * as Types from './types.js' + +import {TypedDocumentNode as DocumentNode} from '@graphql-typed-document-node/core' + +export type SchemaDefinitionByTargetQueryVariables = Types.Exact<{ + handle: Types.Scalars['String']['input'] + version: Types.Scalars['String']['input'] +}> + +export type SchemaDefinitionByTargetQuery = {target?: {api: {schema?: {definition: string} | null}} | null} + +export const SchemaDefinitionByTarget = { + kind: 'Document', + definitions: [ + { + kind: 'OperationDefinition', + operation: 'query', + name: {kind: 'Name', value: 'SchemaDefinitionByTarget'}, + variableDefinitions: [ + { + kind: 'VariableDefinition', + variable: {kind: 'Variable', name: {kind: 'Name', value: 'handle'}}, + type: {kind: 'NonNullType', type: {kind: 'NamedType', name: {kind: 'Name', value: 'String'}}}, + }, + { + kind: 'VariableDefinition', + variable: {kind: 'Variable', name: {kind: 'Name', value: 'version'}}, + type: {kind: 'NonNullType', type: {kind: 'NamedType', name: {kind: 'Name', value: 'String'}}}, + }, + ], + selectionSet: { + kind: 'SelectionSet', + selections: [ + { + kind: 'Field', + name: {kind: 'Name', value: 'target'}, + arguments: [ + { + kind: 'Argument', + name: {kind: 'Name', value: 'handle'}, + value: {kind: 'Variable', name: {kind: 'Name', value: 'handle'}}, + }, + ], + selectionSet: { + kind: 'SelectionSet', + selections: [ + { + kind: 'Field', + name: {kind: 'Name', value: 'api'}, + selectionSet: { + kind: 'SelectionSet', + selections: [ + { + kind: 'Field', + name: {kind: 'Name', value: 'schema'}, + arguments: [ + { + kind: 'Argument', + name: {kind: 'Name', value: 'version'}, + value: {kind: 'Variable', name: {kind: 'Name', value: 'version'}}, + }, + ], + selectionSet: { + kind: 'SelectionSet', + selections: [ + {kind: 'Field', name: {kind: 'Name', value: 'definition'}}, + {kind: 'Field', name: {kind: 'Name', value: '__typename'}}, + ], + }, + }, + {kind: 'Field', name: {kind: 'Name', value: '__typename'}}, + ], + }, + }, + {kind: 'Field', name: {kind: 'Name', value: '__typename'}}, + ], + }, + }, + ], + }, + }, + ], +} as unknown as DocumentNode diff --git a/packages/app/src/cli/api/graphql/functions/generated/types.d.ts b/packages/app/src/cli/api/graphql/functions/generated/types.d.ts new file mode 100644 index 0000000000..2d85fa63ee --- /dev/null +++ b/packages/app/src/cli/api/graphql/functions/generated/types.d.ts @@ -0,0 +1,24 @@ +/* eslint-disable @typescript-eslint/consistent-type-definitions, @typescript-eslint/naming-convention */ +export type Maybe = T | null +export type InputMaybe = Maybe +export type Exact = {[K in keyof T]: T[K]} +export type MakeOptional = Omit & {[SubKey in K]?: Maybe} +export type MakeMaybe = Omit & {[SubKey in K]: Maybe} +export type MakeEmpty = {[_ in K]?: never} +export type Incremental = T | {[P in keyof T]?: P extends ' $fragmentName' | '__typename' ? T[P] : never} +/** All built-in and custom scalars, mapped to their actual values */ +export type Scalars = { + ID: {input: string; output: string} + String: {input: string; output: string} + Boolean: {input: boolean; output: boolean} + Int: {input: number; output: number} + Float: {input: number; output: number} + /** + * Represents an [RFC 3986](https://datatracker.ietf.org/doc/html/rfc3986) and + * [RFC 3987](https://datatracker.ietf.org/doc/html/rfc3987)-compliant URI string. + * + * For example, `"https://example.myshopify.com"` is a valid URL. It includes a scheme (`https`) and a host + * (`example.myshopify.com`). + */ + URL: {input: string; output: string} +} diff --git a/packages/app/src/cli/api/graphql/functions/queries/schema-definition-by-api-type.graphql b/packages/app/src/cli/api/graphql/functions/queries/schema-definition-by-api-type.graphql new file mode 100644 index 0000000000..c9b493ba7c --- /dev/null +++ b/packages/app/src/cli/api/graphql/functions/queries/schema-definition-by-api-type.graphql @@ -0,0 +1,7 @@ +query SchemaDefinitionByApiType($type: String!, $version: String!) { + api(type: $type) { + schema(version: $version) { + definition + } + } +} diff --git a/packages/app/src/cli/api/graphql/functions/queries/schema-definition-by-target.graphql b/packages/app/src/cli/api/graphql/functions/queries/schema-definition-by-target.graphql new file mode 100644 index 0000000000..fd220b53fa --- /dev/null +++ b/packages/app/src/cli/api/graphql/functions/queries/schema-definition-by-target.graphql @@ -0,0 +1,9 @@ +query SchemaDefinitionByTarget($handle: String!, $version: String!) { + target(handle: $handle) { + api { + schema(version: $version) { + definition + } + } + } +} diff --git a/packages/app/src/cli/api/graphql/functions/target_schema_definition.ts b/packages/app/src/cli/api/graphql/functions/target_schema_definition.ts index dbf2ee2eac..a957d03d65 100644 --- a/packages/app/src/cli/api/graphql/functions/target_schema_definition.ts +++ b/packages/app/src/cli/api/graphql/functions/target_schema_definition.ts @@ -9,9 +9,3 @@ export const TargetSchemaDefinitionQuery = gql` export interface TargetSchemaDefinitionQuerySchema { definition: string | null } - -export interface TargetSchemaDefinitionQueryVariables { - apiKey: string - version: string - target: string -} diff --git a/packages/app/src/cli/commands/app/function/run.ts b/packages/app/src/cli/commands/app/function/run.ts index 2c8d352f4d..a9f575d1ba 100644 --- a/packages/app/src/cli/commands/app/function/run.ts +++ b/packages/app/src/cli/commands/app/function/run.ts @@ -44,7 +44,7 @@ export default class FunctionRun extends AppCommand { const app = await inFunctionContext({ path: flags.path, userProvidedConfigName: flags.config, - callback: async (app, developerPlatformClient, ourFunction) => { + callback: async (app, developerPlatformClient, ourFunction, orgId) => { let functionExport = DEFAULT_FUNCTION_EXPORT if (flags.export !== undefined) { @@ -80,7 +80,7 @@ export default class FunctionRun extends AppCommand { const inputQueryPath = ourFunction?.configuration.targeting?.[0]?.input_query const queryPath = inputQueryPath && `${ourFunction?.directory}/${inputQueryPath}` - const schemaPath = await getOrGenerateSchemaPath(ourFunction, app, developerPlatformClient) + const schemaPath = await getOrGenerateSchemaPath(ourFunction, app, developerPlatformClient, orgId) await runFunction({ functionExtension: ourFunction, diff --git a/packages/app/src/cli/commands/app/function/schema.ts b/packages/app/src/cli/commands/app/function/schema.ts index 3b6ab78639..f9aeb62cb2 100644 --- a/packages/app/src/cli/commands/app/function/schema.ts +++ b/packages/app/src/cli/commands/app/function/schema.ts @@ -51,13 +51,14 @@ export default class FetchSchema extends AppCommand { path: flags.path, apiKey, userProvidedConfigName: flags.config, - callback: async (app, developerPlatformClient, ourFunction) => { + callback: async (app, developerPlatformClient, ourFunction, orgId) => { await generateSchemaService({ app, extension: ourFunction, developerPlatformClient, stdout: flags.stdout, path: flags.path, + orgId, }) return app }, diff --git a/packages/app/src/cli/models/app/app.test-data.ts b/packages/app/src/cli/models/app/app.test-data.ts index f9933958bf..2fef29cf63 100644 --- a/packages/app/src/cli/models/app/app.test-data.ts +++ b/packages/app/src/cli/models/app/app.test-data.ts @@ -55,8 +55,6 @@ import { } from '../../api/graphql/extension_migrate_flow_extension.js' import {UpdateURLsSchema, UpdateURLsVariables} from '../../api/graphql/update_urls.js' import {CurrentAccountInfoSchema} from '../../api/graphql/current_account_info.js' -import {TargetSchemaDefinitionQueryVariables} from '../../api/graphql/functions/target_schema_definition.js' -import {ApiSchemaDefinitionQueryVariables} from '../../api/graphql/functions/api_schema_definition.js' import { MigrateToUiExtensionSchema, MigrateToUiExtensionVariables, @@ -69,6 +67,8 @@ import { ExtensionUpdateDraftMutation, ExtensionUpdateDraftMutationVariables, } from '../../api/graphql/partners/generated/update-draft.js' +import {SchemaDefinitionByTargetQueryVariables} from '../../api/graphql/functions/generated/schema-definition-by-target.js' +import {SchemaDefinitionByApiTypeQueryVariables} from '../../api/graphql/functions/generated/schema-definition-by-api-type.js' import {vi} from 'vitest' import {joinPath} from '@shopify/cli-kit/node/path' @@ -1328,8 +1328,10 @@ export function testDeveloperPlatformClient(stubs: Partial Promise.resolve(migrateAppModuleResponse), updateURLs: (_input: UpdateURLsVariables) => Promise.resolve(updateURLsResponse), currentAccountInfo: () => Promise.resolve(currentAccountInfoResponse), - targetSchemaDefinition: (_input: TargetSchemaDefinitionQueryVariables) => Promise.resolve('schema'), - apiSchemaDefinition: (_input: ApiSchemaDefinitionQueryVariables) => Promise.resolve('schema'), + targetSchemaDefinition: (_input: SchemaDefinitionByTargetQueryVariables & {apiKey?: string}, _orgId: string) => + Promise.resolve('schema'), + apiSchemaDefinition: (_input: SchemaDefinitionByApiTypeQueryVariables & {apiKey?: string}, _orgId: string) => + Promise.resolve('schema'), migrateToUiExtension: (_input: MigrateToUiExtensionVariables) => Promise.resolve(migrateToUiExtensionResponse), toExtensionGraphQLType: (input: string) => input, subscribeToAppLogs: (_input: AppLogsSubscribeVariables) => Promise.resolve(appLogsSubscribeResponse), diff --git a/packages/app/src/cli/services/function/common.test.ts b/packages/app/src/cli/services/function/common.test.ts index 1fd0719823..c338f1356e 100644 --- a/packages/app/src/cli/services/function/common.test.ts +++ b/packages/app/src/cli/services/function/common.test.ts @@ -118,7 +118,7 @@ describe('getOrGenerateSchemaPath', () => { vi.mocked(fileExists).mockResolvedValue(true) // When - const result = await getOrGenerateSchemaPath(extension, app, developerPlatformClient) + const result = await getOrGenerateSchemaPath(extension, app, developerPlatformClient, '123') // Then expect(result).toBe(expectedPath) @@ -133,7 +133,7 @@ describe('getOrGenerateSchemaPath', () => { vi.mocked(fileExists).mockResolvedValueOnce(true) // When - const result = await getOrGenerateSchemaPath(extension, app, developerPlatformClient) + const result = await getOrGenerateSchemaPath(extension, app, developerPlatformClient, '123') // Then expect(result).toBe(expectedPath) diff --git a/packages/app/src/cli/services/function/common.ts b/packages/app/src/cli/services/function/common.ts index 49afb3c1eb..ff76db8874 100644 --- a/packages/app/src/cli/services/function/common.ts +++ b/packages/app/src/cli/services/function/common.ts @@ -35,9 +35,10 @@ export async function inFunctionContext({ app: AppLinkedInterface, developerPlatformClient: DeveloperPlatformClient, ourFunction: ExtensionInstance, + orgId: string, ) => Promise }) { - const {app, developerPlatformClient} = await linkedAppContext({ + const {app, developerPlatformClient, organization} = await linkedAppContext({ directory: path, clientId: apiKey, forceRelink: false, @@ -50,14 +51,14 @@ export async function inFunctionContext({ const ourFunction = allFunctions.find((fun) => fun.directory === path) if (ourFunction) { - return callback(app, developerPlatformClient, ourFunction) + return callback(app, developerPlatformClient, ourFunction, organization.id) } else if (isTerminalInteractive()) { const selectedFunction = await renderAutocompletePrompt({ message: 'Which function?', choices: allFunctions.map((shopifyFunction) => ({label: shopifyFunction.handle, value: shopifyFunction})), }) - return callback(app, developerPlatformClient, selectedFunction) + return callback(app, developerPlatformClient, selectedFunction, organization.id) } else { throw new AbortError('Run this command from a function directory or use `--path` to specify a function directory.') } @@ -67,6 +68,7 @@ export async function getOrGenerateSchemaPath( extension: ExtensionInstance, app: AppLinkedInterface, developerPlatformClient: DeveloperPlatformClient, + orgId: string, ): Promise { const path = joinPath(extension.directory, 'schema.graphql') if (await fileExists(path)) { @@ -79,6 +81,7 @@ export async function getOrGenerateSchemaPath( extension, stdout: false, path: extension.directory, + orgId, }) return (await fileExists(path)) ? path : undefined diff --git a/packages/app/src/cli/services/generate-schema.test.ts b/packages/app/src/cli/services/generate-schema.test.ts index 7a20343e8b..fdfbcaa47a 100644 --- a/packages/app/src/cli/services/generate-schema.test.ts +++ b/packages/app/src/cli/services/generate-schema.test.ts @@ -1,9 +1,9 @@ import {generateSchemaService} from './generate-schema.js' import {testAppLinked, testDeveloperPlatformClient, testFunctionExtension} from '../models/app/app.test-data.js' -import {ApiSchemaDefinitionQueryVariables} from '../api/graphql/functions/api_schema_definition.js' +import {SchemaDefinitionByApiTypeQueryVariables} from '../api/graphql/functions/generated/schema-definition-by-api-type.js' import {describe, expect, vi, test} from 'vitest' import {AbortError} from '@shopify/cli-kit/node/error' -import {inTemporaryDirectory, readFile} from '@shopify/cli-kit/node/fs' +import {inTemporaryDirectory, readFile, mkdir} from '@shopify/cli-kit/node/fs' import {joinPath} from '@shopify/cli-kit/node/path' import * as output from '@shopify/cli-kit/node/output' @@ -28,22 +28,27 @@ describe('generateSchemaService', () => { test('Save the latest GraphQL schema to ./[extension]/schema.graphql when stdout flag is ABSENT', async () => { await inTemporaryDirectory(async (tmpDir) => { // Given + const orgId = 'test' + const extensionDir = joinPath(tmpDir, 'extensions', 'my-function') + await mkdir(extensionDir) + const app = testAppLinked() - const extension = await testFunctionExtension({}) - const apiKey = 'api-key' - const path = tmpDir + const extension = await testFunctionExtension({ + dir: tmpDir, + }) // When await generateSchemaService({ app, extension, - path, + path: tmpDir, stdout: false, developerPlatformClient: testDeveloperPlatformClient(), + orgId, }) // Then - const outputFile = await readFile(joinPath(tmpDir, 'schema.graphql')) + const outputFile = await readFile(joinPath(extension.directory, 'schema.graphql')) expect(outputFile).toEqual('schema') }) }) @@ -55,6 +60,7 @@ describe('generateSchemaService', () => { const extension = await testFunctionExtension() const path = tmpDir const stdout = true + const orgId = '123' const mockOutput = vi.fn() vi.spyOn(output, 'outputInfo').mockImplementation(mockOutput) @@ -65,6 +71,7 @@ describe('generateSchemaService', () => { path, stdout, developerPlatformClient: testDeveloperPlatformClient(), + orgId, }) // Then @@ -75,13 +82,24 @@ describe('generateSchemaService', () => { describe('GraphQL query', () => { test('Uses ApiSchemaDefinitionQuery when not using targets', async () => { await inTemporaryDirectory(async (tmpDir) => { - // Given + const extensionDir = joinPath(tmpDir, 'extensions', 'my-function') + await mkdir(extensionDir) + const app = testAppLinked() const extension = await testFunctionExtension({ + dir: tmpDir, config: { name: 'test function extension', description: 'description', - type: 'api_type', + type: 'function', + targeting: [ + { + target: 'first', + }, + { + target: 'second', + }, + ], build: { command: 'echo "hello world"', }, @@ -90,37 +108,41 @@ describe('generateSchemaService', () => { metafields: [], }, }) - const apiKey = 'api-key' + + const orgId = 'test' const path = tmpDir - const { - configuration: {api_version: version}, - type, - } = extension + const expectedTarget = extension.configuration.targeting![0]!.target + const version = extension.configuration.api_version const developerPlatformClient = testDeveloperPlatformClient() - // When await generateSchemaService({ app, extension, path, stdout: false, developerPlatformClient, + orgId, }) - // Then - expect(developerPlatformClient.apiSchemaDefinition).toHaveBeenCalledWith({ - apiKey, - version, - type, - }) + expect(developerPlatformClient.targetSchemaDefinition).toHaveBeenCalledWith( + { + apiKey: app.configuration.client_id, + version, + handle: expectedTarget, + }, + orgId, + ) }) }) test('Uses TargetSchemaDefinitionQuery when targets present', async () => { await inTemporaryDirectory(async (tmpDir) => { - // Given + const extensionDir = joinPath(tmpDir, 'extensions', 'my-function') + await mkdir(extensionDir) + const app = testAppLinked() const extension = await testFunctionExtension({ + dir: tmpDir, config: { name: 'test function extension', description: 'description', @@ -141,27 +163,30 @@ describe('generateSchemaService', () => { metafields: [], }, }) - const apiKey = 'api-key' + const path = tmpDir const expectedTarget = extension.configuration.targeting![0]!.target const version = extension.configuration.api_version + const orgId = 'test' const developerPlatformClient = testDeveloperPlatformClient() - // When await generateSchemaService({ app, extension, path, stdout: false, developerPlatformClient, + orgId, }) - // Then - expect(developerPlatformClient.targetSchemaDefinition).toHaveBeenCalledWith({ - apiKey, - version, - target: expectedTarget, - }) + expect(developerPlatformClient.targetSchemaDefinition).toHaveBeenCalledWith( + { + apiKey: app.configuration.client_id, + version, + handle: expectedTarget, + }, + orgId, + ) }) }) }) @@ -170,9 +195,10 @@ describe('generateSchemaService', () => { // Given const app = testAppLinked() const extension = await testFunctionExtension() - const apiKey = 'api-key' + const orgId = '123' const developerPlatformClient = testDeveloperPlatformClient({ - apiSchemaDefinition: (_input: ApiSchemaDefinitionQueryVariables) => Promise.resolve(null), + apiSchemaDefinition: (_input: SchemaDefinitionByApiTypeQueryVariables & {apiKey?: string}, _orgId: string) => + Promise.resolve(null), }) // When @@ -182,6 +208,7 @@ describe('generateSchemaService', () => { path: '', stdout: true, developerPlatformClient, + orgId, }) // Then diff --git a/packages/app/src/cli/services/generate-schema.ts b/packages/app/src/cli/services/generate-schema.ts index 8544d50b5b..305bf4d680 100644 --- a/packages/app/src/cli/services/generate-schema.ts +++ b/packages/app/src/cli/services/generate-schema.ts @@ -1,6 +1,6 @@ import {DeveloperPlatformClient} from '../utilities/developer-platform-client.js' -import {ApiSchemaDefinitionQueryVariables} from '../api/graphql/functions/api_schema_definition.js' -import {TargetSchemaDefinitionQueryVariables} from '../api/graphql/functions/target_schema_definition.js' +import {SchemaDefinitionByApiTypeQueryVariables} from '../api/graphql/functions/generated/schema-definition-by-api-type.js' +import {SchemaDefinitionByTargetQueryVariables} from '../api/graphql/functions/generated/schema-definition-by-target.js' import {ExtensionInstance} from '../models/extensions/extension-instance.js' import {FunctionConfigType} from '../models/extensions/specifications/function.js' import {AppLinkedInterface} from '../models/app/app.js' @@ -15,10 +15,11 @@ interface GenerateSchemaOptions { stdout: boolean path: string developerPlatformClient: DeveloperPlatformClient + orgId: string } export async function generateSchemaService(options: GenerateSchemaOptions) { - const {extension, stdout, developerPlatformClient, app} = options + const {extension, stdout, developerPlatformClient, app, orgId} = options const apiKey = app.configuration.client_id const {api_version: version, type, targeting} = extension.configuration @@ -31,19 +32,21 @@ export async function generateSchemaService(options: GenerateSchemaOptions) { // eslint-disable-next-line @typescript-eslint/no-non-null-assertion target: targeting![0]!.target, version, + orgId, }) - : generateSchemaFromType({ + : generateSchemaFromApiType({ localIdentifier: extension.localIdentifier, developerPlatformClient, apiKey, type, version, + orgId, })) if (stdout) { outputInfo(definition) } else { - const outputPath = joinPath(options.path, 'schema.graphql') + const outputPath = joinPath(extension.directory, 'schema.graphql') await writeFile(outputPath, definition) outputInfo(`GraphQL Schema for ${extension.localIdentifier} written to ${outputPath}`) } @@ -54,6 +57,7 @@ interface BaseGenerateSchemaOptions { developerPlatformClient: DeveloperPlatformClient apiKey: string version: string + orgId: string } interface GenerateSchemaFromTargetOptions extends BaseGenerateSchemaOptions { @@ -66,13 +70,20 @@ async function generateSchemaFromTarget({ apiKey, target, version, + orgId, }: GenerateSchemaFromTargetOptions): Promise { - const variables: TargetSchemaDefinitionQueryVariables = { - apiKey, - target, + const variables: SchemaDefinitionByTargetQueryVariables = { + handle: target, version, } - const definition = await developerPlatformClient.targetSchemaDefinition(variables) + // Api key required for partners reqs, can be removed once migrated to AMF + const definition = await developerPlatformClient.targetSchemaDefinition( + { + ...variables, + apiKey, + }, + orgId, + ) if (!definition) { throw new AbortError( @@ -88,19 +99,27 @@ interface GenerateSchemaFromType extends BaseGenerateSchemaOptions { type: string } -async function generateSchemaFromType({ +async function generateSchemaFromApiType({ localIdentifier, developerPlatformClient, apiKey, version, type, + orgId, }: GenerateSchemaFromType): Promise { - const variables: ApiSchemaDefinitionQueryVariables = { - apiKey, + const variables: SchemaDefinitionByApiTypeQueryVariables = { version, type, } - const definition = await developerPlatformClient.apiSchemaDefinition(variables) + + // Api key required for partners reqs, can be removed once migrated to AMF + const definition = await developerPlatformClient.apiSchemaDefinition( + { + ...variables, + apiKey, + }, + orgId, + ) if (!definition) { throw new AbortError( diff --git a/packages/app/src/cli/utilities/developer-platform-client.ts b/packages/app/src/cli/utilities/developer-platform-client.ts index 87bf8a14bb..54064d35d9 100644 --- a/packages/app/src/cli/utilities/developer-platform-client.ts +++ b/packages/app/src/cli/utilities/developer-platform-client.ts @@ -36,8 +36,8 @@ import { import {UpdateURLsSchema, UpdateURLsVariables} from '../api/graphql/update_urls.js' import {CurrentAccountInfoSchema} from '../api/graphql/current_account_info.js' import {ExtensionTemplate} from '../models/app/template.js' -import {TargetSchemaDefinitionQueryVariables} from '../api/graphql/functions/target_schema_definition.js' -import {ApiSchemaDefinitionQueryVariables} from '../api/graphql/functions/api_schema_definition.js' +import {SchemaDefinitionByTargetQueryVariables} from '../api/graphql/functions/generated/schema-definition-by-target.js' +import {SchemaDefinitionByApiTypeQueryVariables} from '../api/graphql/functions/generated/schema-definition-by-api-type.js' import { MigrateToUiExtensionSchema, MigrateToUiExtensionVariables, @@ -248,8 +248,19 @@ export interface DeveloperPlatformClient { migrateAppModule: (input: MigrateAppModuleVariables) => Promise updateURLs: (input: UpdateURLsVariables) => Promise currentAccountInfo: () => Promise - targetSchemaDefinition: (input: TargetSchemaDefinitionQueryVariables) => Promise - apiSchemaDefinition: (input: ApiSchemaDefinitionQueryVariables) => Promise + // Api key required for partners requests, can be removed once migrated to AMF + targetSchemaDefinition: ( + input: SchemaDefinitionByTargetQueryVariables & { + apiKey?: string + }, + organizationId: string, + ) => Promise + apiSchemaDefinition: ( + input: SchemaDefinitionByApiTypeQueryVariables & { + apiKey?: string + }, + organizationId: string, + ) => Promise migrateToUiExtension: (input: MigrateToUiExtensionVariables) => Promise toExtensionGraphQLType: (input: string) => string subscribeToAppLogs: (input: AppLogsSubscribeVariables) => Promise diff --git a/packages/app/src/cli/utilities/developer-platform-client/app-management-client.ts b/packages/app/src/cli/utilities/developer-platform-client/app-management-client.ts index ab87551267..7c242bbe50 100644 --- a/packages/app/src/cli/utilities/developer-platform-client/app-management-client.ts +++ b/packages/app/src/cli/utilities/developer-platform-client/app-management-client.ts @@ -56,8 +56,6 @@ import { import {UpdateURLsSchema, UpdateURLsVariables} from '../../api/graphql/update_urls.js' import {CurrentAccountInfoSchema} from '../../api/graphql/current_account_info.js' import {ExtensionTemplate} from '../../models/app/template.js' -import {TargetSchemaDefinitionQueryVariables} from '../../api/graphql/functions/target_schema_definition.js' -import {ApiSchemaDefinitionQueryVariables} from '../../api/graphql/functions/api_schema_definition.js' import { MigrateToUiExtensionVariables, MigrateToUiExtensionSchema, @@ -100,6 +98,16 @@ import {FetchSpecifications} from '../../api/graphql/app-management/generated/sp import {ListApps} from '../../api/graphql/app-management/generated/apps.js' import {FindOrganizations} from '../../api/graphql/business-platform-destinations/generated/find-organizations.js' import {UserInfo} from '../../api/graphql/business-platform-destinations/generated/user-info.js' +import { + SchemaDefinitionByTarget, + SchemaDefinitionByTargetQuery, + SchemaDefinitionByTargetQueryVariables, +} from '../../api/graphql/functions/generated/schema-definition-by-target.js' +import { + SchemaDefinitionByApiType, + SchemaDefinitionByApiTypeQuery, + SchemaDefinitionByApiTypeQueryVariables, +} from '../../api/graphql/functions/generated/schema-definition-by-api-type.js' import {ensureAuthenticatedAppManagement, ensureAuthenticatedBusinessPlatform} from '@shopify/cli-kit/node/session' import {isUnitTest} from '@shopify/cli-kit/node/context/local' import {AbortError, BugError} from '@shopify/cli-kit/node/error' @@ -115,6 +123,7 @@ import {CLI_KIT_VERSION} from '@shopify/cli-kit/common/version' import {versionSatisfies} from '@shopify/cli-kit/node/node-package-manager' import {outputDebug} from '@shopify/cli-kit/node/output' import {developerDashboardFqdn} from '@shopify/cli-kit/node/context/fqdn' +import {functionsRequestDoc} from '@shopify/cli-kit/node/api/functions' const TEMPLATE_JSON_URL = 'https://cdn.shopify.com/static/cli/extensions/templates.json' @@ -747,12 +756,45 @@ export class AppManagementClient implements DeveloperPlatformClient { throw new BugError('Not implemented: currentAccountInfo') } - async targetSchemaDefinition(_input: TargetSchemaDefinitionQueryVariables): Promise { - throw new BugError('Not implemented: targetSchemaDefinition') + async targetSchemaDefinition( + input: SchemaDefinitionByTargetQueryVariables, + organizationId: string, + ): Promise { + try { + const token = await this.token() + const result = await functionsRequestDoc( + organizationId, + SchemaDefinitionByTarget, + token, + { + handle: input.handle, + version: input.version, + }, + ) + + return result?.target?.api?.schema?.definition ?? null + } catch (error) { + throw new AbortError(`Failed to fetch schema definition: ${error}`) + } } - async apiSchemaDefinition(_input: ApiSchemaDefinitionQueryVariables): Promise { - throw new BugError('Not implemented: apiSchemaDefinition') + async apiSchemaDefinition( + input: SchemaDefinitionByApiTypeQueryVariables, + organizationId: string, + ): Promise { + try { + const token = await this.token() + const result = await functionsRequestDoc( + organizationId, + SchemaDefinitionByApiType, + token, + input, + ) + + return result?.api?.schema?.definition ?? null + } catch (error) { + throw new AbortError(`Failed to fetch schema definition: ${error}`) + } } async migrateToUiExtension(_input: MigrateToUiExtensionVariables): Promise { diff --git a/packages/app/src/cli/utilities/developer-platform-client/partners-client.ts b/packages/app/src/cli/utilities/developer-platform-client/partners-client.ts index f25f50b35d..041f253f1e 100644 --- a/packages/app/src/cli/utilities/developer-platform-client/partners-client.ts +++ b/packages/app/src/cli/utilities/developer-platform-client/partners-client.ts @@ -99,12 +99,10 @@ import { } from '../../api/graphql/template_specifications.js' import {ExtensionTemplate} from '../../models/app/template.js' import { - TargetSchemaDefinitionQueryVariables, TargetSchemaDefinitionQuerySchema, TargetSchemaDefinitionQuery, } from '../../api/graphql/functions/target_schema_definition.js' import { - ApiSchemaDefinitionQueryVariables, ApiSchemaDefinitionQuerySchema, ApiSchemaDefinitionQuery, } from '../../api/graphql/functions/api_schema_definition.js' @@ -153,6 +151,8 @@ import { DevStoresByOrgQuery, DevStoresByOrgQueryVariables, } from '../../api/graphql/partners/generated/dev-stores-by-org.js' +import {SchemaDefinitionByTargetQueryVariables} from '../../api/graphql/functions/generated/schema-definition-by-target.js' +import {SchemaDefinitionByApiTypeQueryVariables} from '../../api/graphql/functions/generated/schema-definition-by-api-type.js' import {TypedDocumentNode} from '@graphql-typed-document-node/core' import {isUnitTest} from '@shopify/cli-kit/node/context/local' import {AbortError} from '@shopify/cli-kit/node/error' @@ -502,13 +502,33 @@ export class PartnersClient implements DeveloperPlatformClient { return this.request(CurrentAccountInfoQuery) } - async targetSchemaDefinition(input: TargetSchemaDefinitionQueryVariables): Promise { - const response: TargetSchemaDefinitionQuerySchema = await this.request(TargetSchemaDefinitionQuery, input) + async targetSchemaDefinition( + input: SchemaDefinitionByTargetQueryVariables & {apiKey?: string}, + _organizationId: string, + ): Promise { + // Ensures backwards compatibility with existing partners requests + // Can remove once migrated to AMF + const transformedInput = { + target: input.handle, + version: input.version, + apiKey: input.apiKey, + } + + const response: TargetSchemaDefinitionQuerySchema = await this.request( + TargetSchemaDefinitionQuery, + transformedInput, + ) return response.definition } - async apiSchemaDefinition(input: ApiSchemaDefinitionQueryVariables): Promise { - const response: ApiSchemaDefinitionQuerySchema = await this.request(ApiSchemaDefinitionQuery, input) + async apiSchemaDefinition( + input: SchemaDefinitionByApiTypeQueryVariables & {apiKey?: string}, + _organizationId: string, + ): Promise { + const response: ApiSchemaDefinitionQuerySchema = await this.request(ApiSchemaDefinitionQuery, { + ...input, + apiKey: input.apiKey, + }) return response.definition } diff --git a/packages/cli-kit/src/public/node/api/functions.ts b/packages/cli-kit/src/public/node/api/functions.ts new file mode 100644 index 0000000000..af2e5d30de --- /dev/null +++ b/packages/cli-kit/src/public/node/api/functions.ts @@ -0,0 +1,59 @@ +import {graphqlRequestDoc} from './graphql.js' +import {handleDeprecations} from './app-management.js' +import {appManagementFqdn} from '../context/fqdn.js' +import {TypedDocumentNode} from '@graphql-typed-document-node/core' +import {Variables} from 'graphql-request' +import Bottleneck from 'bottleneck' + +// API Rate limiter for partners API (Limit is 10 requests per second) +// Jobs are launched every 150ms to add an extra 50ms margin per request. +// Only 10 requests can be executed concurrently. +const limiter = new Bottleneck({ + minTime: 150, + maxConcurrent: 10, +}) + +/** + * Prepares the request configuration for the App Management Functions API. + * + * @param orgId - Organization identifier. + * @param token - Authentication token. + * @returns Request configuration object. + */ +async function setupRequest(orgId: string, token: string) { + const api = 'Functions' + const fqdn = await appManagementFqdn() + const url = `https://${fqdn}/functions/unstable/organizations/${orgId}/graphql` + return { + token, + api, + url, + responseOptions: {onResponse: handleDeprecations}, + } +} + +/** + * Executes a rate-limited GraphQL request against the App Management Functions API. + * + * @param orgId - Organization identifier. + * @param query - Typed GraphQL document node. + * @param token - Authentication token. + * @param variables - Optional query variables. + * @returns Promise resolving to the typed query result. + */ +export async function functionsRequestDoc( + orgId: string, + query: TypedDocumentNode, + token: string, + variables?: TVariables, +): Promise { + const result = await limiter.schedule(async () => { + return graphqlRequestDoc({ + ...(await setupRequest(orgId, token)), + query, + variables, + }) + }) + + return result +} diff --git a/packages/eslint-plugin-cli/rules/no-inline-graphql.js b/packages/eslint-plugin-cli/rules/no-inline-graphql.js index 0c3e5991a3..9287a1ca65 100644 --- a/packages/eslint-plugin-cli/rules/no-inline-graphql.js +++ b/packages/eslint-plugin-cli/rules/no-inline-graphql.js @@ -152,10 +152,10 @@ const knownFailures = { '867f01113c20386d6a438dd56a6d241199e407eab928ab1ad9a7f233cd35c1be', 'packages/app/src/cli/api/graphql/find_store_by_domain.ts': '0824f5baaab1ad419a7fa1d64824e306bd369430da47c7457ed72e74a1e94a9a', - 'packages/app/src/cli/api/graphql/functions/api_schema_definition.ts': - '9ebbab831a66a2e49a8c2dac3185bb58c4688655040c6d14a9b2fb41a004bca8', + 'packages/app/src/cli/api/graphql/functions/api_schema_definition.ts': + 'e71100cf61919831681da1be8f12cd9067c4e3f2faf04c1b88b764fd8a275b82', 'packages/app/src/cli/api/graphql/functions/target_schema_definition.ts': - '0469c1fe724568f031a52c5bda54d56d8c3b23d7e8a54a86a684f72028d71b46', + 'd338c5d187ca8a1e1e68892987d780e540426faeba89df2dd9d8c96e193f5c13', 'packages/app/src/cli/api/graphql/generate_signed_upload_url.ts': '848e40bf6b44331de0fe1dc1b0753593c1d47f9705ebe988a1b8ad5638d267ef', 'packages/app/src/cli/api/graphql/get_variant_id.ts':