// 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 * as deasync from "deasync"; import { InvokeOptions } from "./invoke"; /** * Common code for doing RTTI typechecks. RTTI is done by having a boolean property on an object * with a special name (like "__resource" or "__asset"). This function checks that the object * exists, has a **boolean** property with that name, and that that boolean property has the value * of 'true'. Checking that property is 'boolean' helps ensure that this test works even on proxies * that synthesize properties dynamically (like Output). Checking that the property has the 'true' * value isn't strictly necessary, but works to make sure that the impls are following a common * pattern. * * @internal */ export function isInstance(obj: any, name: keyof T): obj is T { return hasTrueBooleanMember(obj, name); } /** @internal */ export function hasTrueBooleanMember(obj: any, memberName: string | number | symbol): boolean { if (obj === undefined || obj === null) { return false; } const val = obj[memberName]; if (typeof val !== "boolean") { return false; } return val === true; } // Workaround errors we sometimes get on some machines saying that Object.values is not available. /** @internal */ export function values(obj: object): any[] { const result: any[] = []; for (const key of Object.keys(obj)) { result.push((obj)[key]); } return result; } /** @internal */ export function union(set1: Set, set2: Set) { return new Set([...set1, ...set2]); } /** * Synchronously blocks until the result of this promise is computed. If the promise is rejected, * this will throw the error the promise was rejected with. If this promise does not complete this * will block indefinitely. * * Be very careful with this function. Only wait on a promise if you are certain it is safe to do * so. * * @internal */ export function promiseResult(promise: Promise): T { enum State { running, finishedSuccessfully, finishedWithError, } let result: T; let error = undefined; let state = State.running; promise.then( val => { result = val; state = State.finishedSuccessfully; }, err => { error = err; state = State.finishedWithError; }); deasync.loopWhile(() => state === State.running); if (state === State.finishedWithError) { throw error; } return result!; } /** * This is an advanced compat function for libraries and should not generally be used by normal * Pulumi application. * * @deprecated No longer supported. This function is now a no-op and will directly return the promise * passed into it. */ export function liftProperties(promise: Promise, opts: InvokeOptions = {}): Promise & T { return promise; }