Skip to content

Commit

Permalink
feat(aws-cdk): better stack dependency handling (#1511)
Browse files Browse the repository at this point in the history
Two improvements to stack dependency handling in the toolkit:

* Destroy stacks in reverse order. This is necessary to properly dispose of stacks that
  use exports and `Fn::ImportValue`. Fixes #1508.
* Automatically include stacks that have a dependency relationship
  with the requested stacks, unless `--exclusively` (`-e`) is
  passed on the command line. Fixes #1505.
  • Loading branch information
rix0rrr authored Jan 10, 2019
1 parent ca5ee35 commit b4bbaf0
Show file tree
Hide file tree
Showing 5 changed files with 129 additions and 22 deletions.
32 changes: 22 additions & 10 deletions packages/aws-cdk/bin/cdk.ts
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,7 @@ import yargs = require('yargs');

import { bootstrapEnvironment, deployStack, destroyStack, loadToolkitInfo, Mode, SDK } from '../lib';
import { environmentsFromDescriptors, globEnvironmentsFromStacks } from '../lib/api/cxapp/environments';
import { AppStacks, listStackNames } from '../lib/api/cxapp/stacks';
import { AppStacks, ExtendedStackSelection, listStackNames } from '../lib/api/cxapp/stacks';
import { leftPad } from '../lib/api/util/string-manipulation';
import { printSecurityDiff, printStackDiff, RequireApproval } from '../lib/diff';
import { availableInitLanguages, cliInit, printAvailableTemplates } from '../lib/init';
Expand Down Expand Up @@ -53,13 +53,16 @@ async function parseCommandLineArguments() {
.command([ 'list', 'ls' ], 'Lists all stacks in the app', yargs => yargs
.option('long', { type: 'boolean', default: false, alias: 'l', desc: 'display environment information for each stack' }))
.command([ 'synthesize [STACKS..]', 'synth [STACKS..]' ], 'Synthesizes and prints the CloudFormation template for this stack', yargs => yargs
.option('exclusively', { type: 'boolean', alias: 'e', desc: 'only deploy requested stacks, don\'t include dependencies' })
.option('interactive', { type: 'boolean', alias: 'i', desc: 'interactively watch and show template updates' })
.option('output', { type: 'string', alias: 'o', desc: 'write CloudFormation template for requested stacks to the given directory' })
.option('numbered', { type: 'boolean', alias: 'n', desc: 'Prefix filenames with numbers to indicate deployment ordering' }))
.command('bootstrap [ENVIRONMENTS..]', 'Deploys the CDK toolkit stack into an AWS environment')
.command('deploy [STACKS..]', 'Deploys the stack(s) named STACKS into your AWS account', yargs => yargs
.option('exclusively', { type: 'boolean', alias: 'e', desc: 'only deploy requested stacks, don\'t include dependencies' })
.option('require-approval', { type: 'string', choices: [RequireApproval.Never, RequireApproval.AnyChange, RequireApproval.Broadening], desc: 'what security-sensitive changes need manual approval' }))
.command('destroy [STACKS..]', 'Destroy the stack(s) named STACKS', yargs => yargs
.option('exclusively', { type: 'boolean', alias: 'x', desc: 'only deploy requested stacks, don\'t include dependees' })
.option('force', { type: 'boolean', alias: 'f', desc: 'Do not ask for confirmation before destroying the stacks' }))
.command('diff [STACK]', 'Compares the specified stack with the deployed stack or a local template file, and returns with status 1 if any difference is found', yargs => yargs
.option('context-lines', { type: 'number', desc: 'number of context lines to include in arbitrary JSON diff rendering', default: 3 })
Expand Down Expand Up @@ -167,14 +170,14 @@ async function initCommandLine() {
return await cliBootstrap(args.ENVIRONMENTS, toolkitStackName, args.roleArn);

case 'deploy':
return await cliDeploy(args.STACKS, toolkitStackName, args.roleArn, configuration.combined.get(['requireApproval']));
return await cliDeploy(args.STACKS, args.exclusively, toolkitStackName, args.roleArn, configuration.combined.get(['requireApproval']));

case 'destroy':
return await cliDestroy(args.STACKS, args.force, args.roleArn);
return await cliDestroy(args.STACKS, args.exclusively, args.force, args.roleArn);

case 'synthesize':
case 'synth':
return await cliSynthesize(args.STACKS, args.interactive, args.output, args.json, args.numbered);
return await cliSynthesize(args.STACKS, args.exclusively, args.interactive, args.output, args.json, args.numbered);

case 'metadata':
return await cliMetadata(await findStack(args.STACK));
Expand Down Expand Up @@ -239,11 +242,12 @@ async function initCommandLine() {
* should be supplied, where the templates will be written.
*/
async function cliSynthesize(stackNames: string[],
exclusively: boolean,
doInteractive: boolean,
outputDir: string|undefined,
json: boolean,
numbered: boolean): Promise<void> {
const stacks = await appStacks.selectStacks(...stackNames);
const stacks = await appStacks.selectStacks(stackNames, exclusively ? ExtendedStackSelection.None : ExtendedStackSelection.Upstream);
renames.validateSelectedStacks(stacks);

if (doInteractive) {
Expand Down Expand Up @@ -300,10 +304,14 @@ async function initCommandLine() {
return 0; // exit-code
}
async function cliDeploy(stackNames: string[], toolkitStackName: string, roleArn: string | undefined, requireApproval: RequireApproval) {
async function cliDeploy(stackNames: string[],
exclusively: boolean,
toolkitStackName: string,
roleArn: string | undefined,
requireApproval: RequireApproval) {
if (requireApproval === undefined) { requireApproval = RequireApproval.Broadening; }
const stacks = await appStacks.selectStacks(...stackNames);
const stacks = await appStacks.selectStacks(stackNames, exclusively ? ExtendedStackSelection.None : ExtendedStackSelection.Upstream);
renames.validateSelectedStacks(stacks);
for (const stack of stacks) {
Expand Down Expand Up @@ -364,8 +372,12 @@ async function initCommandLine() {
}
}
async function cliDestroy(stackNames: string[], force: boolean, roleArn: string | undefined) {
const stacks = await appStacks.selectStacks(...stackNames);
async function cliDestroy(stackNames: string[], exclusively: boolean, force: boolean, roleArn: string | undefined) {
const stacks = await appStacks.selectStacks(stackNames, exclusively ? ExtendedStackSelection.None : ExtendedStackSelection.Downstream);
// The stacks will have been ordered for deployment, so reverse them for deletion.
stacks.reverse();
renames.validateSelectedStacks(stacks);
if (!force) {
Expand Down Expand Up @@ -434,7 +446,7 @@ async function initCommandLine() {
* Match a single stack from the list of available stacks
*/
async function findStack(name: string): Promise<string> {
const stacks = await appStacks.selectStacks(name);
const stacks = await appStacks.selectStacks([name], ExtendedStackSelection.None);
// Could have been a glob so check that we evaluated to exactly one
if (stacks.length > 1) {
Expand Down
4 changes: 4 additions & 0 deletions packages/aws-cdk/integ-tests/app/app.js
Original file line number Diff line number Diff line change
Expand Up @@ -33,13 +33,17 @@ class IamStack extends cdk.Stack {
class ProvidingStack extends cdk.Stack {
constructor(parent, id) {
super(parent, id);

new sns.Topic(this, 'BogusTopic'); // Some filler
}
}

class ConsumingStack extends cdk.Stack {
constructor(parent, id, providingStack) {
super(parent, id);


new sns.Topic(this, 'BogusTopic'); // Some filler
new cdk.Output(this, 'IConsumedSomething', { value: providingStack.stackName });
}
}
Expand Down
10 changes: 5 additions & 5 deletions packages/aws-cdk/integ-tests/test-cdk-order.sh
Original file line number Diff line number Diff line change
Expand Up @@ -6,10 +6,10 @@ source ${scriptdir}/common.bash

setup

# ls order == synthesis order == provider before consumer
assert "cdk list | grep -- -order-" <<HERE
cdk-toolkit-integration-order-providing
cdk-toolkit-integration-order-consuming
HERE
# Deploy the consuming stack which will include the producing stack
cdk deploy cdk-toolkit-integration-order-consuming

# Destroy the providing stack which will include the consuming stack
cdk destroy -f cdk-toolkit-integration-order-providing

echo "✅ success"
4 changes: 2 additions & 2 deletions packages/aws-cdk/lib/api/cxapp/environments.ts
Original file line number Diff line number Diff line change
@@ -1,12 +1,12 @@
import cxapi = require('@aws-cdk/cx-api');
import minimatch = require('minimatch');
import { AppStacks } from './stacks';
import { AppStacks, ExtendedStackSelection } from './stacks';

export async function globEnvironmentsFromStacks(appStacks: AppStacks, environmentGlobs: string[]): Promise<cxapi.Environment[]> {
if (environmentGlobs.length === 0) {
environmentGlobs = [ '**' ]; // default to ALL
}
const stacks = await appStacks.selectStacks();
const stacks = await appStacks.selectStacks([], ExtendedStackSelection.None);

const availableEnvironments = distinct(stacks.map(stack => stack.environment)
.filter(env => env !== undefined) as cxapi.Environment[]);
Expand Down
101 changes: 96 additions & 5 deletions packages/aws-cdk/lib/api/cxapp/stacks.ts
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
import cxapi = require('@aws-cdk/cx-api');
import colors = require('colors/safe');
import minimatch = require('minimatch');
import yargs = require('yargs');
import contextproviders = require('../../context-providers');
Expand Down Expand Up @@ -30,7 +31,7 @@ export class AppStacks {
* It's an error if there are no stacks to select, or if one of the requested parameters
* refers to a nonexistant stack.
*/
public async selectStacks(...selectors: string[]): Promise<cxapi.SynthesizedStack[]> {
public async selectStacks(selectors: string[], extendedSelection: ExtendedStackSelection): Promise<cxapi.SynthesizedStack[]> {
selectors = selectors.filter(s => s != null); // filter null/undefined

const stacks: cxapi.SynthesizedStack[] = await this.listStacks();
Expand All @@ -43,14 +44,19 @@ export class AppStacks {
return stacks;
}

const allStacks = new Map<string, cxapi.SynthesizedStack>();
for (const stack of stacks) {
allStacks.set(stack.name, stack);
}

// For every selector argument, pick stacks from the list.
const matched = new Set<string>();
const selectedStacks = new Map<string, cxapi.SynthesizedStack>();
for (const pattern of selectors) {
let found = false;

for (const stack of stacks) {
if (minimatch(stack.name, pattern)) {
matched.add(stack.name);
if (minimatch(stack.name, pattern) && !selectedStacks.has(stack.name)) {
selectedStacks.set(stack.name, stack);
found = true;
}
}
Expand All @@ -60,7 +66,17 @@ export class AppStacks {
}
}

return stacks.filter(s => matched.has(s.name));
switch (extendedSelection) {
case ExtendedStackSelection.Downstream:
includeDownstreamStacks(selectedStacks, allStacks);
break;
case ExtendedStackSelection.Upstream:
includeUpstreamStacks(selectedStacks, allStacks);
break;
}

// Filter original array because it is in the right order
return stacks.filter(s => selectedStacks.has(s.name));
}

/**
Expand Down Expand Up @@ -205,4 +221,79 @@ export class AppStacks {
*/
export function listStackNames(stacks: cxapi.SynthesizedStack[]): string {
return stacks.map(s => s.name).join(', ');
}

/**
* When selecting stacks, what other stacks to include because of dependencies
*/
export enum ExtendedStackSelection {
/**
* Don't select any extra stacks
*/
None,

/**
* Include stacks that this stack depends on
*/
Upstream,

/**
* Include stacks that depend on this stack
*/
Downstream
}

/**
* Include stacks that depend on the stacks already in the set
*
* Modifies `selectedStacks` in-place.
*/
function includeDownstreamStacks(selectedStacks: Map<string, cxapi.SynthesizedStack>, allStacks: Map<string, cxapi.SynthesizedStack>) {
const added = new Array<string>();

let madeProgress = true;
while (madeProgress) {
madeProgress = false;

for (const [name, stack] of allStacks) {
// Select this stack if it's not selected yet AND it depends on a stack that's in the selected set
if (!selectedStacks.has(name) && (stack.dependsOn || []).some(dependencyName => selectedStacks.has(dependencyName))) {
selectedStacks.set(name, stack);
added.push(name);
madeProgress = true;
}
}
}

if (added.length > 0) {
print('Including depending stacks: %s', colors.bold(added.join(', ')));
}
}

/**
* Include stacks that that stacks in the set depend on
*
* Modifies `selectedStacks` in-place.
*/
function includeUpstreamStacks(selectedStacks: Map<string, cxapi.SynthesizedStack>, allStacks: Map<string, cxapi.SynthesizedStack>) {
const added = new Array<string>();
let madeProgress = true;
while (madeProgress) {
madeProgress = false;

for (const stack of selectedStacks.values()) {
// Select an additional stack if it's not selected yet and a dependency of a selected stack (and exists, obviously)
for (const dependencyName of (stack.dependsOn || [])) {
if (!selectedStacks.has(dependencyName) && allStacks.has(dependencyName)) {
added.push(dependencyName);
selectedStacks.set(dependencyName, allStacks.get(dependencyName)!);
madeProgress = true;
}
}
}
}

if (added.length > 0) {
print('Including dependency stacks: %s', colors.bold(added.join(', ')));
}
}

0 comments on commit b4bbaf0

Please sign in to comment.