@aws-mdaa/dataops-job-l3-construct 1.5.0 → 1.6.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.jsii +38 -28
- package/README.md +5 -0
- package/lib/dataops-job-l3-construct.d.ts +8 -1
- package/lib/dataops-job-l3-construct.js +40 -13
- package/node_modules/@aws-mdaa/config/README.md +3 -0
- package/node_modules/@aws-mdaa/config/lib/blueprint-value-transformer.d.ts +20 -0
- package/node_modules/@aws-mdaa/config/lib/blueprint-value-transformer.js +70 -0
- package/node_modules/@aws-mdaa/config/lib/blueprint-value-transformer.ts +88 -0
- package/node_modules/@aws-mdaa/config/lib/config.d.ts +87 -0
- package/node_modules/@aws-mdaa/config/lib/config.js +7 -0
- package/node_modules/@aws-mdaa/config/lib/config.ts +92 -0
- package/node_modules/@aws-mdaa/config/lib/index.d.ts +11 -0
- package/node_modules/@aws-mdaa/config/lib/index.js +28 -0
- package/node_modules/@aws-mdaa/config/lib/index.ts +12 -0
- package/node_modules/@aws-mdaa/config/lib/param-transformer.d.ts +49 -0
- package/node_modules/@aws-mdaa/config/lib/param-transformer.js +160 -0
- package/node_modules/@aws-mdaa/config/lib/param-transformer.ts +159 -0
- package/node_modules/@aws-mdaa/config/lib/path-value-transformer.d.ts +10 -0
- package/node_modules/@aws-mdaa/config/lib/path-value-transformer.js +30 -0
- package/node_modules/@aws-mdaa/config/lib/path-value-transformer.ts +27 -0
- package/node_modules/@aws-mdaa/config/lib/ref-value-transformer.d.ts +44 -0
- package/node_modules/@aws-mdaa/config/lib/ref-value-transformer.js +243 -0
- package/node_modules/@aws-mdaa/config/lib/ref-value-transformer.ts +302 -0
- package/node_modules/@aws-mdaa/config/lib/ssm-ref-transformer.d.ts +8 -0
- package/node_modules/@aws-mdaa/config/lib/ssm-ref-transformer.js +22 -0
- package/node_modules/@aws-mdaa/config/lib/ssm-ref-transformer.ts +21 -0
- package/node_modules/@aws-mdaa/config/lib/transformer.d.ts +35 -0
- package/node_modules/@aws-mdaa/config/lib/transformer.js +66 -0
- package/node_modules/@aws-mdaa/config/lib/transformer.ts +74 -0
- package/node_modules/@aws-mdaa/config/package.json +42 -0
- package/node_modules/@aws-mdaa/config/test/blueprint-value-transformer.test.js +224 -0
- package/node_modules/@aws-mdaa/config/test/blueprint-value-transformer.test.ts +259 -0
- package/node_modules/@aws-mdaa/config/test/config-nt.test.d.ts +5 -0
- package/node_modules/@aws-mdaa/config/test/config-nt.test.js +129 -0
- package/node_modules/@aws-mdaa/config/test/config-nt.test.ts +163 -0
- package/node_modules/@aws-mdaa/config/test/config.test.d.ts +5 -0
- package/node_modules/@aws-mdaa/config/test/config.test.js +409 -0
- package/node_modules/@aws-mdaa/config/test/config.test.ts +517 -0
- package/node_modules/@aws-mdaa/config/test/param-transformer.test.d.ts +5 -0
- package/node_modules/@aws-mdaa/config/test/param-transformer.test.js +216 -0
- package/node_modules/@aws-mdaa/config/test/param-transformer.test.ts +234 -0
- package/node_modules/@aws-mdaa/config/test/path-value-transformer.test.d.ts +5 -0
- package/node_modules/@aws-mdaa/config/test/path-value-transformer.test.js +59 -0
- package/node_modules/@aws-mdaa/config/test/path-value-transformer.test.ts +68 -0
- package/node_modules/@aws-mdaa/config/test/ref-value-transformer.test.d.ts +5 -0
- package/node_modules/@aws-mdaa/config/test/ref-value-transformer.test.js +254 -0
- package/node_modules/@aws-mdaa/config/test/ref-value-transformer.test.ts +304 -0
- package/node_modules/@aws-mdaa/config/test/ssm-ref-transformer.test.d.ts +5 -0
- package/node_modules/@aws-mdaa/config/test/ssm-ref-transformer.test.js +66 -0
- package/node_modules/@aws-mdaa/config/test/ssm-ref-transformer.test.ts +79 -0
- package/node_modules/@aws-mdaa/config/tsconfig.tsbuildinfo +1 -0
- package/node_modules/lodash/README.md +2 -2
- package/node_modules/lodash/_baseOrderBy.js +1 -1
- package/node_modules/lodash/_baseUnset.js +7 -20
- package/node_modules/lodash/_setCacheHas.js +1 -1
- package/node_modules/lodash/compact.js +1 -1
- package/node_modules/lodash/core.js +3 -3
- package/node_modules/lodash/core.min.js +26 -25
- package/node_modules/lodash/fromPairs.js +3 -1
- package/node_modules/lodash/lodash.js +38 -27
- package/node_modules/lodash/lodash.min.js +125 -129
- package/node_modules/lodash/package.json +4 -2
- package/node_modules/lodash/random.js +9 -0
- package/node_modules/lodash/template.js +16 -4
- package/node_modules/lodash/templateSettings.js +4 -0
- package/package.json +27 -32
- package/node_modules/@aws-mdaa/s3-bucketpolicy-helper/README.md +0 -185
- package/node_modules/@aws-mdaa/s3-bucketpolicy-helper/lib/index.d.ts +0 -57
- package/node_modules/@aws-mdaa/s3-bucketpolicy-helper/lib/index.js +0 -198
- package/node_modules/@aws-mdaa/s3-bucketpolicy-helper/lib/index.ts +0 -241
- package/node_modules/@aws-mdaa/s3-bucketpolicy-helper/package.json +0 -44
- package/node_modules/@aws-mdaa/s3-bucketpolicy-helper/test/bucketpolicy-helper.test.js +0 -200
- package/node_modules/@aws-mdaa/s3-bucketpolicy-helper/test/bucketpolicy-helper.test.ts +0 -215
- package/node_modules/@aws-mdaa/s3-bucketpolicy-helper/tsconfig.tsbuildinfo +0 -1
- package/node_modules/@aws-mdaa/s3-inventory-helper/.npmignore +0 -34
- package/node_modules/@aws-mdaa/s3-inventory-helper/README.md +0 -3
- package/node_modules/@aws-mdaa/s3-inventory-helper/jest.config.js +0 -5
- package/node_modules/@aws-mdaa/s3-inventory-helper/lib/index.d.ts +0 -48
- package/node_modules/@aws-mdaa/s3-inventory-helper/lib/index.js +0 -213
- package/node_modules/@aws-mdaa/s3-inventory-helper/lib/index.ts +0 -241
- package/node_modules/@aws-mdaa/s3-inventory-helper/package.json +0 -44
- package/node_modules/@aws-mdaa/s3-inventory-helper/test/TODO +0 -0
- package/node_modules/@aws-mdaa/s3-inventory-helper/tsconfig.json +0 -40
- package/node_modules/@aws-mdaa/s3-inventory-helper/tsconfig.tsbuildinfo +0 -1
- package/node_modules/@aws-mdaa/s3-inventory-helper/typedoc.json +0 -7
- /package/node_modules/@aws-mdaa/{s3-bucketpolicy-helper → config}/.npmignore +0 -0
- /package/node_modules/@aws-mdaa/{s3-bucketpolicy-helper → config}/jest.config.js +0 -0
- /package/node_modules/@aws-mdaa/{s3-bucketpolicy-helper/test/bucketpolicy-helper.test.d.ts → config/test/blueprint-value-transformer.test.d.ts} +0 -0
- /package/node_modules/@aws-mdaa/{s3-bucketpolicy-helper → config}/tsconfig.json +0 -0
- /package/node_modules/@aws-mdaa/{s3-bucketpolicy-helper → config}/typedoc.json +0 -0
|
@@ -0,0 +1,224 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/*!
|
|
3
|
+
* Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
|
|
4
|
+
* SPDX-License-Identifier: Apache-2.0
|
|
5
|
+
*/
|
|
6
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
7
|
+
const aws_cdk_lib_1 = require("aws-cdk-lib");
|
|
8
|
+
const lib_1 = require("../lib");
|
|
9
|
+
describe('MdaaConfigBlueprintRefValueTransformer', () => {
|
|
10
|
+
let mockNaming;
|
|
11
|
+
let stack;
|
|
12
|
+
beforeEach(() => {
|
|
13
|
+
stack = new aws_cdk_lib_1.Stack();
|
|
14
|
+
mockNaming = {
|
|
15
|
+
props: {
|
|
16
|
+
org: 'test-org',
|
|
17
|
+
env: 'test-env',
|
|
18
|
+
domain: 'test-domain',
|
|
19
|
+
moduleName: 'test-module',
|
|
20
|
+
},
|
|
21
|
+
withModuleName: jest.fn().mockReturnThis(),
|
|
22
|
+
withDomain: jest.fn().mockReturnThis(),
|
|
23
|
+
withOrg: jest.fn().mockReturnThis(),
|
|
24
|
+
withEnv: jest.fn().mockReturnThis(),
|
|
25
|
+
withSuffix: jest.fn().mockReturnThis(),
|
|
26
|
+
ssmDomainPath: jest.fn().mockReturnValue('/test/domain/path'),
|
|
27
|
+
ssmEnvPath: jest.fn().mockReturnValue('/test/env/path'),
|
|
28
|
+
ssmOrgPath: jest.fn().mockReturnValue('/test-org/'),
|
|
29
|
+
resourceName: jest
|
|
30
|
+
.fn()
|
|
31
|
+
.mockImplementation((suffix) => (suffix ? `test-resource-${suffix}` : 'test-resource')),
|
|
32
|
+
ssmPath: jest.fn().mockImplementation((path) => `/test/${path}`),
|
|
33
|
+
stackName: jest.fn().mockImplementation((name) => (name ? `test-stack-${name}` : 'test-stack')),
|
|
34
|
+
exportName: jest.fn().mockImplementation((path) => `test-export-${path}`),
|
|
35
|
+
};
|
|
36
|
+
});
|
|
37
|
+
test('returns unchanged value without refs', () => {
|
|
38
|
+
const transformer = new lib_1.MdaaConfigBlueprintRefValueTransformer({
|
|
39
|
+
naming: mockNaming,
|
|
40
|
+
scope: stack,
|
|
41
|
+
});
|
|
42
|
+
expect(transformer.transformValue('plain-value')).toBe('plain-value');
|
|
43
|
+
});
|
|
44
|
+
test('returns unchanged value with empty string', () => {
|
|
45
|
+
const transformer = new lib_1.MdaaConfigBlueprintRefValueTransformer({
|
|
46
|
+
naming: mockNaming,
|
|
47
|
+
scope: stack,
|
|
48
|
+
});
|
|
49
|
+
expect(transformer.transformValue('')).toBe('');
|
|
50
|
+
});
|
|
51
|
+
test('transforms blueprint ref', () => {
|
|
52
|
+
const transformer = new lib_1.MdaaConfigBlueprintRefValueTransformer({
|
|
53
|
+
naming: mockNaming,
|
|
54
|
+
scope: stack,
|
|
55
|
+
});
|
|
56
|
+
const result = transformer.transformValue('{{blueprint:/some/path}}');
|
|
57
|
+
expect(result).toContain('{{resolve:ssm:/test-org/');
|
|
58
|
+
expect(result).toContain('/some/path}}');
|
|
59
|
+
expect(mockNaming.ssmOrgPath).toHaveBeenCalledWith('', false);
|
|
60
|
+
});
|
|
61
|
+
test('transforms blueprint ref with prefix and suffix', () => {
|
|
62
|
+
const transformer = new lib_1.MdaaConfigBlueprintRefValueTransformer({
|
|
63
|
+
naming: mockNaming,
|
|
64
|
+
scope: stack,
|
|
65
|
+
});
|
|
66
|
+
const result = transformer.transformValue('prefix-{{blueprint:/path}}-suffix');
|
|
67
|
+
expect(result).toContain('prefix-');
|
|
68
|
+
expect(result).toContain('-suffix');
|
|
69
|
+
expect(result).toContain('{{resolve:ssm:');
|
|
70
|
+
});
|
|
71
|
+
test('handles multiple blueprint refs', () => {
|
|
72
|
+
const transformer = new lib_1.MdaaConfigBlueprintRefValueTransformer({
|
|
73
|
+
naming: mockNaming,
|
|
74
|
+
scope: stack,
|
|
75
|
+
});
|
|
76
|
+
const result = transformer.transformValue('{{blueprint:/path1}}-{{blueprint:/path2}}');
|
|
77
|
+
expect(result).toContain('/path1}}');
|
|
78
|
+
expect(result).toContain('/path2}}');
|
|
79
|
+
});
|
|
80
|
+
test('reuses existing CfnParameter for datazoneEnvironmentProjectId', () => {
|
|
81
|
+
const transformer = new lib_1.MdaaConfigBlueprintRefValueTransformer({
|
|
82
|
+
naming: mockNaming,
|
|
83
|
+
scope: stack,
|
|
84
|
+
});
|
|
85
|
+
// First call creates the parameter
|
|
86
|
+
transformer.transformValue('{{blueprint:/path1}}');
|
|
87
|
+
// Second call should reuse the same parameter
|
|
88
|
+
transformer.transformValue('{{blueprint:/path2}}');
|
|
89
|
+
// Should only have one datazoneEnvironmentProjectId parameter
|
|
90
|
+
const params = stack.node.children.filter(child => child.node.id === 'datazoneEnvironmentProjectId');
|
|
91
|
+
expect(params.length).toBe(1);
|
|
92
|
+
});
|
|
93
|
+
test('ignores non-blueprint refs', () => {
|
|
94
|
+
const transformer = new lib_1.MdaaConfigBlueprintRefValueTransformer({
|
|
95
|
+
naming: mockNaming,
|
|
96
|
+
scope: stack,
|
|
97
|
+
});
|
|
98
|
+
const result = transformer.transformValue('{{other:value}}');
|
|
99
|
+
// Non-blueprint refs should remain unchanged
|
|
100
|
+
expect(result).toBe('{{other:value}}');
|
|
101
|
+
});
|
|
102
|
+
test('handles nested refs in blueprint path', () => {
|
|
103
|
+
const transformer = new lib_1.MdaaConfigBlueprintRefValueTransformer({
|
|
104
|
+
naming: mockNaming,
|
|
105
|
+
scope: stack,
|
|
106
|
+
});
|
|
107
|
+
// Nested refs should be processed recursively
|
|
108
|
+
const result = transformer.transformValue('{{blueprint:/{{other}}/path}}');
|
|
109
|
+
// The inner ref is not a blueprint ref, so it stays as-is in the path
|
|
110
|
+
expect(result).toContain('{{resolve:ssm:');
|
|
111
|
+
});
|
|
112
|
+
test('handles unbalanced braces gracefully', () => {
|
|
113
|
+
const transformer = new lib_1.MdaaConfigBlueprintRefValueTransformer({
|
|
114
|
+
naming: mockNaming,
|
|
115
|
+
scope: stack,
|
|
116
|
+
});
|
|
117
|
+
// Unbalanced braces should be skipped
|
|
118
|
+
expect(transformer.transformValue('{{blueprint:/path')).toBe('{{blueprint:/path');
|
|
119
|
+
expect(transformer.transformValue('blueprint:/path}}')).toBe('blueprint:/path}}');
|
|
120
|
+
});
|
|
121
|
+
test('logs resolved blueprint path', () => {
|
|
122
|
+
const consoleSpy = jest.spyOn(console, 'log').mockImplementation();
|
|
123
|
+
const transformer = new lib_1.MdaaConfigBlueprintRefValueTransformer({
|
|
124
|
+
naming: mockNaming,
|
|
125
|
+
scope: stack,
|
|
126
|
+
});
|
|
127
|
+
transformer.transformValue('{{blueprint:/test/path}}');
|
|
128
|
+
expect(consoleSpy).toHaveBeenCalledWith(expect.stringContaining('Resolving blueprint path:'));
|
|
129
|
+
consoleSpy.mockRestore();
|
|
130
|
+
});
|
|
131
|
+
test('handles blueprint ref with special characters in path', () => {
|
|
132
|
+
const transformer = new lib_1.MdaaConfigBlueprintRefValueTransformer({
|
|
133
|
+
naming: mockNaming,
|
|
134
|
+
scope: stack,
|
|
135
|
+
});
|
|
136
|
+
const result = transformer.transformValue('{{blueprint:/path/with-dashes_and_underscores/123}}');
|
|
137
|
+
expect(result).toContain('{{resolve:ssm:');
|
|
138
|
+
expect(result).toContain('/path/with-dashes_and_underscores/123}}');
|
|
139
|
+
});
|
|
140
|
+
test('handles blueprint ref at start of string', () => {
|
|
141
|
+
const transformer = new lib_1.MdaaConfigBlueprintRefValueTransformer({
|
|
142
|
+
naming: mockNaming,
|
|
143
|
+
scope: stack,
|
|
144
|
+
});
|
|
145
|
+
const result = transformer.transformValue('{{blueprint:/path}}-suffix');
|
|
146
|
+
expect(result).toMatch(/^\{\{resolve:ssm:/);
|
|
147
|
+
expect(result).toContain('-suffix');
|
|
148
|
+
});
|
|
149
|
+
test('handles blueprint ref at end of string', () => {
|
|
150
|
+
const transformer = new lib_1.MdaaConfigBlueprintRefValueTransformer({
|
|
151
|
+
naming: mockNaming,
|
|
152
|
+
scope: stack,
|
|
153
|
+
});
|
|
154
|
+
const result = transformer.transformValue('prefix-{{blueprint:/path}}');
|
|
155
|
+
expect(result).toContain('prefix-');
|
|
156
|
+
expect(result).toMatch(/\}\}$/);
|
|
157
|
+
});
|
|
158
|
+
test('handles multiple different ref types mixed', () => {
|
|
159
|
+
const transformer = new lib_1.MdaaConfigBlueprintRefValueTransformer({
|
|
160
|
+
naming: mockNaming,
|
|
161
|
+
scope: stack,
|
|
162
|
+
});
|
|
163
|
+
const result = transformer.transformValue('{{blueprint:/path1}}-{{other:ref}}-{{blueprint:/path2}}');
|
|
164
|
+
expect(result).toContain('/path1}}');
|
|
165
|
+
expect(result).toContain('{{other:ref}}');
|
|
166
|
+
expect(result).toContain('/path2}}');
|
|
167
|
+
});
|
|
168
|
+
test('handles empty blueprint path', () => {
|
|
169
|
+
const transformer = new lib_1.MdaaConfigBlueprintRefValueTransformer({
|
|
170
|
+
naming: mockNaming,
|
|
171
|
+
scope: stack,
|
|
172
|
+
});
|
|
173
|
+
const result = transformer.transformValue('{{blueprint:}}');
|
|
174
|
+
expect(result).toContain('{{resolve:ssm:');
|
|
175
|
+
});
|
|
176
|
+
test('handles blueprint ref with only slashes', () => {
|
|
177
|
+
const transformer = new lib_1.MdaaConfigBlueprintRefValueTransformer({
|
|
178
|
+
naming: mockNaming,
|
|
179
|
+
scope: stack,
|
|
180
|
+
});
|
|
181
|
+
const result = transformer.transformValue('{{blueprint:/}}');
|
|
182
|
+
expect(result).toContain('{{resolve:ssm:');
|
|
183
|
+
expect(result).toContain('/}}');
|
|
184
|
+
});
|
|
185
|
+
test('creates parameter on first blueprint ref', () => {
|
|
186
|
+
const transformer = new lib_1.MdaaConfigBlueprintRefValueTransformer({
|
|
187
|
+
naming: mockNaming,
|
|
188
|
+
scope: stack,
|
|
189
|
+
});
|
|
190
|
+
const childrenBefore = stack.node.children.length;
|
|
191
|
+
transformer.transformValue('{{blueprint:/first}}');
|
|
192
|
+
const childrenAfter = stack.node.children.length;
|
|
193
|
+
expect(childrenAfter).toBeGreaterThan(childrenBefore);
|
|
194
|
+
});
|
|
195
|
+
test('does not create duplicate parameters', () => {
|
|
196
|
+
const transformer = new lib_1.MdaaConfigBlueprintRefValueTransformer({
|
|
197
|
+
naming: mockNaming,
|
|
198
|
+
scope: stack,
|
|
199
|
+
});
|
|
200
|
+
transformer.transformValue('{{blueprint:/first}}');
|
|
201
|
+
const childrenAfterFirst = stack.node.children.length;
|
|
202
|
+
transformer.transformValue('{{blueprint:/second}}');
|
|
203
|
+
const childrenAfterSecond = stack.node.children.length;
|
|
204
|
+
// Should not create additional parameters
|
|
205
|
+
expect(childrenAfterSecond).toBe(childrenAfterFirst);
|
|
206
|
+
});
|
|
207
|
+
test('throws when blueprint ref without scope', () => {
|
|
208
|
+
const transformer = new lib_1.MdaaConfigBlueprintRefValueTransformer({
|
|
209
|
+
naming: mockNaming,
|
|
210
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
211
|
+
scope: undefined,
|
|
212
|
+
});
|
|
213
|
+
expect(() => transformer.transformValue('{{blueprint:/path}}')).toThrow('Unable to resolve blueprint params outside of a Construct');
|
|
214
|
+
});
|
|
215
|
+
test('throws when blueprint ref without naming', () => {
|
|
216
|
+
const transformer = new lib_1.MdaaConfigBlueprintRefValueTransformer({
|
|
217
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
218
|
+
naming: undefined,
|
|
219
|
+
scope: stack,
|
|
220
|
+
});
|
|
221
|
+
expect(() => transformer.transformValue('{{blueprint:/path}}')).toThrow('Unable to resolve blueprint params without a naming implementation');
|
|
222
|
+
});
|
|
223
|
+
});
|
|
224
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"blueprint-value-transformer.test.js","sourceRoot":"","sources":["blueprint-value-transformer.test.ts"],"names":[],"mappings":"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","sourcesContent":["/*!\n * Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.\n * SPDX-License-Identifier: Apache-2.0\n */\n\nimport { Stack } from 'aws-cdk-lib';\nimport { MdaaConfigBlueprintRefValueTransformer } from '../lib';\nimport { IMdaaResourceNaming } from '@aws-mdaa/naming';\n\ndescribe('MdaaConfigBlueprintRefValueTransformer', () => {\n  let mockNaming: IMdaaResourceNaming;\n  let stack: Stack;\n\n  beforeEach(() => {\n    stack = new Stack();\n    mockNaming = {\n      props: {\n        org: 'test-org',\n        env: 'test-env',\n        domain: 'test-domain',\n        moduleName: 'test-module',\n      },\n      withModuleName: jest.fn().mockReturnThis(),\n      withDomain: jest.fn().mockReturnThis(),\n      withOrg: jest.fn().mockReturnThis(),\n      withEnv: jest.fn().mockReturnThis(),\n      withSuffix: jest.fn().mockReturnThis(),\n      ssmDomainPath: jest.fn().mockReturnValue('/test/domain/path'),\n      ssmEnvPath: jest.fn().mockReturnValue('/test/env/path'),\n      ssmOrgPath: jest.fn().mockReturnValue('/test-org/'),\n      resourceName: jest\n        .fn()\n        .mockImplementation((suffix?: string) => (suffix ? `test-resource-${suffix}` : 'test-resource')),\n      ssmPath: jest.fn().mockImplementation((path: string) => `/test/${path}`),\n      stackName: jest.fn().mockImplementation((name?: string) => (name ? `test-stack-${name}` : 'test-stack')),\n      exportName: jest.fn().mockImplementation((path: string) => `test-export-${path}`),\n    } as unknown as IMdaaResourceNaming;\n  });\n\n  test('returns unchanged value without refs', () => {\n    const transformer = new MdaaConfigBlueprintRefValueTransformer({\n      naming: mockNaming,\n      scope: stack,\n    });\n    expect(transformer.transformValue('plain-value')).toBe('plain-value');\n  });\n\n  test('returns unchanged value with empty string', () => {\n    const transformer = new MdaaConfigBlueprintRefValueTransformer({\n      naming: mockNaming,\n      scope: stack,\n    });\n    expect(transformer.transformValue('')).toBe('');\n  });\n\n  test('transforms blueprint ref', () => {\n    const transformer = new MdaaConfigBlueprintRefValueTransformer({\n      naming: mockNaming,\n      scope: stack,\n    });\n    const result = transformer.transformValue('{{blueprint:/some/path}}');\n    expect(result).toContain('{{resolve:ssm:/test-org/');\n    expect(result).toContain('/some/path}}');\n    expect(mockNaming.ssmOrgPath).toHaveBeenCalledWith('', false);\n  });\n\n  test('transforms blueprint ref with prefix and suffix', () => {\n    const transformer = new MdaaConfigBlueprintRefValueTransformer({\n      naming: mockNaming,\n      scope: stack,\n    });\n    const result = transformer.transformValue('prefix-{{blueprint:/path}}-suffix');\n    expect(result).toContain('prefix-');\n    expect(result).toContain('-suffix');\n    expect(result).toContain('{{resolve:ssm:');\n  });\n\n  test('handles multiple blueprint refs', () => {\n    const transformer = new MdaaConfigBlueprintRefValueTransformer({\n      naming: mockNaming,\n      scope: stack,\n    });\n    const result = transformer.transformValue('{{blueprint:/path1}}-{{blueprint:/path2}}');\n    expect(result).toContain('/path1}}');\n    expect(result).toContain('/path2}}');\n  });\n\n  test('reuses existing CfnParameter for datazoneEnvironmentProjectId', () => {\n    const transformer = new MdaaConfigBlueprintRefValueTransformer({\n      naming: mockNaming,\n      scope: stack,\n    });\n\n    // First call creates the parameter\n    transformer.transformValue('{{blueprint:/path1}}');\n\n    // Second call should reuse the same parameter\n    transformer.transformValue('{{blueprint:/path2}}');\n\n    // Should only have one datazoneEnvironmentProjectId parameter\n    const params = stack.node.children.filter(child => child.node.id === 'datazoneEnvironmentProjectId');\n    expect(params.length).toBe(1);\n  });\n\n  test('ignores non-blueprint refs', () => {\n    const transformer = new MdaaConfigBlueprintRefValueTransformer({\n      naming: mockNaming,\n      scope: stack,\n    });\n    const result = transformer.transformValue('{{other:value}}');\n    // Non-blueprint refs should remain unchanged\n    expect(result).toBe('{{other:value}}');\n  });\n\n  test('handles nested refs in blueprint path', () => {\n    const transformer = new MdaaConfigBlueprintRefValueTransformer({\n      naming: mockNaming,\n      scope: stack,\n    });\n    // Nested refs should be processed recursively\n    const result = transformer.transformValue('{{blueprint:/{{other}}/path}}');\n    // The inner ref is not a blueprint ref, so it stays as-is in the path\n    expect(result).toContain('{{resolve:ssm:');\n  });\n\n  test('handles unbalanced braces gracefully', () => {\n    const transformer = new MdaaConfigBlueprintRefValueTransformer({\n      naming: mockNaming,\n      scope: stack,\n    });\n    // Unbalanced braces should be skipped\n    expect(transformer.transformValue('{{blueprint:/path')).toBe('{{blueprint:/path');\n    expect(transformer.transformValue('blueprint:/path}}')).toBe('blueprint:/path}}');\n  });\n\n  test('logs resolved blueprint path', () => {\n    const consoleSpy = jest.spyOn(console, 'log').mockImplementation();\n    const transformer = new MdaaConfigBlueprintRefValueTransformer({\n      naming: mockNaming,\n      scope: stack,\n    });\n\n    transformer.transformValue('{{blueprint:/test/path}}');\n\n    expect(consoleSpy).toHaveBeenCalledWith(expect.stringContaining('Resolving blueprint path:'));\n    consoleSpy.mockRestore();\n  });\n\n  test('handles blueprint ref with special characters in path', () => {\n    const transformer = new MdaaConfigBlueprintRefValueTransformer({\n      naming: mockNaming,\n      scope: stack,\n    });\n    const result = transformer.transformValue('{{blueprint:/path/with-dashes_and_underscores/123}}');\n    expect(result).toContain('{{resolve:ssm:');\n    expect(result).toContain('/path/with-dashes_and_underscores/123}}');\n  });\n\n  test('handles blueprint ref at start of string', () => {\n    const transformer = new MdaaConfigBlueprintRefValueTransformer({\n      naming: mockNaming,\n      scope: stack,\n    });\n    const result = transformer.transformValue('{{blueprint:/path}}-suffix');\n    expect(result).toMatch(/^\\{\\{resolve:ssm:/);\n    expect(result).toContain('-suffix');\n  });\n\n  test('handles blueprint ref at end of string', () => {\n    const transformer = new MdaaConfigBlueprintRefValueTransformer({\n      naming: mockNaming,\n      scope: stack,\n    });\n    const result = transformer.transformValue('prefix-{{blueprint:/path}}');\n    expect(result).toContain('prefix-');\n    expect(result).toMatch(/\\}\\}$/);\n  });\n\n  test('handles multiple different ref types mixed', () => {\n    const transformer = new MdaaConfigBlueprintRefValueTransformer({\n      naming: mockNaming,\n      scope: stack,\n    });\n    const result = transformer.transformValue('{{blueprint:/path1}}-{{other:ref}}-{{blueprint:/path2}}');\n    expect(result).toContain('/path1}}');\n    expect(result).toContain('{{other:ref}}');\n    expect(result).toContain('/path2}}');\n  });\n\n  test('handles empty blueprint path', () => {\n    const transformer = new MdaaConfigBlueprintRefValueTransformer({\n      naming: mockNaming,\n      scope: stack,\n    });\n    const result = transformer.transformValue('{{blueprint:}}');\n    expect(result).toContain('{{resolve:ssm:');\n  });\n\n  test('handles blueprint ref with only slashes', () => {\n    const transformer = new MdaaConfigBlueprintRefValueTransformer({\n      naming: mockNaming,\n      scope: stack,\n    });\n    const result = transformer.transformValue('{{blueprint:/}}');\n    expect(result).toContain('{{resolve:ssm:');\n    expect(result).toContain('/}}');\n  });\n\n  test('creates parameter on first blueprint ref', () => {\n    const transformer = new MdaaConfigBlueprintRefValueTransformer({\n      naming: mockNaming,\n      scope: stack,\n    });\n\n    const childrenBefore = stack.node.children.length;\n    transformer.transformValue('{{blueprint:/first}}');\n    const childrenAfter = stack.node.children.length;\n\n    expect(childrenAfter).toBeGreaterThan(childrenBefore);\n  });\n\n  test('does not create duplicate parameters', () => {\n    const transformer = new MdaaConfigBlueprintRefValueTransformer({\n      naming: mockNaming,\n      scope: stack,\n    });\n\n    transformer.transformValue('{{blueprint:/first}}');\n    const childrenAfterFirst = stack.node.children.length;\n\n    transformer.transformValue('{{blueprint:/second}}');\n    const childrenAfterSecond = stack.node.children.length;\n\n    // Should not create additional parameters\n    expect(childrenAfterSecond).toBe(childrenAfterFirst);\n  });\n\n  test('throws when blueprint ref without scope', () => {\n    const transformer = new MdaaConfigBlueprintRefValueTransformer({\n      naming: mockNaming,\n      // eslint-disable-next-line @typescript-eslint/no-explicit-any\n      scope: undefined as any,\n    });\n    expect(() => transformer.transformValue('{{blueprint:/path}}')).toThrow(\n      'Unable to resolve blueprint params outside of a Construct',\n    );\n  });\n\n  test('throws when blueprint ref without naming', () => {\n    const transformer = new MdaaConfigBlueprintRefValueTransformer({\n      // eslint-disable-next-line @typescript-eslint/no-explicit-any\n      naming: undefined as any,\n      scope: stack,\n    });\n    expect(() => transformer.transformValue('{{blueprint:/path}}')).toThrow(\n      'Unable to resolve blueprint params without a naming implementation',\n    );\n  });\n});\n"]}
|
|
@@ -0,0 +1,259 @@
|
|
|
1
|
+
/*!
|
|
2
|
+
* Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
|
|
3
|
+
* SPDX-License-Identifier: Apache-2.0
|
|
4
|
+
*/
|
|
5
|
+
|
|
6
|
+
import { Stack } from 'aws-cdk-lib';
|
|
7
|
+
import { MdaaConfigBlueprintRefValueTransformer } from '../lib';
|
|
8
|
+
import { IMdaaResourceNaming } from '@aws-mdaa/naming';
|
|
9
|
+
|
|
10
|
+
describe('MdaaConfigBlueprintRefValueTransformer', () => {
|
|
11
|
+
let mockNaming: IMdaaResourceNaming;
|
|
12
|
+
let stack: Stack;
|
|
13
|
+
|
|
14
|
+
beforeEach(() => {
|
|
15
|
+
stack = new Stack();
|
|
16
|
+
mockNaming = {
|
|
17
|
+
props: {
|
|
18
|
+
org: 'test-org',
|
|
19
|
+
env: 'test-env',
|
|
20
|
+
domain: 'test-domain',
|
|
21
|
+
moduleName: 'test-module',
|
|
22
|
+
},
|
|
23
|
+
withModuleName: jest.fn().mockReturnThis(),
|
|
24
|
+
withDomain: jest.fn().mockReturnThis(),
|
|
25
|
+
withOrg: jest.fn().mockReturnThis(),
|
|
26
|
+
withEnv: jest.fn().mockReturnThis(),
|
|
27
|
+
withSuffix: jest.fn().mockReturnThis(),
|
|
28
|
+
ssmDomainPath: jest.fn().mockReturnValue('/test/domain/path'),
|
|
29
|
+
ssmEnvPath: jest.fn().mockReturnValue('/test/env/path'),
|
|
30
|
+
ssmOrgPath: jest.fn().mockReturnValue('/test-org/'),
|
|
31
|
+
resourceName: jest
|
|
32
|
+
.fn()
|
|
33
|
+
.mockImplementation((suffix?: string) => (suffix ? `test-resource-${suffix}` : 'test-resource')),
|
|
34
|
+
ssmPath: jest.fn().mockImplementation((path: string) => `/test/${path}`),
|
|
35
|
+
stackName: jest.fn().mockImplementation((name?: string) => (name ? `test-stack-${name}` : 'test-stack')),
|
|
36
|
+
exportName: jest.fn().mockImplementation((path: string) => `test-export-${path}`),
|
|
37
|
+
} as unknown as IMdaaResourceNaming;
|
|
38
|
+
});
|
|
39
|
+
|
|
40
|
+
test('returns unchanged value without refs', () => {
|
|
41
|
+
const transformer = new MdaaConfigBlueprintRefValueTransformer({
|
|
42
|
+
naming: mockNaming,
|
|
43
|
+
scope: stack,
|
|
44
|
+
});
|
|
45
|
+
expect(transformer.transformValue('plain-value')).toBe('plain-value');
|
|
46
|
+
});
|
|
47
|
+
|
|
48
|
+
test('returns unchanged value with empty string', () => {
|
|
49
|
+
const transformer = new MdaaConfigBlueprintRefValueTransformer({
|
|
50
|
+
naming: mockNaming,
|
|
51
|
+
scope: stack,
|
|
52
|
+
});
|
|
53
|
+
expect(transformer.transformValue('')).toBe('');
|
|
54
|
+
});
|
|
55
|
+
|
|
56
|
+
test('transforms blueprint ref', () => {
|
|
57
|
+
const transformer = new MdaaConfigBlueprintRefValueTransformer({
|
|
58
|
+
naming: mockNaming,
|
|
59
|
+
scope: stack,
|
|
60
|
+
});
|
|
61
|
+
const result = transformer.transformValue('{{blueprint:/some/path}}');
|
|
62
|
+
expect(result).toContain('{{resolve:ssm:/test-org/');
|
|
63
|
+
expect(result).toContain('/some/path}}');
|
|
64
|
+
expect(mockNaming.ssmOrgPath).toHaveBeenCalledWith('', false);
|
|
65
|
+
});
|
|
66
|
+
|
|
67
|
+
test('transforms blueprint ref with prefix and suffix', () => {
|
|
68
|
+
const transformer = new MdaaConfigBlueprintRefValueTransformer({
|
|
69
|
+
naming: mockNaming,
|
|
70
|
+
scope: stack,
|
|
71
|
+
});
|
|
72
|
+
const result = transformer.transformValue('prefix-{{blueprint:/path}}-suffix');
|
|
73
|
+
expect(result).toContain('prefix-');
|
|
74
|
+
expect(result).toContain('-suffix');
|
|
75
|
+
expect(result).toContain('{{resolve:ssm:');
|
|
76
|
+
});
|
|
77
|
+
|
|
78
|
+
test('handles multiple blueprint refs', () => {
|
|
79
|
+
const transformer = new MdaaConfigBlueprintRefValueTransformer({
|
|
80
|
+
naming: mockNaming,
|
|
81
|
+
scope: stack,
|
|
82
|
+
});
|
|
83
|
+
const result = transformer.transformValue('{{blueprint:/path1}}-{{blueprint:/path2}}');
|
|
84
|
+
expect(result).toContain('/path1}}');
|
|
85
|
+
expect(result).toContain('/path2}}');
|
|
86
|
+
});
|
|
87
|
+
|
|
88
|
+
test('reuses existing CfnParameter for datazoneEnvironmentProjectId', () => {
|
|
89
|
+
const transformer = new MdaaConfigBlueprintRefValueTransformer({
|
|
90
|
+
naming: mockNaming,
|
|
91
|
+
scope: stack,
|
|
92
|
+
});
|
|
93
|
+
|
|
94
|
+
// First call creates the parameter
|
|
95
|
+
transformer.transformValue('{{blueprint:/path1}}');
|
|
96
|
+
|
|
97
|
+
// Second call should reuse the same parameter
|
|
98
|
+
transformer.transformValue('{{blueprint:/path2}}');
|
|
99
|
+
|
|
100
|
+
// Should only have one datazoneEnvironmentProjectId parameter
|
|
101
|
+
const params = stack.node.children.filter(child => child.node.id === 'datazoneEnvironmentProjectId');
|
|
102
|
+
expect(params.length).toBe(1);
|
|
103
|
+
});
|
|
104
|
+
|
|
105
|
+
test('ignores non-blueprint refs', () => {
|
|
106
|
+
const transformer = new MdaaConfigBlueprintRefValueTransformer({
|
|
107
|
+
naming: mockNaming,
|
|
108
|
+
scope: stack,
|
|
109
|
+
});
|
|
110
|
+
const result = transformer.transformValue('{{other:value}}');
|
|
111
|
+
// Non-blueprint refs should remain unchanged
|
|
112
|
+
expect(result).toBe('{{other:value}}');
|
|
113
|
+
});
|
|
114
|
+
|
|
115
|
+
test('handles nested refs in blueprint path', () => {
|
|
116
|
+
const transformer = new MdaaConfigBlueprintRefValueTransformer({
|
|
117
|
+
naming: mockNaming,
|
|
118
|
+
scope: stack,
|
|
119
|
+
});
|
|
120
|
+
// Nested refs should be processed recursively
|
|
121
|
+
const result = transformer.transformValue('{{blueprint:/{{other}}/path}}');
|
|
122
|
+
// The inner ref is not a blueprint ref, so it stays as-is in the path
|
|
123
|
+
expect(result).toContain('{{resolve:ssm:');
|
|
124
|
+
});
|
|
125
|
+
|
|
126
|
+
test('handles unbalanced braces gracefully', () => {
|
|
127
|
+
const transformer = new MdaaConfigBlueprintRefValueTransformer({
|
|
128
|
+
naming: mockNaming,
|
|
129
|
+
scope: stack,
|
|
130
|
+
});
|
|
131
|
+
// Unbalanced braces should be skipped
|
|
132
|
+
expect(transformer.transformValue('{{blueprint:/path')).toBe('{{blueprint:/path');
|
|
133
|
+
expect(transformer.transformValue('blueprint:/path}}')).toBe('blueprint:/path}}');
|
|
134
|
+
});
|
|
135
|
+
|
|
136
|
+
test('logs resolved blueprint path', () => {
|
|
137
|
+
const consoleSpy = jest.spyOn(console, 'log').mockImplementation();
|
|
138
|
+
const transformer = new MdaaConfigBlueprintRefValueTransformer({
|
|
139
|
+
naming: mockNaming,
|
|
140
|
+
scope: stack,
|
|
141
|
+
});
|
|
142
|
+
|
|
143
|
+
transformer.transformValue('{{blueprint:/test/path}}');
|
|
144
|
+
|
|
145
|
+
expect(consoleSpy).toHaveBeenCalledWith(expect.stringContaining('Resolving blueprint path:'));
|
|
146
|
+
consoleSpy.mockRestore();
|
|
147
|
+
});
|
|
148
|
+
|
|
149
|
+
test('handles blueprint ref with special characters in path', () => {
|
|
150
|
+
const transformer = new MdaaConfigBlueprintRefValueTransformer({
|
|
151
|
+
naming: mockNaming,
|
|
152
|
+
scope: stack,
|
|
153
|
+
});
|
|
154
|
+
const result = transformer.transformValue('{{blueprint:/path/with-dashes_and_underscores/123}}');
|
|
155
|
+
expect(result).toContain('{{resolve:ssm:');
|
|
156
|
+
expect(result).toContain('/path/with-dashes_and_underscores/123}}');
|
|
157
|
+
});
|
|
158
|
+
|
|
159
|
+
test('handles blueprint ref at start of string', () => {
|
|
160
|
+
const transformer = new MdaaConfigBlueprintRefValueTransformer({
|
|
161
|
+
naming: mockNaming,
|
|
162
|
+
scope: stack,
|
|
163
|
+
});
|
|
164
|
+
const result = transformer.transformValue('{{blueprint:/path}}-suffix');
|
|
165
|
+
expect(result).toMatch(/^\{\{resolve:ssm:/);
|
|
166
|
+
expect(result).toContain('-suffix');
|
|
167
|
+
});
|
|
168
|
+
|
|
169
|
+
test('handles blueprint ref at end of string', () => {
|
|
170
|
+
const transformer = new MdaaConfigBlueprintRefValueTransformer({
|
|
171
|
+
naming: mockNaming,
|
|
172
|
+
scope: stack,
|
|
173
|
+
});
|
|
174
|
+
const result = transformer.transformValue('prefix-{{blueprint:/path}}');
|
|
175
|
+
expect(result).toContain('prefix-');
|
|
176
|
+
expect(result).toMatch(/\}\}$/);
|
|
177
|
+
});
|
|
178
|
+
|
|
179
|
+
test('handles multiple different ref types mixed', () => {
|
|
180
|
+
const transformer = new MdaaConfigBlueprintRefValueTransformer({
|
|
181
|
+
naming: mockNaming,
|
|
182
|
+
scope: stack,
|
|
183
|
+
});
|
|
184
|
+
const result = transformer.transformValue('{{blueprint:/path1}}-{{other:ref}}-{{blueprint:/path2}}');
|
|
185
|
+
expect(result).toContain('/path1}}');
|
|
186
|
+
expect(result).toContain('{{other:ref}}');
|
|
187
|
+
expect(result).toContain('/path2}}');
|
|
188
|
+
});
|
|
189
|
+
|
|
190
|
+
test('handles empty blueprint path', () => {
|
|
191
|
+
const transformer = new MdaaConfigBlueprintRefValueTransformer({
|
|
192
|
+
naming: mockNaming,
|
|
193
|
+
scope: stack,
|
|
194
|
+
});
|
|
195
|
+
const result = transformer.transformValue('{{blueprint:}}');
|
|
196
|
+
expect(result).toContain('{{resolve:ssm:');
|
|
197
|
+
});
|
|
198
|
+
|
|
199
|
+
test('handles blueprint ref with only slashes', () => {
|
|
200
|
+
const transformer = new MdaaConfigBlueprintRefValueTransformer({
|
|
201
|
+
naming: mockNaming,
|
|
202
|
+
scope: stack,
|
|
203
|
+
});
|
|
204
|
+
const result = transformer.transformValue('{{blueprint:/}}');
|
|
205
|
+
expect(result).toContain('{{resolve:ssm:');
|
|
206
|
+
expect(result).toContain('/}}');
|
|
207
|
+
});
|
|
208
|
+
|
|
209
|
+
test('creates parameter on first blueprint ref', () => {
|
|
210
|
+
const transformer = new MdaaConfigBlueprintRefValueTransformer({
|
|
211
|
+
naming: mockNaming,
|
|
212
|
+
scope: stack,
|
|
213
|
+
});
|
|
214
|
+
|
|
215
|
+
const childrenBefore = stack.node.children.length;
|
|
216
|
+
transformer.transformValue('{{blueprint:/first}}');
|
|
217
|
+
const childrenAfter = stack.node.children.length;
|
|
218
|
+
|
|
219
|
+
expect(childrenAfter).toBeGreaterThan(childrenBefore);
|
|
220
|
+
});
|
|
221
|
+
|
|
222
|
+
test('does not create duplicate parameters', () => {
|
|
223
|
+
const transformer = new MdaaConfigBlueprintRefValueTransformer({
|
|
224
|
+
naming: mockNaming,
|
|
225
|
+
scope: stack,
|
|
226
|
+
});
|
|
227
|
+
|
|
228
|
+
transformer.transformValue('{{blueprint:/first}}');
|
|
229
|
+
const childrenAfterFirst = stack.node.children.length;
|
|
230
|
+
|
|
231
|
+
transformer.transformValue('{{blueprint:/second}}');
|
|
232
|
+
const childrenAfterSecond = stack.node.children.length;
|
|
233
|
+
|
|
234
|
+
// Should not create additional parameters
|
|
235
|
+
expect(childrenAfterSecond).toBe(childrenAfterFirst);
|
|
236
|
+
});
|
|
237
|
+
|
|
238
|
+
test('throws when blueprint ref without scope', () => {
|
|
239
|
+
const transformer = new MdaaConfigBlueprintRefValueTransformer({
|
|
240
|
+
naming: mockNaming,
|
|
241
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
242
|
+
scope: undefined as any,
|
|
243
|
+
});
|
|
244
|
+
expect(() => transformer.transformValue('{{blueprint:/path}}')).toThrow(
|
|
245
|
+
'Unable to resolve blueprint params outside of a Construct',
|
|
246
|
+
);
|
|
247
|
+
});
|
|
248
|
+
|
|
249
|
+
test('throws when blueprint ref without naming', () => {
|
|
250
|
+
const transformer = new MdaaConfigBlueprintRefValueTransformer({
|
|
251
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
252
|
+
naming: undefined as any,
|
|
253
|
+
scope: stack,
|
|
254
|
+
});
|
|
255
|
+
expect(() => transformer.transformValue('{{blueprint:/path}}')).toThrow(
|
|
256
|
+
'Unable to resolve blueprint params without a naming implementation',
|
|
257
|
+
);
|
|
258
|
+
});
|
|
259
|
+
});
|