@trojs/logger 2.3.0 → 2.5.0

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/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "@trojs/logger",
3
3
  "description": "Winston logger for TroJS",
4
- "version": "2.3.0",
4
+ "version": "2.5.0",
5
5
  "author": {
6
6
  "name": "Pieter Wigboldus",
7
7
  "url": "https://trojs.org/"
package/src/logger.js CHANGED
@@ -29,6 +29,7 @@ const levels = {
29
29
  * @param {Array<{[key: string]: string}>} [options.loggers=defaultLoggers] - Array of logger transport configurations.
30
30
  * @param {string} [options.level='info'] - Minimum log level for the logger.
31
31
  * @param {object} [options.meta={}] - Default metadata to include in all log messages.
32
+ * @param {boolean} [options.exitOnError=false] - Whether the logger should exit on error.
32
33
  * @returns {LoggerType} Winston logger instance with custom level wrappers.
33
34
  * These handlers will log errors and warnings using the logger, and are only attached once per process.
34
35
  * @example
@@ -36,14 +37,15 @@ const levels = {
36
37
  * const logger = createLogger({ level: 'debug', meta: { service: 'api' } });
37
38
  * logger.info('Service started');
38
39
  */
39
- export default ({ loggers = defaultLoggers, level = 'info', meta = {} } = {}) => {
40
+ export default ({ loggers = defaultLoggers, level = 'info', meta = {}, exitOnError = false } = {}) => {
40
41
  const winstonLoggers = makeLoggers({ winston, loggers })
41
42
 
42
43
  const logger = winston.createLogger({
43
44
  level,
44
45
  levels,
45
46
  defaultMeta: meta,
46
- transports: winstonLoggers
47
+ transports: winstonLoggers,
48
+ exitOnError
47
49
  })
48
50
 
49
51
  const wrapLevel = (lvl) => {
@@ -85,6 +85,26 @@ export class SentryTransport extends TransportStream {
85
85
 
86
86
  const sentryLevel = this.levelsMap[winstonLevel]
87
87
 
88
+ // Normalize message to string
89
+ let normalizedMessage = message
90
+ if (!normalizedMessage || normalizedMessage === '') {
91
+ // Try to get message from stack or error
92
+ if (meta.stack) {
93
+ normalizedMessage = (meta.stack.split('\n')[0] || '').trim()
94
+ } else if (meta.error instanceof Error) {
95
+ normalizedMessage = meta.error.message || meta.error.toString()
96
+ } else {
97
+ normalizedMessage = 'Empty log message'
98
+ }
99
+ } else if (typeof normalizedMessage !== 'string') {
100
+ // Stringify non-string messages
101
+ try {
102
+ normalizedMessage = JSON.stringify(normalizedMessage)
103
+ } catch {
104
+ normalizedMessage = String(normalizedMessage)
105
+ }
106
+ }
107
+
88
108
  return Sentry.withScope((scope) => {
89
109
  if (tags !== undefined && SentryTransport.isObject(tags)) {
90
110
  scope.setTags(tags)
@@ -100,11 +120,11 @@ export class SentryTransport extends TransportStream {
100
120
  if (SentryTransport.shouldLogException(sentryLevel)) {
101
121
  const error
102
122
  = Object.values(info).find((value) => value instanceof Error)
103
- ?? new ExtendedError(info)
123
+ ?? new ExtendedError({ ...info, message: normalizedMessage })
104
124
  Sentry.captureException(error, { tags, level: sentryLevel })
105
125
  } else {
106
126
  // Capturing Messages
107
- Sentry.captureMessage(message, sentryLevel)
127
+ Sentry.captureMessage(normalizedMessage, sentryLevel)
108
128
  }
109
129
  })
110
130
  }