@aws-amplify/graphql-api-construct
Version:
AppSync GraphQL Api Construct using Amplify GraphQL Transformer.
226 lines • 30.9 kB
JavaScript
;
Object.defineProperty(exports, "__esModule", { value: true });
exports.getGeneratedFunctionSlots = exports.getGeneratedResources = void 0;
const aws_appsync_1 = require("aws-cdk-lib/aws-appsync");
const aws_dynamodb_1 = require("aws-cdk-lib/aws-dynamodb");
const aws_iam_1 = require("aws-cdk-lib/aws-iam");
const aws_cdk_lib_1 = require("aws-cdk-lib");
const graphql_transformer_core_1 = require("@aws-amplify/graphql-transformer-core");
const aws_lambda_1 = require("aws-cdk-lib/aws-lambda");
const amplify_dynamodb_table_wrapper_1 = require("../amplify-dynamodb-table-wrapper");
const construct_tree_1 = require("./construct-tree");
/**
* Check if a resource is implementing table interface
* The required properties need to be present in the input
* https://docs.aws.amazon.com/cdk/api/v2/docs/aws-cdk-lib.aws_dynamodb.ITable.html#properties
* @param table table resource
* @returns whether the resource is a ITable or not
*/
const isITable = (table) => {
return 'env' in table && 'node' in table && 'stack' in table && 'tableArn' in table && 'tableName' in table;
};
/**
* Everything below here is intended to help us gather the
* output values and render out the L1 resources for access.
*
* This is done by recursing along the construct tree, and classifying the generated resources.
*
* @param scope root to search for generated resource against
* @returns a mapping of L1 and L2 constructs generated by the Graphql Transformer.
*/
const getGeneratedResources = (scope) => {
let cfnGraphqlApi;
let cfnGraphqlSchema;
let cfnApiKey;
const cfnResolvers = {};
const cfnFunctionConfigurations = {};
const cfnDataSources = {};
const tables = {};
const cfnTables = {};
const amplifyDynamoDbTables = {};
const roles = {};
const cfnRoles = {};
const functions = {};
const cfnFunctions = {};
const additionalCfnResources = {};
const classifyConstruct = (currentScope) => {
if (currentScope instanceof aws_appsync_1.CfnGraphQLApi) {
cfnGraphqlApi = currentScope;
return;
}
if (currentScope instanceof aws_appsync_1.CfnGraphQLSchema) {
cfnGraphqlSchema = currentScope;
return;
}
if (currentScope instanceof aws_appsync_1.CfnApiKey) {
cfnApiKey = currentScope;
return;
}
// Retrieve reference name for indexed resources, and bail if none is found.
const resourceName = (0, graphql_transformer_core_1.getResourceName)(currentScope);
if (!resourceName)
return;
if (currentScope instanceof aws_appsync_1.CfnDataSource) {
cfnDataSources[resourceName] = currentScope;
return;
}
if (currentScope instanceof aws_appsync_1.CfnResolver) {
cfnResolvers[resourceName] = currentScope;
return;
}
if (currentScope instanceof aws_appsync_1.CfnFunctionConfiguration) {
cfnFunctionConfigurations[resourceName] = currentScope;
return;
}
if (currentScope instanceof aws_dynamodb_1.Table || isITable(currentScope)) {
tables[resourceName] = currentScope;
return;
}
if (currentScope instanceof aws_dynamodb_1.CfnTable) {
cfnTables[resourceName] = currentScope;
return;
}
if (amplify_dynamodb_table_wrapper_1.AmplifyDynamoDbTableWrapper.isAmplifyDynamoDbTableResource(currentScope)) {
amplifyDynamoDbTables[resourceName] = new amplify_dynamodb_table_wrapper_1.AmplifyDynamoDbTableWrapper(currentScope);
return;
}
if (currentScope instanceof aws_iam_1.Role) {
roles[resourceName] = currentScope;
return;
}
if (currentScope instanceof aws_iam_1.CfnRole) {
cfnRoles[resourceName] = currentScope;
return;
}
if (currentScope instanceof aws_lambda_1.Function) {
functions[resourceName] = currentScope;
return;
}
if (currentScope instanceof aws_lambda_1.CfnFunction) {
cfnFunctions[resourceName] = currentScope;
return;
}
if (currentScope instanceof aws_cdk_lib_1.CfnResource) {
additionalCfnResources[resourceName] = currentScope;
return;
}
};
scope.node.children.forEach((child) => (0, construct_tree_1.walkAndProcessNodes)(child, classifyConstruct));
if (!cfnGraphqlApi) {
throw new Error('Expected to find AWS::AppSync::GraphQLApi in the generated resource scope.');
}
if (!cfnGraphqlSchema) {
throw new Error('Expected to find AWS::AppSync::GraphQLSchema in the generated resource scope.');
}
const nestedStacks = Object.fromEntries(scope.node.children.filter(aws_cdk_lib_1.NestedStack.isNestedStack).map((nestedStack) => [nestedStack.node.id, nestedStack]));
const proxiedApiAttributes = graphqlApiAttributesFromCfnGraphQLApi(cfnGraphqlApi);
return {
graphqlApi: aws_appsync_1.GraphqlApi.fromGraphqlApiAttributes(scope, 'L2GraphqlApi', proxiedApiAttributes),
tables,
roles,
functions,
nestedStacks,
cfnResources: {
cfnGraphqlApi,
cfnGraphqlSchema,
cfnApiKey,
cfnResolvers,
cfnFunctionConfigurations,
cfnDataSources,
cfnTables,
amplifyDynamoDbTables,
cfnRoles,
cfnFunctions,
additionalCfnResources,
},
};
};
exports.getGeneratedResources = getGeneratedResources;
/**
* Creates a set of L2 {@link GraphqlApiAttributes} from a CfnGraphqlApi L1 construct. Allows for getGeneratedResources to easily pass
* attributes of the CfnGraphqlApi to the `resources` member. Without this the `resources.graphqlApi` member has no properties except for
* the API ID.
*/
const graphqlApiAttributesFromCfnGraphQLApi = (cfnGraphqlApi) => {
const visiblityStruct = {};
if (typeof cfnGraphqlApi.visibility === 'string') {
switch (cfnGraphqlApi.visibility) {
case 'GLOBAL':
visiblityStruct.visibility = aws_appsync_1.Visibility.GLOBAL;
break;
case 'PRIVATE':
visiblityStruct.visibility = aws_appsync_1.Visibility.PRIVATE;
break;
default:
console.warn(`Unsupported AppSync API Visibility setting: ${cfnGraphqlApi.visibility}`);
}
}
const proxiedApiAttributes = {
graphqlApiId: cfnGraphqlApi.attrApiId,
graphqlApiArn: cfnGraphqlApi.attrArn,
graphQLEndpointArn: cfnGraphqlApi.attrGraphQlEndpointArn,
...visiblityStruct,
modes: authenticationTypesFromCfnGraphQLApi(cfnGraphqlApi),
};
return proxiedApiAttributes;
};
const authenticationTypesFromCfnGraphQLApi = (cfnGraphqlApi) => {
const additionalAuthenticationProviders = cfnGraphqlApi.additionalAuthenticationProviders;
if (!additionalAuthenticationProviders) {
return [];
}
// If this is a deploy-time value rather than an array, we can't convert accurately.
if ((0, aws_cdk_lib_1.isResolvableObject)(additionalAuthenticationProviders)) {
return [];
}
const unfilteredAuthorizationTypes = additionalAuthenticationProviders
.filter((additionalAuthProvider) => !(0, aws_cdk_lib_1.isResolvableObject)(additionalAuthProvider))
.map((provider) => provider.authenticationType)
.map(l2AuthorizationTypeFromL1AuthenticationProvider);
const authorizationTypes = unfilteredAuthorizationTypes.filter((type) => typeof type !== 'undefined');
const defaultAuthorizationType = l2AuthorizationTypeFromL1AuthenticationProvider(cfnGraphqlApi.authenticationType);
if (defaultAuthorizationType) {
authorizationTypes.push(defaultAuthorizationType);
}
return authorizationTypes;
};
const l2AuthorizationTypeFromL1AuthenticationProvider = (authenticationType) => {
switch (authenticationType) {
case 'API_KEY':
return aws_appsync_1.AuthorizationType.API_KEY;
case 'AWS_IAM':
return aws_appsync_1.AuthorizationType.IAM;
case 'OPENID_CONNECT':
return aws_appsync_1.AuthorizationType.OIDC;
case 'AMAZON_COGNITO_USER_POOLS':
return aws_appsync_1.AuthorizationType.USER_POOL;
case 'AWS_LAMBDA':
return aws_appsync_1.AuthorizationType.LAMBDA;
default:
console.warn(`Unrecognized Authentication type ${authenticationType}`);
return undefined;
}
};
/**
* Get the function slots generated by the Graphql transform operation, adhering to the FunctionSlot interface.
* @param generatedResolvers the resolvers generated by the transformer to spit back out.
* @returns the list of generated function slots in the transformer, in order to facilitate overrides.
*/
const getGeneratedFunctionSlots = (generatedResolvers) => Object.entries(generatedResolvers)
.filter(([name]) => name.split('.').length === 6)
.map(([name, resolverCode]) => {
const [typeName, fieldName, slotName, slotIndex, templateType] = name.split('.');
return {
typeName,
fieldName,
slotName,
slotIndex: Number.parseInt(slotIndex, 10),
function: {
// TODO: this should consolidate req/req values back together
...(templateType === 'req' ? { requestMappingTemplate: resolverCode } : {}),
...(templateType === 'res' ? { responseMappingTemplate: resolverCode } : {}),
},
};
});
exports.getGeneratedFunctionSlots = getGeneratedFunctionSlots;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"construct-exports.js","sourceRoot":"","sources":["../../src/internal/construct-exports.ts"],"names":[],"mappings":"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","sourcesContent":["import { Construct } from 'constructs';\nimport {\n  AuthorizationType,\n  CfnGraphQLApi,\n  CfnGraphQLSchema,\n  CfnApiKey,\n  CfnResolver,\n  CfnFunctionConfiguration,\n  CfnDataSource,\n  GraphqlApi,\n  GraphqlApiAttributes,\n  Visibility,\n} from 'aws-cdk-lib/aws-appsync';\nimport { CfnTable, Table, ITable } from 'aws-cdk-lib/aws-dynamodb';\nimport { CfnRole, Role } from 'aws-cdk-lib/aws-iam';\nimport { CfnResource, isResolvableObject, NestedStack } from 'aws-cdk-lib';\nimport { getResourceName } from '@aws-amplify/graphql-transformer-core';\nimport { CfnFunction, Function as LambdaFunction } from 'aws-cdk-lib/aws-lambda';\nimport { AmplifyGraphqlApiResources, FunctionSlot } from '../types';\nimport { AmplifyDynamoDbTableWrapper } from '../amplify-dynamodb-table-wrapper';\nimport { walkAndProcessNodes } from './construct-tree';\n\n/**\n * Check if a resource is implementing table interface\n * The required properties need to be present in the input\n * https://docs.aws.amazon.com/cdk/api/v2/docs/aws-cdk-lib.aws_dynamodb.ITable.html#properties\n * @param table table resource\n * @returns whether the resource is a ITable or not\n */\nconst isITable = (table: any): table is ITable => {\n  return 'env' in table && 'node' in table && 'stack' in table && 'tableArn' in table && 'tableName' in table;\n};\n\n/**\n * Everything below here is intended to help us gather the\n * output values and render out the L1 resources for access.\n *\n * This is done by recursing along the construct tree, and classifying the generated resources.\n *\n * @param scope root to search for generated resource against\n * @returns a mapping of L1 and L2 constructs generated by the Graphql Transformer.\n */\nexport const getGeneratedResources = (scope: Construct): AmplifyGraphqlApiResources => {\n  let cfnGraphqlApi: CfnGraphQLApi | undefined;\n  let cfnGraphqlSchema: CfnGraphQLSchema | undefined;\n  let cfnApiKey: CfnApiKey | undefined;\n  const cfnResolvers: Record<string, CfnResolver> = {};\n  const cfnFunctionConfigurations: Record<string, CfnFunctionConfiguration> = {};\n  const cfnDataSources: Record<string, CfnDataSource> = {};\n  const tables: Record<string, ITable> = {};\n  const cfnTables: Record<string, CfnTable> = {};\n  const amplifyDynamoDbTables: Record<string, AmplifyDynamoDbTableWrapper> = {};\n  const roles: Record<string, Role> = {};\n  const cfnRoles: Record<string, CfnRole> = {};\n  const functions: Record<string, LambdaFunction> = {};\n  const cfnFunctions: Record<string, CfnFunction> = {};\n  const additionalCfnResources: Record<string, CfnResource> = {};\n\n  const classifyConstruct = (currentScope: Construct): void => {\n    if (currentScope instanceof CfnGraphQLApi) {\n      cfnGraphqlApi = currentScope;\n      return;\n    }\n    if (currentScope instanceof CfnGraphQLSchema) {\n      cfnGraphqlSchema = currentScope;\n      return;\n    }\n    if (currentScope instanceof CfnApiKey) {\n      cfnApiKey = currentScope;\n      return;\n    }\n\n    // Retrieve reference name for indexed resources, and bail if none is found.\n    const resourceName = getResourceName(currentScope);\n    if (!resourceName) return;\n\n    if (currentScope instanceof CfnDataSource) {\n      cfnDataSources[resourceName] = currentScope;\n      return;\n    }\n    if (currentScope instanceof CfnResolver) {\n      cfnResolvers[resourceName] = currentScope;\n      return;\n    }\n    if (currentScope instanceof CfnFunctionConfiguration) {\n      cfnFunctionConfigurations[resourceName] = currentScope;\n      return;\n    }\n    if (currentScope instanceof Table || isITable(currentScope)) {\n      tables[resourceName] = currentScope;\n      return;\n    }\n    if (currentScope instanceof CfnTable) {\n      cfnTables[resourceName] = currentScope;\n      return;\n    }\n    if (AmplifyDynamoDbTableWrapper.isAmplifyDynamoDbTableResource(currentScope)) {\n      amplifyDynamoDbTables[resourceName] = new AmplifyDynamoDbTableWrapper(currentScope);\n      return;\n    }\n    if (currentScope instanceof Role) {\n      roles[resourceName] = currentScope;\n      return;\n    }\n    if (currentScope instanceof CfnRole) {\n      cfnRoles[resourceName] = currentScope;\n      return;\n    }\n    if (currentScope instanceof LambdaFunction) {\n      functions[resourceName] = currentScope;\n      return;\n    }\n    if (currentScope instanceof CfnFunction) {\n      cfnFunctions[resourceName] = currentScope;\n      return;\n    }\n    if (currentScope instanceof CfnResource) {\n      additionalCfnResources[resourceName] = currentScope;\n      return;\n    }\n  };\n\n  scope.node.children.forEach((child) => walkAndProcessNodes(child, classifyConstruct));\n\n  if (!cfnGraphqlApi) {\n    throw new Error('Expected to find AWS::AppSync::GraphQLApi in the generated resource scope.');\n  }\n\n  if (!cfnGraphqlSchema) {\n    throw new Error('Expected to find AWS::AppSync::GraphQLSchema in the generated resource scope.');\n  }\n\n  const nestedStacks: Record<string, NestedStack> = Object.fromEntries(\n    scope.node.children.filter(NestedStack.isNestedStack).map((nestedStack: NestedStack) => [nestedStack.node.id, nestedStack]),\n  );\n\n  const proxiedApiAttributes = graphqlApiAttributesFromCfnGraphQLApi(cfnGraphqlApi);\n\n  return {\n    graphqlApi: GraphqlApi.fromGraphqlApiAttributes(scope, 'L2GraphqlApi', proxiedApiAttributes),\n    tables,\n    roles,\n    functions,\n    nestedStacks,\n    cfnResources: {\n      cfnGraphqlApi,\n      cfnGraphqlSchema,\n      cfnApiKey,\n      cfnResolvers,\n      cfnFunctionConfigurations,\n      cfnDataSources,\n      cfnTables,\n      amplifyDynamoDbTables,\n      cfnRoles,\n      cfnFunctions,\n      additionalCfnResources,\n    },\n  };\n};\n\n/**\n * Creates a set of L2 {@link GraphqlApiAttributes} from a CfnGraphqlApi L1 construct. Allows for getGeneratedResources to easily pass\n * attributes of the CfnGraphqlApi to the `resources` member. Without this the `resources.graphqlApi` member has no properties except for\n * the API ID.\n */\nconst graphqlApiAttributesFromCfnGraphQLApi = (cfnGraphqlApi: CfnGraphQLApi): GraphqlApiAttributes => {\n  const visiblityStruct: { visibility?: Visibility } = {};\n  if (typeof cfnGraphqlApi.visibility === 'string') {\n    switch (cfnGraphqlApi.visibility) {\n      case 'GLOBAL':\n        visiblityStruct.visibility = Visibility.GLOBAL;\n        break;\n      case 'PRIVATE':\n        visiblityStruct.visibility = Visibility.PRIVATE;\n        break;\n      default:\n        console.warn(`Unsupported AppSync API Visibility setting: ${cfnGraphqlApi.visibility}`);\n    }\n  }\n\n  const proxiedApiAttributes: GraphqlApiAttributes = {\n    graphqlApiId: cfnGraphqlApi.attrApiId,\n    graphqlApiArn: cfnGraphqlApi.attrArn,\n    graphQLEndpointArn: cfnGraphqlApi.attrGraphQlEndpointArn,\n    ...visiblityStruct,\n    modes: authenticationTypesFromCfnGraphQLApi(cfnGraphqlApi),\n  };\n\n  return proxiedApiAttributes;\n};\n\nconst authenticationTypesFromCfnGraphQLApi = (cfnGraphqlApi: CfnGraphQLApi): AuthorizationType[] => {\n  const additionalAuthenticationProviders = cfnGraphqlApi.additionalAuthenticationProviders;\n  if (!additionalAuthenticationProviders) {\n    return [];\n  }\n\n  // If this is a deploy-time value rather than an array, we can't convert accurately.\n  if (isResolvableObject(additionalAuthenticationProviders)) {\n    return [];\n  }\n\n  const unfilteredAuthorizationTypes: (AuthorizationType | undefined)[] = additionalAuthenticationProviders\n    .filter(\n      (additionalAuthProvider): additionalAuthProvider is CfnGraphQLApi.AdditionalAuthenticationProviderProperty =>\n        !isResolvableObject(additionalAuthProvider),\n    )\n    .map((provider) => provider.authenticationType)\n    .map(l2AuthorizationTypeFromL1AuthenticationProvider);\n\n  const authorizationTypes = unfilteredAuthorizationTypes.filter((type): type is AuthorizationType => typeof type !== 'undefined');\n\n  const defaultAuthorizationType = l2AuthorizationTypeFromL1AuthenticationProvider(cfnGraphqlApi.authenticationType);\n  if (defaultAuthorizationType) {\n    authorizationTypes.push(defaultAuthorizationType);\n  }\n  return authorizationTypes;\n};\n\nconst l2AuthorizationTypeFromL1AuthenticationProvider = (authenticationType: string): AuthorizationType | undefined => {\n  switch (authenticationType) {\n    case 'API_KEY':\n      return AuthorizationType.API_KEY;\n    case 'AWS_IAM':\n      return AuthorizationType.IAM;\n    case 'OPENID_CONNECT':\n      return AuthorizationType.OIDC;\n    case 'AMAZON_COGNITO_USER_POOLS':\n      return AuthorizationType.USER_POOL;\n    case 'AWS_LAMBDA':\n      return AuthorizationType.LAMBDA;\n    default:\n      console.warn(`Unrecognized Authentication type ${authenticationType}`);\n      return undefined;\n  }\n};\n\n/**\n * Get the function slots generated by the Graphql transform operation, adhering to the FunctionSlot interface.\n * @param generatedResolvers the resolvers generated by the transformer to spit back out.\n * @returns the list of generated function slots in the transformer, in order to facilitate overrides.\n */\nexport const getGeneratedFunctionSlots = (generatedResolvers: Record<string, string>): FunctionSlot[] =>\n  Object.entries(generatedResolvers)\n    .filter(([name]) => name.split('.').length === 6)\n    .map(([name, resolverCode]) => {\n      const [typeName, fieldName, slotName, slotIndex, templateType] = name.split('.');\n      return {\n        typeName,\n        fieldName,\n        slotName,\n        slotIndex: Number.parseInt(slotIndex, 10),\n        function: {\n          // TODO: this should consolidate req/req values back together\n          ...(templateType === 'req' ? { requestMappingTemplate: resolverCode } : {}),\n          ...(templateType === 'res' ? { responseMappingTemplate: resolverCode } : {}),\n        },\n      } as FunctionSlot;\n    });\n"]}