@futdevpro/nts-dynamo 1.15.20 → 1.15.21

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.
Files changed (34) hide show
  1. package/_specifications/BACKLOG.md +2 -2
  2. package/build/_modules/logs/_models/file-log-entry.interface.d.ts +14 -0
  3. package/build/_modules/logs/_models/file-log-entry.interface.d.ts.map +1 -0
  4. package/build/_modules/logs/_models/file-log-entry.interface.js +3 -0
  5. package/build/_modules/logs/_models/file-log-entry.interface.js.map +1 -0
  6. package/build/_modules/logs/_models/file-log-read-result.interface.d.ts +36 -0
  7. package/build/_modules/logs/_models/file-log-read-result.interface.d.ts.map +1 -0
  8. package/build/_modules/logs/_models/file-log-read-result.interface.js +3 -0
  9. package/build/_modules/logs/_models/file-log-read-result.interface.js.map +1 -0
  10. package/build/_modules/logs/file-log.service.d.ts +46 -0
  11. package/build/_modules/logs/file-log.service.d.ts.map +1 -1
  12. package/build/_modules/logs/file-log.service.js +178 -0
  13. package/build/_modules/logs/file-log.service.js.map +1 -1
  14. package/build/_modules/logs/file-logs.controller.d.ts +41 -0
  15. package/build/_modules/logs/file-logs.controller.d.ts.map +1 -0
  16. package/build/_modules/logs/file-logs.controller.js +139 -0
  17. package/build/_modules/logs/file-logs.controller.js.map +1 -0
  18. package/build/_modules/logs/get-file-logs-routing-module.util.d.ts +32 -0
  19. package/build/_modules/logs/get-file-logs-routing-module.util.d.ts.map +1 -0
  20. package/build/_modules/logs/get-file-logs-routing-module.util.js +38 -0
  21. package/build/_modules/logs/get-file-logs-routing-module.util.js.map +1 -0
  22. package/build/_modules/logs/index.d.ts +4 -0
  23. package/build/_modules/logs/index.d.ts.map +1 -1
  24. package/build/_modules/logs/index.js +5 -1
  25. package/build/_modules/logs/index.js.map +1 -1
  26. package/package.json +1 -1
  27. package/src/_modules/logs/_models/file-log-entry.interface.ts +13 -0
  28. package/src/_modules/logs/_models/file-log-read-result.interface.ts +37 -0
  29. package/src/_modules/logs/file-log.service.spec.ts +139 -0
  30. package/src/_modules/logs/file-log.service.ts +183 -0
  31. package/src/_modules/logs/file-logs.controller.spec.ts +245 -0
  32. package/src/_modules/logs/file-logs.controller.ts +165 -0
  33. package/src/_modules/logs/get-file-logs-routing-module.util.ts +51 -0
  34. package/src/_modules/logs/index.ts +7 -0
