Skip to content

Commit 1a0d68b

Browse files
committed
Modified some system logs to flow through gRPC.
1 parent fcc8b45 commit 1a0d68b

File tree

5 files changed

+37
-19
lines changed

5 files changed

+37
-19
lines changed

src/Context.ts

+6-7
Original file line numberDiff line numberDiff line change
@@ -4,8 +4,8 @@ import { AzureFunctionsRpcMessages as rpc } from '../azure-functions-language-wo
44
import { Request, RequestProperties } from './http/Request';
55
import { Response } from './http/Response';
66
import LogLevel = rpc.RpcLog.Level;
7+
import LogCategory = rpc.RpcLog.RpcLogCategory;
78
import { Context, ExecutionContext, Logger, BindingDefinition, HttpRequest } from './public/Interfaces'
8-
import { systemWarn } from './utils/Logger';
99

1010
export function CreateContextAndInputs(info: FunctionInfo, request: rpc.IInvocationRequest, logCallback: LogCallback, callback: ResultCallback) {
1111
let context = new InvocationContext(info, request, logCallback, callback);
@@ -79,9 +79,9 @@ class InvocationContext implements Context {
7979
_promise = isPromise === true;
8080
if (_done) {
8181
if (_promise) {
82-
logCallback(LogLevel.Error, "Error: Choose either to return a promise or call 'done'. Do not use both in your script.");
82+
logCallback(LogLevel.Error, LogCategory.User, "Error: Choose either to return a promise or call 'done'. Do not use both in your script.");
8383
} else {
84-
logCallback(LogLevel.Error, "Error: 'done' has already been called. Please check your script for extraneous calls to 'done'.");
84+
logCallback(LogLevel.Error, LogCategory.User, "Error: 'done' has already been called. Please check your script for extraneous calls to 'done'.");
8585
}
8686
return;
8787
}
@@ -106,10 +106,9 @@ function logWithAsyncCheck(done: boolean, log: LogCallback, level: LogLevel, exe
106106
let badAsyncMsg = "Warning: Unexpected call to 'log' on the context object after function execution has completed. Please check for asynchronous calls that are not awaited or calls to 'done' made before function execution completes. ";
107107
badAsyncMsg += `Function name: ${executionContext.functionName}. Invocation Id: ${executionContext.invocationId}. `;
108108
badAsyncMsg += `Learn more: https://go.microsoft.com/fwlink/?linkid=2097909 `;
109-
log(LogLevel.Warning, badAsyncMsg);
110-
systemWarn(badAsyncMsg);
109+
log(LogLevel.Warning, LogCategory.System, badAsyncMsg);
111110
}
112-
return log(level, ...args);
111+
return log(level, LogCategory.User, ...args);
113112
}
114113

115114
export interface InvocationResult {
@@ -119,7 +118,7 @@ export interface InvocationResult {
119118

120119
export type DoneCallback = (err?: Error | string, result?: any) => void;
121120

122-
export type LogCallback = (level: LogLevel, ...args: any[]) => void;
121+
export type LogCallback = (level: LogLevel, category: rpc.RpcLog.RpcLogCategory, ...args: any[]) => void;
123122

124123
export type ResultCallback = (err?: any, result?: InvocationResult) => void;
125124

src/Worker.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -57,4 +57,4 @@ export function startNodeWorker(args) {
5757
process.on('exit', code => {
5858
systemLog(`Worker ${workerId} exited with code ${code}`);
5959
});
60-
}
60+
}

src/WorkerChannel.ts

+21-7
Original file line numberDiff line numberDiff line change
@@ -1,13 +1,14 @@
11
import { format, isFunction } from 'util';
22
import { AzureFunctionsRpcMessages as rpc } from '../azure-functions-language-worker-protobuf/src/rpc';
3-
import Status = rpc.StatusResult.Status;
43
import { IFunctionLoader } from './FunctionLoader';
54
import { CreateContextAndInputs, LogCallback, ResultCallback } from './Context';
65
import { IEventStream } from './GrpcService';
76
import { toTypedData } from './converters';
87
import { augmentTriggerMetadata } from './augmenters';
9-
import { systemError, systemLog } from './utils/Logger';
8+
import { systemError } from './utils/Logger';
109
import { InternalException } from './utils/InternalException';
10+
import LogCategory = rpc.RpcLog.RpcLogCategory;
11+
import LogLevel = rpc.RpcLog.Level;
1112

1213
/**
1314
* The worker channel should have a way to handle all incoming gRPC messages.
@@ -46,7 +47,11 @@ export class WorkerChannel implements IWorkerChannel {
4647
if (eventHandler) {
4748
eventHandler.apply(this, [msg.requestId, msg[event]]);
4849
} else {
49-
systemError(`Worker ${workerId} had no handler for message '${event}'`)
50+
this.log({
51+
message: `Worker ${workerId} had no handler for message '${event}'`,
52+
level: LogLevel.Error,
53+
logCategory: LogCategory.System
54+
});
5055
}
5156
});
5257
eventStream.on('error', function (err) {
@@ -109,7 +114,11 @@ export class WorkerChannel implements IWorkerChannel {
109114
}
110115
catch(exception) {
111116
errorMessage = `Worker was unable to load function ${msg.metadata.name}: '${exception}'`;
112-
systemError(errorMessage)
117+
this.log({
118+
message: errorMessage,
119+
level: LogLevel.Error,
120+
logCategory: LogCategory.System
121+
});
113122
err = exception;
114123
}
115124

@@ -133,12 +142,13 @@ export class WorkerChannel implements IWorkerChannel {
133142
augmentTriggerMetadata(msg);
134143

135144
let info = this._functionLoader.getInfo(<string>msg.functionId);
136-
let logCallback: LogCallback = (level, ...args) => {
145+
let logCallback: LogCallback = (level, category, ...args) => {
137146
this.log({
138147
invocationId: msg.invocationId,
139148
category: `${info.name}.Invocation`,
140149
message: format.apply(null, args),
141-
level: level
150+
level: level,
151+
logCategory: category
142152
});
143153
}
144154

@@ -238,7 +248,11 @@ export class WorkerChannel implements IWorkerChannel {
238248
public functionEnvironmentReloadRequest(requestId: string, msg: rpc.IFunctionEnvironmentReloadRequest): void {
239249
// Add environment variables from incoming
240250
let numVariables = (msg.environmentVariables && Object.keys(msg.environmentVariables).length) || 0;
241-
systemLog(`Reloading environment variables. Found ${numVariables} variables to reload.`);
251+
this.log({
252+
message: `Reloading environment variables. Found ${numVariables} variables to reload.`,
253+
level: LogLevel.Information,
254+
logCategory: LogCategory.System
255+
});
242256

243257
let error = null;
244258
try {

src/utils/Logger.ts

+5
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,8 @@
1+
/**
2+
* Use these methods only if you want to guarantee the messages reach the host despite potential performance impact.
3+
Otherwise, please stick to utilizing the gRPC channel to propagate these messages with category: RpcLogCategory.System
4+
**/
5+
16
const logPrefix = "LanguageWorkerConsoleLog";
27

38
export function systemLog(message?: any, ...optionalParams: any[]) {

test/ContextTests.ts

+4-4
Original file line numberDiff line numberDiff line change
@@ -28,7 +28,7 @@ describe('Context', () => {
2828
it ('async function logs error on calling context.done', async () => {
2929
await callUserFunc(BasicAsync.asyncAndCallback, _context);
3030
sinon.assert.calledOnce(_logger);
31-
sinon.assert.calledWith(_logger, rpc.RpcLog.Level.Error, "Error: Choose either to return a promise or call 'done'. Do not use both in your script.");
31+
sinon.assert.calledWith(_logger, rpc.RpcLog.Level.Error, rpc.RpcLog.RpcLogCategory.User, "Error: Choose either to return a promise or call 'done'. Do not use both in your script.");
3232
});
3333

3434
it ('async function calls callback and returns value without context.done', async () => {
@@ -40,21 +40,21 @@ describe('Context', () => {
4040
it ('function logs error on calling context.done more than once', () => {
4141
callUserFunc(BasicCallback.callbackTwice, _context);
4242
sinon.assert.calledOnce(_logger);
43-
sinon.assert.calledWith(_logger, rpc.RpcLog.Level.Error, "Error: 'done' has already been called. Please check your script for extraneous calls to 'done'.");
43+
sinon.assert.calledWith(_logger, rpc.RpcLog.Level.Error, rpc.RpcLog.RpcLogCategory.User, "Error: 'done' has already been called. Please check your script for extraneous calls to 'done'.");
4444
});
4545

4646
it ('function logs error on calling context.log after context.done() called', () => {
4747
callUserFunc(BasicCallback.callbackOnce, _context);
4848
_context.log("");
4949
sinon.assert.calledTwice(_logger);
50-
sinon.assert.calledWith(_logger, rpc.RpcLog.Level.Warning, "Warning: Unexpected call to 'log' on the context object after function execution has completed. Please check for asynchronous calls that are not awaited or calls to 'done' made before function execution completes. Function name: test. Invocation Id: 1. Learn more: https://go.microsoft.com/fwlink/?linkid=2097909 ");
50+
sinon.assert.calledWith(_logger, rpc.RpcLog.Level.Warning, rpc.RpcLog.RpcLogCategory.System, "Warning: Unexpected call to 'log' on the context object after function execution has completed. Please check for asynchronous calls that are not awaited or calls to 'done' made before function execution completes. Function name: test. Invocation Id: 1. Learn more: https://go.microsoft.com/fwlink/?linkid=2097909 ");
5151
});
5252

5353
it ('function logs error on calling context.log from non-awaited async call', async () => {
5454
await callUserFunc(BasicAsync.asyncPlainFunction, _context);
5555
_context.log("");
5656
sinon.assert.calledTwice(_logger);
57-
sinon.assert.calledWith(_logger, rpc.RpcLog.Level.Warning, "Warning: Unexpected call to 'log' on the context object after function execution has completed. Please check for asynchronous calls that are not awaited or calls to 'done' made before function execution completes. Function name: test. Invocation Id: 1. Learn more: https://go.microsoft.com/fwlink/?linkid=2097909 ");
57+
sinon.assert.calledWith(_logger, rpc.RpcLog.Level.Warning, rpc.RpcLog.RpcLogCategory.System, "Warning: Unexpected call to 'log' on the context object after function execution has completed. Please check for asynchronous calls that are not awaited or calls to 'done' made before function execution completes. Function name: test. Invocation Id: 1. Learn more: https://go.microsoft.com/fwlink/?linkid=2097909 ");
5858
});
5959

6060
it ('function calls callback correctly with bindings', () => {

0 commit comments

Comments
 (0)