pulumi/sdk/nodejs/runtime/mocks.ts
2020-04-03 08:33:40 +02:00

134 lines
5.8 KiB
TypeScript

// Copyright 2016-2018, Pulumi Corporation.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
import { deserializeProperties, serializeProperties } from "./rpc";
import { getProject, getStack, setMockOptions } from "./settings";
const provproto = require("../proto/provider_pb.js");
const resproto = require("../proto/resource_pb.js");
const structproto = require("google-protobuf/google/protobuf/struct_pb.js");
/**
* Mocks is an abstract class that allows subclasses to replace operations normally implemented by the Pulumi engine with
* their own implementations. This can be used during testing to ensure that calls to provider functions and resource constructors
* return predictable values.
*/
export interface Mocks {
/**
* call mocks provider-implemented function calls (e.g. aws.get_availability_zones).
*
* @param token: The token that indicates which function is being called. This token is of the form "package:module:function".
* @param args: The arguments provided to the function call.
* @param provider: If provided, the identifier of the provider instance being used to make the call.
*/
call(token: string, args: any, provider?: string): Record<string, any>;
/**
* new_resource mocks resource construction calls. This function should return the physical identifier and the output properties
* for the resource being constructed.
*
* @param type_: The token that indicates which resource type is being constructed. This token is of the form "package:module:type".
* @param name: The logical name of the resource instance.
* @param inputs: The inputs for the resource.
* @param provider: If provided, the identifier of the provider instnace being used to manage this resource.
* @param id_: If provided, the physical identifier of an existing resource to read or import.
*/
newResource(type: string, name: string, inputs: any, provider?: string, id?: string): { id: string, state: Record<string, any> };
}
export class MockMonitor {
mocks: Mocks;
constructor(mocks: Mocks) {
this.mocks = mocks;
}
private newUrn(parent: string, type: string, name: string): string {
if (parent) {
const qualifiedType = parent.split("::")[2];
const parentType = qualifiedType.split("$").pop();
type = parentType + "$" + type;
}
return "urn:pulumi:" + [getStack(), getProject(), type, name].join("::");
}
public async invoke(req: any, callback: (err: any, innerResponse: any) => void) {
try {
const result = this.mocks.call(req.getTok(), deserializeProperties(req.getArgs()), req.getProvider());
const response = new provproto.InvokeResponse();
response.setReturn(structproto.Struct.fromJavaScript(await serializeProperties("", result)));
callback(null, response);
} catch (err) {
callback(err, undefined);
}
}
public async readResource(req: any, callback: (err: any, innterResponse: any) => void) {
try {
const result = this.mocks.newResource(
req.getType(),
req.getName(),
deserializeProperties(req.getProperties()),
req.getProvider(),
req.getId());
const response = new resproto.ReadResourceResponse();
response.setUrn(this.newUrn(req.getParent(), req.getType(), req.getName()));
response.setProperties(structproto.Struct.fromJavaScript(await serializeProperties("", result.state)));
callback(null, response);
} catch (err) {
callback(err, undefined);
}
}
public async registerResource(req: any, callback: (err: any, innerResponse: any) => void) {
try {
const result = this.mocks.newResource(
req.getType(),
req.getName(),
deserializeProperties(req.getObject()),
req.getProvider(),
req.getImportid());
const response = new resproto.RegisterResourceResponse();
response.setUrn(this.newUrn(req.getParent(), req.getType(), req.getName()));
response.setId(result.id);
response.setObject(structproto.Struct.fromJavaScript(await serializeProperties("", result.state)));
callback(null, response);
} catch (err) {
callback(err, undefined);
}
}
public registerResourceOutputs(req: any, callback: (err: any, innerResponse: any) => void) {
callback(null, {});
}
public supportsFeature(req: any, callback: (err: any, innerResponse: any) => void) {
callback(null, {
getHassupport: () => true,
});
}
}
/**
* setMocks configures the Pulumi runtime to use the given mocks for testing.
*
* @param mocks: The mocks to use for calls to provider functions and resource consrtuction.
* @param project: If provided, the name of the Pulumi project. Defaults to "project".
* @param stack: If provided, the name of the Pulumi stack. Defaults to "stack".
* @param preview: If provided, indicates whether or not the program is running a preview. Defaults to false.
*/
export function setMocks(mocks: Mocks, project?: string, stack?: string, preview?: boolean) {
setMockOptions(new MockMonitor(mocks), project, stack, preview);
}