@degel_lc/logger 0.1.3 → 0.1.5

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/README.md CHANGED
@@ -199,17 +199,34 @@ import { initializeLogger, setSentryModule } from '@degel_lc/logger';
199
199
  import * as Sentry from '@sentry/nextjs';
200
200
 
201
201
  export async function register() {
202
- await initializeLogger({
203
- logDir: './logs',
204
- enableSentry: true,
205
- });
206
-
207
- // @sentry/nextjs も対応
208
- setSentryModule(Sentry);
202
+ if (process.env.NEXT_RUNTIME === 'nodejs') {
203
+ await initializeLogger({
204
+ logDir: './logs',
205
+ enableSentry: true,
206
+ });
207
+
208
+ // @sentry/nextjs も対応
209
+ setSentryModule(Sentry);
210
+ }
209
211
  }
210
212
  ```
211
213
 
212
- **注意**: `@sentry/node` または `@sentry/nextjs` はオプショナルなpeerDependencyです。Sentry連携を使用する場合は別途インストールが必要です。
214
+ **注意**:
215
+ - `@sentry/node` または `@sentry/nextjs` はオプショナルなpeerDependencyです。Sentry連携を使用する場合は別途インストールが必要です。
216
+ - `initializeLogger()` は Node.js ランタイムでのみ呼び出してください(`process.env.NEXT_RUNTIME === 'nodejs'`)。
217
+
218
+ ## Next.js バンドラー対応
219
+
220
+ このパッケージは Next.js(webpack/Turbopack)との互換性を考慮して設計されています。
221
+
222
+ - **コード分割**: サーバー専用コード(`fs`/`path`モジュール使用)は別チャンクに分離
223
+ - **動的インポート**: `createServerLogger` は内部で動的インポートを使用
224
+ - **静的インポート可**: `instrumentation.ts` などで静的インポートが可能
225
+
226
+ ```typescript
227
+ // ✅ 静的インポートOK(Next.js 16対応)
228
+ import { initializeLogger, setSentryModule } from '@degel_lc/logger';
229
+ ```
213
230
 
214
231
  ## 開発
215
232
 
@@ -0,0 +1,96 @@
1
+ // src/utils.ts
2
+ var DEFAULT_TIMEZONE_OFFSET = 540;
3
+ function getCurrentDate(timezoneOffset = DEFAULT_TIMEZONE_OFFSET) {
4
+ const now = /* @__PURE__ */ new Date();
5
+ now.setTime(now.getTime() + timezoneOffset * 60 * 1e3);
6
+ const year = now.getUTCFullYear();
7
+ const month = String(now.getUTCMonth() + 1).padStart(2, "0");
8
+ const day = String(now.getUTCDate()).padStart(2, "0");
9
+ return `${year}_${month}_${day}`;
10
+ }
11
+ function getFormattedDateTime(timezoneOffset = DEFAULT_TIMEZONE_OFFSET) {
12
+ const now = /* @__PURE__ */ new Date();
13
+ now.setTime(now.getTime() + timezoneOffset * 60 * 1e3);
14
+ const year = now.getUTCFullYear();
15
+ const month = String(now.getUTCMonth() + 1).padStart(2, "0");
16
+ const day = String(now.getUTCDate()).padStart(2, "0");
17
+ const hours = String(now.getUTCHours()).padStart(2, "0");
18
+ const minutes = String(now.getUTCMinutes()).padStart(2, "0");
19
+ const seconds = String(now.getUTCSeconds()).padStart(2, "0");
20
+ return `${year}_${month}_${day} ${hours}:${minutes}:${seconds}`;
21
+ }
22
+ function formatLogMessage(level, message, location, originalError, timezoneOffset = DEFAULT_TIMEZONE_OFFSET) {
23
+ const formattedDate = getFormattedDateTime(timezoneOffset);
24
+ const errorType = level.toUpperCase();
25
+ const locationInfo = location ?? "Unknown";
26
+ const original = originalError ? originalError.toString() : originalError === null ? message : "\u306A\u3057";
27
+ return `${formattedDate}\uFF1A${errorType}\uFF1A${message}\uFF1A${locationInfo}\uFF1A${original}`;
28
+ }
29
+ function isLogContext(arg) {
30
+ return typeof arg === "object" && arg !== null && !Array.isArray(arg) && typeof arg !== "string";
31
+ }
32
+ function formatContext(ctx) {
33
+ const parts = [];
34
+ if (ctx.path) parts.push(`path=${ctx.path}`);
35
+ if (ctx.method) parts.push(`method=${ctx.method}`);
36
+ if (ctx.userId) parts.push(`userId=${ctx.userId}`);
37
+ if (ctx.tenantId) parts.push(`tenantId=${ctx.tenantId}`);
38
+ if (ctx.location) parts.push(`location=${ctx.location}`);
39
+ for (const [key, value] of Object.entries(ctx)) {
40
+ if (!["path", "method", "userId", "tenantId", "location"].includes(key) && value !== void 0) {
41
+ parts.push(`${key}=${String(value)}`);
42
+ }
43
+ }
44
+ return parts.length > 0 ? `[${parts.join(", ")}]` : "";
45
+ }
46
+ function formatLogMessageWithContext(level, message, ctx, error, timezoneOffset = DEFAULT_TIMEZONE_OFFSET) {
47
+ const formattedDate = getFormattedDateTime(timezoneOffset);
48
+ const errorType = level.toUpperCase();
49
+ const contextInfo = formatContext(ctx);
50
+ let errorInfo = "";
51
+ if (error instanceof Error) {
52
+ errorInfo = ` | Error: ${error.message}`;
53
+ if (error.stack) {
54
+ errorInfo += `
55
+ ${error.stack}`;
56
+ }
57
+ } else if (error !== void 0 && error !== null) {
58
+ errorInfo = ` | Error: ${String(error)}`;
59
+ }
60
+ const contextPart = contextInfo ? ` ${contextInfo}` : "";
61
+ return `${formattedDate}\uFF1A${errorType}\uFF1A${message}${contextPart}${errorInfo}`;
62
+ }
63
+
64
+ // src/sentry.ts
65
+ var sentryModule = null;
66
+ function setSentryModule(sentry) {
67
+ sentryModule = sentry;
68
+ }
69
+ function isSentryEnabled() {
70
+ return sentryModule !== null;
71
+ }
72
+ function captureError(ctx, message, error) {
73
+ if (!sentryModule) return;
74
+ if (error instanceof Error) {
75
+ sentryModule.captureException(error, {
76
+ extra: {
77
+ ...ctx,
78
+ message
79
+ }
80
+ });
81
+ } else if (error) {
82
+ sentryModule.captureMessage(`${message}: ${String(error)}`, {
83
+ level: "error",
84
+ extra: ctx
85
+ });
86
+ } else {
87
+ sentryModule.captureMessage(message, {
88
+ level: "error",
89
+ extra: ctx
90
+ });
91
+ }
92
+ }
93
+
94
+ export { captureError, formatLogMessage, formatLogMessageWithContext, getCurrentDate, isLogContext, isSentryEnabled, setSentryModule };
95
+ //# sourceMappingURL=chunk-34MNPN6V.js.map
96
+ //# sourceMappingURL=chunk-34MNPN6V.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/utils.ts","../src/sentry.ts"],"names":[],"mappings":"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","file":"chunk-34MNPN6V.js","sourcesContent":["import type { LogContext } from './types.js';\n\nconst DEFAULT_TIMEZONE_OFFSET = 540; // JST = UTC+9 (in minutes)\n\n/**\n * 現在の日付を YYYY_MM_DD 形式で取得\n */\nexport function getCurrentDate(timezoneOffset = DEFAULT_TIMEZONE_OFFSET): string {\n const now = new Date();\n now.setTime(now.getTime() + timezoneOffset * 60 * 1000);\n\n const year = now.getUTCFullYear();\n const month = String(now.getUTCMonth() + 1).padStart(2, '0');\n const day = String(now.getUTCDate()).padStart(2, '0');\n\n return `${year}_${month}_${day}`;\n}\n\n/**\n * 現在の日時をフォーマットする(タイムゾーン調整付き)\n */\nexport function getFormattedDateTime(timezoneOffset = DEFAULT_TIMEZONE_OFFSET): string {\n const now = new Date();\n now.setTime(now.getTime() + timezoneOffset * 60 * 1000);\n\n const year = now.getUTCFullYear();\n const month = String(now.getUTCMonth() + 1).padStart(2, '0');\n const day = String(now.getUTCDate()).padStart(2, '0');\n const hours = String(now.getUTCHours()).padStart(2, '0');\n const minutes = String(now.getUTCMinutes()).padStart(2, '0');\n const seconds = String(now.getUTCSeconds()).padStart(2, '0');\n\n return `${year}_${month}_${day} ${hours}:${minutes}:${seconds}`;\n}\n\n/**\n * ログメッセージをフォーマットする\n */\nexport function formatLogMessage(\n level: string,\n message: string,\n location: string | null,\n originalError: Error | string | null,\n timezoneOffset = DEFAULT_TIMEZONE_OFFSET\n): string {\n const formattedDate = getFormattedDateTime(timezoneOffset);\n const errorType = level.toUpperCase();\n const locationInfo = location ?? 'Unknown';\n const original = originalError\n ? originalError.toString()\n : originalError === null\n ? message\n : 'なし';\n\n return `${formattedDate}:${errorType}:${message}:${locationInfo}:${original}`;\n}\n\n/**\n * 引数がLogContextかどうかを判定\n */\nexport function isLogContext(arg: unknown): arg is LogContext {\n return typeof arg === 'object' && arg !== null && !Array.isArray(arg) && typeof arg !== 'string';\n}\n\n/**\n * コンテキスト情報をフォーマットする\n */\nfunction formatContext(ctx: LogContext): string {\n const parts: string[] = [];\n\n if (ctx.path) parts.push(`path=${ctx.path}`);\n if (ctx.method) parts.push(`method=${ctx.method}`);\n if (ctx.userId) parts.push(`userId=${ctx.userId}`);\n if (ctx.tenantId) parts.push(`tenantId=${ctx.tenantId}`);\n if (ctx.location) parts.push(`location=${ctx.location}`);\n\n // その他のカスタムフィールド\n for (const [key, value] of Object.entries(ctx)) {\n if (\n !['path', 'method', 'userId', 'tenantId', 'location'].includes(key) &&\n value !== undefined\n ) {\n parts.push(`${key}=${String(value)}`);\n }\n }\n\n return parts.length > 0 ? `[${parts.join(', ')}]` : '';\n}\n\n/**\n * 構造化コンテキスト対応のログメッセージをフォーマットする\n */\nexport function formatLogMessageWithContext(\n level: string,\n message: string,\n ctx: LogContext,\n error: unknown,\n timezoneOffset = DEFAULT_TIMEZONE_OFFSET\n): string {\n const formattedDate = getFormattedDateTime(timezoneOffset);\n const errorType = level.toUpperCase();\n const contextInfo = formatContext(ctx);\n\n let errorInfo = '';\n if (error instanceof Error) {\n errorInfo = ` | Error: ${error.message}`;\n if (error.stack) {\n errorInfo += `\\n${error.stack}`;\n }\n } else if (error !== undefined && error !== null) {\n errorInfo = ` | Error: ${String(error)}`;\n }\n\n const contextPart = contextInfo ? ` ${contextInfo}` : '';\n return `${formattedDate}:${errorType}:${message}${contextPart}${errorInfo}`;\n}\n","/**\n * Sentry連携ヘルパー\n */\n\nimport type { LogContext, SentryModule } from './types.js';\n\nlet sentryModule: SentryModule | null = null;\n\n/**\n * Sentryモジュールを設定\n */\nexport function setSentryModule(sentry: SentryModule): void {\n sentryModule = sentry;\n}\n\n/**\n * Sentryが有効かどうか\n */\nexport function isSentryEnabled(): boolean {\n return sentryModule !== null;\n}\n\n/**\n * Sentryモジュールをクリア(テスト用)\n */\nexport function clearSentryModule(): void {\n sentryModule = null;\n}\n\n/**\n * エラーをSentryに送信\n */\nexport function captureError(ctx: LogContext, message: string, error?: unknown): void {\n if (!sentryModule) return;\n\n if (error instanceof Error) {\n sentryModule.captureException(error, {\n extra: {\n ...ctx,\n message,\n },\n });\n } else if (error) {\n sentryModule.captureMessage(`${message}: ${String(error)}`, {\n level: 'error',\n extra: ctx,\n });\n } else {\n sentryModule.captureMessage(message, {\n level: 'error',\n extra: ctx,\n });\n }\n}\n"]}
package/dist/index.d.ts CHANGED
@@ -110,11 +110,6 @@ interface LoggerConfig {
110
110
  */
111
111
  declare function createClientLogger(config?: LoggerConfig): Logger;
112
112
 
113
- /**
114
- * サーバーサイド用ロガーを作成
115
- */
116
- declare function createServerLogger(config?: LoggerConfig): Promise<Logger>;
117
-
118
113
  /**
119
114
  * Sentry連携ヘルパー
120
115
  */
@@ -128,6 +123,12 @@ declare function setSentryModule(sentry: SentryModule): void;
128
123
  */
129
124
  declare function isSentryEnabled(): boolean;
130
125
 
126
+ /**
127
+ * サーバーサイド用ロガー作成関数(動的import)
128
+ * Next.jsなどのバンドラーでクライアント用にバンドルされる際に
129
+ * fsモジュールの解決エラーを防ぐため、動的importで提供
130
+ */
131
+ declare function createServerLogger(config?: LoggerConfig): Promise<Logger>;
131
132
  /**
132
133
  * グローバルシングルトン用の型定義
133
134
  * Next.js開発モードでのモジュール分離問題を解決するため、
package/dist/index.js CHANGED
@@ -1,67 +1,5 @@
1
- import pino from 'pino';
2
-
3
- // src/utils.ts
4
- var DEFAULT_TIMEZONE_OFFSET = 540;
5
- function getCurrentDate(timezoneOffset = DEFAULT_TIMEZONE_OFFSET) {
6
- const now = /* @__PURE__ */ new Date();
7
- now.setTime(now.getTime() + timezoneOffset * 60 * 1e3);
8
- const year = now.getUTCFullYear();
9
- const month = String(now.getUTCMonth() + 1).padStart(2, "0");
10
- const day = String(now.getUTCDate()).padStart(2, "0");
11
- return `${year}_${month}_${day}`;
12
- }
13
- function getFormattedDateTime(timezoneOffset = DEFAULT_TIMEZONE_OFFSET) {
14
- const now = /* @__PURE__ */ new Date();
15
- now.setTime(now.getTime() + timezoneOffset * 60 * 1e3);
16
- const year = now.getUTCFullYear();
17
- const month = String(now.getUTCMonth() + 1).padStart(2, "0");
18
- const day = String(now.getUTCDate()).padStart(2, "0");
19
- const hours = String(now.getUTCHours()).padStart(2, "0");
20
- const minutes = String(now.getUTCMinutes()).padStart(2, "0");
21
- const seconds = String(now.getUTCSeconds()).padStart(2, "0");
22
- return `${year}_${month}_${day} ${hours}:${minutes}:${seconds}`;
23
- }
24
- function formatLogMessage(level, message, location, originalError, timezoneOffset = DEFAULT_TIMEZONE_OFFSET) {
25
- const formattedDate = getFormattedDateTime(timezoneOffset);
26
- const errorType = level.toUpperCase();
27
- const locationInfo = location ?? "Unknown";
28
- const original = originalError ? originalError.toString() : originalError === null ? message : "\u306A\u3057";
29
- return `${formattedDate}\uFF1A${errorType}\uFF1A${message}\uFF1A${locationInfo}\uFF1A${original}`;
30
- }
31
- function isLogContext(arg) {
32
- return typeof arg === "object" && arg !== null && !Array.isArray(arg) && typeof arg !== "string";
33
- }
34
- function formatContext(ctx) {
35
- const parts = [];
36
- if (ctx.path) parts.push(`path=${ctx.path}`);
37
- if (ctx.method) parts.push(`method=${ctx.method}`);
38
- if (ctx.userId) parts.push(`userId=${ctx.userId}`);
39
- if (ctx.tenantId) parts.push(`tenantId=${ctx.tenantId}`);
40
- if (ctx.location) parts.push(`location=${ctx.location}`);
41
- for (const [key, value] of Object.entries(ctx)) {
42
- if (!["path", "method", "userId", "tenantId", "location"].includes(key) && value !== void 0) {
43
- parts.push(`${key}=${String(value)}`);
44
- }
45
- }
46
- return parts.length > 0 ? `[${parts.join(", ")}]` : "";
47
- }
48
- function formatLogMessageWithContext(level, message, ctx, error, timezoneOffset = DEFAULT_TIMEZONE_OFFSET) {
49
- const formattedDate = getFormattedDateTime(timezoneOffset);
50
- const errorType = level.toUpperCase();
51
- const contextInfo = formatContext(ctx);
52
- let errorInfo = "";
53
- if (error instanceof Error) {
54
- errorInfo = ` | Error: ${error.message}`;
55
- if (error.stack) {
56
- errorInfo += `
57
- ${error.stack}`;
58
- }
59
- } else if (error !== void 0 && error !== null) {
60
- errorInfo = ` | Error: ${String(error)}`;
61
- }
62
- const contextPart = contextInfo ? ` ${contextInfo}` : "";
63
- return `${formattedDate}\uFF1A${errorType}\uFF1A${message}${contextPart}${errorInfo}`;
64
- }
1
+ import { isLogContext, formatLogMessageWithContext, formatLogMessage } from './chunk-34MNPN6V.js';
2
+ export { isSentryEnabled, setSentryModule } from './chunk-34MNPN6V.js';
65
3
 
66
4
  // src/client.ts
67
5
  var LOG_LEVEL_PRIORITY = {
@@ -123,191 +61,11 @@ function createClientLogger(config = {}) {
123
61
  };
124
62
  }
125
63
 
126
- // src/sentry.ts
127
- var sentryModule = null;
128
- function setSentryModule(sentry) {
129
- sentryModule = sentry;
130
- }
131
- function isSentryEnabled() {
132
- return sentryModule !== null;
133
- }
134
- function captureError(ctx, message, error) {
135
- if (!sentryModule) return;
136
- if (error instanceof Error) {
137
- sentryModule.captureException(error, {
138
- extra: {
139
- ...ctx,
140
- message
141
- }
142
- });
143
- } else if (error) {
144
- sentryModule.captureMessage(`${message}: ${String(error)}`, {
145
- level: "error",
146
- extra: ctx
147
- });
148
- } else {
149
- sentryModule.captureMessage(message, {
150
- level: "error",
151
- extra: ctx
152
- });
153
- }
154
- }
155
-
156
- // src/server.ts
157
- var LOG_LEVEL_PRIORITY2 = {
158
- debug: 0,
159
- info: 1,
160
- warn: 2,
161
- error: 3,
162
- fatal: 4
163
- };
164
- function shouldLog2(logLevel, minLevel) {
165
- return LOG_LEVEL_PRIORITY2[logLevel] >= LOG_LEVEL_PRIORITY2[minLevel];
166
- }
167
- async function createServerLogger(config = {}) {
168
- const {
169
- // biome-ignore lint/complexity/useLiteralKeys: TypeScript requires bracket notation for process.env
170
- logDir = process.env["LOG_DIR"] ?? `${process.cwd()}/logs`,
171
- // biome-ignore lint/complexity/useLiteralKeys: TypeScript requires bracket notation for process.env
172
- level = process.env["LOG_LEVEL"] ?? "debug",
173
- timezoneOffset = 540,
174
- rotationCheckInterval = 60 * 60 * 1e3,
175
- // 1時間
176
- enableSentry = false
177
- } = config;
178
- const validLevels = ["debug", "info", "warn", "error", "fatal"];
179
- const minLevel = validLevels.includes(level) ? level : "debug";
180
- const fs = await import('fs');
181
- const path = await import('path');
182
- if (!fs.existsSync(logDir)) {
183
- fs.mkdirSync(logDir, { recursive: true });
184
- }
185
- const logTypes = {
186
- debug: { fileNamePrefix: "debug_log", level: "debug" },
187
- info: { fileNamePrefix: "info_log", level: "info" },
188
- error: { fileNamePrefix: "error_log", level: "error" },
189
- fatal: { fileNamePrefix: "fatal_log", level: "fatal" },
190
- sqlError: { fileNamePrefix: "sql_error_log", level: "error" }
191
- };
192
- function createLogStream(fileNamePrefix) {
193
- const dateStr = getCurrentDate(timezoneOffset);
194
- const logFilePath = path.join(logDir, `${fileNamePrefix}_${dateStr}.log`);
195
- return fs.createWriteStream(logFilePath, { flags: "a" });
196
- }
197
- function createPinoLogger(fileNamePrefix) {
198
- return pino(
199
- {
200
- base: null,
201
- timestamp: false,
202
- messageKey: "message",
203
- formatters: {
204
- level: () => ({}),
205
- bindings: () => ({})
206
- }
207
- },
208
- createLogStream(fileNamePrefix)
209
- );
210
- }
211
- const state = {
212
- loggers: {},
213
- rotationFunction: (() => {
214
- }),
215
- intervalId: void 0
216
- };
217
- for (const [type, logConfig] of Object.entries(logTypes)) {
218
- state.loggers[type] = createPinoLogger(logConfig.fileNamePrefix);
219
- }
220
- const checkAndRotateLogFiles = () => {
221
- const currentDate = getCurrentDate(timezoneOffset);
222
- if (checkAndRotateLogFiles.lastDate !== currentDate) {
223
- checkAndRotateLogFiles.lastDate = currentDate;
224
- for (const [type, logConfig] of Object.entries(logTypes)) {
225
- state.loggers[type] = createPinoLogger(logConfig.fileNamePrefix);
226
- }
227
- }
228
- };
229
- checkAndRotateLogFiles.lastDate = getCurrentDate(timezoneOffset);
230
- state.rotationFunction = checkAndRotateLogFiles;
231
- if (process.env["NODE_ENV"] !== "test") {
232
- state.intervalId = setInterval(checkAndRotateLogFiles, rotationCheckInterval);
233
- }
234
- checkAndRotateLogFiles();
235
- function formatAndLog(loggerType, logLevel, message, location, originalError) {
236
- if (!shouldLog2(logLevel, minLevel)) {
237
- return;
238
- }
239
- checkAndRotateLogFiles();
240
- const formattedMessage = formatLogMessage(
241
- logLevel,
242
- message,
243
- location,
244
- originalError,
245
- timezoneOffset
246
- );
247
- const logger2 = state.loggers[loggerType];
248
- if (logger2) {
249
- logger2[logLevel](formattedMessage);
250
- }
251
- }
252
- function formatAndLogWithContext(loggerType, logLevel, ctx, message, error, sendToSentry = false) {
253
- if (!shouldLog2(logLevel, minLevel)) {
254
- return;
255
- }
256
- checkAndRotateLogFiles();
257
- const formattedMessage = formatLogMessageWithContext(
258
- logLevel,
259
- message,
260
- ctx,
261
- error,
262
- timezoneOffset
263
- );
264
- const logger2 = state.loggers[loggerType];
265
- if (logger2) {
266
- logger2[logLevel](formattedMessage);
267
- }
268
- if (sendToSentry && enableSentry && isSentryEnabled()) {
269
- captureError(ctx, message, error);
270
- }
271
- }
272
- function createLogMethod(loggerType, logLevel, sendToSentry = false) {
273
- return (ctxOrMessage, messageOrLocation, errorOrOriginal) => {
274
- if (isLogContext(ctxOrMessage)) {
275
- formatAndLogWithContext(
276
- loggerType,
277
- logLevel,
278
- ctxOrMessage,
279
- messageOrLocation ?? "",
280
- errorOrOriginal,
281
- sendToSentry
282
- );
283
- } else {
284
- formatAndLog(
285
- loggerType,
286
- logLevel,
287
- ctxOrMessage,
288
- messageOrLocation ?? null,
289
- errorOrOriginal ?? null
290
- );
291
- if (sendToSentry && enableSentry && isSentryEnabled()) {
292
- const ctx = messageOrLocation ? { location: messageOrLocation } : {};
293
- captureError(ctx, ctxOrMessage, errorOrOriginal);
294
- }
295
- }
296
- };
297
- }
298
- return {
299
- debug: createLogMethod("debug", "debug"),
300
- info: createLogMethod("info", "info"),
301
- warn: createLogMethod("info", "warn"),
302
- error: createLogMethod("error", "error", true),
303
- fatal: createLogMethod("fatal", "fatal", true),
304
- sqlError: (message, location, originalError) => {
305
- formatAndLog("sqlError", "error", message, location ?? null, originalError ?? null);
306
- }
307
- };
308
- }
309
-
310
64
  // src/index.ts
65
+ async function createServerLogger(config) {
66
+ const { createServerLogger: create } = await import('./server-J5UIJDX7.js');
67
+ return create(config);
68
+ }
311
69
  function getLogger() {
312
70
  if (globalThis.__DEGEL_LOGGER__) {
313
71
  return globalThis.__DEGEL_LOGGER__;
@@ -351,6 +109,6 @@ var logger = {
351
109
  };
352
110
  var index_default = logger;
353
111
 
354
- export { createClientLogger, createServerLogger, index_default as default, getLogger, initializeLogger, isSentryEnabled, logger, setSentryModule };
112
+ export { createClientLogger, createServerLogger, index_default as default, getLogger, initializeLogger, logger };
355
113
  //# sourceMappingURL=index.js.map
356
114
  //# sourceMappingURL=index.js.map
package/dist/index.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/utils.ts","../src/client.ts","../src/sentry.ts","../src/server.ts","../src/index.ts"],"names":["LOG_LEVEL_PRIORITY","shouldLog","logger"],"mappings":"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","file":"index.js","sourcesContent":["import type { LogContext } from './types.js';\n\nconst DEFAULT_TIMEZONE_OFFSET = 540; // JST = UTC+9 (in minutes)\n\n/**\n * 現在の日付を YYYY_MM_DD 形式で取得\n */\nexport function getCurrentDate(timezoneOffset = DEFAULT_TIMEZONE_OFFSET): string {\n const now = new Date();\n now.setTime(now.getTime() + timezoneOffset * 60 * 1000);\n\n const year = now.getUTCFullYear();\n const month = String(now.getUTCMonth() + 1).padStart(2, '0');\n const day = String(now.getUTCDate()).padStart(2, '0');\n\n return `${year}_${month}_${day}`;\n}\n\n/**\n * 現在の日時をフォーマットする(タイムゾーン調整付き)\n */\nexport function getFormattedDateTime(timezoneOffset = DEFAULT_TIMEZONE_OFFSET): string {\n const now = new Date();\n now.setTime(now.getTime() + timezoneOffset * 60 * 1000);\n\n const year = now.getUTCFullYear();\n const month = String(now.getUTCMonth() + 1).padStart(2, '0');\n const day = String(now.getUTCDate()).padStart(2, '0');\n const hours = String(now.getUTCHours()).padStart(2, '0');\n const minutes = String(now.getUTCMinutes()).padStart(2, '0');\n const seconds = String(now.getUTCSeconds()).padStart(2, '0');\n\n return `${year}_${month}_${day} ${hours}:${minutes}:${seconds}`;\n}\n\n/**\n * ログメッセージをフォーマットする\n */\nexport function formatLogMessage(\n level: string,\n message: string,\n location: string | null,\n originalError: Error | string | null,\n timezoneOffset = DEFAULT_TIMEZONE_OFFSET\n): string {\n const formattedDate = getFormattedDateTime(timezoneOffset);\n const errorType = level.toUpperCase();\n const locationInfo = location ?? 'Unknown';\n const original = originalError\n ? originalError.toString()\n : originalError === null\n ? message\n : 'なし';\n\n return `${formattedDate}:${errorType}:${message}:${locationInfo}:${original}`;\n}\n\n/**\n * 引数がLogContextかどうかを判定\n */\nexport function isLogContext(arg: unknown): arg is LogContext {\n return typeof arg === 'object' && arg !== null && !Array.isArray(arg) && typeof arg !== 'string';\n}\n\n/**\n * コンテキスト情報をフォーマットする\n */\nfunction formatContext(ctx: LogContext): string {\n const parts: string[] = [];\n\n if (ctx.path) parts.push(`path=${ctx.path}`);\n if (ctx.method) parts.push(`method=${ctx.method}`);\n if (ctx.userId) parts.push(`userId=${ctx.userId}`);\n if (ctx.tenantId) parts.push(`tenantId=${ctx.tenantId}`);\n if (ctx.location) parts.push(`location=${ctx.location}`);\n\n // その他のカスタムフィールド\n for (const [key, value] of Object.entries(ctx)) {\n if (\n !['path', 'method', 'userId', 'tenantId', 'location'].includes(key) &&\n value !== undefined\n ) {\n parts.push(`${key}=${String(value)}`);\n }\n }\n\n return parts.length > 0 ? `[${parts.join(', ')}]` : '';\n}\n\n/**\n * 構造化コンテキスト対応のログメッセージをフォーマットする\n */\nexport function formatLogMessageWithContext(\n level: string,\n message: string,\n ctx: LogContext,\n error: unknown,\n timezoneOffset = DEFAULT_TIMEZONE_OFFSET\n): string {\n const formattedDate = getFormattedDateTime(timezoneOffset);\n const errorType = level.toUpperCase();\n const contextInfo = formatContext(ctx);\n\n let errorInfo = '';\n if (error instanceof Error) {\n errorInfo = ` | Error: ${error.message}`;\n if (error.stack) {\n errorInfo += `\\n${error.stack}`;\n }\n } else if (error !== undefined && error !== null) {\n errorInfo = ` | Error: ${String(error)}`;\n }\n\n const contextPart = contextInfo ? ` ${contextInfo}` : '';\n return `${formattedDate}:${errorType}:${message}${contextPart}${errorInfo}`;\n}\n","import type { LogContext, LogLevel, Logger, LoggerConfig } from './types.js';\nimport { formatLogMessage, formatLogMessageWithContext, isLogContext } from './utils.js';\n\n// ログレベルの優先度マップ\nconst LOG_LEVEL_PRIORITY: Record<LogLevel, number> = {\n debug: 0,\n info: 1,\n warn: 2,\n error: 3,\n fatal: 4,\n};\n\n/**\n * 指定されたログレベルが最小レベル以上かどうかを判定\n */\nfunction shouldLog(logLevel: LogLevel, minLevel: LogLevel): boolean {\n return LOG_LEVEL_PRIORITY[logLevel] >= LOG_LEVEL_PRIORITY[minLevel];\n}\n\n/**\n * クライアントサイド用ロガーを作成\n */\nexport function createClientLogger(config: LoggerConfig = {}): Logger {\n const { timezoneOffset = 540, enableConsole = true, level = 'debug' } = config;\n\n // 有効なログレベルかどうかを検証\n const validLevels: LogLevel[] = ['debug', 'info', 'warn', 'error', 'fatal'];\n const minLevel: LogLevel = validLevels.includes(level) ? level : 'debug';\n\n const logToConsole = (\n logLevel: LogLevel,\n displayLevel: string,\n consoleMethod: (...args: unknown[]) => void,\n ctxOrMessage: LogContext | string,\n messageOrLocation?: string | null,\n errorOrOriginal?: unknown\n ): void => {\n if (!enableConsole) return;\n if (!shouldLog(logLevel, minLevel)) return;\n\n let formattedMessage: string;\n\n if (isLogContext(ctxOrMessage)) {\n // 構造化コンテキスト形式: (ctx, message, error?)\n formattedMessage = formatLogMessageWithContext(\n displayLevel,\n messageOrLocation ?? '',\n ctxOrMessage,\n errorOrOriginal,\n timezoneOffset\n );\n } else {\n // レガシー形式: (message, location?, originalError?)\n formattedMessage = formatLogMessage(\n displayLevel,\n ctxOrMessage,\n (messageOrLocation as string) ?? null,\n (errorOrOriginal as Error | string) ?? null,\n timezoneOffset\n );\n }\n\n consoleMethod(formattedMessage);\n };\n\n return {\n debug: (\n ctxOrMessage: LogContext | string,\n messageOrLocation?: string | null,\n originalError?: unknown\n ) => {\n logToConsole('debug', 'debug', console.debug, ctxOrMessage, messageOrLocation, originalError);\n },\n info: (\n ctxOrMessage: LogContext | string,\n messageOrLocation?: string | null,\n originalError?: unknown\n ) => {\n logToConsole('info', 'info', console.info, ctxOrMessage, messageOrLocation, originalError);\n },\n warn: (\n ctxOrMessage: LogContext | string,\n messageOrLocation?: string | null,\n originalError?: unknown\n ) => {\n logToConsole('warn', 'warn', console.warn, ctxOrMessage, messageOrLocation, originalError);\n },\n error: (\n ctxOrMessage: LogContext | string,\n messageOrLocation?: string | null,\n originalError?: unknown\n ) => {\n logToConsole('error', 'error', console.error, ctxOrMessage, messageOrLocation, originalError);\n },\n fatal: (\n ctxOrMessage: LogContext | string,\n messageOrLocation?: string | null,\n originalError?: unknown\n ) => {\n logToConsole('fatal', 'fatal', console.error, ctxOrMessage, messageOrLocation, originalError);\n },\n sqlError: (\n message: string,\n location?: string | null,\n originalError?: Error | string | null\n ) => {\n // sqlErrorはerrorレベルとしてフィルタリング、表示はsql_error\n logToConsole('error', 'sql_error', console.error, message, location, originalError);\n },\n } as Logger;\n}\n","/**\n * Sentry連携ヘルパー\n */\n\nimport type { LogContext, SentryModule } from './types.js';\n\nlet sentryModule: SentryModule | null = null;\n\n/**\n * Sentryモジュールを設定\n */\nexport function setSentryModule(sentry: SentryModule): void {\n sentryModule = sentry;\n}\n\n/**\n * Sentryが有効かどうか\n */\nexport function isSentryEnabled(): boolean {\n return sentryModule !== null;\n}\n\n/**\n * Sentryモジュールをクリア(テスト用)\n */\nexport function clearSentryModule(): void {\n sentryModule = null;\n}\n\n/**\n * エラーをSentryに送信\n */\nexport function captureError(ctx: LogContext, message: string, error?: unknown): void {\n if (!sentryModule) return;\n\n if (error instanceof Error) {\n sentryModule.captureException(error, {\n extra: {\n ...ctx,\n message,\n },\n });\n } else if (error) {\n sentryModule.captureMessage(`${message}: ${String(error)}`, {\n level: 'error',\n extra: ctx,\n });\n } else {\n sentryModule.captureMessage(message, {\n level: 'error',\n extra: ctx,\n });\n }\n}\n","import pino from 'pino';\nimport { captureError, isSentryEnabled } from './sentry.js';\nimport type {\n LogContext,\n LogLevel,\n LogRotationFunction,\n LogTypeConfig,\n Logger,\n LoggerConfig,\n} from './types.js';\nimport {\n formatLogMessage,\n formatLogMessageWithContext,\n getCurrentDate,\n isLogContext,\n} from './utils.js';\n\n// ログレベルの優先度マップ\nconst LOG_LEVEL_PRIORITY: Record<LogLevel, number> = {\n debug: 0,\n info: 1,\n warn: 2,\n error: 3,\n fatal: 4,\n};\n\n/**\n * 指定されたログレベルが最小レベル以上かどうかを判定\n */\nfunction shouldLog(logLevel: LogLevel, minLevel: LogLevel): boolean {\n return LOG_LEVEL_PRIORITY[logLevel] >= LOG_LEVEL_PRIORITY[minLevel];\n}\n\ninterface LoggerState {\n loggers: Record<string, pino.Logger>;\n rotationFunction: LogRotationFunction;\n intervalId: ReturnType<typeof setInterval> | undefined;\n}\n\n/**\n * サーバーサイド用ロガーを作成\n */\nexport async function createServerLogger(config: LoggerConfig = {}): Promise<Logger> {\n const {\n // biome-ignore lint/complexity/useLiteralKeys: TypeScript requires bracket notation for process.env\n logDir = process.env['LOG_DIR'] ?? `${process.cwd()}/logs`,\n // biome-ignore lint/complexity/useLiteralKeys: TypeScript requires bracket notation for process.env\n level = (process.env['LOG_LEVEL'] as LogLevel) ?? 'debug',\n timezoneOffset = 540,\n rotationCheckInterval = 60 * 60 * 1000, // 1時間\n enableSentry = false,\n } = config;\n\n // 有効なログレベルかどうかを検証\n const validLevels: LogLevel[] = ['debug', 'info', 'warn', 'error', 'fatal'];\n const minLevel: LogLevel = validLevels.includes(level) ? level : 'debug';\n\n const fs = await import('node:fs');\n const path = await import('node:path');\n\n // ログフォルダが存在しない場合は作成\n if (!fs.existsSync(logDir)) {\n fs.mkdirSync(logDir, { recursive: true });\n }\n\n // 各ログレベル用の設定\n const logTypes: Record<string, LogTypeConfig> = {\n debug: { fileNamePrefix: 'debug_log', level: 'debug' },\n info: { fileNamePrefix: 'info_log', level: 'info' },\n error: { fileNamePrefix: 'error_log', level: 'error' },\n fatal: { fileNamePrefix: 'fatal_log', level: 'fatal' },\n sqlError: { fileNamePrefix: 'sql_error_log', level: 'error' },\n };\n\n // ログストリームの作成関数\n function createLogStream(fileNamePrefix: string): ReturnType<typeof fs.createWriteStream> {\n const dateStr = getCurrentDate(timezoneOffset);\n const logFilePath = path.join(logDir, `${fileNamePrefix}_${dateStr}.log`);\n return fs.createWriteStream(logFilePath, { flags: 'a' });\n }\n\n // ロガーを作成する関数\n function createPinoLogger(fileNamePrefix: string): pino.Logger {\n return pino(\n {\n base: null,\n timestamp: false,\n messageKey: 'message',\n formatters: {\n level: () => ({}),\n bindings: () => ({}),\n },\n },\n createLogStream(fileNamePrefix)\n );\n }\n\n // 各ロガーの作成\n const state: LoggerState = {\n loggers: {},\n rotationFunction: (() => {}) as LogRotationFunction,\n intervalId: undefined,\n };\n\n for (const [type, logConfig] of Object.entries(logTypes)) {\n state.loggers[type] = createPinoLogger(logConfig.fileNamePrefix);\n }\n\n // ログファイルを日付変更時に切り替えるための関数\n const checkAndRotateLogFiles: LogRotationFunction = () => {\n const currentDate = getCurrentDate(timezoneOffset);\n\n if (checkAndRotateLogFiles.lastDate !== currentDate) {\n checkAndRotateLogFiles.lastDate = currentDate;\n\n for (const [type, logConfig] of Object.entries(logTypes)) {\n state.loggers[type] = createPinoLogger(logConfig.fileNamePrefix);\n }\n }\n };\n\n // 初期日付設定\n checkAndRotateLogFiles.lastDate = getCurrentDate(timezoneOffset);\n state.rotationFunction = checkAndRotateLogFiles;\n\n // ログローテーションのインターバル設定(テスト環境以外)\n // biome-ignore lint/complexity/useLiteralKeys: TypeScript requires bracket notation for process.env\n if (process.env['NODE_ENV'] !== 'test') {\n state.intervalId = setInterval(checkAndRotateLogFiles, rotationCheckInterval);\n }\n\n // 起動時に一度チェック\n checkAndRotateLogFiles();\n\n // カスタムフォーマットでログを出力する関数(レガシー形式)\n function formatAndLog(\n loggerType: string,\n logLevel: pino.Level,\n message: string,\n location: string | null,\n originalError: Error | string | null\n ): void {\n // ログレベルが最小レベル未満の場合はスキップ\n if (!shouldLog(logLevel as LogLevel, minLevel)) {\n return;\n }\n\n checkAndRotateLogFiles();\n\n const formattedMessage = formatLogMessage(\n logLevel,\n message,\n location,\n originalError,\n timezoneOffset\n );\n const logger = state.loggers[loggerType];\n\n if (logger) {\n logger[logLevel](formattedMessage);\n }\n }\n\n // 構造化コンテキスト対応のログ出力関数\n function formatAndLogWithContext(\n loggerType: string,\n logLevel: pino.Level,\n ctx: LogContext,\n message: string,\n error?: unknown,\n sendToSentry = false\n ): void {\n // ログレベルが最小レベル未満の場合はスキップ\n if (!shouldLog(logLevel as LogLevel, minLevel)) {\n return;\n }\n\n checkAndRotateLogFiles();\n\n const formattedMessage = formatLogMessageWithContext(\n logLevel,\n message,\n ctx,\n error,\n timezoneOffset\n );\n const logger = state.loggers[loggerType];\n\n if (logger) {\n logger[logLevel](formattedMessage);\n }\n\n // Sentry連携\n if (sendToSentry && enableSentry && isSentryEnabled()) {\n captureError(ctx, message, error);\n }\n }\n\n // オーバーロード対応のログ関数\n function createLogMethod(loggerType: string, logLevel: pino.Level, sendToSentry = false) {\n return (\n ctxOrMessage: LogContext | string,\n messageOrLocation?: string | null,\n errorOrOriginal?: unknown\n ) => {\n if (isLogContext(ctxOrMessage)) {\n // 構造化コンテキスト形式\n formatAndLogWithContext(\n loggerType,\n logLevel,\n ctxOrMessage,\n messageOrLocation ?? '',\n errorOrOriginal,\n sendToSentry\n );\n } else {\n // レガシー形式\n formatAndLog(\n loggerType,\n logLevel,\n ctxOrMessage,\n (messageOrLocation as string) ?? null,\n (errorOrOriginal as Error | string) ?? null\n );\n\n // レガシー形式でもSentry送信(error/fatalの場合)\n if (sendToSentry && enableSentry && isSentryEnabled()) {\n const ctx: LogContext = messageOrLocation ? { location: messageOrLocation } : {};\n captureError(ctx, ctxOrMessage, errorOrOriginal);\n }\n }\n };\n }\n\n return {\n debug: createLogMethod('debug', 'debug'),\n info: createLogMethod('info', 'info'),\n warn: createLogMethod('info', 'warn'),\n error: createLogMethod('error', 'error', true),\n fatal: createLogMethod('fatal', 'fatal', true),\n sqlError: (\n message: string,\n location?: string | null,\n originalError?: Error | string | null\n ) => {\n formatAndLog('sqlError', 'error', message, location ?? null, originalError ?? null);\n },\n } as Logger;\n}\n","import { createClientLogger } from './client.js';\nimport { createServerLogger } from './server.js';\nimport type {\n LogContext,\n LogLevel,\n LogMethod,\n LogMethodWithContext,\n LogTypeConfig,\n Logger,\n LoggerConfig,\n SentryModule,\n} from './types.js';\n\nexport type {\n LogContext,\n Logger,\n LoggerConfig,\n LogLevel,\n LogMethod,\n LogMethodWithContext,\n LogTypeConfig,\n SentryModule,\n};\n\nexport { createClientLogger, createServerLogger };\nexport { isSentryEnabled, setSentryModule } from './sentry.js';\n\n/**\n * グローバルシングルトン用の型定義\n * Next.js開発モードでのモジュール分離問題を解決するため、\n * globalThisを使用してロガーインスタンスを共有する\n */\ndeclare global {\n // eslint-disable-next-line no-var\n var __DEGEL_LOGGER__: Logger | undefined;\n // eslint-disable-next-line no-var\n var __DEGEL_LOGGER_INITIALIZED__: boolean | undefined;\n}\n\n/**\n * 環境を判定してデフォルトのロガーを取得\n * サーバーサイドでは非同期初期化が必要なため、初回呼び出し時はクライアントロガーを返す\n */\nexport function getLogger(): Logger {\n // グローバルに初期化済みのロガーがあれば返す\n if (globalThis.__DEGEL_LOGGER__) {\n return globalThis.__DEGEL_LOGGER__;\n }\n\n // クライアントサイドの場合\n if (typeof window !== 'undefined') {\n globalThis.__DEGEL_LOGGER__ = createClientLogger();\n return globalThis.__DEGEL_LOGGER__;\n }\n\n // サーバーサイドだが未初期化の場合は一時的にクライアントロガーを返す\n // initializeLogger()を呼び出すことで正しく初期化される\n if (!globalThis.__DEGEL_LOGGER_INITIALIZED__) {\n return createClientLogger();\n }\n\n return globalThis.__DEGEL_LOGGER__ ?? createClientLogger();\n}\n\n/**\n * サーバーサイド用の非同期初期化\n * Next.js等で使用する場合、アプリケーション起動時に呼び出す\n */\nexport async function initializeLogger(config?: LoggerConfig): Promise<Logger> {\n // 既に初期化済みの場合は既存のロガーを返す\n if (globalThis.__DEGEL_LOGGER__ && globalThis.__DEGEL_LOGGER_INITIALIZED__) {\n return globalThis.__DEGEL_LOGGER__;\n }\n\n if (typeof window !== 'undefined') {\n globalThis.__DEGEL_LOGGER__ = createClientLogger(config);\n } else {\n globalThis.__DEGEL_LOGGER__ = await createServerLogger(config);\n }\n globalThis.__DEGEL_LOGGER_INITIALIZED__ = true;\n return globalThis.__DEGEL_LOGGER__;\n}\n\n/**\n * 共通ロガーオブジェクト(後方互換性のため)\n * サーバー/クライアントで自動的に適切なロガーを選択\n */\nexport const logger: Logger = {\n debug: (ctxOrMessage, messageOrLocation, originalError) =>\n getLogger().debug(ctxOrMessage as string, messageOrLocation, originalError),\n info: (ctxOrMessage, messageOrLocation, originalError) =>\n getLogger().info(ctxOrMessage as string, messageOrLocation, originalError),\n warn: (ctxOrMessage, messageOrLocation, originalError) =>\n getLogger().warn(ctxOrMessage as string, messageOrLocation, originalError),\n error: (ctxOrMessage, messageOrLocation, originalError) =>\n getLogger().error(\n ctxOrMessage as string,\n messageOrLocation,\n originalError as Error | string | null | undefined\n ),\n fatal: (ctxOrMessage, messageOrLocation, originalError) =>\n getLogger().fatal(\n ctxOrMessage as string,\n messageOrLocation,\n originalError as Error | string | null | undefined\n ),\n sqlError: (message, location, originalError) =>\n getLogger().sqlError(message, location, originalError),\n} as Logger;\n\nexport default logger;\n"]}
1
+ {"version":3,"sources":["../src/client.ts","../src/index.ts"],"names":[],"mappings":"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","file":"index.js","sourcesContent":["import type { LogContext, LogLevel, Logger, LoggerConfig } from './types.js';\nimport { formatLogMessage, formatLogMessageWithContext, isLogContext } from './utils.js';\n\n// ログレベルの優先度マップ\nconst LOG_LEVEL_PRIORITY: Record<LogLevel, number> = {\n debug: 0,\n info: 1,\n warn: 2,\n error: 3,\n fatal: 4,\n};\n\n/**\n * 指定されたログレベルが最小レベル以上かどうかを判定\n */\nfunction shouldLog(logLevel: LogLevel, minLevel: LogLevel): boolean {\n return LOG_LEVEL_PRIORITY[logLevel] >= LOG_LEVEL_PRIORITY[minLevel];\n}\n\n/**\n * クライアントサイド用ロガーを作成\n */\nexport function createClientLogger(config: LoggerConfig = {}): Logger {\n const { timezoneOffset = 540, enableConsole = true, level = 'debug' } = config;\n\n // 有効なログレベルかどうかを検証\n const validLevels: LogLevel[] = ['debug', 'info', 'warn', 'error', 'fatal'];\n const minLevel: LogLevel = validLevels.includes(level) ? level : 'debug';\n\n const logToConsole = (\n logLevel: LogLevel,\n displayLevel: string,\n consoleMethod: (...args: unknown[]) => void,\n ctxOrMessage: LogContext | string,\n messageOrLocation?: string | null,\n errorOrOriginal?: unknown\n ): void => {\n if (!enableConsole) return;\n if (!shouldLog(logLevel, minLevel)) return;\n\n let formattedMessage: string;\n\n if (isLogContext(ctxOrMessage)) {\n // 構造化コンテキスト形式: (ctx, message, error?)\n formattedMessage = formatLogMessageWithContext(\n displayLevel,\n messageOrLocation ?? '',\n ctxOrMessage,\n errorOrOriginal,\n timezoneOffset\n );\n } else {\n // レガシー形式: (message, location?, originalError?)\n formattedMessage = formatLogMessage(\n displayLevel,\n ctxOrMessage,\n (messageOrLocation as string) ?? null,\n (errorOrOriginal as Error | string) ?? null,\n timezoneOffset\n );\n }\n\n consoleMethod(formattedMessage);\n };\n\n return {\n debug: (\n ctxOrMessage: LogContext | string,\n messageOrLocation?: string | null,\n originalError?: unknown\n ) => {\n logToConsole('debug', 'debug', console.debug, ctxOrMessage, messageOrLocation, originalError);\n },\n info: (\n ctxOrMessage: LogContext | string,\n messageOrLocation?: string | null,\n originalError?: unknown\n ) => {\n logToConsole('info', 'info', console.info, ctxOrMessage, messageOrLocation, originalError);\n },\n warn: (\n ctxOrMessage: LogContext | string,\n messageOrLocation?: string | null,\n originalError?: unknown\n ) => {\n logToConsole('warn', 'warn', console.warn, ctxOrMessage, messageOrLocation, originalError);\n },\n error: (\n ctxOrMessage: LogContext | string,\n messageOrLocation?: string | null,\n originalError?: unknown\n ) => {\n logToConsole('error', 'error', console.error, ctxOrMessage, messageOrLocation, originalError);\n },\n fatal: (\n ctxOrMessage: LogContext | string,\n messageOrLocation?: string | null,\n originalError?: unknown\n ) => {\n logToConsole('fatal', 'fatal', console.error, ctxOrMessage, messageOrLocation, originalError);\n },\n sqlError: (\n message: string,\n location?: string | null,\n originalError?: Error | string | null\n ) => {\n // sqlErrorはerrorレベルとしてフィルタリング、表示はsql_error\n logToConsole('error', 'sql_error', console.error, message, location, originalError);\n },\n } as Logger;\n}\n","import { createClientLogger } from './client.js';\nimport type {\n LogContext,\n LogLevel,\n LogMethod,\n LogMethodWithContext,\n LogTypeConfig,\n Logger,\n LoggerConfig,\n SentryModule,\n} from './types.js';\n\nexport type {\n LogContext,\n Logger,\n LoggerConfig,\n LogLevel,\n LogMethod,\n LogMethodWithContext,\n LogTypeConfig,\n SentryModule,\n};\n\nexport { createClientLogger };\nexport { isSentryEnabled, setSentryModule } from './sentry.js';\n\n/**\n * サーバーサイド用ロガー作成関数(動的import)\n * Next.jsなどのバンドラーでクライアント用にバンドルされる際に\n * fsモジュールの解決エラーを防ぐため、動的importで提供\n */\nexport async function createServerLogger(\n config?: LoggerConfig\n): Promise<Logger> {\n const { createServerLogger: create } = await import('./server.js');\n return create(config);\n}\n\n/**\n * グローバルシングルトン用の型定義\n * Next.js開発モードでのモジュール分離問題を解決するため、\n * globalThisを使用してロガーインスタンスを共有する\n */\ndeclare global {\n // eslint-disable-next-line no-var\n var __DEGEL_LOGGER__: Logger | undefined;\n // eslint-disable-next-line no-var\n var __DEGEL_LOGGER_INITIALIZED__: boolean | undefined;\n}\n\n/**\n * 環境を判定してデフォルトのロガーを取得\n * サーバーサイドでは非同期初期化が必要なため、初回呼び出し時はクライアントロガーを返す\n */\nexport function getLogger(): Logger {\n // グローバルに初期化済みのロガーがあれば返す\n if (globalThis.__DEGEL_LOGGER__) {\n return globalThis.__DEGEL_LOGGER__;\n }\n\n // クライアントサイドの場合\n if (typeof window !== 'undefined') {\n globalThis.__DEGEL_LOGGER__ = createClientLogger();\n return globalThis.__DEGEL_LOGGER__;\n }\n\n // サーバーサイドだが未初期化の場合は一時的にクライアントロガーを返す\n // initializeLogger()を呼び出すことで正しく初期化される\n if (!globalThis.__DEGEL_LOGGER_INITIALIZED__) {\n return createClientLogger();\n }\n\n return globalThis.__DEGEL_LOGGER__ ?? createClientLogger();\n}\n\n/**\n * サーバーサイド用の非同期初期化\n * Next.js等で使用する場合、アプリケーション起動時に呼び出す\n */\nexport async function initializeLogger(config?: LoggerConfig): Promise<Logger> {\n // 既に初期化済みの場合は既存のロガーを返す\n if (globalThis.__DEGEL_LOGGER__ && globalThis.__DEGEL_LOGGER_INITIALIZED__) {\n return globalThis.__DEGEL_LOGGER__;\n }\n\n if (typeof window !== 'undefined') {\n globalThis.__DEGEL_LOGGER__ = createClientLogger(config);\n } else {\n globalThis.__DEGEL_LOGGER__ = await createServerLogger(config);\n }\n globalThis.__DEGEL_LOGGER_INITIALIZED__ = true;\n return globalThis.__DEGEL_LOGGER__;\n}\n\n/**\n * 共通ロガーオブジェクト(後方互換性のため)\n * サーバー/クライアントで自動的に適切なロガーを選択\n */\nexport const logger: Logger = {\n debug: (ctxOrMessage, messageOrLocation, originalError) =>\n getLogger().debug(ctxOrMessage as string, messageOrLocation, originalError),\n info: (ctxOrMessage, messageOrLocation, originalError) =>\n getLogger().info(ctxOrMessage as string, messageOrLocation, originalError),\n warn: (ctxOrMessage, messageOrLocation, originalError) =>\n getLogger().warn(ctxOrMessage as string, messageOrLocation, originalError),\n error: (ctxOrMessage, messageOrLocation, originalError) =>\n getLogger().error(\n ctxOrMessage as string,\n messageOrLocation,\n originalError as Error | string | null | undefined\n ),\n fatal: (ctxOrMessage, messageOrLocation, originalError) =>\n getLogger().fatal(\n ctxOrMessage as string,\n messageOrLocation,\n originalError as Error | string | null | undefined\n ),\n sqlError: (message, location, originalError) =>\n getLogger().sqlError(message, location, originalError),\n} as Logger;\n\nexport default logger;\n"]}
@@ -0,0 +1,159 @@
1
+ import { getCurrentDate, formatLogMessage, isLogContext, isSentryEnabled, captureError, formatLogMessageWithContext } from './chunk-34MNPN6V.js';
2
+ import pino from 'pino';
3
+
4
+ var LOG_LEVEL_PRIORITY = {
5
+ debug: 0,
6
+ info: 1,
7
+ warn: 2,
8
+ error: 3,
9
+ fatal: 4
10
+ };
11
+ function shouldLog(logLevel, minLevel) {
12
+ return LOG_LEVEL_PRIORITY[logLevel] >= LOG_LEVEL_PRIORITY[minLevel];
13
+ }
14
+ async function createServerLogger(config = {}) {
15
+ const {
16
+ // biome-ignore lint/complexity/useLiteralKeys: TypeScript requires bracket notation for process.env
17
+ logDir = process.env["LOG_DIR"] ?? `${process.cwd()}/logs`,
18
+ // biome-ignore lint/complexity/useLiteralKeys: TypeScript requires bracket notation for process.env
19
+ level = process.env["LOG_LEVEL"] ?? "debug",
20
+ timezoneOffset = 540,
21
+ rotationCheckInterval = 60 * 60 * 1e3,
22
+ // 1時間
23
+ enableSentry = false
24
+ } = config;
25
+ const validLevels = ["debug", "info", "warn", "error", "fatal"];
26
+ const minLevel = validLevels.includes(level) ? level : "debug";
27
+ const fs = await import('fs');
28
+ const path = await import('path');
29
+ if (!fs.existsSync(logDir)) {
30
+ fs.mkdirSync(logDir, { recursive: true });
31
+ }
32
+ const logTypes = {
33
+ debug: { fileNamePrefix: "debug_log", level: "debug" },
34
+ info: { fileNamePrefix: "info_log", level: "info" },
35
+ error: { fileNamePrefix: "error_log", level: "error" },
36
+ fatal: { fileNamePrefix: "fatal_log", level: "fatal" },
37
+ sqlError: { fileNamePrefix: "sql_error_log", level: "error" }
38
+ };
39
+ function createLogStream(fileNamePrefix) {
40
+ const dateStr = getCurrentDate(timezoneOffset);
41
+ const logFilePath = path.join(logDir, `${fileNamePrefix}_${dateStr}.log`);
42
+ return fs.createWriteStream(logFilePath, { flags: "a" });
43
+ }
44
+ function createPinoLogger(fileNamePrefix) {
45
+ return pino(
46
+ {
47
+ base: null,
48
+ timestamp: false,
49
+ messageKey: "message",
50
+ formatters: {
51
+ level: () => ({}),
52
+ bindings: () => ({})
53
+ }
54
+ },
55
+ createLogStream(fileNamePrefix)
56
+ );
57
+ }
58
+ const state = {
59
+ loggers: {},
60
+ rotationFunction: (() => {
61
+ }),
62
+ intervalId: void 0
63
+ };
64
+ for (const [type, logConfig] of Object.entries(logTypes)) {
65
+ state.loggers[type] = createPinoLogger(logConfig.fileNamePrefix);
66
+ }
67
+ const checkAndRotateLogFiles = () => {
68
+ const currentDate = getCurrentDate(timezoneOffset);
69
+ if (checkAndRotateLogFiles.lastDate !== currentDate) {
70
+ checkAndRotateLogFiles.lastDate = currentDate;
71
+ for (const [type, logConfig] of Object.entries(logTypes)) {
72
+ state.loggers[type] = createPinoLogger(logConfig.fileNamePrefix);
73
+ }
74
+ }
75
+ };
76
+ checkAndRotateLogFiles.lastDate = getCurrentDate(timezoneOffset);
77
+ state.rotationFunction = checkAndRotateLogFiles;
78
+ if (process.env["NODE_ENV"] !== "test") {
79
+ state.intervalId = setInterval(checkAndRotateLogFiles, rotationCheckInterval);
80
+ }
81
+ checkAndRotateLogFiles();
82
+ function formatAndLog(loggerType, logLevel, message, location, originalError) {
83
+ if (!shouldLog(logLevel, minLevel)) {
84
+ return;
85
+ }
86
+ checkAndRotateLogFiles();
87
+ const formattedMessage = formatLogMessage(
88
+ logLevel,
89
+ message,
90
+ location,
91
+ originalError,
92
+ timezoneOffset
93
+ );
94
+ const logger = state.loggers[loggerType];
95
+ if (logger) {
96
+ logger[logLevel](formattedMessage);
97
+ }
98
+ }
99
+ function formatAndLogWithContext(loggerType, logLevel, ctx, message, error, sendToSentry = false) {
100
+ if (!shouldLog(logLevel, minLevel)) {
101
+ return;
102
+ }
103
+ checkAndRotateLogFiles();
104
+ const formattedMessage = formatLogMessageWithContext(
105
+ logLevel,
106
+ message,
107
+ ctx,
108
+ error,
109
+ timezoneOffset
110
+ );
111
+ const logger = state.loggers[loggerType];
112
+ if (logger) {
113
+ logger[logLevel](formattedMessage);
114
+ }
115
+ if (sendToSentry && enableSentry && isSentryEnabled()) {
116
+ captureError(ctx, message, error);
117
+ }
118
+ }
119
+ function createLogMethod(loggerType, logLevel, sendToSentry = false) {
120
+ return (ctxOrMessage, messageOrLocation, errorOrOriginal) => {
121
+ if (isLogContext(ctxOrMessage)) {
122
+ formatAndLogWithContext(
123
+ loggerType,
124
+ logLevel,
125
+ ctxOrMessage,
126
+ messageOrLocation ?? "",
127
+ errorOrOriginal,
128
+ sendToSentry
129
+ );
130
+ } else {
131
+ formatAndLog(
132
+ loggerType,
133
+ logLevel,
134
+ ctxOrMessage,
135
+ messageOrLocation ?? null,
136
+ errorOrOriginal ?? null
137
+ );
138
+ if (sendToSentry && enableSentry && isSentryEnabled()) {
139
+ const ctx = messageOrLocation ? { location: messageOrLocation } : {};
140
+ captureError(ctx, ctxOrMessage, errorOrOriginal);
141
+ }
142
+ }
143
+ };
144
+ }
145
+ return {
146
+ debug: createLogMethod("debug", "debug"),
147
+ info: createLogMethod("info", "info"),
148
+ warn: createLogMethod("info", "warn"),
149
+ error: createLogMethod("error", "error", true),
150
+ fatal: createLogMethod("fatal", "fatal", true),
151
+ sqlError: (message, location, originalError) => {
152
+ formatAndLog("sqlError", "error", message, location ?? null, originalError ?? null);
153
+ }
154
+ };
155
+ }
156
+
157
+ export { createServerLogger };
158
+ //# sourceMappingURL=server-J5UIJDX7.js.map
159
+ //# sourceMappingURL=server-J5UIJDX7.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/server.ts"],"names":[],"mappings":"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","file":"server-J5UIJDX7.js","sourcesContent":["import pino from 'pino';\nimport { captureError, isSentryEnabled } from './sentry.js';\nimport type {\n LogContext,\n LogLevel,\n LogRotationFunction,\n LogTypeConfig,\n Logger,\n LoggerConfig,\n} from './types.js';\nimport {\n formatLogMessage,\n formatLogMessageWithContext,\n getCurrentDate,\n isLogContext,\n} from './utils.js';\n\n// ログレベルの優先度マップ\nconst LOG_LEVEL_PRIORITY: Record<LogLevel, number> = {\n debug: 0,\n info: 1,\n warn: 2,\n error: 3,\n fatal: 4,\n};\n\n/**\n * 指定されたログレベルが最小レベル以上かどうかを判定\n */\nfunction shouldLog(logLevel: LogLevel, minLevel: LogLevel): boolean {\n return LOG_LEVEL_PRIORITY[logLevel] >= LOG_LEVEL_PRIORITY[minLevel];\n}\n\ninterface LoggerState {\n loggers: Record<string, pino.Logger>;\n rotationFunction: LogRotationFunction;\n intervalId: ReturnType<typeof setInterval> | undefined;\n}\n\n/**\n * サーバーサイド用ロガーを作成\n */\nexport async function createServerLogger(config: LoggerConfig = {}): Promise<Logger> {\n const {\n // biome-ignore lint/complexity/useLiteralKeys: TypeScript requires bracket notation for process.env\n logDir = process.env['LOG_DIR'] ?? `${process.cwd()}/logs`,\n // biome-ignore lint/complexity/useLiteralKeys: TypeScript requires bracket notation for process.env\n level = (process.env['LOG_LEVEL'] as LogLevel) ?? 'debug',\n timezoneOffset = 540,\n rotationCheckInterval = 60 * 60 * 1000, // 1時間\n enableSentry = false,\n } = config;\n\n // 有効なログレベルかどうかを検証\n const validLevels: LogLevel[] = ['debug', 'info', 'warn', 'error', 'fatal'];\n const minLevel: LogLevel = validLevels.includes(level) ? level : 'debug';\n\n const fs = await import('node:fs');\n const path = await import('node:path');\n\n // ログフォルダが存在しない場合は作成\n if (!fs.existsSync(logDir)) {\n fs.mkdirSync(logDir, { recursive: true });\n }\n\n // 各ログレベル用の設定\n const logTypes: Record<string, LogTypeConfig> = {\n debug: { fileNamePrefix: 'debug_log', level: 'debug' },\n info: { fileNamePrefix: 'info_log', level: 'info' },\n error: { fileNamePrefix: 'error_log', level: 'error' },\n fatal: { fileNamePrefix: 'fatal_log', level: 'fatal' },\n sqlError: { fileNamePrefix: 'sql_error_log', level: 'error' },\n };\n\n // ログストリームの作成関数\n function createLogStream(fileNamePrefix: string): ReturnType<typeof fs.createWriteStream> {\n const dateStr = getCurrentDate(timezoneOffset);\n const logFilePath = path.join(logDir, `${fileNamePrefix}_${dateStr}.log`);\n return fs.createWriteStream(logFilePath, { flags: 'a' });\n }\n\n // ロガーを作成する関数\n function createPinoLogger(fileNamePrefix: string): pino.Logger {\n return pino(\n {\n base: null,\n timestamp: false,\n messageKey: 'message',\n formatters: {\n level: () => ({}),\n bindings: () => ({}),\n },\n },\n createLogStream(fileNamePrefix)\n );\n }\n\n // 各ロガーの作成\n const state: LoggerState = {\n loggers: {},\n rotationFunction: (() => {}) as LogRotationFunction,\n intervalId: undefined,\n };\n\n for (const [type, logConfig] of Object.entries(logTypes)) {\n state.loggers[type] = createPinoLogger(logConfig.fileNamePrefix);\n }\n\n // ログファイルを日付変更時に切り替えるための関数\n const checkAndRotateLogFiles: LogRotationFunction = () => {\n const currentDate = getCurrentDate(timezoneOffset);\n\n if (checkAndRotateLogFiles.lastDate !== currentDate) {\n checkAndRotateLogFiles.lastDate = currentDate;\n\n for (const [type, logConfig] of Object.entries(logTypes)) {\n state.loggers[type] = createPinoLogger(logConfig.fileNamePrefix);\n }\n }\n };\n\n // 初期日付設定\n checkAndRotateLogFiles.lastDate = getCurrentDate(timezoneOffset);\n state.rotationFunction = checkAndRotateLogFiles;\n\n // ログローテーションのインターバル設定(テスト環境以外)\n // biome-ignore lint/complexity/useLiteralKeys: TypeScript requires bracket notation for process.env\n if (process.env['NODE_ENV'] !== 'test') {\n state.intervalId = setInterval(checkAndRotateLogFiles, rotationCheckInterval);\n }\n\n // 起動時に一度チェック\n checkAndRotateLogFiles();\n\n // カスタムフォーマットでログを出力する関数(レガシー形式)\n function formatAndLog(\n loggerType: string,\n logLevel: pino.Level,\n message: string,\n location: string | null,\n originalError: Error | string | null\n ): void {\n // ログレベルが最小レベル未満の場合はスキップ\n if (!shouldLog(logLevel as LogLevel, minLevel)) {\n return;\n }\n\n checkAndRotateLogFiles();\n\n const formattedMessage = formatLogMessage(\n logLevel,\n message,\n location,\n originalError,\n timezoneOffset\n );\n const logger = state.loggers[loggerType];\n\n if (logger) {\n logger[logLevel](formattedMessage);\n }\n }\n\n // 構造化コンテキスト対応のログ出力関数\n function formatAndLogWithContext(\n loggerType: string,\n logLevel: pino.Level,\n ctx: LogContext,\n message: string,\n error?: unknown,\n sendToSentry = false\n ): void {\n // ログレベルが最小レベル未満の場合はスキップ\n if (!shouldLog(logLevel as LogLevel, minLevel)) {\n return;\n }\n\n checkAndRotateLogFiles();\n\n const formattedMessage = formatLogMessageWithContext(\n logLevel,\n message,\n ctx,\n error,\n timezoneOffset\n );\n const logger = state.loggers[loggerType];\n\n if (logger) {\n logger[logLevel](formattedMessage);\n }\n\n // Sentry連携\n if (sendToSentry && enableSentry && isSentryEnabled()) {\n captureError(ctx, message, error);\n }\n }\n\n // オーバーロード対応のログ関数\n function createLogMethod(loggerType: string, logLevel: pino.Level, sendToSentry = false) {\n return (\n ctxOrMessage: LogContext | string,\n messageOrLocation?: string | null,\n errorOrOriginal?: unknown\n ) => {\n if (isLogContext(ctxOrMessage)) {\n // 構造化コンテキスト形式\n formatAndLogWithContext(\n loggerType,\n logLevel,\n ctxOrMessage,\n messageOrLocation ?? '',\n errorOrOriginal,\n sendToSentry\n );\n } else {\n // レガシー形式\n formatAndLog(\n loggerType,\n logLevel,\n ctxOrMessage,\n (messageOrLocation as string) ?? null,\n (errorOrOriginal as Error | string) ?? null\n );\n\n // レガシー形式でもSentry送信(error/fatalの場合)\n if (sendToSentry && enableSentry && isSentryEnabled()) {\n const ctx: LogContext = messageOrLocation ? { location: messageOrLocation } : {};\n captureError(ctx, ctxOrMessage, errorOrOriginal);\n }\n }\n };\n }\n\n return {\n debug: createLogMethod('debug', 'debug'),\n info: createLogMethod('info', 'info'),\n warn: createLogMethod('info', 'warn'),\n error: createLogMethod('error', 'error', true),\n fatal: createLogMethod('fatal', 'fatal', true),\n sqlError: (\n message: string,\n location?: string | null,\n originalError?: Error | string | null\n ) => {\n formatAndLog('sqlError', 'error', message, location ?? null, originalError ?? null);\n },\n } as Logger;\n}\n"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@degel_lc/logger",
3
- "version": "0.1.3",
3
+ "version": "0.1.5",
4
4
  "description": "phsk関連プロジェクト共通のpino-basedロギングライブラリ",
5
5
  "type": "module",
6
6
  "main": "./dist/index.js",