@@ -0,0 +1,38 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.DyNTS_getFileLogsRoutingModule = DyNTS_getFileLogsRoutingModule;
4
+ const routing_module_service_1 = require("../../_services/route/routing-module.service");
5
+ const file_logs_controller_1 = require("./file-logs.controller");
6
+ /**
7
+ * Letrehoz egy routing module-t a `DyNTS_FileLogs_Controller`-rel.
8
+ *
9
+ * **Hasznalat (auth nelkul, fejlesztoi env):**
10
+ * ```ts
11
+ * app.use(DyNTS_getFileLogsRoutingModule());
12
+ * ```
13
+ *
14
+ * **Hasznalat (admin API key auth-tal — javasolt produkcios env-re):**
15
+ * ```ts
16
+ * const adminAuth = DyNTS_AdminApiKey_AuthService.getInstance();
17
+ * app.use(DyNTS_getFileLogsRoutingModule({
18
+ * authPreProcess: adminAuth.verify,
19
+ * route: '/admin/file-logs', // opcionalis override
20
+ * }));
21
+ * ```
22
+ *
23
+ * **Elofeltetel:** `DyNTS_FileLog_Service.getInstance().install()` lefutott
24
+ * a szerver startup-on (es a `log_settings.file_log.enabled = true`).
25
+ * Egyebkent a controller endpointjai 503-at adnak.
26
+ */
27
+ function DyNTS_getFileLogsRoutingModule(config) {
28
+ // Controller config atadasa (static — getInstance() elott)
29
+ if (config) {
30
+ file_logs_controller_1.DyNTS_FileLogs_Controller.configure({ authPreProcess: config.authPreProcess });
31
+ }
32
+ const settings = {
33
+ route: config?.route ?? '/file-logs',
34
+ controllers: [file_logs_controller_1.DyNTS_FileLogs_Controller.getInstance()],
35
+ };
36
+ return new routing_module_service_1.DyNTS_RoutingModule(settings);
37
+ }
38
+ //# sourceMappingURL=get-file-logs-routing-module.util.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"get-file-logs-routing-module.util.js","sourceRoot":"","sources":["../../../src/_modules/logs/get-file-logs-routing-module.util.ts"],"names":[],"mappings":";;AAoCA,wEAcC;AAjDD,yFAAmF;AAEnF,iEAAoG;AAYpG;;;;;;;;;;;;;;;;;;;;GAoBG;AACH,SAAgB,8BAA8B,CAC5C,MAA2C;IAE3C,2DAA2D;IAC3D,IAAI,MAAM,EAAE,CAAC;QACX,gDAAyB,CAAC,SAAS,CAAC,EAAE,cAAc,EAAE,MAAM,CAAC,cAAc,EAAE,CAAC,CAAC;IACjF,CAAC;IAED,MAAM,QAAQ,GAAiC;QAC7C,KAAK,EAAE,MAAM,EAAE,KAAK,IAAI,YAAY;QACpC,WAAW,EAAE,CAAC,gDAAyB,CAAC,WAAW,EAAE,CAAC;KACvD,CAAC;IAEF,OAAO,IAAI,4CAAmB,CAAC,QAAQ,CAAC,CAAC;AAC3C,CAAC"}
@@ -2,4 +2,8 @@ export { DyNTS_Logs_Service } from './logs.service';
2
2
  export { DyNTS_Logs_Controller, DyNTS_LogsController_Config } from './logs.controller';
3
3
  export { DyNTS_getLogsRoutingModule } from './get-logs-routing-module.util';
4
4
  export { DyNTS_FileLog_Service } from './file-log.service';
5
+ export { DyNTS_FileLogs_Controller, DyNTS_FileLogsController_Config } from './file-logs.controller';
6
+ export { DyNTS_getFileLogsRoutingModule, DyNTS_FileLogsRoutingModule_Config, } from './get-file-logs-routing-module.util';
7
+ export { DyNTS_FileLog_Entry } from './_models/file-log-entry.interface';
8
+ export { DyNTS_FileLog_ReadResult, DyNTS_FileLog_ReadOptions } from './_models/file-log-read-result.interface';
5
9
  //# sourceMappingURL=index.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../src/_modules/logs/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,kBAAkB,EAAE,MAAM,gBAAgB,CAAC;AACpD,OAAO,EAAE,qBAAqB,EAAE,2BAA2B,EAAE,MAAM,mBAAmB,CAAC;AACvF,OAAO,EAAE,0BAA0B,EAAE,MAAM,gCAAgC,CAAC;AAC5E,OAAO,EAAE,qBAAqB,EAAE,MAAM,oBAAoB,CAAC"}
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../src/_modules/logs/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,kBAAkB,EAAE,MAAM,gBAAgB,CAAC;AACpD,OAAO,EAAE,qBAAqB,EAAE,2BAA2B,EAAE,MAAM,mBAAmB,CAAC;AACvF,OAAO,EAAE,0BAA0B,EAAE,MAAM,gCAAgC,CAAC;AAC5E,OAAO,EAAE,qBAAqB,EAAE,MAAM,oBAAoB,CAAC;AAC3D,OAAO,EAAE,yBAAyB,EAAE,+BAA+B,EAAE,MAAM,wBAAwB,CAAC;AACpG,OAAO,EACL,8BAA8B,EAC9B,kCAAkC,GACnC,MAAM,qCAAqC,CAAC;AAC7C,OAAO,EAAE,mBAAmB,EAAE,MAAM,oCAAoC,CAAC;AACzE,OAAO,EAAE,wBAAwB,EAAE,yBAAyB,EAAE,MAAM,0CAA0C,CAAC"}
@@ -1,6 +1,6 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.DyNTS_FileLog_Service = exports.DyNTS_getLogsRoutingModule = exports.DyNTS_Logs_Controller = exports.DyNTS_Logs_Service = void 0;
3
+ exports.DyNTS_getFileLogsRoutingModule = exports.DyNTS_FileLogs_Controller = exports.DyNTS_FileLog_Service = exports.DyNTS_getLogsRoutingModule = exports.DyNTS_Logs_Controller = exports.DyNTS_Logs_Service = void 0;
4
4
  var logs_service_1 = require("./logs.service");
