From 76837c08b76731b770fc8b461911f9d00c76f749 Mon Sep 17 00:00:00 2001 From: Pierre Gayvallet Date: Mon, 3 Jun 2024 10:12:21 +0200 Subject: [PATCH] browser-side logging: allow per-logger level configuration (#184568) ## Summary Part of https://github.com/elastic/kibana/issues/144276 Somewhat related to https://github.com/elastic/kibana/pull/184520 Allow to configure level of individual browser-side loggers **Example:** ```yaml logging.browser: loggers: - name: analytics level: debug ``` --- .../src/logging_system.test.ts | 85 +++++++++++++++++-- .../src/logging_system.ts | 43 ++++++++-- .../core-logging-common-internal/index.ts | 6 +- .../src/browser_config.ts | 6 ++ .../src/logging_config.test.ts | 31 +++++++ .../src/logging_config.ts | 21 +++-- 6 files changed, 172 insertions(+), 20 deletions(-) diff --git a/packages/core/logging/core-logging-browser-internal/src/logging_system.test.ts b/packages/core/logging/core-logging-browser-internal/src/logging_system.test.ts index 0058104083f2b..c8723bdf08570 100644 --- a/packages/core/logging/core-logging-browser-internal/src/logging_system.test.ts +++ b/packages/core/logging/core-logging-browser-internal/src/logging_system.test.ts @@ -6,21 +6,23 @@ * Side Public License, v 1. */ -import type { BrowserLoggingConfig } from '@kbn/core-logging-common-internal'; +import type { LogLevelId, Logger } from '@kbn/logging'; import { unsafeConsole } from '@kbn/security-hardening'; +import type { BrowserLoggingConfig } from '@kbn/core-logging-common-internal'; import { BrowserLoggingSystem } from './logging_system'; +import type { BaseLogger } from './logger'; describe('BrowserLoggingSystem', () => { const timestamp = new Date(Date.UTC(2012, 1, 1, 14, 33, 22, 11)); let mockConsoleLog: jest.SpyInstance; - let loggingSystem: BrowserLoggingSystem; const createLoggingConfig = (parts: Partial = {}): BrowserLoggingConfig => { return { root: { level: 'warn', }, + loggers: [], ...parts, }; }; @@ -28,7 +30,6 @@ describe('BrowserLoggingSystem', () => { beforeEach(() => { mockConsoleLog = jest.spyOn(unsafeConsole, 'log').mockReturnValue(undefined); jest.spyOn(global, 'Date').mockImplementation(() => timestamp); - loggingSystem = new BrowserLoggingSystem(createLoggingConfig()); }); afterEach(() => { @@ -37,20 +38,23 @@ describe('BrowserLoggingSystem', () => { describe('#get', () => { it('returns the same logger for same context', () => { + const loggingSystem = new BrowserLoggingSystem(createLoggingConfig()); const loggerA = loggingSystem.get('same.logger'); const loggerB = loggingSystem.get('same.logger'); expect(loggerA).toBe(loggerB); }); it('returns different loggers for different contexts', () => { + const loggingSystem = new BrowserLoggingSystem(createLoggingConfig()); const loggerA = loggingSystem.get('some.logger'); const loggerB = loggingSystem.get('another.logger'); expect(loggerA).not.toBe(loggerB); }); }); - describe('logger configuration', () => { + describe('root logger configuration', () => { it('properly configure the logger to use the correct context and pattern', () => { + const loggingSystem = new BrowserLoggingSystem(createLoggingConfig()); const logger = loggingSystem.get('foo.bar'); logger.warn('some message'); @@ -62,6 +66,7 @@ describe('BrowserLoggingSystem', () => { }); it('properly configure the logger to use the correct level', () => { + const loggingSystem = new BrowserLoggingSystem(createLoggingConfig()); const logger = loggingSystem.get('foo.bar'); logger.trace('some trace message'); logger.debug('some debug message'); @@ -86,9 +91,12 @@ describe('BrowserLoggingSystem', () => { }); it('allows to override the root logger level', () => { - loggingSystem = new BrowserLoggingSystem(createLoggingConfig({ root: { level: 'debug' } })); + const loggingSystem = new BrowserLoggingSystem( + createLoggingConfig({ root: { level: 'debug' } }) + ); const logger = loggingSystem.get('foo.bar'); + logger.trace('some trace message'); logger.debug('some debug message'); logger.info('some info message'); @@ -117,4 +125,71 @@ describe('BrowserLoggingSystem', () => { `); }); }); + + describe('loggers configuration', () => { + it('uses the logger config if specified', () => { + const loggingSystem = new BrowserLoggingSystem( + createLoggingConfig({ + root: { level: 'debug' }, + loggers: [{ name: 'foo.bar', level: 'warn' }], + }) + ); + + const logger = loggingSystem.get('foo.bar') as BaseLogger; + + expect(getLoggerLevel(logger)).toBe('warn'); + }); + + it('uses the parent config if present and logger config is not', () => { + const loggingSystem = new BrowserLoggingSystem( + createLoggingConfig({ + root: { level: 'debug' }, + loggers: [{ name: 'foo', level: 'warn' }], + }) + ); + + const logger = loggingSystem.get('foo.bar') as BaseLogger; + + expect(getLoggerLevel(logger)).toBe('warn'); + }); + + it('uses the closest parent config', () => { + const loggingSystem = new BrowserLoggingSystem( + createLoggingConfig({ + root: { level: 'debug' }, + loggers: [ + { name: 'foo', level: 'warn' }, + { name: 'foo.bar', level: 'error' }, + ], + }) + ); + + const logger = loggingSystem.get('foo.bar.hello') as BaseLogger; + + expect(getLoggerLevel(logger)).toBe('error'); + }); + + it('uses the root logger config by default', () => { + const loggingSystem = new BrowserLoggingSystem( + createLoggingConfig({ + root: { level: 'debug' }, + loggers: [], + }) + ); + + const logger = loggingSystem.get('foo.bar.hello') as BaseLogger; + + expect(getLoggerLevel(logger)).toBe('debug'); + }); + }); }); + +const getLoggerLevel = (logger: Logger): LogLevelId => { + const levels: LogLevelId[] = ['all', 'trace', 'debug', 'info', 'warn', 'error', 'fatal', 'all']; + for (const level of levels) { + if (logger.isLevelEnabled(level)) { + return level; + } + } + return 'off'; +}; diff --git a/packages/core/logging/core-logging-browser-internal/src/logging_system.ts b/packages/core/logging/core-logging-browser-internal/src/logging_system.ts index 155146dce772c..3132f442f1a17 100644 --- a/packages/core/logging/core-logging-browser-internal/src/logging_system.ts +++ b/packages/core/logging/core-logging-browser-internal/src/logging_system.ts @@ -7,7 +7,13 @@ */ import { LogLevel, Logger, LoggerFactory, DisposableAppender } from '@kbn/logging'; -import { getLoggerContext, BrowserLoggingConfig } from '@kbn/core-logging-common-internal'; +import { + ROOT_CONTEXT_NAME, + getLoggerContext, + getParentLoggerContext, + BrowserLoggingConfig, + BrowserLoggerConfig, +} from '@kbn/core-logging-common-internal'; import type { LoggerConfigType } from './types'; import { BaseLogger } from './logger'; import { PatternLayout } from './layouts'; @@ -22,15 +28,20 @@ export interface IBrowserLoggingSystem extends LoggerFactory { asLoggerFactory(): LoggerFactory; } +interface ComputedLoggerConfig { + loggers: Map; +} + /** * @internal */ export class BrowserLoggingSystem implements IBrowserLoggingSystem { + private readonly computedConfig: ComputedLoggerConfig; private readonly loggers: Map = new Map(); private readonly appenders: Map = new Map(); - constructor(private readonly loggingConfig: BrowserLoggingConfig) { - this.setupSystem(loggingConfig); + constructor(loggingConfig: BrowserLoggingConfig) { + this.computedConfig = this.setupSystem(loggingConfig); } public get(...contextParts: string[]): Logger { @@ -49,16 +60,32 @@ export class BrowserLoggingSystem implements IBrowserLoggingSystem { } private getLoggerConfigByContext(context: string): LoggerConfigType { - return { - level: this.loggingConfig.root.level, - appenders: [CONSOLE_APPENDER_ID], - name: context, - }; + const loggerConfig = this.computedConfig.loggers.get(context); + if (loggerConfig !== undefined) { + return { + name: loggerConfig.name, + level: loggerConfig.level, + appenders: [CONSOLE_APPENDER_ID], + }; + } + + // If we don't have configuration for the specified context, we move up to the parent context, up to `root` + return this.getLoggerConfigByContext(getParentLoggerContext(context)); } private setupSystem(loggingConfig: BrowserLoggingConfig) { const consoleAppender = new ConsoleAppender(new PatternLayout()); this.appenders.set(CONSOLE_APPENDER_ID, consoleAppender); + + const loggerConfigs = loggingConfig.loggers.reduce((loggers, logger) => { + loggers.set(logger.name, logger); + return loggers; + }, new Map()); + loggerConfigs.set(ROOT_CONTEXT_NAME, { name: ROOT_CONTEXT_NAME, ...loggingConfig.root }); + + return { + loggers: loggerConfigs, + }; } /** diff --git a/packages/core/logging/core-logging-common-internal/index.ts b/packages/core/logging/core-logging-common-internal/index.ts index 6c9f1c510a512..341ac510c0ff2 100644 --- a/packages/core/logging/core-logging-common-internal/index.ts +++ b/packages/core/logging/core-logging-common-internal/index.ts @@ -22,4 +22,8 @@ export { ROOT_CONTEXT_NAME, DEFAULT_APPENDER_NAME, } from './src'; -export type { BrowserLoggingConfig, BrowserRootLoggerConfig } from './src/browser_config'; +export type { + BrowserLoggingConfig, + BrowserRootLoggerConfig, + BrowserLoggerConfig, +} from './src/browser_config'; diff --git a/packages/core/logging/core-logging-common-internal/src/browser_config.ts b/packages/core/logging/core-logging-common-internal/src/browser_config.ts index ccdc57b9369b3..05e3993e095e6 100644 --- a/packages/core/logging/core-logging-common-internal/src/browser_config.ts +++ b/packages/core/logging/core-logging-common-internal/src/browser_config.ts @@ -13,6 +13,12 @@ import type { LogLevelId } from '@kbn/logging'; */ export interface BrowserLoggingConfig { root: BrowserRootLoggerConfig; + loggers: BrowserLoggerConfig[]; +} + +export interface BrowserLoggerConfig { + name: string; + level: LogLevelId; } /** diff --git a/packages/core/logging/core-logging-server-internal/src/logging_config.test.ts b/packages/core/logging/core-logging-server-internal/src/logging_config.test.ts index 764cce1b34dd5..de1b4d91a277a 100644 --- a/packages/core/logging/core-logging-server-internal/src/logging_config.test.ts +++ b/packages/core/logging/core-logging-server-internal/src/logging_config.test.ts @@ -13,6 +13,7 @@ test('`schema` creates correct schema with defaults.', () => { Object { "appenders": Map {}, "browser": Object { + "loggers": Array [], "root": Object { "level": "info", }, @@ -165,6 +166,36 @@ test('correctly fills in custom `loggers` config.', () => { }); }); +test('correctly fills in custom browser-side `loggers` config.', () => { + const configValue = config.schema.validate({ + browser: { + loggers: [ + { + name: 'plugins', + level: 'warn', + }, + { + name: 'http', + level: 'error', + }, + ], + }, + }); + + expect(configValue.browser.loggers).toMatchInlineSnapshot(` + Array [ + Object { + "level": "warn", + "name": "plugins", + }, + Object { + "level": "error", + "name": "http", + }, + ] + `); +}); + test('fails if loggers use unknown appenders.', () => { const validateConfig = config.schema.validate({ loggers: [ diff --git a/packages/core/logging/core-logging-server-internal/src/logging_config.ts b/packages/core/logging/core-logging-server-internal/src/logging_config.ts index 191e859a0fe6f..e887e3e0dbfa3 100644 --- a/packages/core/logging/core-logging-server-internal/src/logging_config.ts +++ b/packages/core/logging/core-logging-server-internal/src/logging_config.ts @@ -36,6 +36,21 @@ const levelSchema = schema.oneOf( } ); +// until we have feature parity between browser and server logging, we need to define distinct logger schemas +const browserLoggerSchema = schema.object({ + name: schema.string(), + level: levelSchema, +}); + +const browserConfig = schema.object({ + root: schema.object({ + level: levelSchema, + }), + loggers: schema.arrayOf(browserLoggerSchema, { + defaultValue: [], + }), +}); + /** * Config schema for validating the `loggers` key in {@link LoggerContextConfigType} or {@link LoggingConfigType}. * @@ -47,12 +62,6 @@ export const loggerSchema = schema.object({ level: levelSchema, }); -const browserConfig = schema.object({ - root: schema.object({ - level: levelSchema, - }), -}); - export const config = { path: 'logging', schema: schema.object({