mirror of https://github.com/pulumi/pulumi.git
250 lines
8.8 KiB
TypeScript
250 lines
8.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";
|
|
|
|
import * as structproto from "google-protobuf/google/protobuf/struct_pb";
|
|
import * as provproto from "../proto/provider_pb";
|
|
import * as resproto from "../proto/resource_pb";
|
|
|
|
/**
|
|
* MockResourceArgs is used to construct a newResource Mock.
|
|
*/
|
|
export interface MockResourceArgs {
|
|
/**
|
|
* The token that indicates which resource type is being constructed. This token is of the form "package:module:type".
|
|
*/
|
|
type: string;
|
|
|
|
/**
|
|
* The logical name of the resource instance.
|
|
*/
|
|
name: string;
|
|
|
|
/**
|
|
* The inputs for the resource.
|
|
*/
|
|
inputs: any;
|
|
|
|
/**
|
|
* If provided, the identifier of the provider instance being used to manage this resource.
|
|
*/
|
|
provider?: string;
|
|
|
|
/**
|
|
* Specifies whether or not the resource is Custom (i.e. managed by a resource provider).
|
|
*/
|
|
custom?: boolean;
|
|
|
|
/**
|
|
* If provided, the physical identifier of an existing resource to read or import.
|
|
*/
|
|
id?: string;
|
|
}
|
|
|
|
/**
|
|
* MockResourceResult is the result of a newResource Mock, returning a physical identifier and the output properties
|
|
* for the resource being constructed.
|
|
*/
|
|
export type MockResourceResult = {
|
|
id: string | undefined;
|
|
state: Record<string, any>;
|
|
};
|
|
|
|
/**
|
|
* MockResourceArgs is used to construct call Mock.
|
|
*/
|
|
export interface MockCallArgs {
|
|
/**
|
|
* The token that indicates which function is being called. This token is of the form "package:module:function".
|
|
*/
|
|
token: string;
|
|
|
|
/**
|
|
* The arguments provided to the function call.
|
|
*/
|
|
inputs: any;
|
|
|
|
/**
|
|
* If provided, the identifier of the provider instance being used to make the call.
|
|
*/
|
|
provider?: string;
|
|
}
|
|
|
|
/**
|
|
* MockCallResult is the result of a call Mock.
|
|
*/
|
|
export type MockCallResult = Record<string, any>;
|
|
|
|
/**
|
|
* Mocks allows implementations 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 {
|
|
/**
|
|
* Mocks provider-implemented function calls (e.g. aws.get_availability_zones).
|
|
*
|
|
* @param args: MockCallArgs
|
|
*/
|
|
call(args: MockCallArgs): MockCallResult | Promise<MockCallResult>;
|
|
/**
|
|
* Mocks resource construction calls. This function should return the physical identifier and the output properties
|
|
* for the resource being constructed.
|
|
*
|
|
* @param args: MockResourceArgs
|
|
*/
|
|
newResource(args: MockResourceArgs): MockResourceResult | Promise<MockResourceResult>;
|
|
}
|
|
|
|
export class MockMonitor {
|
|
readonly resources = new Map<string, { urn: string; id: string | null; state: any }>();
|
|
|
|
constructor(readonly 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 tok = req.getTok();
|
|
const inputs = deserializeProperties(req.getArgs());
|
|
|
|
if (tok === "pulumi:pulumi:getResource") {
|
|
const registeredResource = this.resources.get(inputs.urn);
|
|
if (!registeredResource) {
|
|
throw new Error(`unknown resource ${inputs.urn}`);
|
|
}
|
|
const resp = new provproto.InvokeResponse();
|
|
resp.setReturn(structproto.Struct.fromJavaScript(registeredResource));
|
|
callback(null, resp);
|
|
return;
|
|
}
|
|
|
|
const result: MockCallResult = await this.mocks.call({
|
|
token: tok,
|
|
inputs: inputs,
|
|
provider: 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 {
|
|
let custom = false;
|
|
if (typeof req.getCustom === "function") {
|
|
custom = req.getCustom();
|
|
}
|
|
|
|
const result: MockResourceResult = await this.mocks.newResource({
|
|
type: req.getType(),
|
|
name: req.getName(),
|
|
inputs: deserializeProperties(req.getProperties()),
|
|
provider: req.getProvider(),
|
|
custom: custom,
|
|
id: req.getId(),
|
|
});
|
|
|
|
const urn = this.newUrn(req.getParent(), req.getType(), req.getName());
|
|
const serializedState = await serializeProperties("", result.state);
|
|
|
|
this.resources.set(urn, { urn, id: result.id ?? null, state: serializedState });
|
|
|
|
const response = new resproto.ReadResourceResponse();
|
|
response.setUrn(urn);
|
|
response.setProperties(structproto.Struct.fromJavaScript(serializedState));
|
|
callback(null, response);
|
|
} catch (err) {
|
|
callback(err, undefined);
|
|
}
|
|
}
|
|
|
|
public async registerResource(req: any, callback: (err: any, innerResponse: any) => void) {
|
|
try {
|
|
const result: MockResourceResult = await this.mocks.newResource({
|
|
type: req.getType(),
|
|
name: req.getName(),
|
|
inputs: deserializeProperties(req.getObject()),
|
|
provider: req.getProvider(),
|
|
custom: req.getCustom(),
|
|
id: req.getImportid(),
|
|
});
|
|
|
|
const urn = this.newUrn(req.getParent(), req.getType(), req.getName());
|
|
const serializedState = await serializeProperties("", result.state);
|
|
|
|
this.resources.set(urn, { urn, id: result.id ?? null, state: serializedState });
|
|
|
|
const response = new resproto.RegisterResourceResponse();
|
|
response.setUrn(urn);
|
|
response.setId(result.id || "");
|
|
response.setObject(structproto.Struct.fromJavaScript(serializedState));
|
|
callback(null, response);
|
|
} catch (err) {
|
|
callback(err, undefined);
|
|
}
|
|
}
|
|
|
|
public registerResourceOutputs(req: any, callback: (err: any, innerResponse: any) => void) {
|
|
try {
|
|
const registeredResource = this.resources.get(req.getUrn());
|
|
if (!registeredResource) {
|
|
throw new Error(`unknown resource ${req.getUrn()}`);
|
|
}
|
|
registeredResource.state = req.getOutputs();
|
|
|
|
callback(null, {});
|
|
} catch (err) {
|
|
callback(err, undefined);
|
|
}
|
|
}
|
|
|
|
public supportsFeature(req: any, callback: (err: any, innerResponse: any) => void) {
|
|
const id = req.getId();
|
|
|
|
// Support for "outputValues" is deliberately disabled for the mock monitor so
|
|
// instances of `Output` don't show up in `MockResourceArgs` inputs.
|
|
const hasSupport = id !== "outputValues";
|
|
|
|
callback(null, {
|
|
getHassupport: () => hasSupport,
|
|
});
|
|
}
|
|
}
|
|
|
|
/**
|
|
* 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 construction.
|
|
* @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.
|
|
* @param organization: If provided, the name of the Pulumi organization. Defaults to nothing.
|
|
*/
|
|
export function setMocks(mocks: Mocks, project?: string, stack?: string, preview?: boolean, organization?: string) {
|
|
setMockOptions(new MockMonitor(mocks), project, stack, preview, organization);
|
|
}
|