5
5
  Object.defineProperty(exports, "DyNTS_Logs_Service", { enumerable: true, get: function () { return logs_service_1.DyNTS_Logs_Service; } });
6
6
  var logs_controller_1 = require("./logs.controller");
@@ -9,4 +9,8 @@ var get_logs_routing_module_util_1 = require("./get-logs-routing-module.util");
9
9
  Object.defineProperty(exports, "DyNTS_getLogsRoutingModule", { enumerable: true, get: function () { return get_logs_routing_module_util_1.DyNTS_getLogsRoutingModule; } });
10
10
  var file_log_service_1 = require("./file-log.service");
11
11
  Object.defineProperty(exports, "DyNTS_FileLog_Service", { enumerable: true, get: function () { return file_log_service_1.DyNTS_FileLog_Service; } });
12
+ var file_logs_controller_1 = require("./file-logs.controller");
13
+ Object.defineProperty(exports, "DyNTS_FileLogs_Controller", { enumerable: true, get: function () { return file_logs_controller_1.DyNTS_FileLogs_Controller; } });
14
+ var get_file_logs_routing_module_util_1 = require("./get-file-logs-routing-module.util");
15
+ Object.defineProperty(exports, "DyNTS_getFileLogsRoutingModule", { enumerable: true, get: function () { return get_file_logs_routing_module_util_1.DyNTS_getFileLogsRoutingModule; } });
12
16
  //# sourceMappingURL=index.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"index.js","sourceRoot":"","sources":["../../../src/_modules/logs/index.ts"],"names":[],"mappings":";;;AAAA,+CAAoD;AAA3C,kHAAA,kBAAkB,OAAA;AAC3B,qDAAuF;AAA9E,wHAAA,qBAAqB,OAAA;AAC9B,+EAA4E;AAAnE,0IAAA,0BAA0B,OAAA;AACnC,uDAA2D;AAAlD,yHAAA,qBAAqB,OAAA"}
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../../src/_modules/logs/index.ts"],"names":[],"mappings":";;;AAAA,+CAAoD;AAA3C,kHAAA,kBAAkB,OAAA;AAC3B,qDAAuF;AAA9E,wHAAA,qBAAqB,OAAA;AAC9B,+EAA4E;AAAnE,0IAAA,0BAA0B,OAAA;AACnC,uDAA2D;AAAlD,yHAAA,qBAAqB,OAAA;AAC9B,+DAAoG;AAA3F,iIAAA,yBAAyB,OAAA;AAClC,yFAG6C;AAF3C,mJAAA,8BAA8B,OAAA"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@futdevpro/nts-dynamo",
3
- "version": "01.15.20",
3
+ "version": "01.15.21",
4
4
  "description": "Dynamic NodeTS (NodeJS-Typescript), MongoDB Backend System Framework by Future Development Program Ltd.",
