@dxos/log 0.8.4-main.dedc0f3 → 0.8.4-main.e8ec1fe
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/lib/browser/index.mjs +96 -68
- package/dist/lib/browser/index.mjs.map +3 -3
- package/dist/lib/browser/meta.json +1 -1
- package/dist/lib/node-esm/index.mjs +96 -68
- package/dist/lib/node-esm/index.mjs.map +3 -3
- package/dist/lib/node-esm/meta.json +1 -1
- package/dist/types/src/config.d.ts.map +1 -1
- package/dist/types/src/context.d.ts.map +1 -1
- package/dist/types/src/decorators.d.ts +1 -1
- package/dist/types/src/decorators.d.ts.map +1 -1
- package/dist/types/src/log.d.ts +12 -18
- package/dist/types/src/log.d.ts.map +1 -1
- package/dist/types/src/options.d.ts +1 -6
- package/dist/types/src/options.d.ts.map +1 -1
- package/dist/types/src/processors/browser-processor.d.ts.map +1 -1
- package/dist/types/src/processors/file-processor.d.ts.map +1 -1
- package/dist/types/tsconfig.tsbuildinfo +1 -1
- package/package.json +3 -3
- package/src/config.ts +1 -0
- package/src/context.ts +36 -5
- package/src/decorators.ts +3 -3
- package/src/log.test.ts +48 -18
- package/src/log.ts +101 -57
- package/src/options.ts +24 -10
- package/src/processors/browser-processor.ts +2 -0
- package/src/processors/file-processor.ts +2 -0
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@dxos/log",
|
|
3
|
-
"version": "0.8.4-main.
|
|
3
|
+
"version": "0.8.4-main.e8ec1fe",
|
|
4
4
|
"description": "Logger",
|
|
5
5
|
"homepage": "https://dxos.org",
|
|
6
6
|
"bugs": "https://github.com/dxos/dxos/issues",
|
|
@@ -35,8 +35,8 @@
|
|
|
35
35
|
"js-yaml": "^4.1.0",
|
|
36
36
|
"lodash.defaultsdeep": "^4.6.1",
|
|
37
37
|
"lodash.omit": "^4.5.0",
|
|
38
|
-
"@dxos/node-std": "0.8.4-main.
|
|
39
|
-
"@dxos/util": "0.8.4-main.
|
|
38
|
+
"@dxos/node-std": "0.8.4-main.e8ec1fe",
|
|
39
|
+
"@dxos/util": "0.8.4-main.e8ec1fe"
|
|
40
40
|
},
|
|
41
41
|
"devDependencies": {
|
|
42
42
|
"@types/js-yaml": "^4.0.5",
|
package/src/config.ts
CHANGED
package/src/context.ts
CHANGED
|
@@ -27,8 +27,33 @@ export interface LogEntry {
|
|
|
27
27
|
*/
|
|
28
28
|
export type LogProcessor = (config: LogConfig, entry: LogEntry) => void;
|
|
29
29
|
|
|
30
|
-
|
|
31
|
-
|
|
30
|
+
/**
|
|
31
|
+
* Returns:
|
|
32
|
+
* true if the log entry matches the filter,
|
|
33
|
+
* false if should be excluded, or
|
|
34
|
+
* undefined if it the filter doesn't match the level.
|
|
35
|
+
*/
|
|
36
|
+
const matchFilter = (filter: LogFilter, level: LogLevel, path?: string): boolean | undefined => {
|
|
37
|
+
// TODO(burdon): Support regexp.
|
|
38
|
+
if (filter.pattern?.startsWith('-')) {
|
|
39
|
+
// Exclude.
|
|
40
|
+
if (path?.includes(filter.pattern.slice(1))) {
|
|
41
|
+
if (level >= filter.level) {
|
|
42
|
+
return false;
|
|
43
|
+
}
|
|
44
|
+
}
|
|
45
|
+
} else {
|
|
46
|
+
// Include.
|
|
47
|
+
if (filter.pattern?.length) {
|
|
48
|
+
if (path?.includes(filter.pattern)) {
|
|
49
|
+
return level >= filter.level;
|
|
50
|
+
}
|
|
51
|
+
} else {
|
|
52
|
+
if (level >= filter.level) {
|
|
53
|
+
return true;
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
}
|
|
32
57
|
};
|
|
33
58
|
|
|
34
59
|
/**
|
|
@@ -36,10 +61,16 @@ const matchFilter = (filter: LogFilter, level: LogLevel, path: string) => {
|
|
|
36
61
|
*/
|
|
37
62
|
export const shouldLog = (entry: LogEntry, filters?: LogFilter[]): boolean => {
|
|
38
63
|
if (filters === undefined) {
|
|
39
|
-
return
|
|
40
|
-
} else {
|
|
41
|
-
return filters.some((filter) => matchFilter(filter, entry.level, entry.meta?.F ?? ''));
|
|
64
|
+
return false;
|
|
42
65
|
}
|
|
66
|
+
|
|
67
|
+
const results = filters
|
|
68
|
+
.map((filter) => matchFilter(filter, entry.level, entry.meta?.F))
|
|
69
|
+
.filter((result): result is boolean => result !== undefined);
|
|
70
|
+
|
|
71
|
+
// Skip if any are explicitely false.
|
|
72
|
+
// console.log({ level: entry.level, path: entry.meta?.F }, filters, results, results.length);
|
|
73
|
+
return results.length > 0 && !results.some((results) => results === false);
|
|
43
74
|
};
|
|
44
75
|
|
|
45
76
|
export const getContextFromEntry = (entry: LogEntry): Record<string, any> | undefined => {
|
package/src/decorators.ts
CHANGED
|
@@ -6,7 +6,7 @@ import { inspect } from 'node:util';
|
|
|
6
6
|
|
|
7
7
|
import chalk from 'chalk';
|
|
8
8
|
|
|
9
|
-
import type
|
|
9
|
+
import { type LogMethods } from './log';
|
|
10
10
|
import { type CallMetadata } from './meta';
|
|
11
11
|
|
|
12
12
|
let nextPromiseId = 0;
|
|
@@ -191,13 +191,13 @@ const logAsyncRejected = (
|
|
|
191
191
|
);
|
|
192
192
|
};
|
|
193
193
|
|
|
194
|
+
const COLOR_FUNCTION = [220, 220, 170] as const;
|
|
195
|
+
|
|
194
196
|
// https://github.com/dxos/dxos/issues/7286
|
|
195
197
|
const greenCheck = typeof chalk.green === 'function' ? chalk.green('✔') : '✔';
|
|
196
198
|
|
|
197
199
|
const formatTimeElapsed = (startTime: number) => chalk.gray(`${(performance.now() - startTime).toFixed(0)}ms`);
|
|
198
200
|
|
|
199
|
-
const COLOR_FUNCTION = [220, 220, 170] as const;
|
|
200
|
-
|
|
201
201
|
const formatFunction = (name: string) => chalk.bold(chalk.rgb(...COLOR_FUNCTION)(name));
|
|
202
202
|
|
|
203
203
|
const formatPromise = (id: number) => chalk.blue(`Promise#${id}`);
|
package/src/log.test.ts
CHANGED
|
@@ -4,10 +4,11 @@
|
|
|
4
4
|
|
|
5
5
|
import path from 'node:path';
|
|
6
6
|
|
|
7
|
-
import { describe, test } from 'vitest';
|
|
7
|
+
import { beforeEach, describe, test } from 'vitest';
|
|
8
8
|
|
|
9
9
|
import { LogLevel } from './config';
|
|
10
|
-
import {
|
|
10
|
+
import { shouldLog } from './context';
|
|
11
|
+
import { type Log, createLog } from './log';
|
|
11
12
|
|
|
12
13
|
class LogError extends Error {
|
|
13
14
|
constructor(
|
|
@@ -25,13 +26,52 @@ class LogError extends Error {
|
|
|
25
26
|
}
|
|
26
27
|
}
|
|
27
28
|
|
|
28
|
-
log
|
|
29
|
-
|
|
30
|
-
});
|
|
29
|
+
describe('log', () => {
|
|
30
|
+
let log!: Log;
|
|
31
31
|
|
|
32
|
-
|
|
32
|
+
beforeEach(() => {
|
|
33
|
+
log = createLog();
|
|
34
|
+
log.config({
|
|
35
|
+
filter: LogLevel.DEBUG,
|
|
36
|
+
});
|
|
37
|
+
});
|
|
38
|
+
|
|
39
|
+
test('filters', ({ expect }) => {
|
|
40
|
+
const tests = [
|
|
41
|
+
{ expected: 0, filter: 'ERROR' },
|
|
42
|
+
{ expected: 2, filter: 'INFO' },
|
|
43
|
+
{ expected: 1, filter: 'foo:INFO' },
|
|
44
|
+
{ expected: 4, filter: 'DEBUG' },
|
|
45
|
+
{ expected: 2, filter: 'DEBUG,-foo:*' },
|
|
46
|
+
{ expected: 1, filter: 'INFO,-foo:*' },
|
|
47
|
+
{ expected: 3, filter: 'DEBUG,-foo:INFO' },
|
|
48
|
+
{ expected: 3, filter: 'foo:DEBUG,bar:INFO' },
|
|
49
|
+
];
|
|
50
|
+
|
|
51
|
+
for (const test of tests) {
|
|
52
|
+
let count = 0;
|
|
53
|
+
const log = createLog();
|
|
54
|
+
const remove = log.addProcessor((config, entry) => {
|
|
55
|
+
if (shouldLog(entry, config.filters)) {
|
|
56
|
+
count++;
|
|
57
|
+
}
|
|
58
|
+
});
|
|
59
|
+
log.config({
|
|
60
|
+
filter: test.filter,
|
|
61
|
+
});
|
|
62
|
+
|
|
63
|
+
console.group(`Filter: "${test.filter}"`);
|
|
64
|
+
log.debug('line 1', {}, { F: 'foo.ts', L: 1, S: undefined });
|
|
65
|
+
log.info('line 2', {}, { F: 'foo.ts', L: 2, S: undefined });
|
|
66
|
+
log.debug('line 3', {}, { F: 'bar.ts', L: 3, S: undefined });
|
|
67
|
+
log.info('line 4', {}, { F: 'bar.ts', L: 4, S: undefined });
|
|
68
|
+
console.groupEnd();
|
|
69
|
+
|
|
70
|
+
expect(count, `Filter: "${test.filter}"`).toBe(test.expected);
|
|
71
|
+
remove();
|
|
72
|
+
}
|
|
73
|
+
});
|
|
33
74
|
|
|
34
|
-
describe('log', () => {
|
|
35
75
|
test('throws an error', () => {
|
|
36
76
|
try {
|
|
37
77
|
throw new LogError('Test failed', { value: 1 });
|
|
@@ -56,16 +96,6 @@ describe('log', () => {
|
|
|
56
96
|
}
|
|
57
97
|
});
|
|
58
98
|
|
|
59
|
-
test('config', () => {
|
|
60
|
-
log.config({
|
|
61
|
-
filter: LogLevel.INFO,
|
|
62
|
-
});
|
|
63
|
-
|
|
64
|
-
log.debug('Debug level log message');
|
|
65
|
-
log.info('Info level log message');
|
|
66
|
-
log.warn('Warn level log message');
|
|
67
|
-
});
|
|
68
|
-
|
|
69
99
|
test('config file', () => {
|
|
70
100
|
log.config({
|
|
71
101
|
file: path.join('packages/common/log/test-config.yml'),
|
|
@@ -91,7 +121,7 @@ describe('log', () => {
|
|
|
91
121
|
});
|
|
92
122
|
});
|
|
93
123
|
|
|
94
|
-
test('error',
|
|
124
|
+
test('error', () => {
|
|
95
125
|
const myError = new Error('Test error', { cause: new Error('Cause') });
|
|
96
126
|
log.catch(myError);
|
|
97
127
|
});
|
package/src/log.ts
CHANGED
|
@@ -6,7 +6,14 @@ import { type LogConfig, LogLevel, type LogOptions } from './config';
|
|
|
6
6
|
import { type LogContext, type LogProcessor } from './context';
|
|
7
7
|
import { createFunctionLogDecorator, createMethodLogDecorator } from './decorators';
|
|
8
8
|
import { type CallMetadata } from './meta';
|
|
9
|
-
import {
|
|
9
|
+
import { createConfig } from './options';
|
|
10
|
+
|
|
11
|
+
/**
|
|
12
|
+
* Accessible from browser console.
|
|
13
|
+
*/
|
|
14
|
+
declare global {
|
|
15
|
+
const DX_LOG: Log;
|
|
16
|
+
}
|
|
10
17
|
|
|
11
18
|
/**
|
|
12
19
|
* Logging function.
|
|
@@ -17,6 +24,9 @@ type LogFunction = (message: string, context?: LogContext, meta?: CallMetadata)
|
|
|
17
24
|
* Logging methods.
|
|
18
25
|
*/
|
|
19
26
|
export interface LogMethods {
|
|
27
|
+
config: (options: LogOptions) => Log;
|
|
28
|
+
addProcessor: (processor: LogProcessor, addDefault?: boolean) => () => void;
|
|
29
|
+
|
|
20
30
|
trace: LogFunction;
|
|
21
31
|
debug: LogFunction;
|
|
22
32
|
verbose: LogFunction;
|
|
@@ -24,70 +34,56 @@ export interface LogMethods {
|
|
|
24
34
|
warn: LogFunction;
|
|
25
35
|
error: LogFunction;
|
|
26
36
|
catch: (error: Error | any, context?: LogContext, meta?: CallMetadata) => void;
|
|
27
|
-
|
|
28
|
-
stack: (message?: string, context?: never, meta?: CallMetadata) => void;
|
|
37
|
+
|
|
29
38
|
method: (arg0?: never, arg1?: never, meta?: CallMetadata) => MethodDecorator;
|
|
30
|
-
|
|
39
|
+
function: <F extends (...args: any[]) => any>(
|
|
31
40
|
name: string,
|
|
32
41
|
fn: F,
|
|
33
|
-
opts?: {
|
|
42
|
+
opts?: {
|
|
43
|
+
transformOutput?: (result: ReturnType<F>) => Promise<any> | any;
|
|
44
|
+
},
|
|
34
45
|
) => F;
|
|
46
|
+
|
|
47
|
+
break: () => void;
|
|
48
|
+
stack: (message?: string, context?: never, meta?: CallMetadata) => void;
|
|
35
49
|
}
|
|
36
50
|
|
|
37
51
|
/**
|
|
38
52
|
* Properties accessible on the logging function.
|
|
53
|
+
* @internal
|
|
39
54
|
*/
|
|
40
|
-
interface Log extends
|
|
41
|
-
|
|
42
|
-
addProcessor: (processor: LogProcessor) => void;
|
|
43
|
-
runtimeConfig: LogConfig;
|
|
55
|
+
export interface Log extends LogFunction, LogMethods {
|
|
56
|
+
readonly runtimeConfig: LogConfig;
|
|
44
57
|
}
|
|
45
58
|
|
|
59
|
+
/**
|
|
60
|
+
* @internal
|
|
61
|
+
*/
|
|
46
62
|
interface LogImp extends Log {
|
|
63
|
+
_id: string;
|
|
47
64
|
_config: LogConfig;
|
|
48
65
|
}
|
|
49
66
|
|
|
50
|
-
|
|
51
|
-
const log: LogImp = ((...params) => processLog(LogLevel.DEBUG, ...params)) as LogImp;
|
|
52
|
-
|
|
53
|
-
log._config = getConfig();
|
|
54
|
-
Object.defineProperty(log, 'runtimeConfig', { get: () => log._config });
|
|
55
|
-
|
|
56
|
-
log.addProcessor = (processor: LogProcessor) => {
|
|
57
|
-
if (DEFAULT_PROCESSORS.filter((p) => p === processor).length === 0) {
|
|
58
|
-
DEFAULT_PROCESSORS.push(processor);
|
|
59
|
-
}
|
|
60
|
-
if (log._config.processors.filter((p) => p === processor).length === 0) {
|
|
61
|
-
log._config.processors.push(processor);
|
|
62
|
-
}
|
|
63
|
-
};
|
|
64
|
-
|
|
65
|
-
// Set config.
|
|
66
|
-
log.config = (options: LogOptions) => {
|
|
67
|
-
log._config = getConfig(options);
|
|
68
|
-
};
|
|
69
|
-
|
|
70
|
-
// TODO(burdon): API to set context and separate error object.
|
|
71
|
-
// E.g., log.warn('failed', { key: 123 }, err);
|
|
72
|
-
|
|
73
|
-
log.trace = (...params) => processLog(LogLevel.TRACE, ...params);
|
|
74
|
-
log.debug = (...params) => processLog(LogLevel.DEBUG, ...params);
|
|
75
|
-
log.verbose = (...params) => processLog(LogLevel.VERBOSE, ...params);
|
|
76
|
-
log.info = (...params) => processLog(LogLevel.INFO, ...params);
|
|
77
|
-
log.warn = (...params) => processLog(LogLevel.WARN, ...params);
|
|
78
|
-
log.error = (...params) => processLog(LogLevel.ERROR, ...params);
|
|
67
|
+
let logCount = 0;
|
|
79
68
|
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
69
|
+
/**
|
|
70
|
+
* Create a logging function with properties.
|
|
71
|
+
* @internal
|
|
72
|
+
*/
|
|
73
|
+
export const createLog = (): LogImp => {
|
|
74
|
+
// Default function.
|
|
75
|
+
const log: LogImp = ((...params) => processLog(LogLevel.DEBUG, ...params)) as LogImp;
|
|
85
76
|
|
|
86
|
-
|
|
87
|
-
|
|
77
|
+
// Add private properties.
|
|
78
|
+
Object.assign<LogImp, Partial<LogImp>>(log, {
|
|
79
|
+
_id: `log-${++logCount}`,
|
|
80
|
+
_config: createConfig(),
|
|
81
|
+
});
|
|
88
82
|
|
|
89
|
-
|
|
90
|
-
log
|
|
83
|
+
// TODO(burdon): Document.
|
|
84
|
+
Object.defineProperty(log, 'runtimeConfig', {
|
|
85
|
+
get: () => log._config,
|
|
86
|
+
});
|
|
91
87
|
|
|
92
88
|
/**
|
|
93
89
|
* Process the current log call.
|
|
@@ -99,16 +95,72 @@ const createLog = (): LogImp => {
|
|
|
99
95
|
meta?: CallMetadata,
|
|
100
96
|
error?: Error,
|
|
101
97
|
) => {
|
|
102
|
-
|
|
98
|
+
// TODO(burdon): Do the filter matching upstream (here) rather than in each processor?
|
|
99
|
+
log._config.processors.forEach((processor) =>
|
|
100
|
+
processor(log._config, {
|
|
101
|
+
level,
|
|
102
|
+
message,
|
|
103
|
+
context,
|
|
104
|
+
meta,
|
|
105
|
+
error,
|
|
106
|
+
}),
|
|
107
|
+
);
|
|
103
108
|
};
|
|
104
109
|
|
|
110
|
+
/**
|
|
111
|
+
* API.
|
|
112
|
+
*/
|
|
113
|
+
Object.assign<Log, LogMethods>(log, {
|
|
114
|
+
/**
|
|
115
|
+
* Update config.
|
|
116
|
+
* NOTE: Preserves any processors that were already added to this logger instance
|
|
117
|
+
* unless an explicit processor option is provided.
|
|
118
|
+
*/
|
|
119
|
+
config: ({ processor, ...options }) => {
|
|
120
|
+
const config = createConfig(options);
|
|
121
|
+
// TODO(burdon): This could be buggy since the behavior is not reentrant.
|
|
122
|
+
const processors = processor ? config.processors : log._config.processors;
|
|
123
|
+
log._config = { ...config, processors };
|
|
124
|
+
return log;
|
|
125
|
+
},
|
|
126
|
+
|
|
127
|
+
/**
|
|
128
|
+
* Adds a processor to the logger.
|
|
129
|
+
*/
|
|
130
|
+
addProcessor: (processor) => {
|
|
131
|
+
if (log._config.processors.filter((p) => p === processor).length === 0) {
|
|
132
|
+
log._config.processors.push(processor);
|
|
133
|
+
}
|
|
134
|
+
|
|
135
|
+
return () => {
|
|
136
|
+
log._config.processors = log._config.processors.filter((p) => p !== processor);
|
|
137
|
+
};
|
|
138
|
+
},
|
|
139
|
+
|
|
140
|
+
trace: (...params) => processLog(LogLevel.TRACE, ...params),
|
|
141
|
+
debug: (...params) => processLog(LogLevel.DEBUG, ...params),
|
|
142
|
+
verbose: (...params) => processLog(LogLevel.VERBOSE, ...params),
|
|
143
|
+
info: (...params) => processLog(LogLevel.INFO, ...params),
|
|
144
|
+
warn: (...params) => processLog(LogLevel.WARN, ...params),
|
|
145
|
+
error: (...params) => processLog(LogLevel.ERROR, ...params),
|
|
146
|
+
catch: (error, context, meta) => processLog(LogLevel.ERROR, undefined, context, meta, error),
|
|
147
|
+
|
|
148
|
+
method: createMethodLogDecorator(log),
|
|
149
|
+
function: createFunctionLogDecorator(log),
|
|
150
|
+
|
|
151
|
+
break: () => log.info('-'.repeat(80)),
|
|
152
|
+
stack: (message, context, meta) => {
|
|
153
|
+
return processLog(LogLevel.INFO, `${message ?? 'Stack Dump'}\n${getFormattedStackTrace()}`, context, meta);
|
|
154
|
+
},
|
|
155
|
+
});
|
|
156
|
+
|
|
105
157
|
return log;
|
|
106
158
|
};
|
|
107
159
|
|
|
108
160
|
/**
|
|
109
161
|
* Global logging function.
|
|
110
162
|
*/
|
|
111
|
-
export const log: Log = ((globalThis as any).
|
|
163
|
+
export const log: Log = ((globalThis as any).DX_LOG ??= createLog());
|
|
112
164
|
|
|
113
165
|
const start = Date.now();
|
|
114
166
|
let last = start;
|
|
@@ -128,12 +180,4 @@ export const debug = (label?: any, args?: any) => {
|
|
|
128
180
|
last = Date.now();
|
|
129
181
|
};
|
|
130
182
|
|
|
131
|
-
/**
|
|
132
|
-
* Accessible from browser console.
|
|
133
|
-
*/
|
|
134
|
-
declare global {
|
|
135
|
-
// eslint-disable-next-line camelcase
|
|
136
|
-
const dx_log: Log;
|
|
137
|
-
}
|
|
138
|
-
|
|
139
183
|
const getFormattedStackTrace = () => new Error().stack!.split('\n').slice(3).join('\n');
|
package/src/options.ts
CHANGED
|
@@ -12,32 +12,46 @@ import { BROWSER_PROCESSOR, CONSOLE_PROCESSOR, DEBUG_PROCESSOR } from './process
|
|
|
12
12
|
/**
|
|
13
13
|
* Processor variants.
|
|
14
14
|
*/
|
|
15
|
-
export const processors:
|
|
15
|
+
export const processors: Record<string, LogProcessor> = {
|
|
16
16
|
[LogProcessorType.CONSOLE]: CONSOLE_PROCESSOR,
|
|
17
17
|
[LogProcessorType.BROWSER]: BROWSER_PROCESSOR,
|
|
18
18
|
[LogProcessorType.DEBUG]: DEBUG_PROCESSOR,
|
|
19
19
|
};
|
|
20
20
|
|
|
21
|
-
const
|
|
21
|
+
const browser = typeof window !== 'undefined' || typeof navigator !== 'undefined';
|
|
22
22
|
|
|
23
|
-
export const DEFAULT_PROCESSORS = [
|
|
23
|
+
export const DEFAULT_PROCESSORS = [browser ? BROWSER_PROCESSOR : CONSOLE_PROCESSOR];
|
|
24
24
|
|
|
25
|
+
const parseLogLevel = (level: string, defValue = LogLevel.WARN) => levels[level.toLowerCase()] ?? defValue;
|
|
26
|
+
|
|
27
|
+
/**
|
|
28
|
+
* @internal
|
|
29
|
+
*/
|
|
25
30
|
export const parseFilter = (filter: string | string[] | LogLevel): LogFilter[] => {
|
|
26
31
|
if (typeof filter === 'number') {
|
|
27
32
|
return [{ level: filter }];
|
|
28
33
|
}
|
|
29
34
|
|
|
30
|
-
const parseLogLevel = (level: string, defValue = LogLevel.WARN) => levels[level.toLowerCase()] ?? defValue;
|
|
31
|
-
|
|
32
35
|
const lines = typeof filter === 'string' ? filter.split(/,\s*/) : filter;
|
|
33
36
|
return lines.map((filter) => {
|
|
34
37
|
const [pattern, level] = filter.split(':');
|
|
35
|
-
return level
|
|
38
|
+
return level
|
|
39
|
+
? {
|
|
40
|
+
level: parseLogLevel(level),
|
|
41
|
+
pattern,
|
|
42
|
+
}
|
|
43
|
+
: {
|
|
44
|
+
level: parseLogLevel(pattern),
|
|
45
|
+
};
|
|
36
46
|
});
|
|
37
47
|
};
|
|
38
48
|
|
|
39
|
-
|
|
40
|
-
|
|
49
|
+
/**
|
|
50
|
+
* @internal
|
|
51
|
+
*/
|
|
52
|
+
export const createConfig = (options?: LogOptions): LogConfig => {
|
|
53
|
+
// Node only.
|
|
54
|
+
const envOptions: LogOptions | undefined =
|
|
41
55
|
'process' in globalThis
|
|
42
56
|
? {
|
|
43
57
|
file: process!.env.LOG_CONFIG,
|
|
@@ -46,12 +60,12 @@ export const getConfig = (options?: LogOptions): LogConfig => {
|
|
|
46
60
|
}
|
|
47
61
|
: undefined;
|
|
48
62
|
|
|
49
|
-
const mergedOptions: LogOptions = defaultsDeep({}, loadOptions(
|
|
63
|
+
const mergedOptions: LogOptions = defaultsDeep({}, loadOptions(envOptions?.file), envOptions, options);
|
|
50
64
|
return {
|
|
51
65
|
options: mergedOptions,
|
|
52
66
|
filters: parseFilter(mergedOptions.filter ?? LogLevel.INFO),
|
|
53
67
|
captureFilters: parseFilter(mergedOptions.captureFilter ?? LogLevel.WARN),
|
|
54
|
-
processors: mergedOptions.processor ? [processors[mergedOptions.processor]] : DEFAULT_PROCESSORS,
|
|
68
|
+
processors: mergedOptions.processor ? [processors[mergedOptions.processor]] : [...DEFAULT_PROCESSORS],
|
|
55
69
|
prefix: mergedOptions.prefix,
|
|
56
70
|
};
|
|
57
71
|
};
|
|
@@ -74,6 +74,8 @@ const APP_BROWSER_PROCESSOR: LogProcessor = (config, entry) => {
|
|
|
74
74
|
if (context) {
|
|
75
75
|
if (Object.keys(context).length === 1 && 'error' in context) {
|
|
76
76
|
args.push(context.error);
|
|
77
|
+
} else if (Object.keys(context).length === 1 && 'err' in context) {
|
|
78
|
+
args.push(context.err);
|
|
77
79
|
} else {
|
|
78
80
|
args.push(context);
|
|
79
81
|
}
|
|
@@ -14,6 +14,7 @@ import { getRelativeFilename } from './common';
|
|
|
14
14
|
|
|
15
15
|
// Amount of time to retry writing after encountering EAGAIN before giving up.
|
|
16
16
|
const EAGAIN_MAX_DURATION = 1000;
|
|
17
|
+
|
|
17
18
|
/**
|
|
18
19
|
* Create a file processor.
|
|
19
20
|
* @param path - Path to log file to create or append to, or existing open file descriptor e.g. stdout.
|
|
@@ -38,6 +39,7 @@ export const createFileProcessor = ({
|
|
|
38
39
|
if (!shouldLog(entry, filters)) {
|
|
39
40
|
return;
|
|
40
41
|
}
|
|
42
|
+
|
|
41
43
|
if (typeof pathOrFd === 'number') {
|
|
42
44
|
fd = pathOrFd;
|
|
43
45
|
} else {
|