codify-plugin-lib 1.0.48 → 1.0.50
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/dist/entities/plugin.js +3 -3
- package/dist/entities/resource-types.d.ts +2 -0
- package/dist/entities/resource.d.ts +3 -0
- package/dist/entities/resource.js +17 -0
- package/dist/entities/transform-parameter.d.ts +4 -0
- package/dist/entities/transform-parameter.js +2 -0
- package/package.json +1 -1
- package/src/entities/plugin.ts +3 -3
- package/src/entities/resource-parameters.test.ts +43 -0
- package/src/entities/resource-types.ts +2 -0
- package/src/entities/resource.ts +23 -1
- package/src/entities/transform-parameter.ts +7 -0
package/dist/entities/plugin.js
CHANGED
|
@@ -65,10 +65,10 @@ export class Plugin {
|
|
|
65
65
|
}
|
|
66
66
|
return this.planStorage.get(planId);
|
|
67
67
|
}
|
|
68
|
-
if (!planRequest?.
|
|
69
|
-
throw new Error('Malformed plan. Resource
|
|
68
|
+
if (!planRequest?.resourceType || !this.resources.has(planRequest.resourceType)) {
|
|
69
|
+
throw new Error('Malformed plan. Resource type must be supplied');
|
|
70
70
|
}
|
|
71
|
-
const resource = this.resources.get(planRequest.
|
|
71
|
+
const resource = this.resources.get(planRequest.resourceType);
|
|
72
72
|
return Plan.fromResponse(data.plan, resource?.defaultValues);
|
|
73
73
|
}
|
|
74
74
|
async crossValidateResources(configs) { }
|
|
@@ -1,5 +1,6 @@
|
|
|
1
1
|
import { StatefulParameter } from './stateful-parameter.js';
|
|
2
2
|
import { ResourceOperation, StringIndexedObject } from 'codify-schemas';
|
|
3
|
+
import { TransformParameter } from './transform-parameter.js';
|
|
3
4
|
export type ErrorMessage = string;
|
|
4
5
|
export interface ResourceParameterConfiguration {
|
|
5
6
|
planOperation?: ResourceOperation.MODIFY | ResourceOperation.RECREATE;
|
|
@@ -11,6 +12,7 @@ export interface ResourceConfiguration<T extends StringIndexedObject> {
|
|
|
11
12
|
callStatefulParameterRemoveOnDestroy?: boolean;
|
|
12
13
|
dependencies?: string[];
|
|
13
14
|
statefulParameters?: Array<StatefulParameter<T, T[keyof T]>>;
|
|
15
|
+
transformParameters?: Partial<Record<keyof T, TransformParameter<T>>>;
|
|
14
16
|
parameterConfigurations?: Partial<Record<keyof T, ResourceParameterConfiguration>>;
|
|
15
17
|
}
|
|
16
18
|
export interface ResourceDefinition {
|
|
@@ -3,9 +3,11 @@ import { Plan } from './plan.js';
|
|
|
3
3
|
import { StatefulParameter } from './stateful-parameter.js';
|
|
4
4
|
import { ResourceConfiguration, ValidationResult } from './resource-types.js';
|
|
5
5
|
import { ParameterConfiguration } from './plan-types.js';
|
|
6
|
+
import { TransformParameter } from './transform-parameter.js';
|
|
6
7
|
export declare abstract class Resource<T extends StringIndexedObject> {
|
|
7
8
|
readonly typeId: string;
|
|
8
9
|
readonly statefulParameters: Map<keyof T, StatefulParameter<T, T[keyof T]>>;
|
|
10
|
+
readonly transformParameters: Map<keyof T, TransformParameter<T>>;
|
|
9
11
|
readonly dependencies: string[];
|
|
10
12
|
readonly parameterConfigurations: Record<keyof T, ParameterConfiguration>;
|
|
11
13
|
readonly configuration: ResourceConfiguration<T>;
|
|
@@ -21,6 +23,7 @@ export declare abstract class Resource<T extends StringIndexedObject> {
|
|
|
21
23
|
private initializeDefaultValues;
|
|
22
24
|
private validateResourceConfiguration;
|
|
23
25
|
private validateRefreshResults;
|
|
26
|
+
private applyTransformParameters;
|
|
24
27
|
private addDefaultValues;
|
|
25
28
|
abstract validate(parameters: unknown): Promise<ValidationResult>;
|
|
26
29
|
abstract refresh(keys: Map<keyof T, T[keyof T]>): Promise<Partial<T> | null>;
|
|
@@ -4,6 +4,7 @@ import { setsEqual, splitUserConfig } from '../utils/utils.js';
|
|
|
4
4
|
export class Resource {
|
|
5
5
|
typeId;
|
|
6
6
|
statefulParameters;
|
|
7
|
+
transformParameters;
|
|
7
8
|
dependencies;
|
|
8
9
|
parameterConfigurations;
|
|
9
10
|
configuration;
|
|
@@ -12,6 +13,7 @@ export class Resource {
|
|
|
12
13
|
this.validateResourceConfiguration(configuration);
|
|
13
14
|
this.typeId = configuration.type;
|
|
14
15
|
this.statefulParameters = new Map(configuration.statefulParameters?.map((sp) => [sp.name, sp]));
|
|
16
|
+
this.transformParameters = new Map(Object.entries(configuration.transformParameters ?? {}));
|
|
15
17
|
this.parameterConfigurations = this.initializeParameterConfigurations(configuration);
|
|
16
18
|
this.defaultValues = this.initializeDefaultValues(configuration);
|
|
17
19
|
this.dependencies = configuration.dependencies ?? [];
|
|
@@ -25,6 +27,7 @@ export class Resource {
|
|
|
25
27
|
};
|
|
26
28
|
const { resourceMetadata, parameters: desiredParameters } = splitUserConfig(desiredConfig);
|
|
27
29
|
this.addDefaultValues(desiredParameters);
|
|
30
|
+
await this.applyTransformParameters(desiredParameters);
|
|
28
31
|
const resourceParameters = Object.fromEntries([
|
|
29
32
|
...Object.entries(desiredParameters).filter(([key]) => !this.statefulParameters.has(key)),
|
|
30
33
|
]);
|
|
@@ -174,6 +177,20 @@ Missing: ${[...desiredKeys].filter((k) => !refreshKeys.has(k))};
|
|
|
174
177
|
Additional: ${[...refreshKeys].filter(k => !desiredKeys.has(k))};`);
|
|
175
178
|
}
|
|
176
179
|
}
|
|
180
|
+
async applyTransformParameters(desired) {
|
|
181
|
+
for (const [key, tp] of this.transformParameters.entries()) {
|
|
182
|
+
if (desired[key] !== null) {
|
|
183
|
+
const transformedValue = await tp.transform(desired[key]);
|
|
184
|
+
if (Object.keys(transformedValue).some((k) => desired[k] !== undefined)) {
|
|
185
|
+
throw new Error(`Transform parameter ${key} is attempting to override existing value ${desired[key]}`);
|
|
186
|
+
}
|
|
187
|
+
Object.entries(transformedValue).forEach(([tvKey, tvValue]) => {
|
|
188
|
+
desired[tvKey] = tvValue;
|
|
189
|
+
});
|
|
190
|
+
delete desired[key];
|
|
191
|
+
}
|
|
192
|
+
}
|
|
193
|
+
}
|
|
177
194
|
addDefaultValues(desired) {
|
|
178
195
|
Object.entries(this.defaultValues)
|
|
179
196
|
.forEach(([key, defaultValue]) => {
|
package/package.json
CHANGED
package/src/entities/plugin.ts
CHANGED
|
@@ -94,11 +94,11 @@ export class Plugin {
|
|
|
94
94
|
return this.planStorage.get(planId)!
|
|
95
95
|
}
|
|
96
96
|
|
|
97
|
-
if (!planRequest?.
|
|
98
|
-
throw new Error('Malformed plan. Resource
|
|
97
|
+
if (!planRequest?.resourceType || !this.resources.has(planRequest.resourceType)) {
|
|
98
|
+
throw new Error('Malformed plan. Resource type must be supplied');
|
|
99
99
|
}
|
|
100
100
|
|
|
101
|
-
const resource = this.resources.get(planRequest.
|
|
101
|
+
const resource = this.resources.get(planRequest.resourceType);
|
|
102
102
|
return Plan.fromResponse(data.plan, resource?.defaultValues!);
|
|
103
103
|
}
|
|
104
104
|
|
|
@@ -4,6 +4,7 @@ import { Plan } from './plan.js';
|
|
|
4
4
|
import { spy } from 'sinon';
|
|
5
5
|
import { ResourceOperation } from 'codify-schemas';
|
|
6
6
|
import { TestConfig, TestResource } from './resource.test.js';
|
|
7
|
+
import { TransformParameter } from './transform-parameter.js';
|
|
7
8
|
|
|
8
9
|
class TestParameter extends StatefulParameter<TestConfig, string> {
|
|
9
10
|
constructor(configuration?: StatefulParameterConfiguration<TestConfig>) {
|
|
@@ -198,4 +199,46 @@ describe('Resource parameters tests', () => {
|
|
|
198
199
|
}
|
|
199
200
|
})
|
|
200
201
|
})
|
|
202
|
+
|
|
203
|
+
it('Supports transform parameters', async () => {
|
|
204
|
+
const transformParameter = new class extends TransformParameter<TestConfig> {
|
|
205
|
+
async transform(value: any): Promise<Partial<TestConfig>> {
|
|
206
|
+
return {
|
|
207
|
+
propA: 'propA',
|
|
208
|
+
propB: 10,
|
|
209
|
+
}
|
|
210
|
+
}
|
|
211
|
+
}
|
|
212
|
+
|
|
213
|
+
const resource = spy(new class extends TestResource {
|
|
214
|
+
constructor() {
|
|
215
|
+
super({
|
|
216
|
+
type: 'resourceType',
|
|
217
|
+
transformParameters: {
|
|
218
|
+
propC: transformParameter
|
|
219
|
+
},
|
|
220
|
+
});
|
|
221
|
+
}
|
|
222
|
+
|
|
223
|
+
async refresh(): Promise<Partial<TestConfig> | null> {
|
|
224
|
+
return {
|
|
225
|
+
propA: 'propA',
|
|
226
|
+
propB: 10,
|
|
227
|
+
}
|
|
228
|
+
}
|
|
229
|
+
});
|
|
230
|
+
|
|
231
|
+
const plan = await resource.plan({ type: 'resourceType', propC: 'abc' } as any);
|
|
232
|
+
|
|
233
|
+
expect(resource.refresh.called).to.be.true;
|
|
234
|
+
expect(resource.refresh.getCall(0).firstArg.has('propA')).to.be.true;
|
|
235
|
+
expect(resource.refresh.getCall(0).firstArg.has('propB')).to.be.true;
|
|
236
|
+
expect(resource.refresh.getCall(0).firstArg.has('propC')).to.be.false;
|
|
237
|
+
|
|
238
|
+
expect(plan.desiredConfig.propA).to.eq('propA');
|
|
239
|
+
expect(plan.desiredConfig.propB).to.eq(10);
|
|
240
|
+
expect(plan.desiredConfig.propC).to.be.undefined;
|
|
241
|
+
|
|
242
|
+
expect(plan.changeSet.operation).to.eq(ResourceOperation.NOOP);
|
|
243
|
+
})
|
|
201
244
|
})
|
|
@@ -1,5 +1,6 @@
|
|
|
1
1
|
import { StatefulParameter } from './stateful-parameter.js';
|
|
2
2
|
import { ResourceOperation, StringIndexedObject } from 'codify-schemas';
|
|
3
|
+
import { TransformParameter } from './transform-parameter.js';
|
|
3
4
|
|
|
4
5
|
export type ErrorMessage = string;
|
|
5
6
|
|
|
@@ -35,6 +36,7 @@ export interface ResourceConfiguration<T extends StringIndexedObject> {
|
|
|
35
36
|
callStatefulParameterRemoveOnDestroy?: boolean,
|
|
36
37
|
dependencies?: string[];
|
|
37
38
|
statefulParameters?: Array<StatefulParameter<T, T[keyof T]>>;
|
|
39
|
+
transformParameters?: Partial<Record<keyof T, TransformParameter<T>>>
|
|
38
40
|
parameterConfigurations?: Partial<Record<keyof T, ResourceParameterConfiguration>>
|
|
39
41
|
}
|
|
40
42
|
|
package/src/entities/resource.ts
CHANGED
|
@@ -5,6 +5,7 @@ import { StatefulParameter } from './stateful-parameter.js';
|
|
|
5
5
|
import { ResourceConfiguration, ValidationResult } from './resource-types.js';
|
|
6
6
|
import { setsEqual, splitUserConfig } from '../utils/utils.js';
|
|
7
7
|
import { ParameterConfiguration, PlanConfiguration } from './plan-types.js';
|
|
8
|
+
import { TransformParameter } from './transform-parameter.js';
|
|
8
9
|
|
|
9
10
|
/**
|
|
10
11
|
* Description of resource here
|
|
@@ -17,6 +18,7 @@ export abstract class Resource<T extends StringIndexedObject> {
|
|
|
17
18
|
|
|
18
19
|
readonly typeId: string;
|
|
19
20
|
readonly statefulParameters: Map<keyof T, StatefulParameter<T, T[keyof T]>>;
|
|
21
|
+
readonly transformParameters: Map<keyof T, TransformParameter<T>>
|
|
20
22
|
readonly dependencies: string[]; // TODO: Change this to a string
|
|
21
23
|
readonly parameterConfigurations: Record<keyof T, ParameterConfiguration>
|
|
22
24
|
readonly configuration: ResourceConfiguration<T>;
|
|
@@ -27,6 +29,7 @@ export abstract class Resource<T extends StringIndexedObject> {
|
|
|
27
29
|
|
|
28
30
|
this.typeId = configuration.type;
|
|
29
31
|
this.statefulParameters = new Map(configuration.statefulParameters?.map((sp) => [sp.name, sp]));
|
|
32
|
+
this.transformParameters = new Map(Object.entries(configuration.transformParameters ?? {})) as Map<keyof T, TransformParameter<T>>;
|
|
30
33
|
this.parameterConfigurations = this.initializeParameterConfigurations(configuration);
|
|
31
34
|
this.defaultValues = this.initializeDefaultValues(configuration);
|
|
32
35
|
|
|
@@ -50,6 +53,7 @@ export abstract class Resource<T extends StringIndexedObject> {
|
|
|
50
53
|
const { resourceMetadata, parameters: desiredParameters } = splitUserConfig(desiredConfig);
|
|
51
54
|
|
|
52
55
|
this.addDefaultValues(desiredParameters);
|
|
56
|
+
await this.applyTransformParameters(desiredParameters);
|
|
53
57
|
|
|
54
58
|
const resourceParameters = Object.fromEntries([
|
|
55
59
|
...Object.entries(desiredParameters).filter(([key]) => !this.statefulParameters.has(key)),
|
|
@@ -210,7 +214,6 @@ export abstract class Resource<T extends StringIndexedObject> {
|
|
|
210
214
|
...resourceParameters,
|
|
211
215
|
...statefulParameters,
|
|
212
216
|
}
|
|
213
|
-
|
|
214
217
|
}
|
|
215
218
|
|
|
216
219
|
private initializeDefaultValues(
|
|
@@ -258,6 +261,25 @@ Additional: ${[...refreshKeys].filter(k => !desiredKeys.has(k))};`
|
|
|
258
261
|
}
|
|
259
262
|
}
|
|
260
263
|
|
|
264
|
+
private async applyTransformParameters(desired: Partial<T>): Promise<void> {
|
|
265
|
+
for (const [key, tp] of this.transformParameters.entries()) {
|
|
266
|
+
if (desired[key] !== null) {
|
|
267
|
+
const transformedValue = await tp.transform(desired[key]);
|
|
268
|
+
|
|
269
|
+
if (Object.keys(transformedValue).some((k) => desired[k] !== undefined)) {
|
|
270
|
+
throw new Error(`Transform parameter ${key as string} is attempting to override existing value ${desired[key]}`);
|
|
271
|
+
}
|
|
272
|
+
|
|
273
|
+
Object.entries(transformedValue).forEach(([tvKey, tvValue]) => {
|
|
274
|
+
// @ts-ignore
|
|
275
|
+
desired[tvKey] = tvValue;
|
|
276
|
+
})
|
|
277
|
+
|
|
278
|
+
delete desired[key];
|
|
279
|
+
}
|
|
280
|
+
}
|
|
281
|
+
}
|
|
282
|
+
|
|
261
283
|
private addDefaultValues(desired: Partial<T>): void {
|
|
262
284
|
Object.entries(this.defaultValues)
|
|
263
285
|
.forEach(([key, defaultValue]) => {
|