import winston from 'winston'; import 'winston-daily-rotate-file'; import { redactFormat, redactMessage, debugTraverse, jsonTruncateFormat } from './parsers'; import { getLogDirectory } from './utils'; const logDir = getLogDirectory(); const { NODE_ENV, DEBUG_LOGGING, CONSOLE_JSON, DEBUG_CONSOLE } = process.env; const useConsoleJson = typeof CONSOLE_JSON === 'string' && CONSOLE_JSON.toLowerCase() === 'true'; const useDebugConsole = typeof DEBUG_CONSOLE === 'string' && DEBUG_CONSOLE.toLowerCase() === 'true'; const useDebugLogging = typeof DEBUG_LOGGING === 'string' && DEBUG_LOGGING.toLowerCase() === 'true'; const levels: winston.config.AbstractConfigSetLevels = { error: 0, warn: 1, info: 2, http: 3, verbose: 4, debug: 5, activity: 6, silly: 7, }; winston.addColors({ info: 'green', warn: 'italic yellow', error: 'red', debug: 'blue', }); const level = (): string => { const env = NODE_ENV || 'development'; return env === 'development' ? 'debug' : 'warn'; }; const fileFormat = winston.format.combine( redactFormat(), winston.format.timestamp({ format: () => new Date().toISOString() }), winston.format.errors({ stack: true }), winston.format.splat(), ); const transports: winston.transport[] = [ new winston.transports.DailyRotateFile({ level: 'error', filename: `${logDir}/error-%DATE%.log`, datePattern: 'YYYY-MM-DD', zippedArchive: true, maxSize: '20m', maxFiles: '14d', format: winston.format.combine(fileFormat, winston.format.json()), }), ]; if (useDebugLogging) { transports.push( new winston.transports.DailyRotateFile({ level: 'debug', filename: `${logDir}/debug-%DATE%.log`, datePattern: 'YYYY-MM-DD', zippedArchive: true, maxSize: '20m', maxFiles: '14d', format: winston.format.combine(fileFormat, debugTraverse), }), ); } const consoleFormat = winston.format.combine( redactFormat(), winston.format.colorize({ all: true }), winston.format.timestamp({ format: 'YYYY-MM-DD HH:mm:ss' }), winston.format.printf((info) => { const message = `${info.timestamp} ${info.level}: ${info.message}`; return info.level.includes('error') ? redactMessage(message) : message; }), ); let consoleLogLevel: string = 'info'; if (useDebugConsole) { consoleLogLevel = 'debug'; } // Add console transport if (useDebugConsole) { transports.push( new winston.transports.Console({ level: consoleLogLevel, format: useConsoleJson ? winston.format.combine(fileFormat, jsonTruncateFormat(), winston.format.json()) : winston.format.combine(fileFormat, debugTraverse), }), ); } else if (useConsoleJson) { transports.push( new winston.transports.Console({ level: consoleLogLevel, format: winston.format.combine(fileFormat, jsonTruncateFormat(), winston.format.json()), }), ); } else { transports.push( new winston.transports.Console({ level: consoleLogLevel, format: consoleFormat, }), ); } // Create logger const logger = winston.createLogger({ level: level(), levels, transports, }); export default logger;