mirror of
https://github.com/gristlabs/grist-core.git
synced 2024-10-27 20:44:07 +00:00
Fix server crash when client passes malformed JSON (#826)
* Fix server crash when client passes malformed JSON * Take remarks into account --------- Co-authored-by: Florent FAYOLLE <florent.fayolle@beta.gouv.fr>
This commit is contained in:
parent
1f5cd0a9d5
commit
5533b9b7ee
@ -526,11 +526,19 @@ export class Client {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private async _onMessage(message: string): Promise<void> {
|
||||||
|
try {
|
||||||
|
await this._onMessageImpl(message);
|
||||||
|
} catch (err) {
|
||||||
|
this._log.warn(null, 'onMessage error received for message "%s": %s', shortDesc(message), err.stack);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Processes a request from a client. All requests from a client get a response, at least to
|
* Processes a request from a client. All requests from a client get a response, at least to
|
||||||
* indicate success or failure.
|
* indicate success or failure.
|
||||||
*/
|
*/
|
||||||
private async _onMessage(message: string): Promise<void> {
|
private async _onMessageImpl(message: string): Promise<void> {
|
||||||
const request = JSON.parse(message);
|
const request = JSON.parse(message);
|
||||||
if (request.beat) {
|
if (request.beat) {
|
||||||
// this is a heart beat, to keep the websocket alive. No need to reply.
|
// this is a heart beat, to keep the websocket alive. No need to reply.
|
||||||
|
@ -188,6 +188,24 @@ describe('Comm', function() {
|
|||||||
]);
|
]);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('should only log warning for malformed JSON data', async function () {
|
||||||
|
const logMessages = await testUtils.captureLog('warn', async () => {
|
||||||
|
ws.send('foobar');
|
||||||
|
}, {waitForFirstLog: true});
|
||||||
|
testUtils.assertMatchArray(logMessages, [
|
||||||
|
/^warn: Client.* Unexpected token.*/
|
||||||
|
]);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should log warning when null value is passed', async function () {
|
||||||
|
const logMessages = await testUtils.captureLog('warn', async () => {
|
||||||
|
ws.send('null');
|
||||||
|
}, {waitForFirstLog: true});
|
||||||
|
testUtils.assertMatchArray(logMessages, [
|
||||||
|
/^warn: Client.*Cannot read properties of null*/
|
||||||
|
]);
|
||||||
|
});
|
||||||
|
|
||||||
it("should support app-level events correctly", async function() {
|
it("should support app-level events correctly", async function() {
|
||||||
comm!.broadcastMessage('fooType' as any, 'hello');
|
comm!.broadcastMessage('fooType' as any, 'hello');
|
||||||
comm!.broadcastMessage('barType' as any, 'world');
|
comm!.broadcastMessage('barType' as any, 'world');
|
||||||
|
@ -126,16 +126,18 @@ export function setTmpLogLevel(level: string, optCaptureFunc?: (level: string, m
|
|||||||
*/
|
*/
|
||||||
export async function captureLog(
|
export async function captureLog(
|
||||||
minLevel: string, callback: (messages: string[]) => void|Promise<void>,
|
minLevel: string, callback: (messages: string[]) => void|Promise<void>,
|
||||||
options: {timestamp: boolean} = {timestamp: false}
|
options: {timestamp?: boolean, waitForFirstLog?: boolean} = {timestamp: false, waitForFirstLog: false}
|
||||||
): Promise<string[]> {
|
): Promise<string[]> {
|
||||||
const messages: string[] = [];
|
const messages: string[] = [];
|
||||||
const prevLogLevel = log.transports.file.level;
|
const prevLogLevel = log.transports.file.level;
|
||||||
const name = _.uniqueId('CaptureLog');
|
const name = _.uniqueId('CaptureLog');
|
||||||
|
|
||||||
|
const captureFirstLogPromise = new Promise((resolve) => {
|
||||||
function capture(level: string, msg: string, meta: any) {
|
function capture(level: string, msg: string, meta: any) {
|
||||||
if ((log as any).levels[level] <= (log as any).levels[minLevel]) { // winston types are off?
|
if ((log as any).levels[level] <= (log as any).levels[minLevel]) { // winston types are off?
|
||||||
const timePrefix = options.timestamp ? new Date().toISOString() + ' ' : '';
|
const timePrefix = options.timestamp ? new Date().toISOString() + ' ' : '';
|
||||||
messages.push(`${timePrefix}${level}: ${msg}${meta ? ' ' + serialize(meta) : ''}`);
|
messages.push(`${timePrefix}${level}: ${msg}${meta ? ' ' + serialize(meta) : ''}`);
|
||||||
|
resolve(null);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -143,8 +145,13 @@ export async function captureLog(
|
|||||||
log.transports.file.level = -1 as any; // Suppress all log output.
|
log.transports.file.level = -1 as any; // Suppress all log output.
|
||||||
}
|
}
|
||||||
log.add(CaptureTransport as any, { captureFunc: capture, name, level: minLevel}); // types are off.
|
log.add(CaptureTransport as any, { captureFunc: capture, name, level: minLevel}); // types are off.
|
||||||
|
});
|
||||||
|
|
||||||
try {
|
try {
|
||||||
await callback(messages);
|
await callback(messages);
|
||||||
|
if (options.waitForFirstLog) {
|
||||||
|
await captureFirstLogPromise;
|
||||||
|
}
|
||||||
} finally {
|
} finally {
|
||||||
log.remove(name);
|
log.remove(name);
|
||||||
log.transports.file.level = prevLogLevel;
|
log.transports.file.level = prevLogLevel;
|
||||||
|
Loading…
Reference in New Issue
Block a user