5
5
  "DyBu_settings": {
6
6
  "packageType": "server-package",
@@ -0,0 +1,13 @@
1
+ /**
2
+ * Egy log fajl meta-adata a listazasban.
3
+ */
4
+ export interface DyNTS_FileLog_Entry {
5
+ /** Fajlnev (csak basename — NEM teljes path). */
6
+ name: string;
7
+ /** Fajlmeret byte-ban. */
8
+ sizeBytes: number;
9
+ /** Modositasi ido ms-ban (mtimeMs). */
10
+ mtimeMs: number;
11
+ /** Igaz, ha ez az aktualis aktiv session fajlja. */
12
+ isCurrent: boolean;
13
+ }
@@ -0,0 +1,37 @@
1
+ /**
2
+ * `readLogFile()` valaszobjektum.
3
+ */
4
+ export interface DyNTS_FileLog_ReadResult {
5
+ /** Fajlnev (csak basename). */
6
+ name: string;
7
+ /** Teljes fajlmeret byte-ban. */
8
+ sizeBytes: number;
9
+ /** A fajlban talalhato osszes sor szama. */
10
+ totalLines: number;
11
+ /** A visszaadott sorok. */
12
+ lines: string[];
13
+ /** Mod amiben olvastunk: 'tail' | 'head' | 'range'. */
14
+ mode: 'tail' | 'head' | 'range';
15
+ /** Visszaadott tartomany 1-based start (range/tail/head normalizalva). */
16
+ rangeStart: number;
17
+ /** Visszaadott tartomany 1-based end (inclusive). */
18
+ rangeEnd: number;
19
+ }
20
+
21
+
22
+ /**
23
+ * `readLogFile(name, options)` opciok.
24
+ *
25
+ * Egyetlen mod aktiv egyszerre. Prioritas: range > head > tail.
26
+ * Egyik sem megadott → default `tail: 200`.
27
+ */
28
+ export interface DyNTS_FileLog_ReadOptions {
29
+ /** Utolso N sor (default mod ha semmi mas nincs). */
30
+ tail?: number;
31
+ /** Elso N sor. */
32
+ head?: number;
33
+ /** 1-based line range start (inclusive). `rangeEnd` is kell. */
34
+ rangeStart?: number;
35
+ /** 1-based line range end (inclusive). */
36
+ rangeEnd?: number;
37
+ }
@@ -199,4 +199,143 @@ describe('| DyNTS_FileLog_Service', (): void => {
199
199
  expect(DyNTS_FileLog_Service.getInstance().isInstalled()).toBe(false);
200
200
  });
201
201
  });
