@aws-cdk/core
Version:
AWS Cloud Development Kit Core Library
156 lines • 25.6 kB
JavaScript
"use strict";
const fc = require("fast-check");
const _ = require("lodash");
const nodeunit = require("nodeunit");
const lib_1 = require("../lib");
const intrinsic_1 = require("../lib/private/intrinsic");
function asyncTest(cb) {
return async (test) => {
let error;
try {
await cb(test);
}
catch (e) {
error = e;
}
finally {
test.doesNotThrow(() => {
if (error) {
throw error;
}
});
test.done();
}
};
}
const nonEmptyString = fc.string(1, 16);
const tokenish = fc.array(nonEmptyString, 2, 2).map(arr => ({ [arr[0]]: arr[1] }));
const anyValue = fc.oneof(nonEmptyString, tokenish);
function stringListToken(o) {
return lib_1.Token.asList(new intrinsic_1.Intrinsic(o));
}
function stringToken(o) {
return lib_1.Token.asString(new intrinsic_1.Intrinsic(o));
}
module.exports = nodeunit.testCase({
'eager resolution for non-tokens': {
'Fn.select'(test) {
test.deepEqual(lib_1.Fn.select(2, ['hello', 'you', 'dude']), 'dude');
test.done();
},
'Fn.split'(test) {
test.deepEqual(lib_1.Fn.split(':', 'hello:world:yeah'), ['hello', 'world', 'yeah']);
test.done();
}
},
'FnJoin': {
'rejects empty list of arguments to join'(test) {
test.throws(() => lib_1.Fn.join('.', []));
test.done();
},
'collapse nested FnJoins even if they contain tokens'(test) {
const stack = new lib_1.Stack();
const obj = lib_1.Fn.join('', [
'a',
lib_1.Fn.join('', [lib_1.Fn.getAtt('a', 'bc').toString(), 'c']),
'd'
]);
test.deepEqual(stack.resolve(obj), { 'Fn::Join': ["",
[
"a",
{ 'Fn::GetAtt': ['a', 'bc'] },
'cd',
]
] });
test.done();
},
'resolves to the value if only one value is joined': asyncTest(async () => {
const stack = new lib_1.Stack();
await fc.assert(fc.property(fc.string(), anyValue, (delimiter, value) => _.isEqual(stack.resolve(lib_1.Fn.join(delimiter, [value])), value)), { verbose: true });
}),
'pre-concatenates string literals': asyncTest(async () => {
const stack = new lib_1.Stack();
await fc.assert(fc.property(fc.string(), fc.array(nonEmptyString, 1, 15), (delimiter, values) => stack.resolve(lib_1.Fn.join(delimiter, values)) === values.join(delimiter)), { verbose: true });
}),
'pre-concatenates around tokens': asyncTest(async () => {
const stack = new lib_1.Stack();
await fc.assert(fc.property(fc.string(), fc.array(nonEmptyString, 1, 3), tokenish, fc.array(nonEmptyString, 1, 3), (delimiter, prefix, obj, suffix) => _.isEqual(stack.resolve(lib_1.Fn.join(delimiter, [...prefix, stringToken(obj), ...suffix])), { 'Fn::Join': [delimiter, [prefix.join(delimiter), obj, suffix.join(delimiter)]] })), { verbose: true, seed: 1539874645005, path: "0:0:0:0:0:0:0:0:0" });
}),
'flattens joins nested under joins with same delimiter': asyncTest(async () => {
const stack = new lib_1.Stack();
await fc.assert(fc.property(fc.string(), fc.array(anyValue), fc.array(anyValue, 1, 3), fc.array(anyValue), (delimiter, prefix, nested, suffix) =>
// Gonna test
_.isEqual(stack.resolve(lib_1.Fn.join(delimiter, [...prefix, lib_1.Fn.join(delimiter, nested), ...suffix])), stack.resolve(lib_1.Fn.join(delimiter, [...prefix, ...nested, ...suffix])))), { verbose: true });
}),
'does not flatten joins nested under joins with different delimiter': asyncTest(async () => {
const stack = new lib_1.Stack();
await fc.assert(fc.property(fc.string(), fc.string(), fc.array(anyValue, 1, 3), fc.array(tokenish, 2, 3), fc.array(anyValue, 3), (delimiter1, delimiter2, prefix, nested, suffix) => {
fc.pre(delimiter1 !== delimiter2);
const join = lib_1.Fn.join(delimiter1, [...prefix, lib_1.Fn.join(delimiter2, stringListToken(nested)), ...suffix]);
const resolved = stack.resolve(join);
return resolved['Fn::Join'][1].find((e) => typeof e === 'object'
&& ('Fn::Join' in e)
&& e['Fn::Join'][0] === delimiter2) != null;
}), { verbose: true });
}),
'Fn::EachMemberIn': asyncTest(async (test) => {
const stack = new lib_1.Stack();
const eachMemberIn = lib_1.Fn.conditionEachMemberIn(lib_1.Fn.valueOfAll('AWS::EC2::Subnet::Id', 'VpcId'), lib_1.Fn.refAll('AWS::EC2::VPC::Id'));
test.deepEqual(stack.resolve(eachMemberIn), {
'Fn::EachMemberIn': [
{ 'Fn::ValueOfAll': ['AWS::EC2::Subnet::Id', 'VpcId'] },
{ 'Fn::RefAll': 'AWS::EC2::VPC::Id' }
]
});
}),
'cross-stack FnJoin elements are properly resolved': asyncTest(async (test) => {
// GIVEN
const app = new lib_1.App();
const stack1 = new lib_1.Stack(app, 'Stack1');
const stack2 = new lib_1.Stack(app, 'Stack2');
// WHEN
new lib_1.CfnOutput(stack2, 'Stack1Id', {
value: lib_1.Fn.join(' = ', ['Stack1Id', stack1.stackId])
});
// THEN
const template = app.synth().getStackByName('Stack2').template;
test.deepEqual(template, {
Outputs: {
Stack1Id: {
Value: {
'Fn::Join': [' = ', [
'Stack1Id',
{ 'Fn::ImportValue': 'Stack1:ExportsOutputRefAWSStackIdB2DD5BAA' }
]]
}
}
}
});
}),
},
'Ref': {
'returns a reference given a logical name'(test) {
const stack = new lib_1.Stack();
test.deepEqual(stack.resolve(lib_1.Fn.ref('hello')), {
Ref: 'hello'
});
test.done();
}
},
'nested Fn::Join with list token'(test) {
const stack = new lib_1.Stack();
const inner = lib_1.Fn.join(',', lib_1.Token.asList({ NotReallyList: true }));
const outer = lib_1.Fn.join(',', [inner, 'Foo']);
test.deepEqual(stack.resolve(outer), {
'Fn::Join': [
',',
[
{ 'Fn::Join': [',', { NotReallyList: true }] },
'Foo'
]
]
});
test.done();
},
});
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"test.fn.js","sourceRoot":"","sources":["test.fn.ts"],"names":[],"mappings":"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","sourcesContent":["import * as fc from 'fast-check';\nimport * as _ from 'lodash';\nimport * as nodeunit from 'nodeunit';\nimport { App, CfnOutput, Fn, Stack, Token } from '../lib';\nimport { Intrinsic } from '../lib/private/intrinsic';\n\nfunction asyncTest(cb: (test: nodeunit.Test) => Promise<void>): (test: nodeunit.Test) => void {\n  return async (test: nodeunit.Test) => {\n    let error: Error;\n    try {\n      await cb(test);\n    } catch (e) {\n      error = e;\n    } finally {\n      test.doesNotThrow(() => {\n        if (error) { throw error; }\n      });\n      test.done();\n    }\n  };\n}\n\nconst nonEmptyString = fc.string(1, 16);\nconst tokenish = fc.array(nonEmptyString, 2, 2).map(arr => ({ [arr[0]]: arr[1] }));\nconst anyValue = fc.oneof<any>(nonEmptyString, tokenish);\n\nexport = nodeunit.testCase({\n  'eager resolution for non-tokens': {\n    'Fn.select'(test: nodeunit.Test) {\n      test.deepEqual(Fn.select(2, [ 'hello', 'you', 'dude' ]), 'dude');\n      test.done();\n    },\n    'Fn.split'(test: nodeunit.Test) {\n      test.deepEqual(Fn.split(':', 'hello:world:yeah'), [ 'hello', 'world', 'yeah' ]);\n      test.done();\n    }\n  },\n  'FnJoin': {\n    'rejects empty list of arguments to join'(test: nodeunit.Test) {\n      test.throws(() => Fn.join('.', []));\n      test.done();\n    },\n    'collapse nested FnJoins even if they contain tokens'(test: nodeunit.Test) {\n      const stack = new Stack();\n\n      const obj = Fn.join('', [\n        'a',\n        Fn.join('', [Fn.getAtt('a', 'bc').toString(), 'c']),\n        'd'\n      ]);\n\n      test.deepEqual(stack.resolve(obj), { 'Fn::Join': [ \"\",\n        [\n          \"a\",\n          { 'Fn::GetAtt': ['a', 'bc'] },\n          'cd',\n        ]\n      ]});\n\n      test.done();\n    },\n    'resolves to the value if only one value is joined': asyncTest(async () => {\n      const stack = new Stack();\n      await fc.assert(\n        fc.property(\n          fc.string(), anyValue,\n          (delimiter, value) => _.isEqual(stack.resolve(Fn.join(delimiter, [value])), value)\n        ),\n        { verbose: true }\n      );\n    }),\n    'pre-concatenates string literals': asyncTest(async () => {\n      const stack = new Stack();\n      await fc.assert(\n        fc.property(\n          fc.string(), fc.array(nonEmptyString, 1, 15),\n          (delimiter, values) => stack.resolve(Fn.join(delimiter, values)) === values.join(delimiter)\n        ),\n        { verbose: true }\n      );\n    }),\n    'pre-concatenates around tokens': asyncTest(async () => {\n      const stack = new Stack();\n      await fc.assert(\n        fc.property(\n          fc.string(), fc.array(nonEmptyString, 1, 3), tokenish, fc.array(nonEmptyString, 1, 3),\n          (delimiter, prefix, obj, suffix) =>\n            _.isEqual(stack.resolve(Fn.join(delimiter, [...prefix, stringToken(obj), ...suffix])),\n                      { 'Fn::Join': [delimiter, [prefix.join(delimiter), obj, suffix.join(delimiter)]] })\n        ),\n        { verbose: true, seed: 1539874645005, path: \"0:0:0:0:0:0:0:0:0\" }\n      );\n    }),\n    'flattens joins nested under joins with same delimiter': asyncTest(async () => {\n      const stack = new Stack();\n      await fc.assert(\n        fc.property(\n          fc.string(), fc.array(anyValue),\n                      fc.array(anyValue, 1, 3),\n                      fc.array(anyValue),\n          (delimiter, prefix, nested, suffix) =>\n            // Gonna test\n            _.isEqual(stack.resolve(Fn.join(delimiter, [...prefix, Fn.join(delimiter, nested), ...suffix])),\n                      stack.resolve(Fn.join(delimiter, [...prefix, ...nested, ...suffix])))\n        ),\n        { verbose: true }\n      );\n    }),\n    'does not flatten joins nested under joins with different delimiter': asyncTest(async () => {\n      const stack = new Stack();\n      await fc.assert(\n        fc.property(\n          fc.string(), fc.string(),\n          fc.array(anyValue, 1, 3),\n          fc.array(tokenish, 2, 3),\n          fc.array(anyValue, 3),\n          (delimiter1, delimiter2, prefix,  nested, suffix) => {\n            fc.pre(delimiter1 !== delimiter2);\n            const join = Fn.join(delimiter1, [...prefix, Fn.join(delimiter2, stringListToken(nested)), ...suffix]);\n            const resolved = stack.resolve(join);\n            return resolved['Fn::Join'][1].find((e: any) => typeof e === 'object'\n                                                        && ('Fn::Join' in e)\n                                                        && e['Fn::Join'][0] === delimiter2) != null;\n          }\n        ),\n        { verbose: true }\n      );\n    }),\n    'Fn::EachMemberIn': asyncTest(async (test) => {\n      const stack = new Stack();\n      const eachMemberIn = Fn.conditionEachMemberIn(\n        Fn.valueOfAll('AWS::EC2::Subnet::Id', 'VpcId'),\n        Fn.refAll('AWS::EC2::VPC::Id')\n      );\n      test.deepEqual(stack.resolve(eachMemberIn), {\n        'Fn::EachMemberIn': [\n          { 'Fn::ValueOfAll': ['AWS::EC2::Subnet::Id', 'VpcId'] },\n          { 'Fn::RefAll': 'AWS::EC2::VPC::Id'}\n        ]\n      });\n    }),\n\n    'cross-stack FnJoin elements are properly resolved': asyncTest(async (test) => {\n      // GIVEN\n      const app = new App();\n      const stack1 = new Stack(app, 'Stack1');\n      const stack2 = new Stack(app, 'Stack2');\n\n      // WHEN\n      new CfnOutput(stack2, 'Stack1Id', {\n        value: Fn.join(' = ', [ 'Stack1Id', stack1.stackId ])\n      });\n\n      // THEN\n      const template = app.synth().getStackByName('Stack2').template;\n\n      test.deepEqual(template, {\n        Outputs: {\n          Stack1Id: {\n            Value: {\n              'Fn::Join': [' = ', [\n                'Stack1Id',\n                { 'Fn::ImportValue': 'Stack1:ExportsOutputRefAWSStackIdB2DD5BAA' }\n              ]]\n            }\n          }\n        }\n      });\n    }),\n  },\n  'Ref': {\n    'returns a reference given a logical name'(test: nodeunit.Test) {\n      const stack = new Stack();\n      test.deepEqual(stack.resolve(Fn.ref('hello')), {\n        Ref: 'hello'\n      });\n      test.done();\n    }\n  },\n  'nested Fn::Join with list token'(test: nodeunit.Test) {\n    const stack = new Stack();\n    const inner = Fn.join(',', Token.asList({ NotReallyList: true }));\n    const outer = Fn.join(',', [ inner, 'Foo' ]);\n    test.deepEqual(stack.resolve(outer), {\n      'Fn::Join': [\n        ',',\n        [\n          { 'Fn::Join': [ ',', { NotReallyList: true } ] },\n          'Foo'\n        ]\n      ]\n    });\n    test.done();\n  },\n});\n\nfunction stringListToken(o: any): string[] {\n  return Token.asList(new Intrinsic(o));\n}\nfunction stringToken(o: any): string {\n  return Token.asString(new Intrinsic(o));\n}\n"]}