Comparing version 2.167.2 to 2.168.0
{ | ||
"comment": "Generated at 2024-11-19T09:45:32Z by generate.sh", | ||
"commit": "3669dce" | ||
"comment": "Generated at 2024-11-20T13:21:43Z by generate.sh", | ||
"commit": "f4a02ab" | ||
} |
import { FunctionConfiguration, type GetSchemaCreationStatusCommandInput, type GetSchemaCreationStatusCommandOutput, type ListFunctionsCommandInput, type StartSchemaCreationCommandInput, type StartSchemaCreationCommandOutput, type UpdateApiKeyCommandInput, type UpdateApiKeyCommandOutput, type UpdateFunctionCommandInput, type UpdateFunctionCommandOutput, type UpdateResolverCommandInput, type UpdateResolverCommandOutput } from '@aws-sdk/client-appsync'; | ||
import { ContinueUpdateRollbackCommandInput, ContinueUpdateRollbackCommandOutput, type CreateChangeSetCommandInput, type CreateChangeSetCommandOutput, type CreateGeneratedTemplateCommandInput, type CreateGeneratedTemplateCommandOutput, type CreateStackCommandInput, type CreateStackCommandOutput, type DeleteChangeSetCommandInput, type DeleteChangeSetCommandOutput, type DeleteGeneratedTemplateCommandInput, type DeleteGeneratedTemplateCommandOutput, type DeleteStackCommandInput, type DeleteStackCommandOutput, type DescribeChangeSetCommandInput, type DescribeChangeSetCommandOutput, type DescribeGeneratedTemplateCommandInput, type DescribeGeneratedTemplateCommandOutput, type DescribeResourceScanCommandInput, type DescribeResourceScanCommandOutput, type DescribeStackEventsCommandInput, DescribeStackResourcesCommandInput, DescribeStackResourcesCommandOutput, type DescribeStacksCommandInput, type DescribeStacksCommandOutput, type ExecuteChangeSetCommandInput, type ExecuteChangeSetCommandOutput, type GetGeneratedTemplateCommandInput, type GetGeneratedTemplateCommandOutput, type GetTemplateCommandInput, type GetTemplateCommandOutput, type GetTemplateSummaryCommandInput, type GetTemplateSummaryCommandOutput, type ListExportsCommandInput, type ListExportsCommandOutput, type ListResourceScanRelatedResourcesCommandInput, type ListResourceScanRelatedResourcesCommandOutput, type ListResourceScanResourcesCommandInput, type ListResourceScanResourcesCommandOutput, type ListResourceScansCommandInput, type ListResourceScansCommandOutput, type ListStackResourcesCommandInput, ListStacksCommandInput, ListStacksCommandOutput, RollbackStackCommandInput, RollbackStackCommandOutput, StackEvent, StackResourceSummary, type StartResourceScanCommandInput, type StartResourceScanCommandOutput, type UpdateStackCommandInput, type UpdateStackCommandOutput, type UpdateTerminationProtectionCommandInput, type UpdateTerminationProtectionCommandOutput } from '@aws-sdk/client-cloudformation'; | ||
import { ContinueUpdateRollbackCommandInput, ContinueUpdateRollbackCommandOutput, type CreateChangeSetCommandInput, type CreateChangeSetCommandOutput, type CreateGeneratedTemplateCommandInput, type CreateGeneratedTemplateCommandOutput, type CreateStackCommandInput, type CreateStackCommandOutput, type DeleteChangeSetCommandInput, type DeleteChangeSetCommandOutput, type DeleteGeneratedTemplateCommandInput, type DeleteGeneratedTemplateCommandOutput, type DeleteStackCommandInput, type DeleteStackCommandOutput, type DescribeChangeSetCommandInput, type DescribeChangeSetCommandOutput, type DescribeGeneratedTemplateCommandInput, type DescribeGeneratedTemplateCommandOutput, type DescribeResourceScanCommandInput, type DescribeResourceScanCommandOutput, type DescribeStackEventsCommandInput, DescribeStackEventsCommandOutput, DescribeStackResourcesCommandInput, DescribeStackResourcesCommandOutput, type DescribeStacksCommandInput, type DescribeStacksCommandOutput, type ExecuteChangeSetCommandInput, type ExecuteChangeSetCommandOutput, type GetGeneratedTemplateCommandInput, type GetGeneratedTemplateCommandOutput, type GetTemplateCommandInput, type GetTemplateCommandOutput, type GetTemplateSummaryCommandInput, type GetTemplateSummaryCommandOutput, type ListExportsCommandInput, type ListExportsCommandOutput, type ListResourceScanRelatedResourcesCommandInput, type ListResourceScanRelatedResourcesCommandOutput, type ListResourceScanResourcesCommandInput, type ListResourceScanResourcesCommandOutput, type ListResourceScansCommandInput, type ListResourceScansCommandOutput, type ListStackResourcesCommandInput, ListStacksCommandInput, ListStacksCommandOutput, RollbackStackCommandInput, RollbackStackCommandOutput, StackResourceSummary, type StartResourceScanCommandInput, type StartResourceScanCommandOutput, type UpdateStackCommandInput, type UpdateStackCommandOutput, type UpdateTerminationProtectionCommandInput, type UpdateTerminationProtectionCommandOutput } from '@aws-sdk/client-cloudformation'; | ||
import { type DescribeLogGroupsCommandInput, type DescribeLogGroupsCommandOutput, FilterLogEventsCommandInput, FilterLogEventsCommandOutput } from '@aws-sdk/client-cloudwatch-logs'; | ||
@@ -86,3 +86,3 @@ import { type UpdateProjectCommandInput, type UpdateProjectCommandOutput } from '@aws-sdk/client-codebuild'; | ||
updateTerminationProtection(input: UpdateTerminationProtectionCommandInput): Promise<UpdateTerminationProtectionCommandOutput>; | ||
describeStackEvents(input: DescribeStackEventsCommandInput): Promise<StackEvent[]>; | ||
describeStackEvents(input: DescribeStackEventsCommandInput): Promise<DescribeStackEventsCommandOutput>; | ||
listStackResources(input: ListStackResourcesCommandInput): Promise<StackResourceSummary[]>; | ||
@@ -89,0 +89,0 @@ } |
@@ -71,6 +71,6 @@ "use strict"; | ||
async tick() { | ||
// excluding from codecoverage because this | ||
// doesn't always run (depends on timing) | ||
/* istanbul ignore next */ | ||
if (!this.active) { | ||
// excluding from codecoverage because this | ||
// doesn't always run (depends on timing) | ||
/* istanbul ignore next */ | ||
return; | ||
@@ -167,2 +167,2 @@ } | ||
exports.CloudWatchLogEventMonitor = CloudWatchLogEventMonitor; | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"logs-monitor.js","sourceRoot":"","sources":["logs-monitor.ts"],"names":[],"mappings":"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","sourcesContent":["import * as util from 'util';\nimport * as cxapi from '@aws-cdk/cx-api';\nimport * as chalk from 'chalk';\nimport { print, error } from '../../logging';\nimport { flatten } from '../../util/arrays';\nimport type { SDK } from '../aws-auth';\n\n/**\n * After reading events from all CloudWatch log groups\n * how long should we wait to read more events.\n *\n * If there is some error with reading events (i.e. Throttle)\n * then this is also how long we wait until we try again\n */\nconst SLEEP = 2_000;\n\n/**\n * Represents a CloudWatch Log Event that will be\n * printed to the terminal\n */\ninterface CloudWatchLogEvent {\n  /**\n   * The log event message\n   */\n  readonly message: string;\n\n  /**\n   * The name of the log group\n   */\n  readonly logGroupName: string;\n\n  /**\n   * The time at which the event occurred\n   */\n  readonly timestamp: Date;\n}\n\n/**\n * Configuration tracking information on the log groups that are\n * being monitored\n */\ninterface LogGroupsAccessSettings {\n  /**\n   * The SDK for a given environment (account/region)\n   */\n  readonly sdk: SDK;\n\n  /**\n   * A map of log groups and associated startTime in a given account.\n   *\n   * The monitor will read events from the log group starting at the\n   * associated startTime\n   */\n  readonly logGroupsStartTimes: { [logGroupName: string]: number };\n}\n\nexport class CloudWatchLogEventMonitor {\n  /**\n   * Determines which events not to display\n   */\n  private startTime: number;\n\n  /**\n   * Map of environment (account:region) to LogGroupsAccessSettings\n   */\n  private readonly envsLogGroupsAccessSettings = new Map<string, LogGroupsAccessSettings>();\n\n  private active = false;\n\n  constructor(startTime?: Date) {\n    this.startTime = startTime?.getTime() ?? Date.now();\n  }\n\n  /**\n   * resume reading/printing events\n   */\n  public activate(): void {\n    this.active = true;\n    this.scheduleNextTick(0);\n  }\n\n  /**\n   * deactivates the monitor so no new events are read\n   * use case for this is when we are in the middle of performing a deployment\n   * and don't want to interweave all the logs together with the CFN\n   * deployment logs\n   *\n   * Also resets the start time to be when the new deployment was triggered\n   * and clears the list of tracked log groups\n   */\n  public deactivate(): void {\n    this.active = false;\n    this.startTime = Date.now();\n    this.envsLogGroupsAccessSettings.clear();\n  }\n\n  /**\n   * Adds CloudWatch log groups to read log events from.\n   * Since we could be watching multiple stacks that deploy to\n   * multiple environments (account+region), we need to store a list of log groups\n   * per env along with the SDK object that has access to read from\n   * that environment.\n   */\n  public addLogGroups(env: cxapi.Environment, sdk: SDK, logGroupNames: string[]): void {\n    const awsEnv = `${env.account}:${env.region}`;\n    const logGroupsStartTimes = logGroupNames.reduce(\n      (acc, groupName) => {\n        acc[groupName] = this.startTime;\n        return acc;\n      },\n      {} as { [logGroupName: string]: number },\n    );\n    this.envsLogGroupsAccessSettings.set(awsEnv, {\n      sdk,\n      logGroupsStartTimes: {\n        ...this.envsLogGroupsAccessSettings.get(awsEnv)?.logGroupsStartTimes,\n        ...logGroupsStartTimes,\n      },\n    });\n  }\n\n  private scheduleNextTick(sleep: number): void {\n    setTimeout(() => void this.tick(), sleep);\n  }\n\n  private async tick(): Promise<void> {\n    if (!this.active) {\n      // excluding from codecoverage because this\n      // doesn't always run (depends on timing)\n      /* istanbul ignore next */\n      return;\n    }\n    try {\n      const events = flatten(await this.readNewEvents());\n      events.forEach((event) => {\n        this.print(event);\n      });\n    } catch (e) {\n      error('Error occurred while monitoring logs: %s', e);\n    }\n\n    this.scheduleNextTick(SLEEP);\n  }\n\n  /**\n   * Reads all new log events from a set of CloudWatch Log Groups\n   * in parallel\n   */\n  private async readNewEvents(): Promise<Array<Array<CloudWatchLogEvent>>> {\n    const promises: Array<Promise<Array<CloudWatchLogEvent>>> = [];\n    for (const settings of this.envsLogGroupsAccessSettings.values()) {\n      for (const group of Object.keys(settings.logGroupsStartTimes)) {\n        promises.push(this.readEventsFromLogGroup(settings, group));\n      }\n    }\n    // Limited set of log groups\n    // eslint-disable-next-line @cdklabs/promiseall-no-unbounded-parallelism\n    return Promise.all(promises);\n  }\n\n  /**\n   * Print out a cloudwatch event\n   */\n  private print(event: CloudWatchLogEvent): void {\n    print(\n      util.format(\n        '[%s] %s %s',\n        chalk.blue(event.logGroupName),\n        chalk.yellow(event.timestamp.toLocaleTimeString()),\n        event.message.trim(),\n      ),\n    );\n  }\n\n  /**\n   * Reads all new log events from a CloudWatch Log Group\n   * starting at either the time the hotswap was triggered or\n   * when the last event was read on the previous tick\n   */\n  private async readEventsFromLogGroup(\n    logGroupsAccessSettings: LogGroupsAccessSettings,\n    logGroupName: string,\n  ): Promise<Array<CloudWatchLogEvent>> {\n    const events: CloudWatchLogEvent[] = [];\n\n    // log events from some service are ingested faster than others\n    // so we need to track the start/end time for each log group individually\n    // to make sure that we process all events from each log group\n    const startTime = logGroupsAccessSettings.logGroupsStartTimes[logGroupName] ?? this.startTime;\n    let endTime = startTime;\n    try {\n      const response = await logGroupsAccessSettings.sdk.cloudWatchLogs().filterLogEvents({\n        logGroupName: logGroupName,\n        limit: 100,\n        startTime: startTime,\n      });\n      const filteredEvents = response.events ?? [];\n\n      for (const event of filteredEvents) {\n        if (event.message) {\n          events.push({\n            message: event.message,\n            logGroupName,\n            timestamp: event.timestamp ? new Date(event.timestamp) : new Date(),\n          });\n\n          if (event.timestamp && endTime < event.timestamp) {\n            endTime = event.timestamp;\n          }\n        }\n      }\n      // As long as there are _any_ events in the log group `filterLogEvents` will return a nextToken.\n      // This is true even if these events are before `startTime`. So if we have 100 events and a nextToken\n      // then assume that we have hit the limit and let the user know some messages have been supressed.\n      // We are essentially showing them a sampling (10000 events printed out is not very useful)\n      if (filteredEvents.length === 100 && response.nextToken) {\n        events.push({\n          message: '>>> `watch` shows only the first 100 log messages - the rest have been truncated...',\n          logGroupName,\n          timestamp: new Date(endTime),\n        });\n      }\n    } catch (e: any) {\n      // with Lambda functions the CloudWatch is not created\n      // until something is logged, so just keep polling until\n      // there is somthing to find\n      if (e.name === 'ResourceNotFoundException') {\n        return [];\n      }\n      throw e;\n    }\n    logGroupsAccessSettings.logGroupsStartTimes[logGroupName] = endTime + 1;\n    return events;\n  }\n}\n"]} | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"logs-monitor.js","sourceRoot":"","sources":["logs-monitor.ts"],"names":[],"mappings":"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","sourcesContent":["import * as util from 'util';\nimport * as cxapi from '@aws-cdk/cx-api';\nimport * as chalk from 'chalk';\nimport { print, error } from '../../logging';\nimport { flatten } from '../../util/arrays';\nimport type { SDK } from '../aws-auth';\n\n/**\n * After reading events from all CloudWatch log groups\n * how long should we wait to read more events.\n *\n * If there is some error with reading events (i.e. Throttle)\n * then this is also how long we wait until we try again\n */\nconst SLEEP = 2_000;\n\n/**\n * Represents a CloudWatch Log Event that will be\n * printed to the terminal\n */\ninterface CloudWatchLogEvent {\n  /**\n   * The log event message\n   */\n  readonly message: string;\n\n  /**\n   * The name of the log group\n   */\n  readonly logGroupName: string;\n\n  /**\n   * The time at which the event occurred\n   */\n  readonly timestamp: Date;\n}\n\n/**\n * Configuration tracking information on the log groups that are\n * being monitored\n */\ninterface LogGroupsAccessSettings {\n  /**\n   * The SDK for a given environment (account/region)\n   */\n  readonly sdk: SDK;\n\n  /**\n   * A map of log groups and associated startTime in a given account.\n   *\n   * The monitor will read events from the log group starting at the\n   * associated startTime\n   */\n  readonly logGroupsStartTimes: { [logGroupName: string]: number };\n}\n\nexport class CloudWatchLogEventMonitor {\n  /**\n   * Determines which events not to display\n   */\n  private startTime: number;\n\n  /**\n   * Map of environment (account:region) to LogGroupsAccessSettings\n   */\n  private readonly envsLogGroupsAccessSettings = new Map<string, LogGroupsAccessSettings>();\n\n  private active = false;\n\n  constructor(startTime?: Date) {\n    this.startTime = startTime?.getTime() ?? Date.now();\n  }\n\n  /**\n   * resume reading/printing events\n   */\n  public activate(): void {\n    this.active = true;\n    this.scheduleNextTick(0);\n  }\n\n  /**\n   * deactivates the monitor so no new events are read\n   * use case for this is when we are in the middle of performing a deployment\n   * and don't want to interweave all the logs together with the CFN\n   * deployment logs\n   *\n   * Also resets the start time to be when the new deployment was triggered\n   * and clears the list of tracked log groups\n   */\n  public deactivate(): void {\n    this.active = false;\n    this.startTime = Date.now();\n    this.envsLogGroupsAccessSettings.clear();\n  }\n\n  /**\n   * Adds CloudWatch log groups to read log events from.\n   * Since we could be watching multiple stacks that deploy to\n   * multiple environments (account+region), we need to store a list of log groups\n   * per env along with the SDK object that has access to read from\n   * that environment.\n   */\n  public addLogGroups(env: cxapi.Environment, sdk: SDK, logGroupNames: string[]): void {\n    const awsEnv = `${env.account}:${env.region}`;\n    const logGroupsStartTimes = logGroupNames.reduce(\n      (acc, groupName) => {\n        acc[groupName] = this.startTime;\n        return acc;\n      },\n      {} as { [logGroupName: string]: number },\n    );\n    this.envsLogGroupsAccessSettings.set(awsEnv, {\n      sdk,\n      logGroupsStartTimes: {\n        ...this.envsLogGroupsAccessSettings.get(awsEnv)?.logGroupsStartTimes,\n        ...logGroupsStartTimes,\n      },\n    });\n  }\n\n  private scheduleNextTick(sleep: number): void {\n    setTimeout(() => void this.tick(), sleep);\n  }\n\n  private async tick(): Promise<void> {\n    // excluding from codecoverage because this\n    // doesn't always run (depends on timing)\n    /* istanbul ignore next */\n    if (!this.active) {\n      return;\n    }\n    try {\n      const events = flatten(await this.readNewEvents());\n      events.forEach((event) => {\n        this.print(event);\n      });\n    } catch (e) {\n      error('Error occurred while monitoring logs: %s', e);\n    }\n\n    this.scheduleNextTick(SLEEP);\n  }\n\n  /**\n   * Reads all new log events from a set of CloudWatch Log Groups\n   * in parallel\n   */\n  private async readNewEvents(): Promise<Array<Array<CloudWatchLogEvent>>> {\n    const promises: Array<Promise<Array<CloudWatchLogEvent>>> = [];\n    for (const settings of this.envsLogGroupsAccessSettings.values()) {\n      for (const group of Object.keys(settings.logGroupsStartTimes)) {\n        promises.push(this.readEventsFromLogGroup(settings, group));\n      }\n    }\n    // Limited set of log groups\n    // eslint-disable-next-line @cdklabs/promiseall-no-unbounded-parallelism\n    return Promise.all(promises);\n  }\n\n  /**\n   * Print out a cloudwatch event\n   */\n  private print(event: CloudWatchLogEvent): void {\n    print(\n      util.format(\n        '[%s] %s %s',\n        chalk.blue(event.logGroupName),\n        chalk.yellow(event.timestamp.toLocaleTimeString()),\n        event.message.trim(),\n      ),\n    );\n  }\n\n  /**\n   * Reads all new log events from a CloudWatch Log Group\n   * starting at either the time the hotswap was triggered or\n   * when the last event was read on the previous tick\n   */\n  private async readEventsFromLogGroup(\n    logGroupsAccessSettings: LogGroupsAccessSettings,\n    logGroupName: string,\n  ): Promise<Array<CloudWatchLogEvent>> {\n    const events: CloudWatchLogEvent[] = [];\n\n    // log events from some service are ingested faster than others\n    // so we need to track the start/end time for each log group individually\n    // to make sure that we process all events from each log group\n    const startTime = logGroupsAccessSettings.logGroupsStartTimes[logGroupName] ?? this.startTime;\n    let endTime = startTime;\n    try {\n      const response = await logGroupsAccessSettings.sdk.cloudWatchLogs().filterLogEvents({\n        logGroupName: logGroupName,\n        limit: 100,\n        startTime: startTime,\n      });\n      const filteredEvents = response.events ?? [];\n\n      for (const event of filteredEvents) {\n        if (event.message) {\n          events.push({\n            message: event.message,\n            logGroupName,\n            timestamp: event.timestamp ? new Date(event.timestamp) : new Date(),\n          });\n\n          if (event.timestamp && endTime < event.timestamp) {\n            endTime = event.timestamp;\n          }\n        }\n      }\n      // As long as there are _any_ events in the log group `filterLogEvents` will return a nextToken.\n      // This is true even if these events are before `startTime`. So if we have 100 events and a nextToken\n      // then assume that we have hit the limit and let the user know some messages have been supressed.\n      // We are essentially showing them a sampling (10000 events printed out is not very useful)\n      if (filteredEvents.length === 100 && response.nextToken) {\n        events.push({\n          message: '>>> `watch` shows only the first 100 log messages - the rest have been truncated...',\n          logGroupName,\n          timestamp: new Date(endTime),\n        });\n      }\n    } catch (e: any) {\n      // with Lambda functions the CloudWatch is not created\n      // until something is logged, so just keep polling until\n      // there is somthing to find\n      if (e.name === 'ResourceNotFoundException') {\n        return [];\n      }\n      throw e;\n    }\n    logGroupsAccessSettings.logGroupsStartTimes[logGroupName] = endTime + 1;\n    return events;\n  }\n}\n"]} |
@@ -43,36 +43,42 @@ "use strict"; | ||
try { | ||
const eventList = await this.cfn.describeStackEvents({ | ||
StackName: this.props.stackName, | ||
}); | ||
for (const event of eventList) { | ||
// Event from before we were interested in 'em | ||
if (this.props.startTime !== undefined && event.Timestamp.valueOf() < this.props.startTime) { | ||
return events; | ||
let nextToken; | ||
let finished = false; | ||
while (!finished) { | ||
const page = await this.cfn.describeStackEvents({ StackName: this.props.stackName, NextToken: nextToken }); | ||
for (const event of page?.StackEvents ?? []) { | ||
// Event from before we were interested in 'em | ||
if (this.props.startTime !== undefined && event.Timestamp.valueOf() < this.props.startTime) { | ||
return events; | ||
} | ||
// Already seen this one | ||
if (this.eventIds.has(event.EventId)) { | ||
return events; | ||
} | ||
this.eventIds.add(event.EventId); | ||
// The events for the stack itself are also included next to events about resources; we can test for them in this way. | ||
const isParentStackEvent = event.PhysicalResourceId === event.StackId; | ||
if (isParentStackEvent && this.props.stackStatuses?.includes(event.ResourceStatus ?? '')) { | ||
return events; | ||
} | ||
// Fresh event | ||
const resEvent = { | ||
event: event, | ||
parentStackLogicalIds: this.props.parentStackLogicalIds ?? [], | ||
isStackEvent: isParentStackEvent, | ||
}; | ||
events.push(resEvent); | ||
if (!isParentStackEvent && | ||
event.ResourceType === 'AWS::CloudFormation::Stack' && | ||
isStackBeginOperationState(event.ResourceStatus)) { | ||
// If the event is not for `this` stack and has a physical resource Id, recursively call for events in the nested stack | ||
this.trackNestedStack(event, [...(this.props.parentStackLogicalIds ?? []), event.LogicalResourceId ?? '']); | ||
} | ||
if (isParentStackEvent && isStackTerminalState(event.ResourceStatus)) { | ||
this.complete = true; | ||
} | ||
} | ||
// Already seen this one | ||
if (this.eventIds.has(event.EventId)) { | ||
return events; | ||
nextToken = page?.NextToken; | ||
if (nextToken === undefined) { | ||
finished = true; | ||
} | ||
this.eventIds.add(event.EventId); | ||
// The events for the stack itself are also included next to events about resources; we can test for them in this way. | ||
const isParentStackEvent = event.PhysicalResourceId === event.StackId; | ||
if (isParentStackEvent && this.props.stackStatuses?.includes(event.ResourceStatus ?? '')) { | ||
return events; | ||
} | ||
// Fresh event | ||
const resEvent = { | ||
event: event, | ||
parentStackLogicalIds: this.props.parentStackLogicalIds ?? [], | ||
isStackEvent: isParentStackEvent, | ||
}; | ||
events.push(resEvent); | ||
if (!isParentStackEvent && | ||
event.ResourceType === 'AWS::CloudFormation::Stack' && | ||
isStackBeginOperationState(event.ResourceStatus)) { | ||
// If the event is not for `this` stack and has a physical resource Id, recursively call for events in the nested stack | ||
this.trackNestedStack(event, [...(this.props.parentStackLogicalIds ?? []), event.LogicalResourceId ?? '']); | ||
} | ||
if (isParentStackEvent && isStackTerminalState(event.ResourceStatus)) { | ||
this.complete = true; | ||
} | ||
} | ||
@@ -85,12 +91,2 @@ } | ||
} | ||
// // Also poll all nested stacks we're currently tracking | ||
// for (const [logicalId, poller] of Object.entries(this.nestedStackPollers)) { | ||
// events.push(...(await poller.poll())); | ||
// if (poller.complete) { | ||
// delete this.nestedStackPollers[logicalId]; | ||
// } | ||
// } | ||
// // Return what we have so far | ||
// events.sort((a, b) => a.event.Timestamp!.valueOf() - b.event.Timestamp!.valueOf()); | ||
// this.events.push(...events); | ||
return events; | ||
@@ -133,2 +129,2 @@ } | ||
} | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"stack-event-poller.js","sourceRoot":"","sources":["stack-event-poller.ts"],"names":[],"mappings":"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","sourcesContent":["import type { StackEvent } from '@aws-sdk/client-cloudformation';\nimport type { ICloudFormationClient } from '../../aws-auth';\n\nexport interface StackEventPollerProps {\n  /**\n   * The stack to poll\n   */\n  readonly stackName: string;\n\n  /**\n   * IDs of parent stacks of this resource, in case of resources in nested stacks\n   */\n  readonly parentStackLogicalIds?: string[];\n\n  /**\n   * Timestamp for the oldest event we're interested in\n   *\n   * @default - Read all events\n   */\n  readonly startTime?: number;\n\n  /**\n   * Stop reading when we see the stack entering this status\n   *\n   * Should be something like `CREATE_IN_PROGRESS`, `UPDATE_IN_PROGRESS`,\n   * `DELETE_IN_PROGRESS, `ROLLBACK_IN_PROGRESS`.\n   *\n   * @default - Read all events\n   */\n  readonly stackStatuses?: string[];\n}\n\nexport interface ResourceEvent {\n  readonly event: StackEvent;\n  readonly parentStackLogicalIds: string[];\n\n  /**\n   * Whether this event regards the root stack\n   *\n   * @default false\n   */\n  readonly isStackEvent?: boolean;\n}\n\nexport class StackEventPoller {\n  public readonly events: ResourceEvent[] = [];\n  public complete: boolean = false;\n\n  private readonly eventIds = new Set<string>();\n  private readonly nestedStackPollers: Record<string, StackEventPoller> = {};\n\n  constructor(\n    private readonly cfn: ICloudFormationClient,\n    private readonly props: StackEventPollerProps,\n  ) {}\n\n  /**\n   * From all accumulated events, return only the errors\n   */\n  public get resourceErrors(): ResourceEvent[] {\n    return this.events.filter((e) => e.event.ResourceStatus?.endsWith('_FAILED') && !e.isStackEvent);\n  }\n\n  /**\n   * Poll for new stack events\n   *\n   * Will not return events older than events indicated by the constructor filters.\n   *\n   * Recurses into nested stacks, and returns events old-to-new.\n   */\n  public async poll(): Promise<ResourceEvent[]> {\n    const events: ResourceEvent[] = await this.doPoll();\n\n    // Also poll all nested stacks we're currently tracking\n    for (const [logicalId, poller] of Object.entries(this.nestedStackPollers)) {\n      events.push(...(await poller.poll()));\n      if (poller.complete) {\n        delete this.nestedStackPollers[logicalId];\n      }\n    }\n\n    // Return what we have so far\n    events.sort((a, b) => a.event.Timestamp!.valueOf() - b.event.Timestamp!.valueOf());\n    this.events.push(...events);\n    return events;\n  }\n\n  private async doPoll(): Promise<ResourceEvent[]> {\n    const events: ResourceEvent[] = [];\n    try {\n      const eventList = await this.cfn.describeStackEvents({\n        StackName: this.props.stackName,\n      });\n      for (const event of eventList) {\n        // Event from before we were interested in 'em\n        if (this.props.startTime !== undefined && event.Timestamp!.valueOf() < this.props.startTime) {\n          return events;\n        }\n\n        // Already seen this one\n        if (this.eventIds.has(event.EventId!)) {\n          return events;\n        }\n        this.eventIds.add(event.EventId!);\n\n        // The events for the stack itself are also included next to events about resources; we can test for them in this way.\n        const isParentStackEvent = event.PhysicalResourceId === event.StackId;\n\n        if (isParentStackEvent && this.props.stackStatuses?.includes(event.ResourceStatus ?? '')) {\n          return events;\n        }\n\n        // Fresh event\n        const resEvent: ResourceEvent = {\n          event: event,\n          parentStackLogicalIds: this.props.parentStackLogicalIds ?? [],\n          isStackEvent: isParentStackEvent,\n        };\n        events.push(resEvent);\n\n        if (\n          !isParentStackEvent &&\n          event.ResourceType === 'AWS::CloudFormation::Stack' &&\n          isStackBeginOperationState(event.ResourceStatus)\n        ) {\n          // If the event is not for `this` stack and has a physical resource Id, recursively call for events in the nested stack\n          this.trackNestedStack(event, [...(this.props.parentStackLogicalIds ?? []), event.LogicalResourceId ?? '']);\n        }\n\n        if (isParentStackEvent && isStackTerminalState(event.ResourceStatus)) {\n          this.complete = true;\n        }\n      }\n    } catch (e: any) {\n      if (!(e.name === 'ValidationError' && e.message === `Stack [${this.props.stackName}] does not exist`)) {\n        throw e;\n      }\n    }\n    // // Also poll all nested stacks we're currently tracking\n    // for (const [logicalId, poller] of Object.entries(this.nestedStackPollers)) {\n    //   events.push(...(await poller.poll()));\n    //   if (poller.complete) {\n    //     delete this.nestedStackPollers[logicalId];\n    //   }\n    // }\n\n    // // Return what we have so far\n    // events.sort((a, b) => a.event.Timestamp!.valueOf() - b.event.Timestamp!.valueOf());\n    // this.events.push(...events);\n    return events;\n  }\n\n  /**\n   * On the CREATE_IN_PROGRESS, UPDATE_IN_PROGRESS, DELETE_IN_PROGRESS event of a nested stack, poll the nested stack updates\n   */\n  private trackNestedStack(event: StackEvent, parentStackLogicalIds: string[]) {\n    const logicalId = event.LogicalResourceId;\n    const physicalResourceId = event.PhysicalResourceId;\n\n    // The CREATE_IN_PROGRESS event for a Nested Stack is emitted twice; first without a PhysicalResourceId\n    // and then with. Ignore this event if we don't have that property yet.\n    //\n    // (At this point, I also don't trust that logicalId is always going to be there so validate that as well)\n    if (!logicalId || !physicalResourceId) {\n      return;\n    }\n\n    if (!this.nestedStackPollers[logicalId]) {\n      this.nestedStackPollers[logicalId] = new StackEventPoller(this.cfn, {\n        stackName: physicalResourceId,\n        parentStackLogicalIds: parentStackLogicalIds,\n        startTime: event.Timestamp!.valueOf(),\n      });\n    }\n  }\n}\n\nfunction isStackBeginOperationState(state: string | undefined) {\n  return [\n    'CREATE_IN_PROGRESS',\n    'UPDATE_IN_PROGRESS',\n    'DELETE_IN_PROGRESS',\n    'UPDATE_ROLLBACK_IN_PROGRESS',\n    'ROLLBACK_IN_PROGRESS',\n  ].includes(state ?? '');\n}\n\nfunction isStackTerminalState(state: string | undefined) {\n  return !(state ?? '').endsWith('_IN_PROGRESS');\n}\n"]} | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"stack-event-poller.js","sourceRoot":"","sources":["stack-event-poller.ts"],"names":[],"mappings":"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","sourcesContent":["import type { StackEvent } from '@aws-sdk/client-cloudformation';\nimport type { ICloudFormationClient } from '../../aws-auth';\n\nexport interface StackEventPollerProps {\n  /**\n   * The stack to poll\n   */\n  readonly stackName: string;\n\n  /**\n   * IDs of parent stacks of this resource, in case of resources in nested stacks\n   */\n  readonly parentStackLogicalIds?: string[];\n\n  /**\n   * Timestamp for the oldest event we're interested in\n   *\n   * @default - Read all events\n   */\n  readonly startTime?: number;\n\n  /**\n   * Stop reading when we see the stack entering this status\n   *\n   * Should be something like `CREATE_IN_PROGRESS`, `UPDATE_IN_PROGRESS`,\n   * `DELETE_IN_PROGRESS, `ROLLBACK_IN_PROGRESS`.\n   *\n   * @default - Read all events\n   */\n  readonly stackStatuses?: string[];\n}\n\nexport interface ResourceEvent {\n  readonly event: StackEvent;\n  readonly parentStackLogicalIds: string[];\n\n  /**\n   * Whether this event regards the root stack\n   *\n   * @default false\n   */\n  readonly isStackEvent?: boolean;\n}\n\nexport class StackEventPoller {\n  public readonly events: ResourceEvent[] = [];\n  public complete: boolean = false;\n\n  private readonly eventIds = new Set<string>();\n  private readonly nestedStackPollers: Record<string, StackEventPoller> = {};\n\n  constructor(\n    private readonly cfn: ICloudFormationClient,\n    private readonly props: StackEventPollerProps,\n  ) {}\n\n  /**\n   * From all accumulated events, return only the errors\n   */\n  public get resourceErrors(): ResourceEvent[] {\n    return this.events.filter((e) => e.event.ResourceStatus?.endsWith('_FAILED') && !e.isStackEvent);\n  }\n\n  /**\n   * Poll for new stack events\n   *\n   * Will not return events older than events indicated by the constructor filters.\n   *\n   * Recurses into nested stacks, and returns events old-to-new.\n   */\n  public async poll(): Promise<ResourceEvent[]> {\n    const events: ResourceEvent[] = await this.doPoll();\n\n    // Also poll all nested stacks we're currently tracking\n    for (const [logicalId, poller] of Object.entries(this.nestedStackPollers)) {\n      events.push(...(await poller.poll()));\n      if (poller.complete) {\n        delete this.nestedStackPollers[logicalId];\n      }\n    }\n\n    // Return what we have so far\n    events.sort((a, b) => a.event.Timestamp!.valueOf() - b.event.Timestamp!.valueOf());\n    this.events.push(...events);\n    return events;\n  }\n\n  private async doPoll(): Promise<ResourceEvent[]> {\n    const events: ResourceEvent[] = [];\n    try {\n      let nextToken: string | undefined;\n      let finished = false;\n\n      while (!finished) {\n        const page = await this.cfn.describeStackEvents({ StackName: this.props.stackName, NextToken: nextToken });\n        for (const event of page?.StackEvents ?? []) {\n          // Event from before we were interested in 'em\n          if (this.props.startTime !== undefined && event.Timestamp!.valueOf() < this.props.startTime) {\n            return events;\n          }\n\n          // Already seen this one\n          if (this.eventIds.has(event.EventId!)) {\n            return events;\n          }\n          this.eventIds.add(event.EventId!);\n\n          // The events for the stack itself are also included next to events about resources; we can test for them in this way.\n          const isParentStackEvent = event.PhysicalResourceId === event.StackId;\n\n          if (isParentStackEvent && this.props.stackStatuses?.includes(event.ResourceStatus ?? '')) {\n            return events;\n          }\n\n          // Fresh event\n          const resEvent: ResourceEvent = {\n            event: event,\n            parentStackLogicalIds: this.props.parentStackLogicalIds ?? [],\n            isStackEvent: isParentStackEvent,\n          };\n          events.push(resEvent);\n\n          if (\n            !isParentStackEvent &&\n              event.ResourceType === 'AWS::CloudFormation::Stack' &&\n              isStackBeginOperationState(event.ResourceStatus)\n          ) {\n            // If the event is not for `this` stack and has a physical resource Id, recursively call for events in the nested stack\n            this.trackNestedStack(event, [...(this.props.parentStackLogicalIds ?? []), event.LogicalResourceId ?? '']);\n          }\n\n          if (isParentStackEvent && isStackTerminalState(event.ResourceStatus)) {\n            this.complete = true;\n          }\n        }\n\n        nextToken = page?.NextToken;\n        if (nextToken === undefined) {\n          finished = true;\n        }\n\n      }\n    } catch (e: any) {\n      if (!(e.name === 'ValidationError' && e.message === `Stack [${this.props.stackName}] does not exist`)) {\n        throw e;\n      }\n    }\n\n    return events;\n  }\n\n  /**\n   * On the CREATE_IN_PROGRESS, UPDATE_IN_PROGRESS, DELETE_IN_PROGRESS event of a nested stack, poll the nested stack updates\n   */\n  private trackNestedStack(event: StackEvent, parentStackLogicalIds: string[]) {\n    const logicalId = event.LogicalResourceId;\n    const physicalResourceId = event.PhysicalResourceId;\n\n    // The CREATE_IN_PROGRESS event for a Nested Stack is emitted twice; first without a PhysicalResourceId\n    // and then with. Ignore this event if we don't have that property yet.\n    //\n    // (At this point, I also don't trust that logicalId is always going to be there so validate that as well)\n    if (!logicalId || !physicalResourceId) {\n      return;\n    }\n\n    if (!this.nestedStackPollers[logicalId]) {\n      this.nestedStackPollers[logicalId] = new StackEventPoller(this.cfn, {\n        stackName: physicalResourceId,\n        parentStackLogicalIds: parentStackLogicalIds,\n        startTime: event.Timestamp!.valueOf(),\n      });\n    }\n  }\n}\n\nfunction isStackBeginOperationState(state: string | undefined) {\n  return [\n    'CREATE_IN_PROGRESS',\n    'UPDATE_IN_PROGRESS',\n    'DELETE_IN_PROGRESS',\n    'UPDATE_ROLLBACK_IN_PROGRESS',\n    'ROLLBACK_IN_PROGRESS',\n  ].includes(state ?? '');\n}\n\nfunction isStackTerminalState(state: string | undefined) {\n  return !(state ?? '').endsWith('_IN_PROGRESS');\n}\n"]} |
#!/usr/bin/env node | ||
import 'source-map-support/register'; | ||
import * as cdk from 'aws-cdk-lib'; | ||
@@ -4,0 +3,0 @@ import { %name.PascalCased%Stack } from '../lib/%name%-stack'; |
@@ -24,5 +24,4 @@ { | ||
"aws-cdk-lib": "%cdk-version%", | ||
"constructs": "%constructs-version%", | ||
"source-map-support": "^0.5.21" | ||
"constructs": "%constructs-version%" | ||
} | ||
} | ||
} |
{ | ||
"name": "aws-cdk", | ||
"description": "CDK Toolkit, the command line tool for CDK apps", | ||
"version": "2.167.2", | ||
"version": "2.168.0", | ||
"bin": { | ||
@@ -71,5 +71,5 @@ "cdk": "bin/cdk" | ||
"devDependencies": { | ||
"@aws-cdk/cdk-build-tools": "2.167.2-alpha.0", | ||
"@aws-cdk/pkglint": "2.167.2-alpha.0", | ||
"@aws-cdk/yargs-gen": "2.167.2-alpha.0", | ||
"@aws-cdk/cdk-build-tools": "2.168.0-alpha.0", | ||
"@aws-cdk/pkglint": "2.168.0-alpha.0", | ||
"@aws-cdk/yargs-gen": "2.168.0-alpha.0", | ||
"@octokit/rest": "^18.12.0", | ||
@@ -80,2 +80,3 @@ "@types/archiver": "^5.3.4", | ||
"@types/jest": "^29.5.12", | ||
"@types/node": "^18.18.14", | ||
"@types/mockery": "^1.4.33", | ||
@@ -85,3 +86,2 @@ "@types/promptly": "^3.0.5", | ||
"@types/sinon": "^9.0.11", | ||
"@types/source-map-support": "^0.5.10", | ||
"@types/table": "^6.3.2", | ||
@@ -91,3 +91,3 @@ "@types/uuid": "^8.3.4", | ||
"@types/yargs": "^15.0.19", | ||
"aws-cdk-lib": "2.167.2", | ||
"aws-cdk-lib": "2.168.0", | ||
"aws-sdk-client-mock": "^4.0.1", | ||
@@ -108,5 +108,5 @@ "aws-sdk-client-mock-jest": "^4.0.1", | ||
"@aws-cdk/cloud-assembly-schema": "^38.0.0", | ||
"@aws-cdk/cloudformation-diff": "2.167.2", | ||
"@aws-cdk/cx-api": "2.167.2", | ||
"@aws-cdk/region-info": "2.167.2", | ||
"@aws-cdk/cloudformation-diff": "2.168.0", | ||
"@aws-cdk/cx-api": "2.168.0", | ||
"@aws-cdk/region-info": "2.168.0", | ||
"@aws-sdk/client-appsync": "3.632.0", | ||
@@ -155,3 +155,2 @@ "@aws-sdk/client-cloudformation": "3.632.0", | ||
"semver": "^7.6.3", | ||
"source-map-support": "^0.5.21", | ||
"strip-ansi": "^6.0.1", | ||
@@ -158,0 +157,0 @@ "table": "^6.8.2", |
@@ -7,2 +7,3 @@ "use strict"; | ||
const util_1 = require("../util"); | ||
const assembly_versions_1 = require("./assembly-versions"); | ||
// behave like v2 | ||
@@ -217,4 +218,5 @@ process.env.CXAPI_DISABLE_SELECT_BY_ID = '1'; | ||
}); | ||
return cloudExec.synthesize(); | ||
const asm = await cloudExec.synthesize(); | ||
return (0, assembly_versions_1.cliAssemblyWithForcedVersion)(asm, '30.0.0'); | ||
} | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cloud-assembly.test.js","sourceRoot":"","sources":["cloud-assembly.test.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable import/order */\nimport * as cxschema from '@aws-cdk/cloud-assembly-schema';\nimport { DefaultSelection } from '../../lib/api/cxapp/cloud-assembly';\nimport { MockCloudExecutable } from '../util';\n\n// behave like v2\nprocess.env.CXAPI_DISABLE_SELECT_BY_ID = '1';\n\ntest('do not throw when selecting stack without errors', async () => {\n  // GIVEN\n  const cxasm = await testCloudAssembly();\n\n  // WHEN\n  const selected = await cxasm.selectStacks( { patterns: ['withouterrorsNODEPATH'] }, {\n    defaultBehavior: DefaultSelection.AllStacks,\n  });\n  selected.processMetadataMessages();\n\n  // THEN\n  expect(selected.firstStack.template.resource).toBe('noerrorresource');\n});\n\ntest('do throw when selecting stack with errors', async () => {\n  // GIVEN\n  const cxasm = await testCloudAssembly();\n\n  // WHEN\n  const selected = await cxasm.selectStacks({ patterns: ['witherrors'] }, {\n    defaultBehavior: DefaultSelection.AllStacks,\n  });\n\n  // THEN\n  expect(() => selected.processMetadataMessages()).toThrow(/Found errors/);\n});\n\ntest('select all top level stacks in the presence of nested assemblies', async () => {\n  // GIVEN\n  const cxasm = await testNestedCloudAssembly();\n\n  // WHEN\n  const x = await cxasm.selectStacks({ allTopLevel: true, patterns: [] }, { defaultBehavior: DefaultSelection.AllStacks });\n\n  // THEN\n  expect(x.stackCount).toBe(2);\n  expect(x.stackIds).toContain('witherrors');\n  expect(x.stackIds).toContain('withouterrors');\n});\n\ntest('select stacks by glob pattern', async () => {\n  // GIVEN\n  const cxasm = await testCloudAssembly();\n\n  // WHEN\n  const x = await cxasm.selectStacks({ patterns: ['with*'] }, { defaultBehavior: DefaultSelection.AllStacks });\n\n  // THEN\n  expect(x.stackCount).toBe(2);\n  expect(x.stackIds).toContain('witherrors');\n  expect(x.stackIds).toContain('withouterrors');\n});\n\ntest('select behavior: all', async () => {\n  // GIVEN\n  const cxasm = await testCloudAssembly();\n\n  // WHEN\n  const x = await cxasm.selectStacks({ patterns: [] }, { defaultBehavior: DefaultSelection.AllStacks });\n\n  // THEN\n  expect(x.stackCount).toBe(2);\n});\n\ntest('select behavior: none', async () => {\n  // GIVEN\n  const cxasm = await testCloudAssembly();\n\n  // WHEN\n  const x = await cxasm.selectStacks({ patterns: [] }, { defaultBehavior: DefaultSelection.None });\n\n  // THEN\n  expect(x.stackCount).toBe(0);\n});\n\ntest('select behavior: single', async () => {\n  // GIVEN\n  const cxasm = await testCloudAssembly();\n\n  // WHEN\n  await expect(cxasm.selectStacks({ patterns: [] }, { defaultBehavior: DefaultSelection.OnlySingle }))\n    .rejects.toThrow('Since this app includes more than a single stack, specify which stacks to use (wildcards are supported) or specify `--all`');\n});\n\ntest('stack list error contains node paths', async () => {\n  // GIVEN\n  const cxasm = await testCloudAssembly();\n\n  // WHEN\n  await expect(cxasm.selectStacks({ patterns: [] }, { defaultBehavior: DefaultSelection.OnlySingle }))\n    .rejects.toThrow('withouterrorsNODEPATH');\n});\n\ntest('select behavior: repeat', async () => {\n  // GIVEN\n  const cxasm = await testCloudAssembly();\n\n  // WHEN\n  const x = await cxasm.selectStacks({ patterns: ['withouterrorsNODEPATH', 'withouterrorsNODEPATH'] }, {\n    defaultBehavior: DefaultSelection.AllStacks,\n  });\n\n  // THEN\n  expect(x.stackCount).toBe(1);\n});\n\ntest('select behavior with nested assemblies: all', async () => {\n  // GIVEN\n  const cxasm = await testNestedCloudAssembly();\n\n  // WHEN\n  const x = await cxasm.selectStacks({ patterns: [] }, { defaultBehavior: DefaultSelection.AllStacks });\n\n  // THEN\n  expect(x.stackCount).toBe(3);\n});\n\ntest('select behavior with nested assemblies: none', async () => {\n  // GIVEN\n  const cxasm = await testNestedCloudAssembly();\n\n  // WHEN\n  const x = await cxasm.selectStacks({ patterns: [] }, { defaultBehavior: DefaultSelection.None });\n\n  // THEN\n  expect(x.stackCount).toBe(0);\n});\n\ntest('select behavior with nested assemblies: single', async () => {\n  // GIVEN\n  const cxasm = await testNestedCloudAssembly();\n\n  // WHEN\n  await expect(cxasm.selectStacks({ patterns: [] }, { defaultBehavior: DefaultSelection.OnlySingle }))\n    .rejects.toThrow('Since this app includes more than a single stack, specify which stacks to use (wildcards are supported) or specify `--all`');\n});\n\ntest('select behavior with nested assemblies: repeat', async() => {\n  // GIVEN\n  const cxasm = await testNestedCloudAssembly();\n\n  // WHEN\n  const x = await cxasm.selectStacks({ patterns: ['deeply/hidden/withouterrors', 'nested'] }, {\n    defaultBehavior: DefaultSelection.AllStacks,\n  });\n\n  // THEN\n  expect(x.stackCount).toBe(2);\n});\n\ntest('select behavior with no stacks and ignore stacks option', async() => {\n  // GIVEN\n  const cxasm = await testCloudAssemblyNoStacks();\n\n  // WHEN\n  const x = await cxasm.selectStacks({ patterns: [] }, {\n    defaultBehavior: DefaultSelection.AllStacks,\n    ignoreNoStacks: true,\n  });\n\n  // THEN\n  expect(x.stackCount).toBe(0);\n});\n\ntest('select behavior with no stacks and no ignore stacks option', async() => {\n  // GIVEN\n  const cxasm = await testCloudAssemblyNoStacks();\n\n  // WHEN & THEN\n  await expect(cxasm.selectStacks({ patterns: [] }, { defaultBehavior: DefaultSelection.AllStacks, ignoreNoStacks: false }))\n    .rejects.toThrow('This app contains no stacks');\n});\n\ntest('select behavior with no stacks and default ignore stacks options (false)', async() => {\n  // GIVEN\n  const cxasm = await testCloudAssemblyNoStacks();\n\n  // WHEN & THEN\n  await expect(cxasm.selectStacks({ patterns: [] }, { defaultBehavior: DefaultSelection.AllStacks }))\n    .rejects.toThrow('This app contains no stacks');\n});\n\nasync function testCloudAssembly({ env }: { env?: string; versionReporting?: boolean } = {}) {\n  const cloudExec = new MockCloudExecutable({\n    stacks: [{\n      stackName: 'withouterrors',\n      displayName: 'withouterrorsNODEPATH',\n      env,\n      template: { resource: 'noerrorresource' },\n    },\n    {\n      stackName: 'witherrors',\n      env,\n      template: { resource: 'errorresource' },\n      metadata: {\n        '/resource': [\n          {\n            type: cxschema.ArtifactMetadataEntryType.ERROR,\n            data: 'this is an error',\n          },\n        ],\n      },\n    }],\n  });\n\n  return cloudExec.synthesize();\n}\n\nasync function testCloudAssemblyNoStacks() {\n  const cloudExec = new MockCloudExecutable({\n    stacks: [],\n  });\n\n  return cloudExec.synthesize();\n}\n\nasync function testNestedCloudAssembly({ env }: { env?: string; versionReporting?: boolean } = {}) {\n  const cloudExec = new MockCloudExecutable({\n    stacks: [{\n      stackName: 'withouterrors',\n      env,\n      template: { resource: 'noerrorresource' },\n      // The nesting in the path should be independent of the position in the tree\n      displayName: 'deeply/hidden/withouterrors',\n    },\n    {\n      stackName: 'witherrors',\n      env,\n      template: { resource: 'errorresource' },\n      metadata: {\n        '/resource': [\n          {\n            type: cxschema.ArtifactMetadataEntryType.ERROR,\n            data: 'this is an error',\n          },\n        ],\n      },\n    }],\n    nestedAssemblies: [{\n      stacks: [{\n        stackName: 'nested',\n        env,\n        template: { resource: 'nestederror' },\n        metadata: {\n          '/resource': [\n            {\n              type: cxschema.ArtifactMetadataEntryType.ERROR,\n              data: 'this is another error',\n            },\n          ],\n        },\n      }],\n    }],\n  });\n\n  return cloudExec.synthesize();\n}\n"]} | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cloud-assembly.test.js","sourceRoot":"","sources":["cloud-assembly.test.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable import/order */\nimport * as cxschema from '@aws-cdk/cloud-assembly-schema';\nimport { DefaultSelection } from '../../lib/api/cxapp/cloud-assembly';\nimport { MockCloudExecutable } from '../util';\nimport { cliAssemblyWithForcedVersion } from './assembly-versions';\n\n// behave like v2\nprocess.env.CXAPI_DISABLE_SELECT_BY_ID = '1';\n\ntest('do not throw when selecting stack without errors', async () => {\n  // GIVEN\n  const cxasm = await testCloudAssembly();\n\n  // WHEN\n  const selected = await cxasm.selectStacks( { patterns: ['withouterrorsNODEPATH'] }, {\n    defaultBehavior: DefaultSelection.AllStacks,\n  });\n  selected.processMetadataMessages();\n\n  // THEN\n  expect(selected.firstStack.template.resource).toBe('noerrorresource');\n});\n\ntest('do throw when selecting stack with errors', async () => {\n  // GIVEN\n  const cxasm = await testCloudAssembly();\n\n  // WHEN\n  const selected = await cxasm.selectStacks({ patterns: ['witherrors'] }, {\n    defaultBehavior: DefaultSelection.AllStacks,\n  });\n\n  // THEN\n  expect(() => selected.processMetadataMessages()).toThrow(/Found errors/);\n});\n\ntest('select all top level stacks in the presence of nested assemblies', async () => {\n  // GIVEN\n  const cxasm = await testNestedCloudAssembly();\n\n  // WHEN\n  const x = await cxasm.selectStacks({ allTopLevel: true, patterns: [] }, { defaultBehavior: DefaultSelection.AllStacks });\n\n  // THEN\n  expect(x.stackCount).toBe(2);\n  expect(x.stackIds).toContain('witherrors');\n  expect(x.stackIds).toContain('withouterrors');\n});\n\ntest('select stacks by glob pattern', async () => {\n  // GIVEN\n  const cxasm = await testCloudAssembly();\n\n  // WHEN\n  const x = await cxasm.selectStacks({ patterns: ['with*'] }, { defaultBehavior: DefaultSelection.AllStacks });\n\n  // THEN\n  expect(x.stackCount).toBe(2);\n  expect(x.stackIds).toContain('witherrors');\n  expect(x.stackIds).toContain('withouterrors');\n});\n\ntest('select behavior: all', async () => {\n  // GIVEN\n  const cxasm = await testCloudAssembly();\n\n  // WHEN\n  const x = await cxasm.selectStacks({ patterns: [] }, { defaultBehavior: DefaultSelection.AllStacks });\n\n  // THEN\n  expect(x.stackCount).toBe(2);\n});\n\ntest('select behavior: none', async () => {\n  // GIVEN\n  const cxasm = await testCloudAssembly();\n\n  // WHEN\n  const x = await cxasm.selectStacks({ patterns: [] }, { defaultBehavior: DefaultSelection.None });\n\n  // THEN\n  expect(x.stackCount).toBe(0);\n});\n\ntest('select behavior: single', async () => {\n  // GIVEN\n  const cxasm = await testCloudAssembly();\n\n  // WHEN\n  await expect(cxasm.selectStacks({ patterns: [] }, { defaultBehavior: DefaultSelection.OnlySingle }))\n    .rejects.toThrow('Since this app includes more than a single stack, specify which stacks to use (wildcards are supported) or specify `--all`');\n});\n\ntest('stack list error contains node paths', async () => {\n  // GIVEN\n  const cxasm = await testCloudAssembly();\n\n  // WHEN\n  await expect(cxasm.selectStacks({ patterns: [] }, { defaultBehavior: DefaultSelection.OnlySingle }))\n    .rejects.toThrow('withouterrorsNODEPATH');\n});\n\ntest('select behavior: repeat', async () => {\n  // GIVEN\n  const cxasm = await testCloudAssembly();\n\n  // WHEN\n  const x = await cxasm.selectStacks({ patterns: ['withouterrorsNODEPATH', 'withouterrorsNODEPATH'] }, {\n    defaultBehavior: DefaultSelection.AllStacks,\n  });\n\n  // THEN\n  expect(x.stackCount).toBe(1);\n});\n\ntest('select behavior with nested assemblies: all', async () => {\n  // GIVEN\n  const cxasm = await testNestedCloudAssembly();\n\n  // WHEN\n  const x = await cxasm.selectStacks({ patterns: [] }, { defaultBehavior: DefaultSelection.AllStacks });\n\n  // THEN\n  expect(x.stackCount).toBe(3);\n});\n\ntest('select behavior with nested assemblies: none', async () => {\n  // GIVEN\n  const cxasm = await testNestedCloudAssembly();\n\n  // WHEN\n  const x = await cxasm.selectStacks({ patterns: [] }, { defaultBehavior: DefaultSelection.None });\n\n  // THEN\n  expect(x.stackCount).toBe(0);\n});\n\ntest('select behavior with nested assemblies: single', async () => {\n  // GIVEN\n  const cxasm = await testNestedCloudAssembly();\n\n  // WHEN\n  await expect(cxasm.selectStacks({ patterns: [] }, { defaultBehavior: DefaultSelection.OnlySingle }))\n    .rejects.toThrow('Since this app includes more than a single stack, specify which stacks to use (wildcards are supported) or specify `--all`');\n});\n\ntest('select behavior with nested assemblies: repeat', async() => {\n  // GIVEN\n  const cxasm = await testNestedCloudAssembly();\n\n  // WHEN\n  const x = await cxasm.selectStacks({ patterns: ['deeply/hidden/withouterrors', 'nested'] }, {\n    defaultBehavior: DefaultSelection.AllStacks,\n  });\n\n  // THEN\n  expect(x.stackCount).toBe(2);\n});\n\ntest('select behavior with no stacks and ignore stacks option', async() => {\n  // GIVEN\n  const cxasm = await testCloudAssemblyNoStacks();\n\n  // WHEN\n  const x = await cxasm.selectStacks({ patterns: [] }, {\n    defaultBehavior: DefaultSelection.AllStacks,\n    ignoreNoStacks: true,\n  });\n\n  // THEN\n  expect(x.stackCount).toBe(0);\n});\n\ntest('select behavior with no stacks and no ignore stacks option', async() => {\n  // GIVEN\n  const cxasm = await testCloudAssemblyNoStacks();\n\n  // WHEN & THEN\n  await expect(cxasm.selectStacks({ patterns: [] }, { defaultBehavior: DefaultSelection.AllStacks, ignoreNoStacks: false }))\n    .rejects.toThrow('This app contains no stacks');\n});\n\ntest('select behavior with no stacks and default ignore stacks options (false)', async() => {\n  // GIVEN\n  const cxasm = await testCloudAssemblyNoStacks();\n\n  // WHEN & THEN\n  await expect(cxasm.selectStacks({ patterns: [] }, { defaultBehavior: DefaultSelection.AllStacks }))\n    .rejects.toThrow('This app contains no stacks');\n});\n\nasync function testCloudAssembly({ env }: { env?: string; versionReporting?: boolean } = {}) {\n  const cloudExec = new MockCloudExecutable({\n    stacks: [{\n      stackName: 'withouterrors',\n      displayName: 'withouterrorsNODEPATH',\n      env,\n      template: { resource: 'noerrorresource' },\n    },\n    {\n      stackName: 'witherrors',\n      env,\n      template: { resource: 'errorresource' },\n      metadata: {\n        '/resource': [\n          {\n            type: cxschema.ArtifactMetadataEntryType.ERROR,\n            data: 'this is an error',\n          },\n        ],\n      },\n    }],\n  });\n\n  return cloudExec.synthesize();\n}\n\nasync function testCloudAssemblyNoStacks() {\n  const cloudExec = new MockCloudExecutable({\n    stacks: [],\n  });\n\n  return cloudExec.synthesize();\n}\n\nasync function testNestedCloudAssembly({ env }: { env?: string; versionReporting?: boolean } = {}) {\n  const cloudExec = new MockCloudExecutable({\n    stacks: [{\n      stackName: 'withouterrors',\n      env,\n      template: { resource: 'noerrorresource' },\n      // The nesting in the path should be independent of the position in the tree\n      displayName: 'deeply/hidden/withouterrors',\n    },\n    {\n      stackName: 'witherrors',\n      env,\n      template: { resource: 'errorresource' },\n      metadata: {\n        '/resource': [\n          {\n            type: cxschema.ArtifactMetadataEntryType.ERROR,\n            data: 'this is an error',\n          },\n        ],\n      },\n    }],\n    nestedAssemblies: [{\n      stacks: [{\n        stackName: 'nested',\n        env,\n        template: { resource: 'nestederror' },\n        metadata: {\n          '/resource': [\n            {\n              type: cxschema.ArtifactMetadataEntryType.ERROR,\n              data: 'this is another error',\n            },\n          ],\n        },\n      }],\n    }],\n  });\n\n  const asm = await cloudExec.synthesize();\n  return cliAssemblyWithForcedVersion(asm, '30.0.0');\n}\n"]} |
@@ -9,52 +9,63 @@ "use strict"; | ||
const util_1 = require("../util"); | ||
// Apps on this version of the cxschema don't emit their own metadata resources | ||
// yet, so rely on the CLI to add the Metadata resource in. | ||
const SCHEMA_VERSION_THAT_DOESNT_INCLUDE_METADATA_ITSELF = '2.0.0'; | ||
describe('AWS::CDK::Metadata', () => { | ||
test('is generated for relocatable stacks from old frameworks', async () => { | ||
await withFakeCurrentCxVersion('2.0.0', async () => { | ||
const cx = await testCloudExecutable({ env: `aws://${cxapi.UNKNOWN_ACCOUNT}/${cxapi.UNKNOWN_REGION}`, versionReporting: true }); | ||
const cxasm = await cx.synthesize(); | ||
const result = cxasm.stackById('withouterrors').firstStack; | ||
const metadata = result.template.Resources && result.template.Resources.CDKMetadata; | ||
expect(metadata).toEqual({ | ||
Type: 'AWS::CDK::Metadata', | ||
Properties: { | ||
// eslint-disable-next-line @typescript-eslint/no-require-imports | ||
Modules: `${require('../../package.json').name}=${require('../../package.json').version}`, | ||
}, | ||
Condition: 'CDKMetadataAvailable', | ||
}); | ||
expect(result.template.Conditions?.CDKMetadataAvailable).toBeDefined(); | ||
const cx = await testCloudExecutable({ | ||
env: `aws://${cxapi.UNKNOWN_ACCOUNT}/${cxapi.UNKNOWN_REGION}`, | ||
versionReporting: true, | ||
schemaVersion: SCHEMA_VERSION_THAT_DOESNT_INCLUDE_METADATA_ITSELF, | ||
}); | ||
const cxasm = await cx.synthesize(); | ||
const result = cxasm.stackById('withouterrors').firstStack; | ||
const metadata = result.template.Resources && result.template.Resources.CDKMetadata; | ||
expect(metadata).toEqual({ | ||
Type: 'AWS::CDK::Metadata', | ||
Properties: { | ||
// eslint-disable-next-line @typescript-eslint/no-require-imports | ||
Modules: `${require('../../package.json').name}=${require('../../package.json').version}`, | ||
}, | ||
Condition: 'CDKMetadataAvailable', | ||
}); | ||
expect(result.template.Conditions?.CDKMetadataAvailable).toBeDefined(); | ||
}); | ||
test('is generated for stacks in supported regions from old frameworks', async () => { | ||
await withFakeCurrentCxVersion('2.0.0', async () => { | ||
const cx = await testCloudExecutable({ env: 'aws://012345678912/us-east-1', versionReporting: true }); | ||
const cxasm = await cx.synthesize(); | ||
const result = cxasm.stackById('withouterrors').firstStack; | ||
const metadata = result.template.Resources && result.template.Resources.CDKMetadata; | ||
expect(metadata).toEqual({ | ||
Type: 'AWS::CDK::Metadata', | ||
Properties: { | ||
// eslint-disable-next-line @typescript-eslint/no-require-imports | ||
Modules: `${require('../../package.json').name}=${require('../../package.json').version}`, | ||
}, | ||
}); | ||
const cx = await testCloudExecutable({ | ||
env: 'aws://012345678912/us-east-1', | ||
versionReporting: true, | ||
schemaVersion: SCHEMA_VERSION_THAT_DOESNT_INCLUDE_METADATA_ITSELF, | ||
}); | ||
const cxasm = await cx.synthesize(); | ||
const result = cxasm.stackById('withouterrors').firstStack; | ||
const metadata = result.template.Resources && result.template.Resources.CDKMetadata; | ||
expect(metadata).toEqual({ | ||
Type: 'AWS::CDK::Metadata', | ||
Properties: { | ||
// eslint-disable-next-line @typescript-eslint/no-require-imports | ||
Modules: `${require('../../package.json').name}=${require('../../package.json').version}`, | ||
}, | ||
}); | ||
}); | ||
test('is not generated for stacks in unsupported regions from old frameworks', async () => { | ||
await withFakeCurrentCxVersion('2.0.0', async () => { | ||
const cx = await testCloudExecutable({ env: 'aws://012345678912/bermuda-triangle-1337', versionReporting: true }); | ||
const cxasm = await cx.synthesize(); | ||
const result = cxasm.stackById('withouterrors').firstStack; | ||
const metadata = result.template.Resources && result.template.Resources.CDKMetadata; | ||
expect(metadata).toBeUndefined(); | ||
const cx = await testCloudExecutable({ | ||
env: 'aws://012345678912/bermuda-triangle-1337', | ||
versionReporting: true, | ||
schemaVersion: SCHEMA_VERSION_THAT_DOESNT_INCLUDE_METADATA_ITSELF, | ||
}); | ||
const cxasm = await cx.synthesize(); | ||
const result = cxasm.stackById('withouterrors').firstStack; | ||
const metadata = result.template.Resources && result.template.Resources.CDKMetadata; | ||
expect(metadata).toBeUndefined(); | ||
}); | ||
test('is not generated for new frameworks', async () => { | ||
await withFakeCurrentCxVersion('8.0.0', async () => { | ||
const cx = await testCloudExecutable({ env: 'aws://012345678912/us-east-1', versionReporting: true }); | ||
const cxasm = await cx.synthesize(); | ||
const result = cxasm.stackById('withouterrors').firstStack; | ||
const metadata = result.template.Resources && result.template.Resources.CDKMetadata; | ||
expect(metadata).toBeUndefined(); | ||
const cx = await testCloudExecutable({ | ||
env: 'aws://012345678912/us-east-1', | ||
versionReporting: true, | ||
schemaVersion: '8.0.0', | ||
}); | ||
const cxasm = await cx.synthesize(); | ||
const result = cxasm.stackById('withouterrors').firstStack; | ||
const metadata = result.template.Resources && result.template.Resources.CDKMetadata; | ||
expect(metadata).toBeUndefined(); | ||
}); | ||
@@ -99,3 +110,3 @@ }); | ||
}); | ||
async function testCloudExecutable({ env, versionReporting = true } = {}) { | ||
async function testCloudExecutable({ env, versionReporting = true, schemaVersion } = {}) { | ||
const cloudExec = new util_1.MockCloudExecutable({ | ||
@@ -120,2 +131,3 @@ stacks: [{ | ||
}], | ||
schemaVersion, | ||
}); | ||
@@ -125,12 +137,2 @@ cloudExec.configuration.settings.set(['versionReporting'], versionReporting); | ||
} | ||
async function withFakeCurrentCxVersion(version, block) { | ||
const currentVersionFn = cxschema.Manifest.version; | ||
cxschema.Manifest.version = () => version; | ||
try { | ||
return await block(); | ||
} | ||
finally { | ||
cxschema.Manifest.version = currentVersionFn; | ||
} | ||
} | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cloud-executable.test.js","sourceRoot":"","sources":["cloud-executable.test.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable import/order */\nimport * as cxschema from '@aws-cdk/cloud-assembly-schema';\nimport * as cxapi from '@aws-cdk/cx-api';\nimport { DefaultSelection } from '../../lib/api/cxapp/cloud-assembly';\nimport { registerContextProvider } from '../../lib/context-providers';\nimport { MockCloudExecutable } from '../util';\n\ndescribe('AWS::CDK::Metadata', () => {\n  test('is generated for relocatable stacks from old frameworks', async () => {\n    await withFakeCurrentCxVersion('2.0.0', async () => {\n      const cx = await testCloudExecutable({ env: `aws://${cxapi.UNKNOWN_ACCOUNT}/${cxapi.UNKNOWN_REGION}`, versionReporting: true });\n      const cxasm = await cx.synthesize();\n\n      const result = cxasm.stackById('withouterrors').firstStack;\n      const metadata = result.template.Resources && result.template.Resources.CDKMetadata;\n      expect(metadata).toEqual({\n        Type: 'AWS::CDK::Metadata',\n        Properties: {\n          // eslint-disable-next-line @typescript-eslint/no-require-imports\n          Modules: `${require('../../package.json').name}=${require('../../package.json').version}`,\n        },\n        Condition: 'CDKMetadataAvailable',\n      });\n\n      expect(result.template.Conditions?.CDKMetadataAvailable).toBeDefined();\n    });\n  });\n\n  test('is generated for stacks in supported regions from old frameworks', async () => {\n    await withFakeCurrentCxVersion('2.0.0', async () => {\n      const cx = await testCloudExecutable({ env: 'aws://012345678912/us-east-1', versionReporting: true });\n      const cxasm = await cx.synthesize();\n\n      const result = cxasm.stackById('withouterrors').firstStack;\n      const metadata = result.template.Resources && result.template.Resources.CDKMetadata;\n      expect(metadata).toEqual({\n        Type: 'AWS::CDK::Metadata',\n        Properties: {\n          // eslint-disable-next-line @typescript-eslint/no-require-imports\n          Modules: `${require('../../package.json').name}=${require('../../package.json').version}`,\n        },\n      });\n    });\n  });\n\n  test('is not generated for stacks in unsupported regions from old frameworks', async () => {\n    await withFakeCurrentCxVersion('2.0.0', async () => {\n      const cx = await testCloudExecutable({ env: 'aws://012345678912/bermuda-triangle-1337', versionReporting: true });\n      const cxasm = await cx.synthesize();\n\n      const result = cxasm.stackById('withouterrors').firstStack;\n      const metadata = result.template.Resources && result.template.Resources.CDKMetadata;\n      expect(metadata).toBeUndefined();\n    });\n  });\n\n  test('is not generated for new frameworks', async () => {\n    await withFakeCurrentCxVersion('8.0.0', async () => {\n      const cx = await testCloudExecutable({ env: 'aws://012345678912/us-east-1', versionReporting: true });\n      const cxasm = await cx.synthesize();\n\n      const result = cxasm.stackById('withouterrors').firstStack;\n      const metadata = result.template.Resources && result.template.Resources.CDKMetadata;\n      expect(metadata).toBeUndefined();\n    });\n  });\n});\n\ntest('stop executing if context providers are not making progress', async () => {\n  registerContextProvider(cxschema.ContextProvider.AVAILABILITY_ZONE_PROVIDER, {\n    async getValue(_: { [key: string]: any }): Promise<any> {\n      return 'foo';\n    },\n  });\n\n  const cloudExecutable = new MockCloudExecutable({\n    stacks: [{\n      stackName: 'thestack',\n      template: { resource: 'noerrorresource' },\n    }],\n    // Always return the same missing keys, synthesis should still finish.\n    missing: [\n      { key: 'abcdef', props: { account: '1324', region: 'us-east-1' }, provider: cxschema.ContextProvider.AVAILABILITY_ZONE_PROVIDER },\n    ],\n  });\n  const cxasm = await cloudExecutable.synthesize();\n\n  // WHEN\n  await cxasm.selectStacks({ patterns: ['thestack'] }, { defaultBehavior: DefaultSelection.AllStacks });\n\n  // THEN: the test finishes normally});\n});\n\ntest('fails if lookups are disabled and missing context is synthesized', async () => {\n  // GIVEN\n  const cloudExecutable = new MockCloudExecutable({\n    stacks: [{\n      stackName: 'thestack',\n      template: { resource: 'noerrorresource' },\n    }],\n    // Always return the same missing keys, synthesis should still finish.\n    missing: [\n      { key: 'abcdef', props: { account: '1324', region: 'us-east-1' }, provider: cxschema.ContextProvider.AVAILABILITY_ZONE_PROVIDER },\n    ],\n  });\n  cloudExecutable.configuration.settings.set(['lookups'], false);\n\n  // WHEN\n  await expect(cloudExecutable.synthesize()).rejects.toThrow(/Context lookups have been disabled/);\n});\n\nasync function testCloudExecutable({ env, versionReporting = true }: { env?: string; versionReporting?: boolean } = {}) {\n  const cloudExec = new MockCloudExecutable({\n    stacks: [{\n      stackName: 'withouterrors',\n      env,\n      template: { resource: 'noerrorresource' },\n    },\n    {\n      stackName: 'witherrors',\n      env,\n      template: { resource: 'errorresource' },\n      metadata: {\n        '/resource': [\n          {\n            type: cxschema.ArtifactMetadataEntryType.ERROR,\n            data: 'this is an error',\n          },\n        ],\n      },\n    }],\n  });\n  cloudExec.configuration.settings.set(['versionReporting'], versionReporting);\n\n  return cloudExec;\n}\n\nasync function withFakeCurrentCxVersion<A>(version: string, block: () => Promise<A>): Promise<A> {\n  const currentVersionFn = cxschema.Manifest.version;\n  cxschema.Manifest.version = () => version;\n  try {\n    return await block();\n  } finally {\n    cxschema.Manifest.version = currentVersionFn;\n  }\n}"]} | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cloud-executable.test.js","sourceRoot":"","sources":["cloud-executable.test.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable import/order */\nimport * as cxschema from '@aws-cdk/cloud-assembly-schema';\nimport * as cxapi from '@aws-cdk/cx-api';\nimport { DefaultSelection } from '../../lib/api/cxapp/cloud-assembly';\nimport { registerContextProvider } from '../../lib/context-providers';\nimport { MockCloudExecutable } from '../util';\n\n// Apps on this version of the cxschema don't emit their own metadata resources\n// yet, so rely on the CLI to add the Metadata resource in.\nconst SCHEMA_VERSION_THAT_DOESNT_INCLUDE_METADATA_ITSELF = '2.0.0';\n\ndescribe('AWS::CDK::Metadata', () => {\n  test('is generated for relocatable stacks from old frameworks', async () => {\n    const cx = await testCloudExecutable({\n      env: `aws://${cxapi.UNKNOWN_ACCOUNT}/${cxapi.UNKNOWN_REGION}`,\n      versionReporting: true,\n      schemaVersion: SCHEMA_VERSION_THAT_DOESNT_INCLUDE_METADATA_ITSELF,\n    });\n    const cxasm = await cx.synthesize();\n\n    const result = cxasm.stackById('withouterrors').firstStack;\n    const metadata = result.template.Resources && result.template.Resources.CDKMetadata;\n    expect(metadata).toEqual({\n      Type: 'AWS::CDK::Metadata',\n      Properties: {\n        // eslint-disable-next-line @typescript-eslint/no-require-imports\n        Modules: `${require('../../package.json').name}=${require('../../package.json').version}`,\n      },\n      Condition: 'CDKMetadataAvailable',\n    });\n\n    expect(result.template.Conditions?.CDKMetadataAvailable).toBeDefined();\n  });\n\n  test('is generated for stacks in supported regions from old frameworks', async () => {\n    const cx = await testCloudExecutable({\n      env: 'aws://012345678912/us-east-1',\n      versionReporting: true,\n      schemaVersion: SCHEMA_VERSION_THAT_DOESNT_INCLUDE_METADATA_ITSELF,\n    });\n    const cxasm = await cx.synthesize();\n\n    const result = cxasm.stackById('withouterrors').firstStack;\n    const metadata = result.template.Resources && result.template.Resources.CDKMetadata;\n    expect(metadata).toEqual({\n      Type: 'AWS::CDK::Metadata',\n      Properties: {\n        // eslint-disable-next-line @typescript-eslint/no-require-imports\n        Modules: `${require('../../package.json').name}=${require('../../package.json').version}`,\n      },\n    });\n  });\n\n  test('is not generated for stacks in unsupported regions from old frameworks', async () => {\n    const cx = await testCloudExecutable({\n      env: 'aws://012345678912/bermuda-triangle-1337',\n      versionReporting: true,\n      schemaVersion: SCHEMA_VERSION_THAT_DOESNT_INCLUDE_METADATA_ITSELF,\n    });\n    const cxasm = await cx.synthesize();\n\n    const result = cxasm.stackById('withouterrors').firstStack;\n    const metadata = result.template.Resources && result.template.Resources.CDKMetadata;\n    expect(metadata).toBeUndefined();\n  });\n\n  test('is not generated for new frameworks', async () => {\n    const cx = await testCloudExecutable({\n      env: 'aws://012345678912/us-east-1',\n      versionReporting: true,\n      schemaVersion: '8.0.0',\n    });\n    const cxasm = await cx.synthesize();\n\n    const result = cxasm.stackById('withouterrors').firstStack;\n    const metadata = result.template.Resources && result.template.Resources.CDKMetadata;\n    expect(metadata).toBeUndefined();\n  });\n});\n\ntest('stop executing if context providers are not making progress', async () => {\n  registerContextProvider(cxschema.ContextProvider.AVAILABILITY_ZONE_PROVIDER, {\n    async getValue(_: { [key: string]: any }): Promise<any> {\n      return 'foo';\n    },\n  });\n\n  const cloudExecutable = new MockCloudExecutable({\n    stacks: [{\n      stackName: 'thestack',\n      template: { resource: 'noerrorresource' },\n    }],\n    // Always return the same missing keys, synthesis should still finish.\n    missing: [\n      { key: 'abcdef', props: { account: '1324', region: 'us-east-1' }, provider: cxschema.ContextProvider.AVAILABILITY_ZONE_PROVIDER },\n    ],\n  });\n  const cxasm = await cloudExecutable.synthesize();\n\n  // WHEN\n  await cxasm.selectStacks({ patterns: ['thestack'] }, { defaultBehavior: DefaultSelection.AllStacks });\n\n  // THEN: the test finishes normally});\n});\n\ntest('fails if lookups are disabled and missing context is synthesized', async () => {\n  // GIVEN\n  const cloudExecutable = new MockCloudExecutable({\n    stacks: [{\n      stackName: 'thestack',\n      template: { resource: 'noerrorresource' },\n    }],\n    // Always return the same missing keys, synthesis should still finish.\n    missing: [\n      { key: 'abcdef', props: { account: '1324', region: 'us-east-1' }, provider: cxschema.ContextProvider.AVAILABILITY_ZONE_PROVIDER },\n    ],\n  });\n  cloudExecutable.configuration.settings.set(['lookups'], false);\n\n  // WHEN\n  await expect(cloudExecutable.synthesize()).rejects.toThrow(/Context lookups have been disabled/);\n});\n\nasync function testCloudExecutable(\n  { env, versionReporting = true, schemaVersion }:\n  { env?: string; versionReporting?: boolean; schemaVersion?: string } = {},\n) {\n  const cloudExec = new MockCloudExecutable({\n    stacks: [{\n      stackName: 'withouterrors',\n      env,\n      template: { resource: 'noerrorresource' },\n    },\n    {\n      stackName: 'witherrors',\n      env,\n      template: { resource: 'errorresource' },\n      metadata: {\n        '/resource': [\n          {\n            type: cxschema.ArtifactMetadataEntryType.ERROR,\n            data: 'this is an error',\n          },\n        ],\n      },\n    }],\n    schemaVersion,\n  });\n  cloudExec.configuration.settings.set(['versionReporting'], versionReporting);\n\n  return cloudExec;\n}\n"]} |
@@ -6,3 +6,3 @@ "use strict"; | ||
const cdk_build_tools_1 = require("@aws-cdk/cdk-build-tools"); | ||
const cxschema = require("aws-cdk-lib/cloud-assembly-schema"); | ||
const cxschema = require("@aws-cdk/cloud-assembly-schema"); | ||
const cdk = require("aws-cdk-lib"); | ||
@@ -19,2 +19,3 @@ const semver = require("semver"); | ||
const rwlock_1 = require("../../lib/api/util/rwlock"); | ||
const assembly_versions_1 = require("./assembly-versions"); | ||
let sdkProvider; | ||
@@ -70,2 +71,3 @@ let config; | ||
} | ||
(0, assembly_versions_1.rewriteManifestVersion)('cdk.out', `${mockManifestVersion}`); | ||
const expectedError = 'This CDK CLI is not compatible with the CDK library used by your application. Please upgrade the CLI to the latest version.' | ||
@@ -79,2 +81,3 @@ + `\n(Cloud assembly schema version mismatch: Maximum schema version supported is ${semver.major(currentSchemaVersion)}.x.x, but found ${mockManifestVersion})`; | ||
app.synth(); | ||
rewriteManifestVersionToOurs(); | ||
config.settings.set(['app'], 'cdk.out'); | ||
@@ -84,6 +87,14 @@ const { lock } = await (0, exec_1.execProgram)(sdkProvider, config); | ||
}, TEN_SECOND_TIMEOUT); | ||
test('cli does not throw when manifest version < schema version', async () => { | ||
// Why do we have to do something here at all? Because `aws-cdk-lib` has its own version of `cloud-assembly-schema`, | ||
// which will have real version `38.0.0`, different from the `0.0.0` version of `cloud-assembly-schema` that the CLI | ||
// uses. | ||
// | ||
// Since our Cloud Assembly Schema version will be `0.0.0` and there is no such thing as `-1.0.0`, this test doesn't | ||
// make any sense anymore. | ||
// eslint-disable-next-line jest/no-disabled-tests | ||
test.skip('cli does not throw when manifest version < schema version', async () => { | ||
const app = createApp(); | ||
const currentSchemaVersion = cxschema.Manifest.version(); | ||
app.synth(); | ||
rewriteManifestVersionToOurs(); | ||
config.settings.set(['app'], 'cdk.out'); | ||
@@ -109,2 +120,3 @@ // this mock will cause the cli to think its exepcted schema version is | ||
writeOutputAssembly(); | ||
rewriteManifestVersionToOurs(); | ||
// WHEN | ||
@@ -217,3 +229,16 @@ const { assembly: cloudAssembly, lock } = await (0, exec_1.execProgram)(sdkProvider, config); | ||
cdk_build_tools_1.bockfs.write('/home/project/cdk.out/manifest.json', JSON.stringify(asm.manifest)); | ||
rewriteManifestVersionToOurs(cdk_build_tools_1.bockfs.path('/home/project/cdk.out')); | ||
} | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"exec.test.js","sourceRoot":"","sources":["exec.test.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable import/order */\njest.mock('child_process');\nimport { bockfs } from '@aws-cdk/cdk-build-tools';\nimport * as cxschema from 'aws-cdk-lib/cloud-assembly-schema';\nimport * as cdk from 'aws-cdk-lib';\nimport * as semver from 'semver';\nimport * as sinon from 'sinon';\nimport { ImportMock } from 'ts-mock-imports';\nimport { execProgram } from '../../lib/api/cxapp/exec';\nimport { LogLevel, setLogLevel } from '../../lib/logging';\nimport { Configuration } from '../../lib/settings';\nimport { testAssembly } from '../util';\nimport { mockSpawn } from '../util/mock-child_process';\nimport { MockSdkProvider } from '../util/mock-sdk';\nimport { RWLock } from '../../lib/api/util/rwlock';\n\nlet sdkProvider: MockSdkProvider;\nlet config: Configuration;\nbeforeEach(() => {\n  setLogLevel(LogLevel.DEBUG);\n\n  sdkProvider = new MockSdkProvider();\n  config = new Configuration();\n\n  config.settings.set(['output'], 'cdk.out');\n\n  // insert contents in fake filesystem\n  bockfs({\n    '/home/project/cloud-executable': 'ARBITRARY',\n    '/home/project/windows.js': 'ARBITRARY',\n    'home/project/executable-app.js': 'ARBITRARY',\n  });\n  bockfs.workingDirectory('/home/project');\n  bockfs.executable('/home/project/cloud-executable');\n  bockfs.executable('/home/project/executable-app.js');\n});\n\nafterEach(() => {\n  setLogLevel(LogLevel.DEFAULT);\n\n  sinon.restore();\n  bockfs.restore();\n});\n\n// We need to increase the default 5s jest\n// timeout for async tests because the 'execProgram' invocation\n// might take a while :\\\nconst TEN_SECOND_TIMEOUT = 10000;\n\nfunction createApp(): cdk.App {\n  const app = new cdk.App({ outdir: 'cdk.out' });\n  const stack = new cdk.Stack(app, 'Stack');\n\n  new cdk.CfnResource(stack, 'Role', {\n    type: 'AWS::IAM::Role',\n    properties: {\n      RoleName: 'Role',\n    },\n  });\n\n  return app;\n}\n\ntest('cli throws when manifest version > schema version', async () => {\n\n  const app = createApp();\n  const currentSchemaVersion = cxschema.Manifest.version();\n  const mockManifestVersion = semver.inc(currentSchemaVersion, 'major');\n\n  // this mock will cause the framework to use a greater schema version than the real one,\n  // and should cause the CLI to fail.\n  const mockVersionNumber = ImportMock.mockFunction(cxschema.Manifest, 'version', mockManifestVersion);\n  try {\n    app.synth();\n  } finally {\n    mockVersionNumber.restore();\n  }\n\n  const expectedError = 'This CDK CLI is not compatible with the CDK library used by your application. Please upgrade the CLI to the latest version.'\n    + `\\n(Cloud assembly schema version mismatch: Maximum schema version supported is ${semver.major(currentSchemaVersion)}.x.x, but found ${mockManifestVersion})`;\n\n  config.settings.set(['app'], 'cdk.out');\n\n  await expect(execProgram(sdkProvider, config)).rejects.toEqual(new Error(expectedError));\n\n}, TEN_SECOND_TIMEOUT);\n\ntest('cli does not throw when manifest version = schema version', async () => {\n\n  const app = createApp();\n  app.synth();\n\n  config.settings.set(['app'], 'cdk.out');\n\n  const { lock } = await execProgram(sdkProvider, config);\n  await lock.release();\n\n}, TEN_SECOND_TIMEOUT);\n\ntest('cli does not throw when manifest version < schema version', async () => {\n\n  const app = createApp();\n  const currentSchemaVersion = cxschema.Manifest.version();\n\n  app.synth();\n\n  config.settings.set(['app'], 'cdk.out');\n\n  // this mock will cause the cli to think its exepcted schema version is\n  // greater that the version created in the manifest, which is what we are testing for.\n  const mockVersionNumber = ImportMock.mockFunction(cxschema.Manifest, 'version', semver.inc(currentSchemaVersion, 'major'));\n  try {\n    const { lock } = await execProgram(sdkProvider, config);\n    await lock.release();\n  } finally {\n    mockVersionNumber.restore();\n  }\n\n}, TEN_SECOND_TIMEOUT);\n\ntest('validates --app key is present', async () => {\n  // GIVEN no config key for `app`\n  await expect(execProgram(sdkProvider, config)).rejects.toThrow(\n    '--app is required either in command-line, in cdk.json or in ~/.cdk.json',\n  );\n\n});\n\ntest('bypasses synth when app points to a cloud assembly', async () => {\n  // GIVEN\n  config.settings.set(['app'], 'cdk.out');\n  writeOutputAssembly();\n\n  // WHEN\n  const { assembly: cloudAssembly, lock } = await execProgram(sdkProvider, config);\n  expect(cloudAssembly.artifacts).toEqual([]);\n  expect(cloudAssembly.directory).toEqual('cdk.out');\n\n  await lock.release();\n});\n\ntest('the application set in --app is executed', async () => {\n  // GIVEN\n  config.settings.set(['app'], 'cloud-executable');\n  mockSpawn({\n    commandLine: 'cloud-executable',\n    sideEffect: () => writeOutputAssembly(),\n  });\n\n  // WHEN\n  const { lock } = await execProgram(sdkProvider, config);\n  await lock.release();\n});\n\ntest('the application set in --app is executed as-is if it contains a filename that does not exist', async () => {\n  // GIVEN\n  config.settings.set(['app'], 'does-not-exist');\n  mockSpawn({\n    commandLine: 'does-not-exist',\n    sideEffect: () => writeOutputAssembly(),\n  });\n\n  // WHEN\n  const { lock } = await execProgram(sdkProvider, config);\n  await lock.release();\n});\n\ntest('the application set in --app is executed with arguments', async () => {\n  // GIVEN\n  config.settings.set(['app'], 'cloud-executable an-arg');\n  mockSpawn({\n    commandLine: 'cloud-executable an-arg',\n    sideEffect: () => writeOutputAssembly(),\n  });\n\n  // WHEN\n  const { lock } = await execProgram(sdkProvider, config);\n  await lock.release();\n});\n\ntest('application set in --app as `*.js` always uses handler on windows', async () => {\n  // GIVEN\n  sinon.stub(process, 'platform').value('win32');\n  config.settings.set(['app'], 'windows.js');\n  mockSpawn({\n    commandLine: process.execPath + ' windows.js',\n    sideEffect: () => writeOutputAssembly(),\n  });\n\n  // WHEN\n  const { lock } = await execProgram(sdkProvider, config);\n  await lock.release();\n});\n\ntest('application set in --app is `*.js` and executable', async () => {\n  // GIVEN\n  config.settings.set(['app'], 'executable-app.js');\n  mockSpawn({\n    commandLine: 'executable-app.js',\n    sideEffect: () => writeOutputAssembly(),\n  });\n\n  // WHEN\n  const { lock } = await execProgram(sdkProvider, config);\n  await lock.release();\n});\n\ntest('cli throws when the `build` script fails', async () => {\n  // GIVEN\n  config.settings.set(['build'], 'fake-command');\n  mockSpawn({\n    commandLine: 'fake-command',\n    exitCode: 127,\n  });\n\n  // WHEN\n  await expect(execProgram(sdkProvider, config)).rejects.toEqual(new Error('Subprocess exited with error 127'));\n}, TEN_SECOND_TIMEOUT);\n\ntest('cli does not throw when the `build` script succeeds', async () => {\n  // GIVEN\n  config.settings.set(['build'], 'real command');\n  config.settings.set(['app'], 'executable-app.js');\n  mockSpawn({\n    commandLine: 'real command', // `build` key is not split on whitespace\n    exitCode: 0,\n  },\n  {\n    commandLine: 'executable-app.js',\n    sideEffect: () => writeOutputAssembly(),\n  });\n\n  // WHEN\n  const { lock } = await execProgram(sdkProvider, config);\n  await lock.release();\n}, TEN_SECOND_TIMEOUT);\n\ntest('cli releases the outdir lock when execProgram throws', async () => {\n  // GIVEN\n  config.settings.set(['app'], 'cloud-executable');\n  mockSpawn({\n    commandLine: 'fake-command',\n    exitCode: 127,\n  });\n\n  // WHEN\n  await expect(execProgram(sdkProvider, config)).rejects.toThrow();\n\n  const output = config.settings.get(['output']);\n  expect(output).toBeDefined();\n\n  // check that the lock is released\n  const lock = await new RWLock(output).acquireWrite();\n  await lock.release();\n});\n\nfunction writeOutputAssembly() {\n  const asm = testAssembly({\n    stacks: [],\n  });\n  bockfs.write('/home/project/cdk.out/manifest.json', JSON.stringify(asm.manifest));\n}\n"]} | ||
/** | ||
* Rewrite the manifest schema version in the given directory to match the version number we expect (probably `0.0.0`). | ||
* | ||
* Why do we have to do this? Because `aws-cdk-lib` has its own version of `cloud-assembly-schema`, | ||
* which will have real version `38.0.0`, different from the `0.0.0` version of `cloud-assembly-schema` that the CLI | ||
* uses. | ||
* | ||
* If we don't do this, every time we load a Cloud Assembly the code will say "Maximum schema version supported is 0.x.x, but found 30.0.0".0 | ||
*/ | ||
function rewriteManifestVersionToOurs(dir = 'cdk.out') { | ||
(0, assembly_versions_1.rewriteManifestVersion)(dir, cxschema.Manifest.version()); | ||
} | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"exec.test.js","sourceRoot":"","sources":["exec.test.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable import/order */\njest.mock('child_process');\nimport { bockfs } from '@aws-cdk/cdk-build-tools';\nimport * as cxschema from '@aws-cdk/cloud-assembly-schema';\nimport * as cdk from 'aws-cdk-lib';\nimport * as semver from 'semver';\nimport * as sinon from 'sinon';\nimport { ImportMock } from 'ts-mock-imports';\nimport { execProgram } from '../../lib/api/cxapp/exec';\nimport { LogLevel, setLogLevel } from '../../lib/logging';\nimport { Configuration } from '../../lib/settings';\nimport { testAssembly } from '../util';\nimport { mockSpawn } from '../util/mock-child_process';\nimport { MockSdkProvider } from '../util/mock-sdk';\nimport { RWLock } from '../../lib/api/util/rwlock';\nimport { rewriteManifestVersion } from './assembly-versions';\n\nlet sdkProvider: MockSdkProvider;\nlet config: Configuration;\nbeforeEach(() => {\n  setLogLevel(LogLevel.DEBUG);\n\n  sdkProvider = new MockSdkProvider();\n  config = new Configuration();\n\n  config.settings.set(['output'], 'cdk.out');\n\n  // insert contents in fake filesystem\n  bockfs({\n    '/home/project/cloud-executable': 'ARBITRARY',\n    '/home/project/windows.js': 'ARBITRARY',\n    'home/project/executable-app.js': 'ARBITRARY',\n  });\n  bockfs.workingDirectory('/home/project');\n  bockfs.executable('/home/project/cloud-executable');\n  bockfs.executable('/home/project/executable-app.js');\n});\n\nafterEach(() => {\n  setLogLevel(LogLevel.DEFAULT);\n\n  sinon.restore();\n  bockfs.restore();\n});\n\n// We need to increase the default 5s jest\n// timeout for async tests because the 'execProgram' invocation\n// might take a while :\\\nconst TEN_SECOND_TIMEOUT = 10000;\n\nfunction createApp(): cdk.App {\n  const app = new cdk.App({ outdir: 'cdk.out' });\n  const stack = new cdk.Stack(app, 'Stack');\n\n  new cdk.CfnResource(stack, 'Role', {\n    type: 'AWS::IAM::Role',\n    properties: {\n      RoleName: 'Role',\n    },\n  });\n\n  return app;\n}\n\ntest('cli throws when manifest version > schema version', async () => {\n\n  const app = createApp();\n  const currentSchemaVersion = cxschema.Manifest.version();\n  const mockManifestVersion = semver.inc(currentSchemaVersion, 'major');\n\n  // this mock will cause the framework to use a greater schema version than the real one,\n  // and should cause the CLI to fail.\n  const mockVersionNumber = ImportMock.mockFunction(cxschema.Manifest, 'version', mockManifestVersion);\n  try {\n    app.synth();\n  } finally {\n    mockVersionNumber.restore();\n  }\n\n  rewriteManifestVersion('cdk.out', `${mockManifestVersion}`);\n\n  const expectedError = 'This CDK CLI is not compatible with the CDK library used by your application. Please upgrade the CLI to the latest version.'\n    + `\\n(Cloud assembly schema version mismatch: Maximum schema version supported is ${semver.major(currentSchemaVersion)}.x.x, but found ${mockManifestVersion})`;\n\n  config.settings.set(['app'], 'cdk.out');\n\n  await expect(execProgram(sdkProvider, config)).rejects.toEqual(new Error(expectedError));\n\n}, TEN_SECOND_TIMEOUT);\n\ntest('cli does not throw when manifest version = schema version', async () => {\n\n  const app = createApp();\n  app.synth();\n\n  rewriteManifestVersionToOurs();\n\n  config.settings.set(['app'], 'cdk.out');\n\n  const { lock } = await execProgram(sdkProvider, config);\n  await lock.release();\n\n}, TEN_SECOND_TIMEOUT);\n\n// Why do we have to do something here at all? Because `aws-cdk-lib` has its own version of `cloud-assembly-schema`,\n// which will have real version `38.0.0`, different from the `0.0.0` version of `cloud-assembly-schema` that the CLI\n// uses.\n//\n// Since our Cloud Assembly Schema version will be `0.0.0` and there is no such thing as `-1.0.0`, this test doesn't\n// make any sense anymore.\n// eslint-disable-next-line jest/no-disabled-tests\ntest.skip('cli does not throw when manifest version < schema version', async () => {\n\n  const app = createApp();\n  const currentSchemaVersion = cxschema.Manifest.version();\n\n  app.synth();\n\n  rewriteManifestVersionToOurs();\n\n  config.settings.set(['app'], 'cdk.out');\n\n  // this mock will cause the cli to think its exepcted schema version is\n  // greater that the version created in the manifest, which is what we are testing for.\n  const mockVersionNumber = ImportMock.mockFunction(cxschema.Manifest, 'version', semver.inc(currentSchemaVersion, 'major'));\n  try {\n    const { lock } = await execProgram(sdkProvider, config);\n    await lock.release();\n  } finally {\n    mockVersionNumber.restore();\n  }\n\n}, TEN_SECOND_TIMEOUT);\n\ntest('validates --app key is present', async () => {\n  // GIVEN no config key for `app`\n  await expect(execProgram(sdkProvider, config)).rejects.toThrow(\n    '--app is required either in command-line, in cdk.json or in ~/.cdk.json',\n  );\n\n});\n\ntest('bypasses synth when app points to a cloud assembly', async () => {\n  // GIVEN\n  config.settings.set(['app'], 'cdk.out');\n  writeOutputAssembly();\n  rewriteManifestVersionToOurs();\n\n  // WHEN\n  const { assembly: cloudAssembly, lock } = await execProgram(sdkProvider, config);\n  expect(cloudAssembly.artifacts).toEqual([]);\n  expect(cloudAssembly.directory).toEqual('cdk.out');\n\n  await lock.release();\n});\n\ntest('the application set in --app is executed', async () => {\n  // GIVEN\n  config.settings.set(['app'], 'cloud-executable');\n  mockSpawn({\n    commandLine: 'cloud-executable',\n    sideEffect: () => writeOutputAssembly(),\n  });\n\n  // WHEN\n  const { lock } = await execProgram(sdkProvider, config);\n  await lock.release();\n});\n\ntest('the application set in --app is executed as-is if it contains a filename that does not exist', async () => {\n  // GIVEN\n  config.settings.set(['app'], 'does-not-exist');\n  mockSpawn({\n    commandLine: 'does-not-exist',\n    sideEffect: () => writeOutputAssembly(),\n  });\n\n  // WHEN\n  const { lock } = await execProgram(sdkProvider, config);\n  await lock.release();\n});\n\ntest('the application set in --app is executed with arguments', async () => {\n  // GIVEN\n  config.settings.set(['app'], 'cloud-executable an-arg');\n  mockSpawn({\n    commandLine: 'cloud-executable an-arg',\n    sideEffect: () => writeOutputAssembly(),\n  });\n\n  // WHEN\n  const { lock } = await execProgram(sdkProvider, config);\n  await lock.release();\n});\n\ntest('application set in --app as `*.js` always uses handler on windows', async () => {\n  // GIVEN\n  sinon.stub(process, 'platform').value('win32');\n  config.settings.set(['app'], 'windows.js');\n  mockSpawn({\n    commandLine: process.execPath + ' windows.js',\n    sideEffect: () => writeOutputAssembly(),\n  });\n\n  // WHEN\n  const { lock } = await execProgram(sdkProvider, config);\n  await lock.release();\n});\n\ntest('application set in --app is `*.js` and executable', async () => {\n  // GIVEN\n  config.settings.set(['app'], 'executable-app.js');\n  mockSpawn({\n    commandLine: 'executable-app.js',\n    sideEffect: () => writeOutputAssembly(),\n  });\n\n  // WHEN\n  const { lock } = await execProgram(sdkProvider, config);\n  await lock.release();\n});\n\ntest('cli throws when the `build` script fails', async () => {\n  // GIVEN\n  config.settings.set(['build'], 'fake-command');\n  mockSpawn({\n    commandLine: 'fake-command',\n    exitCode: 127,\n  });\n\n  // WHEN\n  await expect(execProgram(sdkProvider, config)).rejects.toEqual(new Error('Subprocess exited with error 127'));\n}, TEN_SECOND_TIMEOUT);\n\ntest('cli does not throw when the `build` script succeeds', async () => {\n  // GIVEN\n  config.settings.set(['build'], 'real command');\n  config.settings.set(['app'], 'executable-app.js');\n  mockSpawn({\n    commandLine: 'real command', // `build` key is not split on whitespace\n    exitCode: 0,\n  },\n  {\n    commandLine: 'executable-app.js',\n    sideEffect: () => writeOutputAssembly(),\n  });\n\n  // WHEN\n  const { lock } = await execProgram(sdkProvider, config);\n  await lock.release();\n}, TEN_SECOND_TIMEOUT);\n\ntest('cli releases the outdir lock when execProgram throws', async () => {\n  // GIVEN\n  config.settings.set(['app'], 'cloud-executable');\n  mockSpawn({\n    commandLine: 'fake-command',\n    exitCode: 127,\n  });\n\n  // WHEN\n  await expect(execProgram(sdkProvider, config)).rejects.toThrow();\n\n  const output = config.settings.get(['output']);\n  expect(output).toBeDefined();\n\n  // check that the lock is released\n  const lock = await new RWLock(output).acquireWrite();\n  await lock.release();\n});\n\nfunction writeOutputAssembly() {\n  const asm = testAssembly({\n    stacks: [],\n  });\n  bockfs.write('/home/project/cdk.out/manifest.json', JSON.stringify(asm.manifest));\n  rewriteManifestVersionToOurs(bockfs.path('/home/project/cdk.out'));\n}\n\n/**\n * Rewrite the manifest schema version in the given directory to match the version number we expect (probably `0.0.0`).\n *\n * Why do we have to do this? Because `aws-cdk-lib` has its own version of `cloud-assembly-schema`,\n * which will have real version `38.0.0`, different from the `0.0.0` version of `cloud-assembly-schema` that the CLI\n * uses.\n *\n * If we don't do this, every time we load a Cloud Assembly the code will say \"Maximum schema version supported is 0.x.x, but found 30.0.0\".0\n */\nfunction rewriteManifestVersionToOurs(dir: string = 'cdk.out') {\n  rewriteManifestVersion(dir, cxschema.Manifest.version());\n}"]} |
@@ -18,5 +18,5 @@ "use strict"; | ||
expect(buildStackAssets).toBeCalledTimes(3); | ||
expect(buildStackAssets).toBeCalledWith(A); | ||
expect(buildStackAssets).toBeCalledWith(B); | ||
expect(buildStackAssets).toBeCalledWith(C); | ||
expect(buildStackAssets).toHaveBeenCalledWith(A); | ||
expect(buildStackAssets).toHaveBeenCalledWith(B); | ||
expect(buildStackAssets).toHaveBeenCalledWith(C); | ||
}); | ||
@@ -32,2 +32,2 @@ test('errors', async () => { | ||
}); | ||
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYnVpbGQudGVzdC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbImJ1aWxkLnRlc3QudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7QUFFQSx3Q0FBbUQ7QUFJbkQsUUFBUSxDQUFDLHFCQUFxQixFQUFFLEdBQUcsRUFBRTtJQUNuQyxNQUFNLENBQUMsR0FBRyxFQUFFLEVBQUUsRUFBRSxHQUFHLEVBQUUsQ0FBQztJQUN0QixNQUFNLENBQUMsR0FBRyxFQUFFLEVBQUUsRUFBRSxHQUFHLEVBQUUsQ0FBQztJQUN0QixNQUFNLENBQUMsR0FBRyxFQUFFLEVBQUUsRUFBRSxHQUFHLEVBQUUsQ0FBQztJQUN0QixNQUFNLFNBQVMsR0FBRyxDQUFDLENBQUMsRUFBRSxDQUFDLEVBQUUsQ0FBQyxDQUF1QixDQUFDO0lBRWxELE1BQU0sS0FBSyxHQUFHLEtBQUssRUFBRSxRQUFnQixFQUFFLEVBQUUsQ0FBQyxJQUFJLE9BQU8sQ0FBTyxDQUFDLE9BQU8sRUFBRSxFQUFFLENBQUMsVUFBVSxDQUFDLEdBQUcsRUFBRSxDQUFDLE9BQU8sRUFBRSxFQUFFLFFBQVEsQ0FBQyxDQUFDLENBQUM7SUFFaEgsSUFBSSxDQUFDLE9BQU8sRUFBRSxLQUFLLElBQUksRUFBRTtRQUN2QixRQUFRO1FBQ1IsTUFBTSxnQkFBZ0IsR0FBRyxJQUFJLENBQUMsRUFBRSxDQUFDLEdBQUcsRUFBRSxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDO1FBRWpELFlBQVk7UUFDWixNQUFNLE1BQU0sQ0FBQyxJQUFBLDJCQUFtQixFQUFDLFNBQVMsRUFBRSxFQUFFLGdCQUFnQixFQUFFLENBQUMsQ0FBQzthQUMvRCxRQUFRO2FBQ1IsYUFBYSxFQUFFLENBQUM7UUFFbkIsTUFBTSxDQUFDLGdCQUFnQixDQUFDLENBQUMsZUFBZSxDQUFDLENBQUMsQ0FBQyxDQUFDO1FBQzVDLE1BQU0sQ0FBQyxnQkFBZ0IsQ0FBQyxDQUFDLGNBQWMsQ0FBQyxDQUFDLENBQUMsQ0FBQztRQUMzQyxNQUFNLENBQUMsZ0JBQWdCLENBQUMsQ0FBQyxjQUFjLENBQUMsQ0FBQyxDQUFDLENBQUM7UUFDM0MsTUFBTSxDQUFDLGdCQUFnQixDQUFDLENBQUMsY0FBYyxDQUFDLENBQUMsQ0FBQyxDQUFDO0lBQzdDLENBQUMsQ0FBQyxDQUFDO0lBRUgsSUFBSSxDQUFDLFFBQVEsRUFBRSxLQUFLLElBQUksRUFBRTtRQUN4QixRQUFRO1FBQ1IsTUFBTSxnQkFBZ0IsR0FBRyxLQUFLLElBQUksRUFBRSxHQUFHLE1BQU0sSUFBSSxLQUFLLENBQUMsU0FBUyxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUM7UUFFckUsWUFBWTtRQUNaLE1BQU0sTUFBTSxDQUFDLElBQUEsMkJBQW1CLEVBQUMsU0FBUyxFQUFFLEVBQUUsZ0JBQWdCLEVBQUUsQ0FBQyxDQUFDO2FBQy9ELE9BQU87YUFDUCxPQUFPLENBQUMsd0VBQXdFLENBQUMsQ0FBQztJQUN2RixDQUFDLENBQUMsQ0FBQztBQUNMLENBQUMsQ0FBQyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyogZXNsaW50LWRpc2FibGUgaW1wb3J0L29yZGVyICovXG5pbXBvcnQgKiBhcyBjeGFwaSBmcm9tICdAYXdzLWNkay9jeC1hcGknO1xuaW1wb3J0IHsgYnVpbGRBbGxTdGFja0Fzc2V0cyB9IGZyb20gJy4uL2xpYi9idWlsZCc7XG5cbnR5cGUgU3RhY2sgPSBjeGFwaS5DbG91ZEZvcm1hdGlvblN0YWNrQXJ0aWZhY3Q7XG5cbmRlc2NyaWJlKCdidWlsZEFsbFN0YWNrQXNzZXRzJywgKCkgPT4ge1xuICBjb25zdCBBID0geyBpZDogJ0EnIH07XG4gIGNvbnN0IEIgPSB7IGlkOiAnQicgfTtcbiAgY29uc3QgQyA9IHsgaWQ6ICdDJyB9O1xuICBjb25zdCB0b1B1Ymxpc2ggPSBbQSwgQiwgQ10gYXMgdW5rbm93biBhcyBTdGFja1tdO1xuXG4gIGNvbnN0IHNsZWVwID0gYXN5bmMgKGR1cmF0aW9uOiBudW1iZXIpID0+IG5ldyBQcm9taXNlPHZvaWQ+KChyZXNvbHZlKSA9PiBzZXRUaW1lb3V0KCgpID0+IHJlc29sdmUoKSwgZHVyYXRpb24pKTtcblxuICB0ZXN0KCdidWlsZCcsIGFzeW5jICgpID0+IHtcbiAgICAvLyBHSVZFTlxuICAgIGNvbnN0IGJ1aWxkU3RhY2tBc3NldHMgPSBqZXN0LmZuKCgpID0+IHNsZWVwKDEpKTtcblxuICAgIC8vIFdIRU4vVEhFTlxuICAgIGF3YWl0IGV4cGVjdChidWlsZEFsbFN0YWNrQXNzZXRzKHRvUHVibGlzaCwgeyBidWlsZFN0YWNrQXNzZXRzIH0pKVxuICAgICAgLnJlc29sdmVzXG4gICAgICAudG9CZVVuZGVmaW5lZCgpO1xuXG4gICAgZXhwZWN0KGJ1aWxkU3RhY2tBc3NldHMpLnRvQmVDYWxsZWRUaW1lcygzKTtcbiAgICBleHBlY3QoYnVpbGRTdGFja0Fzc2V0cykudG9CZUNhbGxlZFdpdGgoQSk7XG4gICAgZXhwZWN0KGJ1aWxkU3RhY2tBc3NldHMpLnRvQmVDYWxsZWRXaXRoKEIpO1xuICAgIGV4cGVjdChidWlsZFN0YWNrQXNzZXRzKS50b0JlQ2FsbGVkV2l0aChDKTtcbiAgfSk7XG5cbiAgdGVzdCgnZXJyb3JzJywgYXN5bmMgKCkgPT4ge1xuICAgIC8vIEdJVkVOXG4gICAgY29uc3QgYnVpbGRTdGFja0Fzc2V0cyA9IGFzeW5jICgpID0+IHsgdGhyb3cgbmV3IEVycm9yKCdNZXNzYWdlJyk7IH07XG5cbiAgICAvLyBXSEVOL1RIRU5cbiAgICBhd2FpdCBleHBlY3QoYnVpbGRBbGxTdGFja0Fzc2V0cyh0b1B1Ymxpc2gsIHsgYnVpbGRTdGFja0Fzc2V0cyB9KSlcbiAgICAgIC5yZWplY3RzXG4gICAgICAudG9UaHJvdygnQnVpbGRpbmcgQXNzZXRzIEZhaWxlZDogRXJyb3I6IE1lc3NhZ2UsIEVycm9yOiBNZXNzYWdlLCBFcnJvcjogTWVzc2FnZScpO1xuICB9KTtcbn0pO1xuIl19 | ||
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYnVpbGQudGVzdC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbImJ1aWxkLnRlc3QudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7QUFFQSx3Q0FBbUQ7QUFJbkQsUUFBUSxDQUFDLHFCQUFxQixFQUFFLEdBQUcsRUFBRTtJQUNuQyxNQUFNLENBQUMsR0FBRyxFQUFFLEVBQUUsRUFBRSxHQUFHLEVBQUUsQ0FBQztJQUN0QixNQUFNLENBQUMsR0FBRyxFQUFFLEVBQUUsRUFBRSxHQUFHLEVBQUUsQ0FBQztJQUN0QixNQUFNLENBQUMsR0FBRyxFQUFFLEVBQUUsRUFBRSxHQUFHLEVBQUUsQ0FBQztJQUN0QixNQUFNLFNBQVMsR0FBRyxDQUFDLENBQUMsRUFBRSxDQUFDLEVBQUUsQ0FBQyxDQUF1QixDQUFDO0lBRWxELE1BQU0sS0FBSyxHQUFHLEtBQUssRUFBRSxRQUFnQixFQUFFLEVBQUUsQ0FBQyxJQUFJLE9BQU8sQ0FBTyxDQUFDLE9BQU8sRUFBRSxFQUFFLENBQUMsVUFBVSxDQUFDLEdBQUcsRUFBRSxDQUFDLE9BQU8sRUFBRSxFQUFFLFFBQVEsQ0FBQyxDQUFDLENBQUM7SUFFaEgsSUFBSSxDQUFDLE9BQU8sRUFBRSxLQUFLLElBQUksRUFBRTtRQUN2QixRQUFRO1FBQ1IsTUFBTSxnQkFBZ0IsR0FBRyxJQUFJLENBQUMsRUFBRSxDQUFDLEdBQUcsRUFBRSxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDO1FBRWpELFlBQVk7UUFDWixNQUFNLE1BQU0sQ0FBQyxJQUFBLDJCQUFtQixFQUFDLFNBQVMsRUFBRSxFQUFFLGdCQUFnQixFQUFFLENBQUMsQ0FBQzthQUMvRCxRQUFRO2FBQ1IsYUFBYSxFQUFFLENBQUM7UUFFbkIsTUFBTSxDQUFDLGdCQUFnQixDQUFDLENBQUMsZUFBZSxDQUFDLENBQUMsQ0FBQyxDQUFDO1FBQzVDLE1BQU0sQ0FBQyxnQkFBZ0IsQ0FBQyxDQUFDLG9CQUFvQixDQUFDLENBQUMsQ0FBQyxDQUFDO1FBQ2pELE1BQU0sQ0FBQyxnQkFBZ0IsQ0FBQyxDQUFDLG9CQUFvQixDQUFDLENBQUMsQ0FBQyxDQUFDO1FBQ2pELE1BQU0sQ0FBQyxnQkFBZ0IsQ0FBQyxDQUFDLG9CQUFvQixDQUFDLENBQUMsQ0FBQyxDQUFDO0lBQ25ELENBQUMsQ0FBQyxDQUFDO0lBRUgsSUFBSSxDQUFDLFFBQVEsRUFBRSxLQUFLLElBQUksRUFBRTtRQUN4QixRQUFRO1FBQ1IsTUFBTSxnQkFBZ0IsR0FBRyxLQUFLLElBQUksRUFBRSxHQUFHLE1BQU0sSUFBSSxLQUFLLENBQUMsU0FBUyxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUM7UUFFckUsWUFBWTtRQUNaLE1BQU0sTUFBTSxDQUFDLElBQUEsMkJBQW1CLEVBQUMsU0FBUyxFQUFFLEVBQUUsZ0JBQWdCLEVBQUUsQ0FBQyxDQUFDO2FBQy9ELE9BQU87YUFDUCxPQUFPLENBQUMsd0VBQXdFLENBQUMsQ0FBQztJQUN2RixDQUFDLENBQUMsQ0FBQztBQUNMLENBQUMsQ0FBQyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyogZXNsaW50LWRpc2FibGUgaW1wb3J0L29yZGVyICovXG5pbXBvcnQgKiBhcyBjeGFwaSBmcm9tICdAYXdzLWNkay9jeC1hcGknO1xuaW1wb3J0IHsgYnVpbGRBbGxTdGFja0Fzc2V0cyB9IGZyb20gJy4uL2xpYi9idWlsZCc7XG5cbnR5cGUgU3RhY2sgPSBjeGFwaS5DbG91ZEZvcm1hdGlvblN0YWNrQXJ0aWZhY3Q7XG5cbmRlc2NyaWJlKCdidWlsZEFsbFN0YWNrQXNzZXRzJywgKCkgPT4ge1xuICBjb25zdCBBID0geyBpZDogJ0EnIH07XG4gIGNvbnN0IEIgPSB7IGlkOiAnQicgfTtcbiAgY29uc3QgQyA9IHsgaWQ6ICdDJyB9O1xuICBjb25zdCB0b1B1Ymxpc2ggPSBbQSwgQiwgQ10gYXMgdW5rbm93biBhcyBTdGFja1tdO1xuXG4gIGNvbnN0IHNsZWVwID0gYXN5bmMgKGR1cmF0aW9uOiBudW1iZXIpID0+IG5ldyBQcm9taXNlPHZvaWQ+KChyZXNvbHZlKSA9PiBzZXRUaW1lb3V0KCgpID0+IHJlc29sdmUoKSwgZHVyYXRpb24pKTtcblxuICB0ZXN0KCdidWlsZCcsIGFzeW5jICgpID0+IHtcbiAgICAvLyBHSVZFTlxuICAgIGNvbnN0IGJ1aWxkU3RhY2tBc3NldHMgPSBqZXN0LmZuKCgpID0+IHNsZWVwKDEpKTtcblxuICAgIC8vIFdIRU4vVEhFTlxuICAgIGF3YWl0IGV4cGVjdChidWlsZEFsbFN0YWNrQXNzZXRzKHRvUHVibGlzaCwgeyBidWlsZFN0YWNrQXNzZXRzIH0pKVxuICAgICAgLnJlc29sdmVzXG4gICAgICAudG9CZVVuZGVmaW5lZCgpO1xuXG4gICAgZXhwZWN0KGJ1aWxkU3RhY2tBc3NldHMpLnRvQmVDYWxsZWRUaW1lcygzKTtcbiAgICBleHBlY3QoYnVpbGRTdGFja0Fzc2V0cykudG9IYXZlQmVlbkNhbGxlZFdpdGgoQSk7XG4gICAgZXhwZWN0KGJ1aWxkU3RhY2tBc3NldHMpLnRvSGF2ZUJlZW5DYWxsZWRXaXRoKEIpO1xuICAgIGV4cGVjdChidWlsZFN0YWNrQXNzZXRzKS50b0hhdmVCZWVuQ2FsbGVkV2l0aChDKTtcbiAgfSk7XG5cbiAgdGVzdCgnZXJyb3JzJywgYXN5bmMgKCkgPT4ge1xuICAgIC8vIEdJVkVOXG4gICAgY29uc3QgYnVpbGRTdGFja0Fzc2V0cyA9IGFzeW5jICgpID0+IHsgdGhyb3cgbmV3IEVycm9yKCdNZXNzYWdlJyk7IH07XG5cbiAgICAvLyBXSEVOL1RIRU5cbiAgICBhd2FpdCBleHBlY3QoYnVpbGRBbGxTdGFja0Fzc2V0cyh0b1B1Ymxpc2gsIHsgYnVpbGRTdGFja0Fzc2V0cyB9KSlcbiAgICAgIC5yZWplY3RzXG4gICAgICAudG9UaHJvdygnQnVpbGRpbmcgQXNzZXRzIEZhaWxlZDogRXJyb3I6IE1lc3NhZ2UsIEVycm9yOiBNZXNzYWdlLCBFcnJvcjogTWVzc2FnZScpO1xuICB9KTtcbn0pO1xuIl19 |
@@ -28,2 +28,3 @@ import { type AssetManifest, type AssetMetadataEntry, type AwsCloudFormationStackProperties, type MissingContext } from '@aws-cdk/cloud-assembly-schema'; | ||
nestedAssemblies?: TestAssembly[]; | ||
schemaVersion?: string; | ||
} | ||
@@ -30,0 +31,0 @@ export declare class MockCloudExecutable extends CloudExecutable { |
@@ -16,3 +16,5 @@ "use strict"; | ||
const settings_1 = require("../lib/settings"); | ||
const assembly_versions_1 = require("./api/assembly-versions"); | ||
exports.DEFAULT_FAKE_TEMPLATE = { No: 'Resources' }; | ||
const SOME_RECENT_SCHEMA_VERSION = '30.0.0'; | ||
class MockCloudExecutable extends cloud_executable_1.CloudExecutable { | ||
@@ -104,3 +106,4 @@ constructor(assembly, sdkProviderArg) { | ||
} | ||
return builder.buildAssembly(); | ||
const asm = builder.buildAssembly(); | ||
return (0, assembly_versions_1.cxapiAssemblyWithForcedVersion)(asm, assembly.schemaVersion ?? SOME_RECENT_SCHEMA_VERSION); | ||
} | ||
@@ -176,2 +179,2 @@ /** | ||
} | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"util.js","sourceRoot":"","sources":["util.ts"],"names":[],"mappings":"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","sourcesContent":["import * as fs from 'fs';\nimport * as path from 'path';\nimport { ArtifactMetadataEntryType, ArtifactType, type AssetManifest, type AssetMetadataEntry, type AwsCloudFormationStackProperties, type MetadataEntry, type MissingContext } from '@aws-cdk/cloud-assembly-schema';\nimport { type CloudAssembly, CloudAssemblyBuilder, type CloudFormationStackArtifact, type StackMetadata } from '@aws-cdk/cx-api';\nimport { MockSdkProvider } from './util/mock-sdk';\nimport { CloudExecutable } from '../lib/api/cxapp/cloud-executable';\nimport { Configuration } from '../lib/settings';\n\nexport const DEFAULT_FAKE_TEMPLATE = { No: 'Resources' };\n\nexport interface TestStackArtifact {\n  stackName: string;\n  template?: any;\n  env?: string;\n  depends?: string[];\n  metadata?: StackMetadata;\n  notificationArns?: string[];\n\n  /** Old-style assets */\n  assets?: AssetMetadataEntry[];\n  properties?: Partial<AwsCloudFormationStackProperties>;\n  terminationProtection?: boolean;\n  displayName?: string;\n\n  /** New-style assets */\n  assetManifest?: AssetManifest;\n}\n\nexport interface TestAssembly {\n  stacks: TestStackArtifact[];\n  missing?: MissingContext[];\n  nestedAssemblies?: TestAssembly[];\n}\n\nexport class MockCloudExecutable extends CloudExecutable {\n  public readonly configuration: Configuration;\n  public readonly sdkProvider: MockSdkProvider;\n\n  constructor(assembly: TestAssembly, sdkProviderArg?: MockSdkProvider) {\n    const configuration = new Configuration();\n    const sdkProvider = sdkProviderArg ?? new MockSdkProvider();\n\n    super({\n      configuration,\n      sdkProvider,\n      synthesizer: () => Promise.resolve(testAssembly(assembly)),\n    });\n\n    this.configuration = configuration;\n    this.sdkProvider = sdkProvider;\n  }\n}\n\nfunction clone(obj: any) {\n  return JSON.parse(JSON.stringify(obj));\n}\n\nfunction addAttributes(assembly: TestAssembly, builder: CloudAssemblyBuilder) {\n  for (const stack of assembly.stacks) {\n    const templateFile = `${stack.stackName}.template.json`;\n    const template = stack.template ?? DEFAULT_FAKE_TEMPLATE;\n    fs.writeFileSync(path.join(builder.outdir, templateFile), JSON.stringify(template, undefined, 2));\n    addNestedStacks(templateFile, builder.outdir, template);\n\n    // we call patchStackTags here to simulate the tags formatter\n    // that is used when building real manifest files.\n    const metadata: { [path: string]: MetadataEntry[] } = patchStackTags({ ...stack.metadata });\n    for (const asset of stack.assets || []) {\n      metadata[asset.id] = [{ type: ArtifactMetadataEntryType.ASSET, data: asset }];\n    }\n\n    for (const missing of assembly.missing || []) {\n      builder.addMissing(missing);\n    }\n\n    const dependencies = [...(stack.depends ?? [])];\n\n    if (stack.assetManifest) {\n      const manifestFile = `${stack.stackName}.assets.json`;\n      fs.writeFileSync(path.join(builder.outdir, manifestFile), JSON.stringify(stack.assetManifest, undefined, 2));\n      dependencies.push(`${stack.stackName}.assets`);\n      builder.addArtifact(`${stack.stackName}.assets`, {\n        type: ArtifactType.ASSET_MANIFEST,\n        environment: stack.env || 'aws://123456789012/here',\n        properties: {\n          file: manifestFile,\n        },\n      });\n    }\n\n    builder.addArtifact(stack.stackName, {\n      type: ArtifactType.AWS_CLOUDFORMATION_STACK,\n      environment: stack.env || 'aws://123456789012/here',\n\n      dependencies,\n      metadata,\n      properties: {\n        ...stack.properties,\n        templateFile,\n        terminationProtection: stack.terminationProtection,\n        notificationArns: stack.notificationArns,\n      },\n      displayName: stack.displayName,\n    });\n  }\n}\n\nfunction addNestedStacks(templatePath: string, outdir: string, rootStackTemplate?: any) {\n  let template = rootStackTemplate;\n\n  if (!template) {\n    const templatePathWithDir = path.join('nested-stack-templates', templatePath);\n    template = JSON.parse(fs.readFileSync(path.join(__dirname, templatePathWithDir)).toString());\n    fs.writeFileSync(path.join(outdir, templatePath), JSON.stringify(template, undefined, 2));\n  }\n\n  for (const logicalId in template.Resources) {\n    if (template.Resources[logicalId].Type === 'AWS::CloudFormation::Stack') {\n      if (template.Resources[logicalId].Metadata && template.Resources[logicalId].Metadata['aws:asset:path']) {\n        const nestedTemplatePath = template.Resources[logicalId].Metadata['aws:asset:path'];\n        addNestedStacks(nestedTemplatePath, outdir);\n      }\n    }\n  }\n}\n\nexport function testAssembly(assembly: TestAssembly): CloudAssembly {\n  const builder = new CloudAssemblyBuilder();\n  addAttributes(assembly, builder);\n\n  if (assembly.nestedAssemblies != null && assembly.nestedAssemblies.length > 0) {\n    assembly.nestedAssemblies?.forEach((nestedAssembly: TestAssembly, i: number) => {\n      const nestedAssemblyBuilder = builder.createNestedAssembly(`nested${i}`, `nested${i}`);\n      addAttributes(nestedAssembly, nestedAssemblyBuilder);\n      nestedAssemblyBuilder.buildAssembly();\n    });\n  }\n\n  return builder.buildAssembly();\n}\n\n/**\n * Transform stack tags from how they are decalred in source code (lower cased)\n * to how they are stored on disk (upper cased). In real synthesis this is done\n * by a special tags formatter.\n *\n * @see aws-cdk-lib/lib/stack.ts\n */\nfunction patchStackTags(metadata: { [path: string]: MetadataEntry[] }): {\n  [path: string]: MetadataEntry[];\n} {\n  const cloned = clone(metadata) as { [path: string]: MetadataEntry[] };\n\n  for (const metadataEntries of Object.values(cloned)) {\n    for (const metadataEntry of metadataEntries) {\n      if (metadataEntry.type === ArtifactMetadataEntryType.STACK_TAGS && metadataEntry.data) {\n        const metadataAny = metadataEntry as any;\n\n        metadataAny.data = metadataAny.data.map((t: any) => {\n          return { Key: t.key, Value: t.value };\n        });\n      }\n    }\n  }\n  return cloned;\n}\n\nexport function testStack(stack: TestStackArtifact): CloudFormationStackArtifact {\n  const assembly = testAssembly({ stacks: [stack] });\n  return assembly.getStackByName(stack.stackName);\n}\n\n/**\n * Return a mocked instance of a class, given its constructor\n *\n * I don't understand why jest doesn't provide this by default,\n * but there you go.\n *\n * FIXME: Currently very limited. Doesn't support inheritance, getters or\n * automatic detection of properties (as those exist on instances, not\n * classes).\n */\nexport function instanceMockFrom<A>(ctr: new (...args: any[]) => A): jest.Mocked<A> {\n  const ret: any = {};\n  for (const methodName of Object.getOwnPropertyNames(ctr.prototype)) {\n    ret[methodName] = jest.fn();\n  }\n  return ret;\n}\n\nexport function withMocked<A extends object, K extends keyof A, B>(\n  obj: A,\n  key: K,\n  block: (fn: jest.Mocked<A>[K]) => B,\n): B {\n  const original = obj[key];\n  const mockFn = jest.fn();\n  (obj as any)[key] = mockFn;\n\n  let asyncFinally: boolean = false;\n  try {\n    const ret = block(mockFn as any);\n    if (!isPromise(ret)) {\n      return ret;\n    }\n\n    asyncFinally = true;\n    return ret.finally(() => {\n      obj[key] = original;\n    }) as any;\n  } finally {\n    if (!asyncFinally) {\n      obj[key] = original;\n    }\n  }\n}\n\nfunction isPromise<A>(object: any): object is Promise<A> {\n  return Promise.resolve(object) === object;\n}\n\nexport async function sleep(ms: number) {\n  return new Promise((ok) => setTimeout(ok, ms));\n}\n"]} | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"util.js","sourceRoot":"","sources":["util.ts"],"names":[],"mappings":";;;AAkIA,oCAcC;AA4BD,8BAGC;AAYD,4CAMC;AAED,gCAyBC;AAMD,sBAEC;AApOD,yBAAyB;AACzB,6BAA6B;AAC7B,0EAAsN;AACtN,4CAAiI;AACjI,8CAAkD;AAClD,wEAAoE;AACpE,8CAAgD;AAChD,+DAAyE;AAE5D,QAAA,qBAAqB,GAAG,EAAE,EAAE,EAAE,WAAW,EAAE,CAAC;AAEzD,MAAM,0BAA0B,GAAG,QAAQ,CAAC;AA2B5C,MAAa,mBAAoB,SAAQ,kCAAe;IAItD,YAAY,QAAsB,EAAE,cAAgC;QAClE,MAAM,aAAa,GAAG,IAAI,wBAAa,EAAE,CAAC;QAC1C,MAAM,WAAW,GAAG,cAAc,IAAI,IAAI,0BAAe,EAAE,CAAC;QAE5D,KAAK,CAAC;YACJ,aAAa;YACb,WAAW;YACX,WAAW,EAAE,GAAG,EAAE,CAAC,OAAO,CAAC,OAAO,CAAC,YAAY,CAAC,QAAQ,CAAC,CAAC;SAC3D,CAAC,CAAC;QAEH,IAAI,CAAC,aAAa,GAAG,aAAa,CAAC;QACnC,IAAI,CAAC,WAAW,GAAG,WAAW,CAAC;IACjC,CAAC;CACF;AAjBD,kDAiBC;AAED,SAAS,KAAK,CAAC,GAAQ;IACrB,OAAO,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC,CAAC;AACzC,CAAC;AAED,SAAS,aAAa,CAAC,QAAsB,EAAE,OAA6B;IAC1E,KAAK,MAAM,KAAK,IAAI,QAAQ,CAAC,MAAM,EAAE,CAAC;QACpC,MAAM,YAAY,GAAG,GAAG,KAAK,CAAC,SAAS,gBAAgB,CAAC;QACxD,MAAM,QAAQ,GAAG,KAAK,CAAC,QAAQ,IAAI,6BAAqB,CAAC;QACzD,EAAE,CAAC,aAAa,CAAC,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,MAAM,EAAE,YAAY,CAAC,EAAE,IAAI,CAAC,SAAS,CAAC,QAAQ,EAAE,SAAS,EAAE,CAAC,CAAC,CAAC,CAAC;QAClG,eAAe,CAAC,YAAY,EAAE,OAAO,CAAC,MAAM,EAAE,QAAQ,CAAC,CAAC;QAExD,6DAA6D;QAC7D,kDAAkD;QAClD,MAAM,QAAQ,GAAwC,cAAc,CAAC,EAAE,GAAG,KAAK,CAAC,QAAQ,EAAE,CAAC,CAAC;QAC5F,KAAK,MAAM,KAAK,IAAI,KAAK,CAAC,MAAM,IAAI,EAAE,EAAE,CAAC;YACvC,QAAQ,CAAC,KAAK,CAAC,EAAE,CAAC,GAAG,CAAC,EAAE,IAAI,EAAE,iDAAyB,CAAC,KAAK,EAAE,IAAI,EAAE,KAAK,EAAE,CAAC,CAAC;QAChF,CAAC;QAED,KAAK,MAAM,OAAO,IAAI,QAAQ,CAAC,OAAO,IAAI,EAAE,EAAE,CAAC;YAC7C,OAAO,CAAC,UAAU,CAAC,OAAO,CAAC,CAAC;QAC9B,CAAC;QAED,MAAM,YAAY,GAAG,CAAC,GAAG,CAAC,KAAK,CAAC,OAAO,IAAI,EAAE,CAAC,CAAC,CAAC;QAEhD,IAAI,KAAK,CAAC,aAAa,EAAE,CAAC;YACxB,MAAM,YAAY,GAAG,GAAG,KAAK,CAAC,SAAS,cAAc,CAAC;YACtD,EAAE,CAAC,aAAa,CAAC,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,MAAM,EAAE,YAAY,CAAC,EAAE,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,aAAa,EAAE,SAAS,EAAE,CAAC,CAAC,CAAC,CAAC;YAC7G,YAAY,CAAC,IAAI,CAAC,GAAG,KAAK,CAAC,SAAS,SAAS,CAAC,CAAC;YAC/C,OAAO,CAAC,WAAW,CAAC,GAAG,KAAK,CAAC,SAAS,SAAS,EAAE;gBAC/C,IAAI,EAAE,oCAAY,CAAC,cAAc;gBACjC,WAAW,EAAE,KAAK,CAAC,GAAG,IAAI,yBAAyB;gBACnD,UAAU,EAAE;oBACV,IAAI,EAAE,YAAY;iBACnB;aACF,CAAC,CAAC;QACL,CAAC;QAED,OAAO,CAAC,WAAW,CAAC,KAAK,CAAC,SAAS,EAAE;YACnC,IAAI,EAAE,oCAAY,CAAC,wBAAwB;YAC3C,WAAW,EAAE,KAAK,CAAC,GAAG,IAAI,yBAAyB;YAEnD,YAAY;YACZ,QAAQ;YACR,UAAU,EAAE;gBACV,GAAG,KAAK,CAAC,UAAU;gBACnB,YAAY;gBACZ,qBAAqB,EAAE,KAAK,CAAC,qBAAqB;gBAClD,gBAAgB,EAAE,KAAK,CAAC,gBAAgB;aACzC;YACD,WAAW,EAAE,KAAK,CAAC,WAAW;SAC/B,CAAC,CAAC;IACL,CAAC;AACH,CAAC;AAED,SAAS,eAAe,CAAC,YAAoB,EAAE,MAAc,EAAE,iBAAuB;IACpF,IAAI,QAAQ,GAAG,iBAAiB,CAAC;IAEjC,IAAI,CAAC,QAAQ,EAAE,CAAC;QACd,MAAM,mBAAmB,GAAG,IAAI,CAAC,IAAI,CAAC,wBAAwB,EAAE,YAAY,CAAC,CAAC;QAC9E,QAAQ,GAAG,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,mBAAmB,CAAC,CAAC,CAAC,QAAQ,EAAE,CAAC,CAAC;QAC7F,EAAE,CAAC,aAAa,CAAC,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE,YAAY,CAAC,EAAE,IAAI,CAAC,SAAS,CAAC,QAAQ,EAAE,SAAS,EAAE,CAAC,CAAC,CAAC,CAAC;IAC5F,CAAC;IAED,KAAK,MAAM,SAAS,IAAI,QAAQ,CAAC,SAAS,EAAE,CAAC;QAC3C,IAAI,QAAQ,CAAC,SAAS,CAAC,SAAS,CAAC,CAAC,IAAI,KAAK,4BAA4B,EAAE,CAAC;YACxE,IAAI,QAAQ,CAAC,SAAS,CAAC,SAAS,CAAC,CAAC,QAAQ,IAAI,QAAQ,CAAC,SAAS,CAAC,SAAS,CAAC,CAAC,QAAQ,CAAC,gBAAgB,CAAC,EAAE,CAAC;gBACvG,MAAM,kBAAkB,GAAG,QAAQ,CAAC,SAAS,CAAC,SAAS,CAAC,CAAC,QAAQ,CAAC,gBAAgB,CAAC,CAAC;gBACpF,eAAe,CAAC,kBAAkB,EAAE,MAAM,CAAC,CAAC;YAC9C,CAAC;QACH,CAAC;IACH,CAAC;AACH,CAAC;AAED,SAAgB,YAAY,CAAC,QAAsB;IACjD,MAAM,OAAO,GAAG,IAAI,6BAAoB,EAAE,CAAC;IAC3C,aAAa,CAAC,QAAQ,EAAE,OAAO,CAAC,CAAC;IAEjC,IAAI,QAAQ,CAAC,gBAAgB,IAAI,IAAI,IAAI,QAAQ,CAAC,gBAAgB,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;QAC9E,QAAQ,CAAC,gBAAgB,EAAE,OAAO,CAAC,CAAC,cAA4B,EAAE,CAAS,EAAE,EAAE;YAC7E,MAAM,qBAAqB,GAAG,OAAO,CAAC,oBAAoB,CAAC,SAAS,CAAC,EAAE,EAAE,SAAS,CAAC,EAAE,CAAC,CAAC;YACvF,aAAa,CAAC,cAAc,EAAE,qBAAqB,CAAC,CAAC;YACrD,qBAAqB,CAAC,aAAa,EAAE,CAAC;QACxC,CAAC,CAAC,CAAC;IACL,CAAC;IAED,MAAM,GAAG,GAAG,OAAO,CAAC,aAAa,EAAE,CAAC;IACpC,OAAO,IAAA,kDAA8B,EAAC,GAAG,EAAE,QAAQ,CAAC,aAAa,IAAI,0BAA0B,CAAC,CAAC;AACnG,CAAC;AAED;;;;;;GAMG;AACH,SAAS,cAAc,CAAC,QAA6C;IAGnE,MAAM,MAAM,GAAG,KAAK,CAAC,QAAQ,CAAwC,CAAC;IAEtE,KAAK,MAAM,eAAe,IAAI,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,EAAE,CAAC;QACpD,KAAK,MAAM,aAAa,IAAI,eAAe,EAAE,CAAC;YAC5C,IAAI,aAAa,CAAC,IAAI,KAAK,iDAAyB,CAAC,UAAU,IAAI,aAAa,CAAC,IAAI,EAAE,CAAC;gBACtF,MAAM,WAAW,GAAG,aAAoB,CAAC;gBAEzC,WAAW,CAAC,IAAI,GAAG,WAAW,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,CAAM,EAAE,EAAE;oBACjD,OAAO,EAAE,GAAG,EAAE,CAAC,CAAC,GAAG,EAAE,KAAK,EAAE,CAAC,CAAC,KAAK,EAAE,CAAC;gBACxC,CAAC,CAAC,CAAC;YACL,CAAC;QACH,CAAC;IACH,CAAC;IACD,OAAO,MAAM,CAAC;AAChB,CAAC;AAED,SAAgB,SAAS,CAAC,KAAwB;IAChD,MAAM,QAAQ,GAAG,YAAY,CAAC,EAAE,MAAM,EAAE,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;IACnD,OAAO,QAAQ,CAAC,cAAc,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;AAClD,CAAC;AAED;;;;;;;;;GASG;AACH,SAAgB,gBAAgB,CAAI,GAA8B;IAChE,MAAM,GAAG,GAAQ,EAAE,CAAC;IACpB,KAAK,MAAM,UAAU,IAAI,MAAM,CAAC,mBAAmB,CAAC,GAAG,CAAC,SAAS,CAAC,EAAE,CAAC;QACnE,GAAG,CAAC,UAAU,CAAC,GAAG,IAAI,CAAC,EAAE,EAAE,CAAC;IAC9B,CAAC;IACD,OAAO,GAAG,CAAC;AACb,CAAC;AAED,SAAgB,UAAU,CACxB,GAAM,EACN,GAAM,EACN,KAAmC;IAEnC,MAAM,QAAQ,GAAG,GAAG,CAAC,GAAG,CAAC,CAAC;IAC1B,MAAM,MAAM,GAAG,IAAI,CAAC,EAAE,EAAE,CAAC;IACxB,GAAW,CAAC,GAAG,CAAC,GAAG,MAAM,CAAC;IAE3B,IAAI,YAAY,GAAY,KAAK,CAAC;IAClC,IAAI,CAAC;QACH,MAAM,GAAG,GAAG,KAAK,CAAC,MAAa,CAAC,CAAC;QACjC,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,EAAE,CAAC;YACpB,OAAO,GAAG,CAAC;QACb,CAAC;QAED,YAAY,GAAG,IAAI,CAAC;QACpB,OAAO,GAAG,CAAC,OAAO,CAAC,GAAG,EAAE;YACtB,GAAG,CAAC,GAAG,CAAC,GAAG,QAAQ,CAAC;QACtB,CAAC,CAAQ,CAAC;IACZ,CAAC;YAAS,CAAC;QACT,IAAI,CAAC,YAAY,EAAE,CAAC;YAClB,GAAG,CAAC,GAAG,CAAC,GAAG,QAAQ,CAAC;QACtB,CAAC;IACH,CAAC;AACH,CAAC;AAED,SAAS,SAAS,CAAI,MAAW;IAC/B,OAAO,OAAO,CAAC,OAAO,CAAC,MAAM,CAAC,KAAK,MAAM,CAAC;AAC5C,CAAC;AAEM,KAAK,UAAU,KAAK,CAAC,EAAU;IACpC,OAAO,IAAI,OAAO,CAAC,CAAC,EAAE,EAAE,EAAE,CAAC,UAAU,CAAC,EAAE,EAAE,EAAE,CAAC,CAAC,CAAC;AACjD,CAAC","sourcesContent":["import * as fs from 'fs';\nimport * as path from 'path';\nimport { ArtifactMetadataEntryType, ArtifactType, type AssetManifest, type AssetMetadataEntry, type AwsCloudFormationStackProperties, type MetadataEntry, type MissingContext } from '@aws-cdk/cloud-assembly-schema';\nimport { type CloudAssembly, CloudAssemblyBuilder, type CloudFormationStackArtifact, type StackMetadata } from '@aws-cdk/cx-api';\nimport { MockSdkProvider } from './util/mock-sdk';\nimport { CloudExecutable } from '../lib/api/cxapp/cloud-executable';\nimport { Configuration } from '../lib/settings';\nimport { cxapiAssemblyWithForcedVersion } from './api/assembly-versions';\n\nexport const DEFAULT_FAKE_TEMPLATE = { No: 'Resources' };\n\nconst SOME_RECENT_SCHEMA_VERSION = '30.0.0';\n\nexport interface TestStackArtifact {\n  stackName: string;\n  template?: any;\n  env?: string;\n  depends?: string[];\n  metadata?: StackMetadata;\n  notificationArns?: string[];\n\n  /** Old-style assets */\n  assets?: AssetMetadataEntry[];\n  properties?: Partial<AwsCloudFormationStackProperties>;\n  terminationProtection?: boolean;\n  displayName?: string;\n\n  /** New-style assets */\n  assetManifest?: AssetManifest;\n}\n\nexport interface TestAssembly {\n  stacks: TestStackArtifact[];\n  missing?: MissingContext[];\n  nestedAssemblies?: TestAssembly[];\n  schemaVersion?: string;\n}\n\nexport class MockCloudExecutable extends CloudExecutable {\n  public readonly configuration: Configuration;\n  public readonly sdkProvider: MockSdkProvider;\n\n  constructor(assembly: TestAssembly, sdkProviderArg?: MockSdkProvider) {\n    const configuration = new Configuration();\n    const sdkProvider = sdkProviderArg ?? new MockSdkProvider();\n\n    super({\n      configuration,\n      sdkProvider,\n      synthesizer: () => Promise.resolve(testAssembly(assembly)),\n    });\n\n    this.configuration = configuration;\n    this.sdkProvider = sdkProvider;\n  }\n}\n\nfunction clone(obj: any) {\n  return JSON.parse(JSON.stringify(obj));\n}\n\nfunction addAttributes(assembly: TestAssembly, builder: CloudAssemblyBuilder) {\n  for (const stack of assembly.stacks) {\n    const templateFile = `${stack.stackName}.template.json`;\n    const template = stack.template ?? DEFAULT_FAKE_TEMPLATE;\n    fs.writeFileSync(path.join(builder.outdir, templateFile), JSON.stringify(template, undefined, 2));\n    addNestedStacks(templateFile, builder.outdir, template);\n\n    // we call patchStackTags here to simulate the tags formatter\n    // that is used when building real manifest files.\n    const metadata: { [path: string]: MetadataEntry[] } = patchStackTags({ ...stack.metadata });\n    for (const asset of stack.assets || []) {\n      metadata[asset.id] = [{ type: ArtifactMetadataEntryType.ASSET, data: asset }];\n    }\n\n    for (const missing of assembly.missing || []) {\n      builder.addMissing(missing);\n    }\n\n    const dependencies = [...(stack.depends ?? [])];\n\n    if (stack.assetManifest) {\n      const manifestFile = `${stack.stackName}.assets.json`;\n      fs.writeFileSync(path.join(builder.outdir, manifestFile), JSON.stringify(stack.assetManifest, undefined, 2));\n      dependencies.push(`${stack.stackName}.assets`);\n      builder.addArtifact(`${stack.stackName}.assets`, {\n        type: ArtifactType.ASSET_MANIFEST,\n        environment: stack.env || 'aws://123456789012/here',\n        properties: {\n          file: manifestFile,\n        },\n      });\n    }\n\n    builder.addArtifact(stack.stackName, {\n      type: ArtifactType.AWS_CLOUDFORMATION_STACK,\n      environment: stack.env || 'aws://123456789012/here',\n\n      dependencies,\n      metadata,\n      properties: {\n        ...stack.properties,\n        templateFile,\n        terminationProtection: stack.terminationProtection,\n        notificationArns: stack.notificationArns,\n      },\n      displayName: stack.displayName,\n    });\n  }\n}\n\nfunction addNestedStacks(templatePath: string, outdir: string, rootStackTemplate?: any) {\n  let template = rootStackTemplate;\n\n  if (!template) {\n    const templatePathWithDir = path.join('nested-stack-templates', templatePath);\n    template = JSON.parse(fs.readFileSync(path.join(__dirname, templatePathWithDir)).toString());\n    fs.writeFileSync(path.join(outdir, templatePath), JSON.stringify(template, undefined, 2));\n  }\n\n  for (const logicalId in template.Resources) {\n    if (template.Resources[logicalId].Type === 'AWS::CloudFormation::Stack') {\n      if (template.Resources[logicalId].Metadata && template.Resources[logicalId].Metadata['aws:asset:path']) {\n        const nestedTemplatePath = template.Resources[logicalId].Metadata['aws:asset:path'];\n        addNestedStacks(nestedTemplatePath, outdir);\n      }\n    }\n  }\n}\n\nexport function testAssembly(assembly: TestAssembly): CloudAssembly {\n  const builder = new CloudAssemblyBuilder();\n  addAttributes(assembly, builder);\n\n  if (assembly.nestedAssemblies != null && assembly.nestedAssemblies.length > 0) {\n    assembly.nestedAssemblies?.forEach((nestedAssembly: TestAssembly, i: number) => {\n      const nestedAssemblyBuilder = builder.createNestedAssembly(`nested${i}`, `nested${i}`);\n      addAttributes(nestedAssembly, nestedAssemblyBuilder);\n      nestedAssemblyBuilder.buildAssembly();\n    });\n  }\n\n  const asm = builder.buildAssembly();\n  return cxapiAssemblyWithForcedVersion(asm, assembly.schemaVersion ?? SOME_RECENT_SCHEMA_VERSION);\n}\n\n/**\n * Transform stack tags from how they are decalred in source code (lower cased)\n * to how they are stored on disk (upper cased). In real synthesis this is done\n * by a special tags formatter.\n *\n * @see aws-cdk-lib/lib/stack.ts\n */\nfunction patchStackTags(metadata: { [path: string]: MetadataEntry[] }): {\n  [path: string]: MetadataEntry[];\n} {\n  const cloned = clone(metadata) as { [path: string]: MetadataEntry[] };\n\n  for (const metadataEntries of Object.values(cloned)) {\n    for (const metadataEntry of metadataEntries) {\n      if (metadataEntry.type === ArtifactMetadataEntryType.STACK_TAGS && metadataEntry.data) {\n        const metadataAny = metadataEntry as any;\n\n        metadataAny.data = metadataAny.data.map((t: any) => {\n          return { Key: t.key, Value: t.value };\n        });\n      }\n    }\n  }\n  return cloned;\n}\n\nexport function testStack(stack: TestStackArtifact): CloudFormationStackArtifact {\n  const assembly = testAssembly({ stacks: [stack] });\n  return assembly.getStackByName(stack.stackName);\n}\n\n/**\n * Return a mocked instance of a class, given its constructor\n *\n * I don't understand why jest doesn't provide this by default,\n * but there you go.\n *\n * FIXME: Currently very limited. Doesn't support inheritance, getters or\n * automatic detection of properties (as those exist on instances, not\n * classes).\n */\nexport function instanceMockFrom<A>(ctr: new (...args: any[]) => A): jest.Mocked<A> {\n  const ret: any = {};\n  for (const methodName of Object.getOwnPropertyNames(ctr.prototype)) {\n    ret[methodName] = jest.fn();\n  }\n  return ret;\n}\n\nexport function withMocked<A extends object, K extends keyof A, B>(\n  obj: A,\n  key: K,\n  block: (fn: jest.Mocked<A>[K]) => B,\n): B {\n  const original = obj[key];\n  const mockFn = jest.fn();\n  (obj as any)[key] = mockFn;\n\n  let asyncFinally: boolean = false;\n  try {\n    const ret = block(mockFn as any);\n    if (!isPromise(ret)) {\n      return ret;\n    }\n\n    asyncFinally = true;\n    return ret.finally(() => {\n      obj[key] = original;\n    }) as any;\n  } finally {\n    if (!asyncFinally) {\n      obj[key] = original;\n    }\n  }\n}\n\nfunction isPromise<A>(object: any): object is Promise<A> {\n  return Promise.resolve(object) === object;\n}\n\nexport async function sleep(ms: number) {\n  return new Promise((ok) => setTimeout(ok, ms));\n}\n"]} |
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is too big to display
Sorry, the diff of this file is too big to display
Sorry, the diff of this file is too big to display
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is too big to display
Sorry, the diff of this file is not supported yet
License Policy Violation
LicenseThis package is not allowed per your license policy. Review the package's license to ensure compliance.
Found 1 instance in 1 package
License Policy Violation
LicenseThis package is not allowed per your license policy. Review the package's license to ensure compliance.
Found 1 instance in 1 package
85
590
50084
49468365
134