202
+
203
+ describe('| listLogFiles()', (): void => {
204
+ it('| ures lista ha nincs installalva', (): void => {
205
+ const svc = DyNTS_FileLog_Service.getInstance();
206
+ // NEM hivunk install-t
207
+ expect(svc.listLogFiles()).toEqual([]);
208
+ });
209
+
210
+ it('| listazza az osszes prefix-elt log fajlt + jelzi az isCurrent flag-et', (): void => {
211
+ // 3 fajl letrehozasa kezzel kulonbozo prefix-szel
212
+ fs.writeFileSync(path.join(tempDir, 'spec-2026-01-01_aaa.log'), 'a');
213
+ fs.writeFileSync(path.join(tempDir, 'spec-2026-01-02_bbb.log'), 'bb');
214
+ fs.writeFileSync(path.join(tempDir, 'spec-2026-01-03_ccc.log'), 'ccc');
215
+ fs.writeFileSync(path.join(tempDir, 'other-2026-01-04_zzz.log'), 'should-be-ignored');
216
+
217
+ DyNTS_global_settings.log_settings.file_log = {
218
+ enabled: true,
219
+ logDir: tempDir,
220
+ filenamePrefix: 'spec-',
221
+ };
222
+ const svc = DyNTS_FileLog_Service.getInstance();
223
+ svc.install();
224
+
225
+ const list = svc.listLogFiles();
226
+ // 3 spec- fajl + 1 aktualis session fajl = 4
227
+ expect(list.length).toBe(4);
228
+ const names: string[] = list.map((e) => e.name);
229
+ expect(names.find((n) => n.startsWith('other-'))).toBeUndefined();
230
+ expect(list.filter((e) => e.isCurrent).length).toBe(1);
231
+ });
232
+ });
233
+
234
+ describe('| readLogFile() — happy paths', (): void => {
235
+ let svc: DyNTS_FileLog_Service;
236
+ let testFile: string;
237
+
238
+ beforeEach((): void => {
239
+ DyNTS_global_settings.log_settings.file_log = {
240
+ enabled: true,
241
+ logDir: tempDir,
242
+ filenamePrefix: 'spec-',
243
+ };
244
+ svc = DyNTS_FileLog_Service.getInstance();
245
+ svc.install();
246
+ // Kulon teszt-fajl
247
+ testFile = `spec-test-${Date.now()}.log`;
248
+ const lines: string[] = [];
249
+ for (let i: number = 1; i <= 50; i++) {
250
+ lines.push(`line-${i}`);
251
+ }
252
+ fs.writeFileSync(path.join(tempDir, testFile), lines.join('\n') + '\n');
253
+ });
254
+
255
+ it('| tail mod: utolso N sor', (): void => {
256
+ const result = svc.readLogFile(testFile, { tail: 5 });
257
+ expect(result.mode).toBe('tail');
258
+ expect(result.lines).toEqual(['line-46', 'line-47', 'line-48', 'line-49', 'line-50']);
259
+ expect(result.totalLines).toBe(50);
260
+ expect(result.rangeStart).toBe(46);
261
+ expect(result.rangeEnd).toBe(50);
262
+ });
263
+
264
+ it('| head mod: elso N sor', (): void => {
265
+ const result = svc.readLogFile(testFile, { head: 3 });
266
+ expect(result.mode).toBe('head');
267
+ expect(result.lines).toEqual(['line-1', 'line-2', 'line-3']);
268
+ expect(result.rangeStart).toBe(1);
269
+ expect(result.rangeEnd).toBe(3);
270
+ });
271
+
272
+ it('| range mod: explicit 10-15 tartomany', (): void => {
273
+ const result = svc.readLogFile(testFile, { rangeStart: 10, rangeEnd: 15 });
274
+ expect(result.mode).toBe('range');
275
+ expect(result.lines).toEqual(['line-10', 'line-11', 'line-12', 'line-13', 'line-14', 'line-15']);
276
+ });
277
+
278
+ it('| default tail 200 ha semmi mod nincs megadva', (): void => {
279
+ const result = svc.readLogFile(testFile, {});
280
+ expect(result.mode).toBe('tail');
281
+ // A teszt fajl 50 sor → mind visszajon
282
+ expect(result.lines.length).toBe(50);
283
+ });
284
+ });
285
+
286
+ describe('| readLogFile() — error paths', (): void => {
287
+ let svc: DyNTS_FileLog_Service;
288
+
289
+ beforeEach((): void => {
290
+ DyNTS_global_settings.log_settings.file_log = {
291
+ enabled: true,
292
+ logDir: tempDir,
293
+ filenamePrefix: 'spec-',
294
+ };
295
+ svc = DyNTS_FileLog_Service.getInstance();
296
+ svc.install();
297
+ });
298
+
299
+ it('| throw ha invalid filename (prefix nem matchel)', (): void => {
300
+ expect(() => svc.readLogFile('other-2026.log')).toThrowError('invalid filename');
301
+ });
302
+
303
+ it('| throw ha path traversal kiserlet (`../`)', (): void => {
304
+ expect(() => svc.readLogFile('../etc/passwd')).toThrowError('invalid filename');
305
+ });
306
+
307
+ it('| throw ha path separator (`/`) van a nevben', (): void => {
308
+ expect(() => svc.readLogFile('spec-foo/bar.log')).toThrowError('invalid filename');
309
+ });
310
+
311
+ it('| throw ha file nem letezik', (): void => {
312
+ expect(() => svc.readLogFile('spec-nonexistent.log')).toThrowError('file not found');
313
+ });
314
+
315
+ it('| NEM installalt service → throw `not installed`', (): void => {
316
+ svc._teardownForTesting();
317
+ expect(() => svc.readLogFile('spec-foo.log')).toThrowError(/not installed/);
318
+ });
319
+ });
320
+
321
+ describe('| getCurrentLogFilename()', (): void => {
322
+ it('| ures string ha nincs installalva', (): void => {
323
+ expect(DyNTS_FileLog_Service.getInstance().getCurrentLogFilename()).toBe('');
324
+ });
325
+
326
+ it('| basename-t ad vissza ha installalva van', (): void => {
327
+ DyNTS_global_settings.log_settings.file_log = {
328
+ enabled: true,
329
+ logDir: tempDir,
330
+ filenamePrefix: 'spec-',
331
+ };
332
+ const svc = DyNTS_FileLog_Service.getInstance();
333
+ svc.install();
334
+ const basename: string = svc.getCurrentLogFilename();
335
+ expect(basename.startsWith('spec-')).toBe(true);
336
+ expect(basename.endsWith('.log')).toBe(true);
337
+ expect(basename.includes('/')).toBe(false);
338
+ expect(basename.includes('\\')).toBe(false);
339
+ });
340
+ });
202
341
  });
