2018-05-22 19:43:36 +00:00
|
|
|
// 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.
|
2017-10-08 19:10:46 +00:00
|
|
|
|
|
|
|
// The log module logs messages in a way that tightly integrates with the resource engine's interface.
|
|
|
|
|
2023-12-04 15:22:44 +00:00
|
|
|
import * as engrpc from "../proto/engine_grpc_pb";
|
|
|
|
import * as engproto from "../proto/engine_pb";
|
2018-04-10 19:03:11 +00:00
|
|
|
import * as resourceTypes from "../resource";
|
2017-11-26 18:04:15 +00:00
|
|
|
import { getEngine, rpcKeepAlive } from "../runtime/settings";
|
2023-12-01 17:35:07 +00:00
|
|
|
import { getStore } from "../runtime/state";
|
2017-10-08 19:10:46 +00:00
|
|
|
|
|
|
|
let lastLog: Promise<any> = Promise.resolve();
|
|
|
|
|
2021-04-07 02:34:15 +00:00
|
|
|
const messageLevels = {
|
|
|
|
[engproto.LogSeverity.DEBUG]: "debug",
|
|
|
|
[engproto.LogSeverity.INFO]: "info",
|
|
|
|
[engproto.LogSeverity.WARNING]: "warn",
|
|
|
|
[engproto.LogSeverity.ERROR]: "error",
|
|
|
|
};
|
|
|
|
|
2017-10-08 19:10:46 +00:00
|
|
|
/**
|
2024-07-15 09:05:28 +00:00
|
|
|
* Returns true if any errors have occurred in the program.
|
2017-10-08 19:10:46 +00:00
|
|
|
*/
|
|
|
|
export function hasErrors(): boolean {
|
2023-12-01 17:35:07 +00:00
|
|
|
return getStore().logErrorCount > 0;
|
2017-10-08 19:10:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2024-07-15 09:05:28 +00:00
|
|
|
* Logs a debug-level message that is generally hidden from end-users.
|
2017-10-08 19:10:46 +00:00
|
|
|
*/
|
2020-09-24 02:06:26 +00:00
|
|
|
export function debug(msg: string, resource?: resourceTypes.Resource, streamId?: number, ephemeral?: boolean) {
|
2023-12-04 15:22:44 +00:00
|
|
|
const engine = getEngine();
|
2017-10-08 19:10:46 +00:00
|
|
|
if (engine) {
|
2020-09-24 02:06:26 +00:00
|
|
|
return log(engine, engproto.LogSeverity.DEBUG, msg, resource, streamId, ephemeral);
|
2023-04-28 22:27:10 +00:00
|
|
|
} else {
|
2020-09-24 02:06:26 +00:00
|
|
|
return Promise.resolve();
|
2017-10-08 19:10:46 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2024-07-15 09:05:28 +00:00
|
|
|
* Logs an informational message that is generally printed to standard output
|
|
|
|
* during resource operations.
|
2017-10-08 19:10:46 +00:00
|
|
|
*/
|
2018-10-12 18:09:09 +00:00
|
|
|
export function info(msg: string, resource?: resourceTypes.Resource, streamId?: number, ephemeral?: boolean) {
|
2023-12-04 15:22:44 +00:00
|
|
|
const engine = getEngine();
|
2017-10-08 19:10:46 +00:00
|
|
|
if (engine) {
|
2018-10-12 18:09:09 +00:00
|
|
|
return log(engine, engproto.LogSeverity.INFO, msg, resource, streamId, ephemeral);
|
2023-04-28 22:27:10 +00:00
|
|
|
} else {
|
2017-10-08 19:10:46 +00:00
|
|
|
console.log(`info: [runtime] ${msg}`);
|
2018-04-10 19:03:11 +00:00
|
|
|
return Promise.resolve();
|
2017-10-08 19:10:46 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2024-07-15 09:05:28 +00:00
|
|
|
* Logs a warning to indicate that something went wrong, but not
|
|
|
|
* catastrophically so.
|
2017-10-08 19:10:46 +00:00
|
|
|
*/
|
2018-10-12 18:09:09 +00:00
|
|
|
export function warn(msg: string, resource?: resourceTypes.Resource, streamId?: number, ephemeral?: boolean) {
|
2023-12-04 15:22:44 +00:00
|
|
|
const engine = getEngine();
|
2017-10-08 19:10:46 +00:00
|
|
|
if (engine) {
|
2018-10-12 18:09:09 +00:00
|
|
|
return log(engine, engproto.LogSeverity.WARNING, msg, resource, streamId, ephemeral);
|
2023-04-28 22:27:10 +00:00
|
|
|
} else {
|
2017-10-08 19:10:46 +00:00
|
|
|
console.warn(`warning: [runtime] ${msg}`);
|
2018-04-10 19:03:11 +00:00
|
|
|
return Promise.resolve();
|
2017-10-08 19:10:46 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2024-07-15 09:05:28 +00:00
|
|
|
* Logs a fatal condition. Consider raising an exception after calling error to
|
|
|
|
* stop the Pulumi program.
|
2017-10-08 19:10:46 +00:00
|
|
|
*/
|
2018-10-12 18:09:09 +00:00
|
|
|
export function error(msg: string, resource?: resourceTypes.Resource, streamId?: number, ephemeral?: boolean) {
|
2023-12-01 17:35:07 +00:00
|
|
|
getStore().logErrorCount++; // remember the error so we can suppress leaks.
|
2017-10-08 19:10:46 +00:00
|
|
|
|
2023-12-04 15:22:44 +00:00
|
|
|
const engine = getEngine();
|
2017-10-08 19:10:46 +00:00
|
|
|
if (engine) {
|
2018-10-12 18:09:09 +00:00
|
|
|
return log(engine, engproto.LogSeverity.ERROR, msg, resource, streamId, ephemeral);
|
2023-04-28 22:27:10 +00:00
|
|
|
} else {
|
2017-10-08 19:10:46 +00:00
|
|
|
console.error(`error: [runtime] ${msg}`);
|
2018-04-10 19:03:11 +00:00
|
|
|
return Promise.resolve();
|
2017-10-08 19:10:46 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-10 19:03:11 +00:00
|
|
|
function log(
|
2023-12-19 14:35:23 +00:00
|
|
|
engine: engrpc.IEngineClient,
|
2023-12-04 15:22:44 +00:00
|
|
|
sev: engproto.LogSeverity,
|
2023-04-28 22:27:10 +00:00
|
|
|
msg: string,
|
2021-08-10 18:31:59 +00:00
|
|
|
resource: resourceTypes.Resource | undefined,
|
|
|
|
streamId: number | undefined,
|
2023-04-28 22:27:10 +00:00
|
|
|
ephemeral: boolean | undefined,
|
|
|
|
): Promise<void> {
|
2017-10-08 19:10:46 +00:00
|
|
|
// Ensure we log everything in serial order.
|
2017-10-18 22:03:56 +00:00
|
|
|
const keepAlive: () => void = rpcKeepAlive();
|
2018-04-10 19:03:11 +00:00
|
|
|
|
2023-04-28 22:27:10 +00:00
|
|
|
const urnPromise = resource ? resource.urn.promise() : Promise.resolve("");
|
2018-04-10 19:03:11 +00:00
|
|
|
|
2019-02-22 04:18:29 +00:00
|
|
|
lastLog = Promise.all([lastLog, urnPromise]).then(([_, urn]) => {
|
2023-12-04 15:22:44 +00:00
|
|
|
return new Promise<void>((resolve, reject) => {
|
2018-04-07 15:02:59 +00:00
|
|
|
try {
|
|
|
|
const req = new engproto.LogRequest();
|
|
|
|
req.setSeverity(sev);
|
|
|
|
req.setMessage(msg);
|
2018-04-10 19:03:11 +00:00
|
|
|
req.setUrn(urn);
|
2018-07-11 22:04:00 +00:00
|
|
|
req.setStreamid(streamId === undefined ? 0 : streamId);
|
2018-10-12 18:09:09 +00:00
|
|
|
req.setEphemeral(ephemeral === true);
|
2018-04-07 15:02:59 +00:00
|
|
|
engine.log(req, () => {
|
|
|
|
resolve(); // let the next log through
|
|
|
|
keepAlive(); // permit RPC channel tear-downs
|
|
|
|
});
|
2023-04-28 22:27:10 +00:00
|
|
|
} catch (err) {
|
2018-04-07 15:02:59 +00:00
|
|
|
reject(err);
|
|
|
|
}
|
2017-10-08 19:10:46 +00:00
|
|
|
});
|
|
|
|
});
|
2018-04-10 19:03:11 +00:00
|
|
|
|
2021-04-07 02:34:15 +00:00
|
|
|
return lastLog.catch((err) => {
|
|
|
|
// debug messages never go to stdout/err
|
|
|
|
if (sev !== engproto.LogSeverity.DEBUG) {
|
|
|
|
// if we're unable to deliver the log message, deliver to stderr instead
|
2023-04-28 22:27:10 +00:00
|
|
|
console.error(
|
|
|
|
`failed to deliver log message. \nerror: ${err} \noriginal message: ${msg}\n message severity: ${messageLevels[sev]}`,
|
|
|
|
);
|
2021-04-07 02:34:15 +00:00
|
|
|
}
|
|
|
|
// we still need to free up the outstanding promise chain, whether or not delivery succeeded.
|
|
|
|
keepAlive();
|
|
|
|
});
|
2017-10-08 19:10:46 +00:00
|
|
|
}
|