@@ -3,6 +3,8 @@ import * as path from 'path';
3
3
 
4
4
  import { DyNTS_SingletonServiceBase } from '../../_services/base/singleton.service-base';
5
5
  import { DyNTS_global_settings } from '../../_collections/global-settings.const';
6
+ import { DyNTS_FileLog_Entry } from './_models/file-log-entry.interface';
7
+ import { DyNTS_FileLog_ReadOptions, DyNTS_FileLog_ReadResult } from './_models/file-log-read-result.interface';
6
8
 
7
9
 
8
10
  /**
@@ -22,6 +24,12 @@ const DEFAULT_INCLUDE_STDERR: boolean = true;
22
24
  /** ANSI escape code regex (szin/formazas kodok). */
23
25
  const ANSI_ESCAPE_REGEX: RegExp = /\x1b\[[0-9;]*[a-zA-Z]/g;
24
26
 
27
+ /** Max sor szam egy `readLogFile()` hivasra — vedi a memory-t nagy fajloknal. */
28
+ const MAX_READ_LINES: number = 10000;
29
+
30
+ /** Default tail sor szam (ha semmi mas mod nincs megadva). */
31
+ const DEFAULT_READ_TAIL: number = 200;
32
+
25
33
 
26
34
  /**
27
35
  * File-based szerver log service — duplikalja a stdout/stderr-t per-session
@@ -149,6 +157,181 @@ export class DyNTS_FileLog_Service extends DyNTS_SingletonServiceBase {
149
157
  return this.currentLogPath;
150
158
  }
151
159
 
160
+ /**
161
+ * Az aktualis aktiv log fajl basename-je (NEM teljes path). Csak akkor
162
+ * nem-ures, ha az install() sikeresen lefutott.
163
+ */
164
+ getCurrentLogFilename(): string {
165
+ if (!this.currentLogPath) { return ''; }
166
+ return path.basename(this.currentLogPath);
167
+ }
168
+
169
+ /**
170
+ * Aktiv log dir abszolut path-ja. Csak akkor nem-ures, ha az install()
171
+ * sikeresen lefutott.
172
+ */
173
+ getActiveLogDir(): string {
174
+ return this.activeLogDir;
175
+ }
176
+
177
+ /**
178
+ * Aktiv filename prefix (a service config-jabol). Hasznalja a controller
179
+ * a request-fajlnev whitelist regex epitesehez.
180
+ */
181
+ getFilenamePrefix(): string {
182
+ return this.filenamePrefix;
183
+ }
184
+
185
+ /**
186
+ * Osszes log fajl listazasa az aktiv log dir-bol. Csak a service prefix-evel
187
+ * matchelo fajlokat veszi (`{prefix}*.log`).
188
+ *
189
+ * Rendezes: mtime DESC (legujabb elol). Hibakat csendben elnyel — ha a dir
190
+ * nem letezik vagy nem olvashato, ures array-t ad vissza.
191
+ */
192
+ listLogFiles(): DyNTS_FileLog_Entry[] {
193
+ if (!this.activeLogDir || !this.installed) { return []; }
194
+ const currentBasename: string = this.getCurrentLogFilename();
195
+ const entries: DyNTS_FileLog_Entry[] = [];
196
+
197
+ try {
198
+ const dirContents: string[] = fs.readdirSync(this.activeLogDir);
199
+ for (const name of dirContents) {
200
+ if (!name.startsWith(this.filenamePrefix) || !name.endsWith(DEFAULT_FILENAME_SUFFIX)) { continue; }
201
+ const fullPath: string = path.join(this.activeLogDir, name);
202
+ try {
203
+ const stats: fs.Stats = fs.statSync(fullPath);
204
+ if (!stats.isFile()) { continue; }
205
+ entries.push({
206
+ name: name,
207
+ sizeBytes: stats.size,
208
+ mtimeMs: stats.mtimeMs,
209
+ isCurrent: name === currentBasename,
210
+ });
211
+ } catch {
212
+ // egy fajlonkenti hiba ne szakitsa meg a listat
213
+ }
214
+ }
215
+ } catch {
216
+ // dir read hiba → ures listat adunk vissza
217
+ return [];
218
+ }
219
+
220
+ // Legujabb elol
221
+ entries.sort((a: DyNTS_FileLog_Entry, b: DyNTS_FileLog_Entry): number => b.mtimeMs - a.mtimeMs);
222
+ return entries;
223
+ }
224
+
225
+ /**
226
+ * Egy konkret log fajl olvasasa, line-based mode-ban (tail / head / range).
227
+ *
228
+ * **Biztonsagi safeguards:**
229
+ * - Filename whitelist: `{prefix}*.log` mintara matchelo nev kotelezo
230
+ * - Path resolution `path.resolve(activeLogDir, name)` + `startsWith(activeLogDir)` (path traversal vedelem)
231
+ * - Sor szam cap: `MAX_READ_LINES` (10000)
232
+ *
233
+ * **Hibak:**
234
+ * - Throw `Error('not installed')` ha a service nincs install-olva
235
+ * - Throw `Error('invalid filename')` ha a name nem matchel a whitelist-re VAGY kilep az activeLogDir-bol
236
+ * - Throw `Error('file not found')` ha a fajl nem letezik
237
+ *
238
+ * A hivot kell vigyaznia hogy ezeket HTTP status code-okra forditja.
239
+ */
240
+ readLogFile(name: string, options: DyNTS_FileLog_ReadOptions = {}): DyNTS_FileLog_ReadResult {
241
+ if (!this.installed || !this.activeLogDir) {
242
+ throw new Error('FileLog service not installed');
243
+ }
244
+
245
+ // Whitelist regex: csak a service prefix-evel matchelo fajlnev
246
+ if (!this.isValidFilename(name)) {
247
+ throw new Error('invalid filename');
248
+ }
249
+
250
+ // Path resolve + traversal check
251
+ const resolvedPath: string = path.resolve(this.activeLogDir, name);
252
+ const resolvedDir: string = path.resolve(this.activeLogDir);
253
+ // path.sep + sufix ellenőrzéssel megelőzzük a hasonló prefixű dir átverést
254
+ // (pl. '/logs' vs '/logs-other'); a resolvedPath az activeLogDir-en BELUL
255
+ // kell legyen.
256
+ if (!resolvedPath.startsWith(resolvedDir + path.sep) && resolvedPath !== resolvedDir) {
257
+ throw new Error('invalid filename');
258
+ }
259
+
260
+ if (!fs.existsSync(resolvedPath)) {
261
+ throw new Error('file not found');
262
+ }
263
+
264
+ const stats: fs.Stats = fs.statSync(resolvedPath);
265
+ if (!stats.isFile()) {
266
+ throw new Error('file not found');
267
+ }
268
+
269
+ // Olvass + split soroka. UTF-8 encoding alapertelmezett. Nagy fajloknal a
270
+ // teljes betoltes nem ideal, de a 10000-soros cap es a Stream-based
271
+ // optimization egy kesobbi iteracio lehet ha kell.
272
+ const fullContent: string = fs.readFileSync(resolvedPath, 'utf-8');
273
+ // Split a sorvegekre — utolso ures stringet kivagjuk ha a fajl
274
+ // sorvégzodessel zarult (regenerált sor szám hibakerulest celozzuk meg)
275
+ const allLines: string[] = fullContent.split(/\r?\n/);
276
+ if (allLines.length > 0 && allLines[allLines.length - 1] === '') {
277
+ allLines.pop();
278
+ }
279
+ const totalLines: number = allLines.length;
280
+
281
+ // Mod meghatarozasa: range > head > tail
282
+ let mode: 'tail' | 'head' | 'range';
283
+ let startIdx: number;
284
+ let endIdx: number; // exclusive
285
+
286
+ if (options.rangeStart !== undefined && options.rangeEnd !== undefined) {
287
+ mode = 'range';
288
+ const start1: number = Math.max(1, Math.floor(options.rangeStart));
289
+ const end1: number = Math.max(start1, Math.floor(options.rangeEnd));
290
+ startIdx = Math.max(0, Math.min(totalLines, start1 - 1));
291
+ endIdx = Math.max(startIdx, Math.min(totalLines, end1));
292
+ // Cap
293
+ if (endIdx - startIdx > MAX_READ_LINES) {
294
+ endIdx = startIdx + MAX_READ_LINES;
295
+ }
296
+ } else if (options.head !== undefined) {
297
+ mode = 'head';
298
+ const headN: number = Math.min(Math.max(1, Math.floor(options.head)), MAX_READ_LINES);
299
+ startIdx = 0;
300
+ endIdx = Math.min(totalLines, headN);
301
+ } else {
302
+ mode = 'tail';
303
+ const tailN: number = Math.min(Math.max(1, Math.floor(options.tail ?? DEFAULT_READ_TAIL)), MAX_READ_LINES);
304
+ startIdx = Math.max(0, totalLines - tailN);
305
+ endIdx = totalLines;
306
+ }
307
+
308
+ const lines: string[] = allLines.slice(startIdx, endIdx);
309
+
310
+ return {
311
+ name: path.basename(resolvedPath),
312
+ sizeBytes: stats.size,
313
+ totalLines: totalLines,
314
+ lines: lines,
315
+ mode: mode,
316
+ rangeStart: startIdx + 1, // 1-based, inclusive
317
+ rangeEnd: Math.max(startIdx, endIdx), // 1-based, inclusive ha lines van; egyebkent 0-szeru
318
+ };
319
+ }
320
+
321
+ /**
322
+ * Filename whitelist ellenorzes. Csak a service prefix-evel matchelo,
323
+ * biztonsagos karakterkeszletu fajlnev megengedett.
324
+ */
325
+ private isValidFilename(name: string): boolean {
326
+ if (typeof name !== 'string' || name.length === 0 || name.length > 256) { return false; }
327
+ // Path-szeparatorokat kifejezetten tiltjuk (mar a regex is, de gyors check)
328
+ if (name.includes('/') || name.includes('\\') || name.includes('..')) { return false; }
329
+ // `{prefix}[\w\-.]+\.log` — szigoru ASCII set
330
+ const escapedPrefix: string = this.filenamePrefix.replace(/[.*+?^${}()|[\]\\]/g, '\\$&');
331
+ const regex: RegExp = new RegExp(`^${escapedPrefix}[\\w\\-.]+\\.log$`);
332
+ return regex.test(name);
333
+ }
334
+
152
335
  /**
153
336
  * Telepitve van-e (csak akkor true, ha az enabled === true es a setup nem bukott).
154
337
  */