@j3r3mcdev/oast-server 1.1.5 → 1.1.7
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/.github/workflows/ci.yml +29 -29
- package/.github/workflows/publish.yml +31 -31
- package/README.md +192 -192
- package/jest.config.js +14 -14
- package/package.json +45 -45
- package/sadmin list shadows +9 -9
- package/src/api/controllers/__tests__/tasks.controller.test.ts +74 -74
- package/src/api/controllers/events.controller.ts +10 -10
- package/src/api/controllers/health.controller.ts +7 -7
- package/src/api/controllers/tasks.controller.ts +41 -41
- package/src/api/dto/__tests__/create-task.dto.test.ts +41 -41
- package/src/api/dto/__tests__/filter-tasks.dto.test.ts +35 -35
- package/src/api/dto/create-task.dto.ts +33 -33
- package/src/api/dto/filter-tasks.dto.ts +33 -33
- package/src/api/services/__tests__/events.service.test.ts +41 -41
- package/src/api/services/__tests__/tasks.service.test.ts +41 -41
- package/src/api/services/events.service.ts +17 -17
- package/src/api/services/tasks.service.ts +79 -79
- package/src/api/sse/events.stream.ts +90 -90
- package/src/bootstrap.ts +89 -89
- package/src/core/__tests__/core-router.test.ts +30 -30
- package/src/core/__tests__/core-server.test.ts +44 -44
- package/src/core/__tests__/event.normalizer.test.ts +56 -56
- package/src/core/__tests__/event.router.test.ts +89 -89
- package/src/core/__tests__/logger.test.ts +32 -32
- package/src/core/__tests__/storage-manager.test.ts +74 -74
- package/src/core/event.normalizer.ts +147 -147
- package/src/core/event.router.ts +13 -13
- package/src/core/http/__tests__/adapter-node.test.ts +52 -52
- package/src/core/http/__tests__/body-parser-multipart.test.ts +41 -41
- package/src/core/http/__tests__/body-parser-raw.test.ts +28 -28
- package/src/core/http/__tests__/body-parser-text.test.ts +28 -28
- package/src/core/http/__tests__/compile-path.test.ts +39 -39
- package/src/core/http/__tests__/middleware-pipeline.test.ts +51 -51
- package/src/core/http/__tests__/request.test.ts +34 -34
- package/src/core/http/__tests__/response.test.ts +35 -35
- package/src/core/http/__tests__/router-match.test.ts +171 -171
- package/src/core/http/adapter-node.ts +51 -51
- package/src/core/http/buildRequest.ts +18 -18
- package/src/core/http/compile-path.ts +32 -32
- package/src/core/http/errors.ts +37 -37
- package/src/core/http/http-server.ts +52 -52
- package/src/core/http/middleware.ts +160 -160
- package/src/core/http/request.ts +55 -55
- package/src/core/http/response.ts +93 -93
- package/src/core/http/router.ts +138 -138
- package/src/core/id-generator.ts +8 -8
- package/src/core/logger.ts +113 -113
- package/src/core/router.ts +44 -44
- package/src/core/server.ts +85 -85
- package/src/core/storage.ts +64 -64
- package/src/index.ts +14 -14
- package/src/listeners/api/__tests__/api.controller.test.ts +116 -116
- package/src/listeners/api/__tests__/api.extractor.test.ts +46 -46
- package/src/listeners/api/__tests__/api.listener.test.ts +82 -82
- package/src/listeners/api/__tests__/api.routes.test.ts +155 -155
- package/src/listeners/api/__tests__/api.sse.test.ts +105 -105
- package/src/listeners/api/api.controllers.ts +67 -67
- package/src/listeners/api/api.extractor.ts +43 -43
- package/src/listeners/api/api.listener.ts +50 -50
- package/src/listeners/api/api.routes.ts +76 -76
- package/src/listeners/api/api.sse.ts +38 -38
- package/src/listeners/dns/__tests__/dns.test.ts +118 -118
- package/src/listeners/dns/dns.extractor.ts +14 -14
- package/src/listeners/dns/dns.listener.ts +61 -61
- package/src/listeners/http/__tests__/http.extractor.test.ts +59 -59
- package/src/listeners/http/__tests__/http.listener.test.ts +133 -133
- package/src/listeners/http/http.extractor.ts +15 -15
- package/src/listeners/http/http.listener.ts +110 -110
- package/src/listeners/listener.interface.ts +4 -4
- package/src/listeners/smtp/__tests__/smtp.extractor.test.ts +69 -69
- package/src/listeners/smtp/__tests__/smtp.listener.test.ts +150 -150
- package/src/listeners/smtp/smtp.extractor.ts +18 -18
- package/src/listeners/smtp/smtp.listener.ts +60 -60
- package/src/listeners/ssrf/__tests__/ssrf.extractor.test.ts +41 -41
- package/src/listeners/ssrf/__tests__/ssrf.listener.test.ts +87 -87
- package/src/listeners/ssrf/ssrf.extractor.ts +14 -14
- package/src/listeners/ssrf/ssrf.listener.ts +37 -37
- package/src/listeners/tcp/tcp.extractor.ts +16 -16
- package/src/listeners/tcp/tcp.listener.ts +61 -61
- package/src/listeners/webhook/__tests__/webhook.extractor.test.ts +35 -35
- package/src/listeners/webhook/__tests__/webhook.listener.test.ts +122 -122
- package/src/listeners/webhook/webhook.extractor.ts +12 -12
- package/src/listeners/webhook/webhook.listener.ts +58 -58
- package/src/listeners/websocket/__tests__/websocket.extractor.test.ts +33 -33
- package/src/listeners/websocket/__tests__/websocket.listener.test.ts +90 -90
- package/src/listeners/websocket/websocket.extractor.ts +11 -11
- package/src/listeners/websocket/websocket.listener.ts +40 -40
- package/src/storage-adapters/adapters/__tests__/memory.storage.test.ts +75 -75
- package/src/storage-adapters/adapters/memory.storage.ts +64 -64
- package/src/storage-adapters/storage.interface.ts +26 -26
- package/src/types/event.types.ts +147 -147
- package/tsconfig.json +20 -21
package/src/core/server.ts
CHANGED
|
@@ -1,85 +1,85 @@
|
|
|
1
|
-
import { Logger } from "./logger";
|
|
2
|
-
import { CoreRouter } from "./router";
|
|
3
|
-
import { StorageManager } from "./storage";
|
|
4
|
-
import { Listener } from "../listeners/listener.interface";
|
|
5
|
-
export interface CoreServerOptions {
|
|
6
|
-
logger?: Logger;
|
|
7
|
-
storage?: StorageManager;
|
|
8
|
-
listeners?: Listener[];
|
|
9
|
-
}
|
|
10
|
-
|
|
11
|
-
export class CoreServer {
|
|
12
|
-
private logger: Logger;
|
|
13
|
-
private storage: StorageManager;
|
|
14
|
-
private router: CoreRouter;
|
|
15
|
-
private listeners: Listener[];
|
|
16
|
-
|
|
17
|
-
constructor(options: CoreServerOptions = {}) {
|
|
18
|
-
this.logger = options.logger ?? new Logger({ context: "CoreServer" });
|
|
19
|
-
this.storage =
|
|
20
|
-
options.storage ?? new StorageManager({ logger: this.logger });
|
|
21
|
-
this.router = new CoreRouter(this.storage, { logger: this.logger });
|
|
22
|
-
this.listeners = options.listeners ?? [];
|
|
23
|
-
}
|
|
24
|
-
|
|
25
|
-
getRouter() {
|
|
26
|
-
return this.router;
|
|
27
|
-
}
|
|
28
|
-
|
|
29
|
-
getStorage() {
|
|
30
|
-
return this.storage;
|
|
31
|
-
}
|
|
32
|
-
|
|
33
|
-
async start(): Promise<void> {
|
|
34
|
-
this.logger.info("Starting OAST server...");
|
|
35
|
-
|
|
36
|
-
// Démarrage des listeners
|
|
37
|
-
for (const listener of this.listeners) {
|
|
38
|
-
try {
|
|
39
|
-
await listener.start();
|
|
40
|
-
this.logger.info(`Listener started`, {
|
|
41
|
-
listener: listener.constructor.name,
|
|
42
|
-
});
|
|
43
|
-
} catch (err: any) {
|
|
44
|
-
this.logger.error("Listener failed to start", {
|
|
45
|
-
listener: listener.constructor.name,
|
|
46
|
-
error: err.message,
|
|
47
|
-
});
|
|
48
|
-
}
|
|
49
|
-
}
|
|
50
|
-
|
|
51
|
-
this.logger.info("OAST server started");
|
|
52
|
-
}
|
|
53
|
-
|
|
54
|
-
async stop(): Promise<void> {
|
|
55
|
-
this.logger.info("Stopping OAST server...");
|
|
56
|
-
|
|
57
|
-
for (const listener of this.listeners) {
|
|
58
|
-
try {
|
|
59
|
-
await listener.stop();
|
|
60
|
-
this.logger.info(`Listener stopped`, {
|
|
61
|
-
listener: listener.constructor.name,
|
|
62
|
-
});
|
|
63
|
-
} catch (err: any) {
|
|
64
|
-
this.logger.error("Listener failed to stop", {
|
|
65
|
-
listener: listener.constructor.name,
|
|
66
|
-
error: err.message,
|
|
67
|
-
});
|
|
68
|
-
}
|
|
69
|
-
}
|
|
70
|
-
|
|
71
|
-
this.logger.info("OAST server stopped");
|
|
72
|
-
}
|
|
73
|
-
|
|
74
|
-
/**
|
|
75
|
-
* Méthode appelée par CoreRouter après chaque event normalisé.
|
|
76
|
-
* Permet de diffuser l’event aux listeners SSE (ApiListener).
|
|
77
|
-
*/
|
|
78
|
-
broadcast(event: any) {
|
|
79
|
-
for (const listener of this.listeners) {
|
|
80
|
-
if (typeof (listener as any).broadcastEvent === "function") {
|
|
81
|
-
(listener as any).broadcastEvent(event);
|
|
82
|
-
}
|
|
83
|
-
}
|
|
84
|
-
}
|
|
85
|
-
}
|
|
1
|
+
import { Logger } from "./logger";
|
|
2
|
+
import { CoreRouter } from "./router";
|
|
3
|
+
import { StorageManager } from "./storage";
|
|
4
|
+
import { Listener } from "../listeners/listener.interface";
|
|
5
|
+
export interface CoreServerOptions {
|
|
6
|
+
logger?: Logger;
|
|
7
|
+
storage?: StorageManager;
|
|
8
|
+
listeners?: Listener[];
|
|
9
|
+
}
|
|
10
|
+
|
|
11
|
+
export class CoreServer {
|
|
12
|
+
private logger: Logger;
|
|
13
|
+
private storage: StorageManager;
|
|
14
|
+
private router: CoreRouter;
|
|
15
|
+
private listeners: Listener[];
|
|
16
|
+
|
|
17
|
+
constructor(options: CoreServerOptions = {}) {
|
|
18
|
+
this.logger = options.logger ?? new Logger({ context: "CoreServer" });
|
|
19
|
+
this.storage =
|
|
20
|
+
options.storage ?? new StorageManager({ logger: this.logger });
|
|
21
|
+
this.router = new CoreRouter(this.storage, { logger: this.logger });
|
|
22
|
+
this.listeners = options.listeners ?? [];
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
getRouter() {
|
|
26
|
+
return this.router;
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
getStorage() {
|
|
30
|
+
return this.storage;
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
async start(): Promise<void> {
|
|
34
|
+
this.logger.info("Starting OAST server...");
|
|
35
|
+
|
|
36
|
+
// Démarrage des listeners
|
|
37
|
+
for (const listener of this.listeners) {
|
|
38
|
+
try {
|
|
39
|
+
await listener.start();
|
|
40
|
+
this.logger.info(`Listener started`, {
|
|
41
|
+
listener: listener.constructor.name,
|
|
42
|
+
});
|
|
43
|
+
} catch (err: any) {
|
|
44
|
+
this.logger.error("Listener failed to start", {
|
|
45
|
+
listener: listener.constructor.name,
|
|
46
|
+
error: err.message,
|
|
47
|
+
});
|
|
48
|
+
}
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
this.logger.info("OAST server started");
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
async stop(): Promise<void> {
|
|
55
|
+
this.logger.info("Stopping OAST server...");
|
|
56
|
+
|
|
57
|
+
for (const listener of this.listeners) {
|
|
58
|
+
try {
|
|
59
|
+
await listener.stop();
|
|
60
|
+
this.logger.info(`Listener stopped`, {
|
|
61
|
+
listener: listener.constructor.name,
|
|
62
|
+
});
|
|
63
|
+
} catch (err: any) {
|
|
64
|
+
this.logger.error("Listener failed to stop", {
|
|
65
|
+
listener: listener.constructor.name,
|
|
66
|
+
error: err.message,
|
|
67
|
+
});
|
|
68
|
+
}
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
this.logger.info("OAST server stopped");
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
/**
|
|
75
|
+
* Méthode appelée par CoreRouter après chaque event normalisé.
|
|
76
|
+
* Permet de diffuser l’event aux listeners SSE (ApiListener).
|
|
77
|
+
*/
|
|
78
|
+
broadcast(event: any) {
|
|
79
|
+
for (const listener of this.listeners) {
|
|
80
|
+
if (typeof (listener as any).broadcastEvent === "function") {
|
|
81
|
+
(listener as any).broadcastEvent(event);
|
|
82
|
+
}
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
}
|
package/src/core/storage.ts
CHANGED
|
@@ -1,64 +1,64 @@
|
|
|
1
|
-
import { Storage } from "../storage-adapters/storage.interface";
|
|
2
|
-
import { AnyNormalizedEvent } from "../types/event.types";
|
|
3
|
-
import { Logger } from "./logger";
|
|
4
|
-
|
|
5
|
-
export class StorageManager implements Storage {
|
|
6
|
-
private events: AnyNormalizedEvent[] = [];
|
|
7
|
-
private logger: Logger;
|
|
8
|
-
|
|
9
|
-
constructor(options: { logger?: Logger } = {}) {
|
|
10
|
-
this.logger = options.logger ?? new Logger({ context: "StorageManager" });
|
|
11
|
-
}
|
|
12
|
-
|
|
13
|
-
async save(event: AnyNormalizedEvent): Promise<void> {
|
|
14
|
-
this.events.push(event);
|
|
15
|
-
}
|
|
16
|
-
|
|
17
|
-
async getEvent(id: string): Promise<AnyNormalizedEvent | null> {
|
|
18
|
-
return this.events.find((e) => e.id === id) ?? null;
|
|
19
|
-
}
|
|
20
|
-
|
|
21
|
-
async listEvents(params: {
|
|
22
|
-
type?: string;
|
|
23
|
-
page?: number;
|
|
24
|
-
limit?: number;
|
|
25
|
-
}): Promise<AnyNormalizedEvent[]> {
|
|
26
|
-
const { type, page = 1, limit = 50 } = params;
|
|
27
|
-
|
|
28
|
-
let filtered = this.events;
|
|
29
|
-
if (type) filtered = filtered.filter((e) => e.type === type);
|
|
30
|
-
|
|
31
|
-
const start = (page - 1) * limit;
|
|
32
|
-
return filtered.slice(start, start + limit);
|
|
33
|
-
}
|
|
34
|
-
|
|
35
|
-
async getAll(): Promise<AnyNormalizedEvent[]> {
|
|
36
|
-
return this.events;
|
|
37
|
-
}
|
|
38
|
-
|
|
39
|
-
async deleteEvent(id: string): Promise<boolean> {
|
|
40
|
-
const before = this.events.length;
|
|
41
|
-
this.events = this.events.filter((e) => e.id !== id);
|
|
42
|
-
return this.events.length !== before;
|
|
43
|
-
}
|
|
44
|
-
|
|
45
|
-
async clearEvents(): Promise<void> {
|
|
46
|
-
this.events = [];
|
|
47
|
-
}
|
|
48
|
-
|
|
49
|
-
async getStats(): Promise<{
|
|
50
|
-
total: number;
|
|
51
|
-
byType: Record<string, number>;
|
|
52
|
-
}> {
|
|
53
|
-
const byType: Record<string, number> = {};
|
|
54
|
-
|
|
55
|
-
for (const e of this.events) {
|
|
56
|
-
byType[e.type] = (byType[e.type] ?? 0) + 1;
|
|
57
|
-
}
|
|
58
|
-
|
|
59
|
-
return {
|
|
60
|
-
total: this.events.length,
|
|
61
|
-
byType,
|
|
62
|
-
};
|
|
63
|
-
}
|
|
64
|
-
}
|
|
1
|
+
import { Storage } from "../storage-adapters/storage.interface";
|
|
2
|
+
import { AnyNormalizedEvent } from "../types/event.types";
|
|
3
|
+
import { Logger } from "./logger";
|
|
4
|
+
|
|
5
|
+
export class StorageManager implements Storage {
|
|
6
|
+
private events: AnyNormalizedEvent[] = [];
|
|
7
|
+
private logger: Logger;
|
|
8
|
+
|
|
9
|
+
constructor(options: { logger?: Logger } = {}) {
|
|
10
|
+
this.logger = options.logger ?? new Logger({ context: "StorageManager" });
|
|
11
|
+
}
|
|
12
|
+
|
|
13
|
+
async save(event: AnyNormalizedEvent): Promise<void> {
|
|
14
|
+
this.events.push(event);
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
async getEvent(id: string): Promise<AnyNormalizedEvent | null> {
|
|
18
|
+
return this.events.find((e) => e.id === id) ?? null;
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
async listEvents(params: {
|
|
22
|
+
type?: string;
|
|
23
|
+
page?: number;
|
|
24
|
+
limit?: number;
|
|
25
|
+
}): Promise<AnyNormalizedEvent[]> {
|
|
26
|
+
const { type, page = 1, limit = 50 } = params;
|
|
27
|
+
|
|
28
|
+
let filtered = this.events;
|
|
29
|
+
if (type) filtered = filtered.filter((e) => e.type === type);
|
|
30
|
+
|
|
31
|
+
const start = (page - 1) * limit;
|
|
32
|
+
return filtered.slice(start, start + limit);
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
async getAll(): Promise<AnyNormalizedEvent[]> {
|
|
36
|
+
return this.events;
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
async deleteEvent(id: string): Promise<boolean> {
|
|
40
|
+
const before = this.events.length;
|
|
41
|
+
this.events = this.events.filter((e) => e.id !== id);
|
|
42
|
+
return this.events.length !== before;
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
async clearEvents(): Promise<void> {
|
|
46
|
+
this.events = [];
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
async getStats(): Promise<{
|
|
50
|
+
total: number;
|
|
51
|
+
byType: Record<string, number>;
|
|
52
|
+
}> {
|
|
53
|
+
const byType: Record<string, number> = {};
|
|
54
|
+
|
|
55
|
+
for (const e of this.events) {
|
|
56
|
+
byType[e.type] = (byType[e.type] ?? 0) + 1;
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
return {
|
|
60
|
+
total: this.events.length,
|
|
61
|
+
byType,
|
|
62
|
+
};
|
|
63
|
+
}
|
|
64
|
+
}
|
package/src/index.ts
CHANGED
|
@@ -1,14 +1,14 @@
|
|
|
1
|
-
export { ApiListener } from "./listeners/api/api.listener";
|
|
2
|
-
export { HttpListener } from "./listeners/http/http.listener";
|
|
3
|
-
export { DnsListener } from "./listeners/dns/dns.listener";
|
|
4
|
-
export { SmtpListener } from "./listeners/smtp/smtp.listener";
|
|
5
|
-
export { TcpListener } from "./listeners/tcp/tcp.listener";
|
|
6
|
-
|
|
7
|
-
export { CoreServer } from "./core/server";
|
|
8
|
-
export { CoreRouter } from "./core/router";
|
|
9
|
-
export { StorageManager } from "./core/storage";
|
|
10
|
-
export { Logger } from "./core/logger";
|
|
11
|
-
|
|
12
|
-
export { EventNormalizer } from "./core/event.normalizer";
|
|
13
|
-
|
|
14
|
-
export * from "./types/event.types";
|
|
1
|
+
export { ApiListener } from "./listeners/api/api.listener";
|
|
2
|
+
export { HttpListener } from "./listeners/http/http.listener";
|
|
3
|
+
export { DnsListener } from "./listeners/dns/dns.listener";
|
|
4
|
+
export { SmtpListener } from "./listeners/smtp/smtp.listener";
|
|
5
|
+
export { TcpListener } from "./listeners/tcp/tcp.listener";
|
|
6
|
+
|
|
7
|
+
export { CoreServer } from "./core/server";
|
|
8
|
+
export { CoreRouter } from "./core/router";
|
|
9
|
+
export { StorageManager } from "./core/storage";
|
|
10
|
+
export { Logger } from "./core/logger";
|
|
11
|
+
|
|
12
|
+
export { EventNormalizer } from "./core/event.normalizer";
|
|
13
|
+
|
|
14
|
+
export * from "./types/event.types";
|
|
@@ -1,116 +1,116 @@
|
|
|
1
|
-
import { ApiController } from "../api.controllers";
|
|
2
|
-
import { StorageManager } from "../../../core/storage";
|
|
3
|
-
import { ServerResponse, IncomingMessage } from "http";
|
|
4
|
-
import { Socket } from "net";
|
|
5
|
-
import { describe, it, expect, beforeEach, jest } from "@jest/globals";
|
|
6
|
-
|
|
7
|
-
// Mock ServerResponse
|
|
8
|
-
function mockRes(): ServerResponse {
|
|
9
|
-
const socket = new Socket();
|
|
10
|
-
const req = new IncomingMessage(socket);
|
|
11
|
-
const res = new ServerResponse(req);
|
|
12
|
-
|
|
13
|
-
jest.spyOn(res, "writeHead");
|
|
14
|
-
jest.spyOn(res, "end");
|
|
15
|
-
|
|
16
|
-
return res;
|
|
17
|
-
}
|
|
18
|
-
|
|
19
|
-
describe("ApiController", () => {
|
|
20
|
-
let storage: StorageManager;
|
|
21
|
-
|
|
22
|
-
beforeEach(() => {
|
|
23
|
-
// On utilise un vrai StorageManager en mémoire
|
|
24
|
-
storage = new StorageManager();
|
|
25
|
-
});
|
|
26
|
-
|
|
27
|
-
it("listEvents renvoie la liste des events (vide par défaut)", async () => {
|
|
28
|
-
const res = mockRes();
|
|
29
|
-
const url = new URL("http://localhost/events?page=2&type=http");
|
|
30
|
-
|
|
31
|
-
await ApiController.listEvents(url, res, storage);
|
|
32
|
-
|
|
33
|
-
expect(storage.listEvents).toBeDefined();
|
|
34
|
-
|
|
35
|
-
expect(res.writeHead).toHaveBeenCalledWith(200, {
|
|
36
|
-
"Content-Type": "application/json",
|
|
37
|
-
});
|
|
38
|
-
|
|
39
|
-
const body = (res.end as jest.Mock).mock.calls[0][0] as string;
|
|
40
|
-
const payload = JSON.parse(body);
|
|
41
|
-
expect(payload.success).toBe(true);
|
|
42
|
-
expect(payload.events).toEqual([]);
|
|
43
|
-
});
|
|
44
|
-
|
|
45
|
-
it("getEvent renvoie 404 si event absent", async () => {
|
|
46
|
-
const res = mockRes();
|
|
47
|
-
|
|
48
|
-
await ApiController.getEvent("unknown", res, storage);
|
|
49
|
-
|
|
50
|
-
expect(res.writeHead).toHaveBeenCalledWith(404, {
|
|
51
|
-
"Content-Type": "application/json",
|
|
52
|
-
});
|
|
53
|
-
|
|
54
|
-
const body = (res.end as jest.Mock).mock.calls[0][0] as string;
|
|
55
|
-
const payload = JSON.parse(body);
|
|
56
|
-
expect(payload.success).toBe(false);
|
|
57
|
-
});
|
|
58
|
-
|
|
59
|
-
it("getEvent renvoie l'event si présent", async () => {
|
|
60
|
-
// On mocke juste getEvent pour ce test
|
|
61
|
-
jest.spyOn(storage, "getEvent").mockResolvedValue({ id: "abc" } as any);
|
|
62
|
-
|
|
63
|
-
const res = mockRes();
|
|
64
|
-
|
|
65
|
-
await ApiController.getEvent("abc", res, storage);
|
|
66
|
-
|
|
67
|
-
expect(res.writeHead).toHaveBeenCalledWith(200, {
|
|
68
|
-
"Content-Type": "application/json",
|
|
69
|
-
});
|
|
70
|
-
|
|
71
|
-
const body = (res.end as jest.Mock).mock.calls[0][0] as string;
|
|
72
|
-
const payload = JSON.parse(body);
|
|
73
|
-
expect(payload.success).toBe(true);
|
|
74
|
-
expect(payload.event.id).toBe("abc");
|
|
75
|
-
});
|
|
76
|
-
|
|
77
|
-
it("deleteAll supprime tous les events", async () => {
|
|
78
|
-
const res = mockRes();
|
|
79
|
-
|
|
80
|
-
await ApiController.deleteAll(res, storage);
|
|
81
|
-
|
|
82
|
-
const body = (res.end as jest.Mock).mock.calls[0][0] as string;
|
|
83
|
-
const payload = JSON.parse(body);
|
|
84
|
-
expect(payload.success).toBe(true);
|
|
85
|
-
});
|
|
86
|
-
|
|
87
|
-
it("deleteOne supprime un event", async () => {
|
|
88
|
-
const res = mockRes();
|
|
89
|
-
|
|
90
|
-
await storage.save({
|
|
91
|
-
id: "xyz",
|
|
92
|
-
type: "http",
|
|
93
|
-
timestamp: Date.now(),
|
|
94
|
-
sourceIp: "127.0.0.1",
|
|
95
|
-
request: { method: "GET", path: "/", headers: {}, query: {}, body: "" },
|
|
96
|
-
});
|
|
97
|
-
|
|
98
|
-
await ApiController.deleteOne("xyz", res, storage);
|
|
99
|
-
|
|
100
|
-
const raw = String((res.end as jest.Mock).mock.calls[0][0]);
|
|
101
|
-
const payload = JSON.parse(raw);
|
|
102
|
-
|
|
103
|
-
expect(payload.success).toBe(true);
|
|
104
|
-
});
|
|
105
|
-
|
|
106
|
-
it("stats renvoie les stats", async () => {
|
|
107
|
-
const res = mockRes();
|
|
108
|
-
|
|
109
|
-
await ApiController.stats(res, storage);
|
|
110
|
-
|
|
111
|
-
const body = (res.end as jest.Mock).mock.calls[0][0] as string;
|
|
112
|
-
const payload = JSON.parse(body);
|
|
113
|
-
expect(payload.success).toBe(true);
|
|
114
|
-
expect(payload.stats).toEqual({ total: 0, byType: {} });
|
|
115
|
-
});
|
|
116
|
-
});
|
|
1
|
+
import { ApiController } from "../api.controllers";
|
|
2
|
+
import { StorageManager } from "../../../core/storage";
|
|
3
|
+
import { ServerResponse, IncomingMessage } from "http";
|
|
4
|
+
import { Socket } from "net";
|
|
5
|
+
import { describe, it, expect, beforeEach, jest } from "@jest/globals";
|
|
6
|
+
|
|
7
|
+
// Mock ServerResponse
|
|
8
|
+
function mockRes(): ServerResponse {
|
|
9
|
+
const socket = new Socket();
|
|
10
|
+
const req = new IncomingMessage(socket);
|
|
11
|
+
const res = new ServerResponse(req);
|
|
12
|
+
|
|
13
|
+
jest.spyOn(res, "writeHead");
|
|
14
|
+
jest.spyOn(res, "end");
|
|
15
|
+
|
|
16
|
+
return res;
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
describe("ApiController", () => {
|
|
20
|
+
let storage: StorageManager;
|
|
21
|
+
|
|
22
|
+
beforeEach(() => {
|
|
23
|
+
// On utilise un vrai StorageManager en mémoire
|
|
24
|
+
storage = new StorageManager();
|
|
25
|
+
});
|
|
26
|
+
|
|
27
|
+
it("listEvents renvoie la liste des events (vide par défaut)", async () => {
|
|
28
|
+
const res = mockRes();
|
|
29
|
+
const url = new URL("http://localhost/events?page=2&type=http");
|
|
30
|
+
|
|
31
|
+
await ApiController.listEvents(url, res, storage);
|
|
32
|
+
|
|
33
|
+
expect(storage.listEvents).toBeDefined();
|
|
34
|
+
|
|
35
|
+
expect(res.writeHead).toHaveBeenCalledWith(200, {
|
|
36
|
+
"Content-Type": "application/json",
|
|
37
|
+
});
|
|
38
|
+
|
|
39
|
+
const body = (res.end as jest.Mock).mock.calls[0][0] as string;
|
|
40
|
+
const payload = JSON.parse(body);
|
|
41
|
+
expect(payload.success).toBe(true);
|
|
42
|
+
expect(payload.events).toEqual([]);
|
|
43
|
+
});
|
|
44
|
+
|
|
45
|
+
it("getEvent renvoie 404 si event absent", async () => {
|
|
46
|
+
const res = mockRes();
|
|
47
|
+
|
|
48
|
+
await ApiController.getEvent("unknown", res, storage);
|
|
49
|
+
|
|
50
|
+
expect(res.writeHead).toHaveBeenCalledWith(404, {
|
|
51
|
+
"Content-Type": "application/json",
|
|
52
|
+
});
|
|
53
|
+
|
|
54
|
+
const body = (res.end as jest.Mock).mock.calls[0][0] as string;
|
|
55
|
+
const payload = JSON.parse(body);
|
|
56
|
+
expect(payload.success).toBe(false);
|
|
57
|
+
});
|
|
58
|
+
|
|
59
|
+
it("getEvent renvoie l'event si présent", async () => {
|
|
60
|
+
// On mocke juste getEvent pour ce test
|
|
61
|
+
jest.spyOn(storage, "getEvent").mockResolvedValue({ id: "abc" } as any);
|
|
62
|
+
|
|
63
|
+
const res = mockRes();
|
|
64
|
+
|
|
65
|
+
await ApiController.getEvent("abc", res, storage);
|
|
66
|
+
|
|
67
|
+
expect(res.writeHead).toHaveBeenCalledWith(200, {
|
|
68
|
+
"Content-Type": "application/json",
|
|
69
|
+
});
|
|
70
|
+
|
|
71
|
+
const body = (res.end as jest.Mock).mock.calls[0][0] as string;
|
|
72
|
+
const payload = JSON.parse(body);
|
|
73
|
+
expect(payload.success).toBe(true);
|
|
74
|
+
expect(payload.event.id).toBe("abc");
|
|
75
|
+
});
|
|
76
|
+
|
|
77
|
+
it("deleteAll supprime tous les events", async () => {
|
|
78
|
+
const res = mockRes();
|
|
79
|
+
|
|
80
|
+
await ApiController.deleteAll(res, storage);
|
|
81
|
+
|
|
82
|
+
const body = (res.end as jest.Mock).mock.calls[0][0] as string;
|
|
83
|
+
const payload = JSON.parse(body);
|
|
84
|
+
expect(payload.success).toBe(true);
|
|
85
|
+
});
|
|
86
|
+
|
|
87
|
+
it("deleteOne supprime un event", async () => {
|
|
88
|
+
const res = mockRes();
|
|
89
|
+
|
|
90
|
+
await storage.save({
|
|
91
|
+
id: "xyz",
|
|
92
|
+
type: "http",
|
|
93
|
+
timestamp: Date.now(),
|
|
94
|
+
sourceIp: "127.0.0.1",
|
|
95
|
+
request: { method: "GET", path: "/", headers: {}, query: {}, body: "" },
|
|
96
|
+
});
|
|
97
|
+
|
|
98
|
+
await ApiController.deleteOne("xyz", res, storage);
|
|
99
|
+
|
|
100
|
+
const raw = String((res.end as jest.Mock).mock.calls[0][0]);
|
|
101
|
+
const payload = JSON.parse(raw);
|
|
102
|
+
|
|
103
|
+
expect(payload.success).toBe(true);
|
|
104
|
+
});
|
|
105
|
+
|
|
106
|
+
it("stats renvoie les stats", async () => {
|
|
107
|
+
const res = mockRes();
|
|
108
|
+
|
|
109
|
+
await ApiController.stats(res, storage);
|
|
110
|
+
|
|
111
|
+
const body = (res.end as jest.Mock).mock.calls[0][0] as string;
|
|
112
|
+
const payload = JSON.parse(body);
|
|
113
|
+
expect(payload.success).toBe(true);
|
|
114
|
+
expect(payload.stats).toEqual({ total: 0, byType: {} });
|
|
115
|
+
});
|
|
116
|
+
});
|
|
@@ -1,46 +1,46 @@
|
|
|
1
|
-
import { ApiExtractor } from "../api.extractor";
|
|
2
|
-
import { IncomingMessage } from "http";
|
|
3
|
-
import { Socket } from "net";
|
|
4
|
-
import { describe, it, expect } from "@jest/globals";
|
|
5
|
-
|
|
6
|
-
function mockReq(url: string, method = "GET", body?: any): IncomingMessage {
|
|
7
|
-
const socket = new Socket();
|
|
8
|
-
const req = new IncomingMessage(socket);
|
|
9
|
-
|
|
10
|
-
req.url = url;
|
|
11
|
-
req.method = method;
|
|
12
|
-
req.headers = { host: "localhost" };
|
|
13
|
-
|
|
14
|
-
if (body !== undefined) {
|
|
15
|
-
const json = JSON.stringify(body);
|
|
16
|
-
process.nextTick(() => {
|
|
17
|
-
req.emit("data", json);
|
|
18
|
-
req.emit("end");
|
|
19
|
-
});
|
|
20
|
-
} else {
|
|
21
|
-
process.nextTick(() => req.emit("end"));
|
|
22
|
-
}
|
|
23
|
-
|
|
24
|
-
return req;
|
|
25
|
-
}
|
|
26
|
-
|
|
27
|
-
describe("ApiExtractor", () => {
|
|
28
|
-
it("extrait un RawEvent GET", async () => {
|
|
29
|
-
const req = mockReq("/events?page=2&type=http");
|
|
30
|
-
|
|
31
|
-
const event = await ApiExtractor.extract(req);
|
|
32
|
-
|
|
33
|
-
expect(event.method).toBe("GET");
|
|
34
|
-
expect(event.path).toBe("/events");
|
|
35
|
-
expect(event.query).toEqual({ page: "2", type: "http" });
|
|
36
|
-
});
|
|
37
|
-
|
|
38
|
-
it("extrait un RawEvent POST avec body JSON", async () => {
|
|
39
|
-
const req = mockReq("/events", "POST", { hello: "world" });
|
|
40
|
-
|
|
41
|
-
const event = await ApiExtractor.extract(req);
|
|
42
|
-
|
|
43
|
-
expect(event.method).toBe("POST");
|
|
44
|
-
expect(event.body).toEqual({ hello: "world" });
|
|
45
|
-
});
|
|
46
|
-
});
|
|
1
|
+
import { ApiExtractor } from "../api.extractor";
|
|
2
|
+
import { IncomingMessage } from "http";
|
|
3
|
+
import { Socket } from "net";
|
|
4
|
+
import { describe, it, expect } from "@jest/globals";
|
|
5
|
+
|
|
6
|
+
function mockReq(url: string, method = "GET", body?: any): IncomingMessage {
|
|
7
|
+
const socket = new Socket();
|
|
8
|
+
const req = new IncomingMessage(socket);
|
|
9
|
+
|
|
10
|
+
req.url = url;
|
|
11
|
+
req.method = method;
|
|
12
|
+
req.headers = { host: "localhost" };
|
|
13
|
+
|
|
14
|
+
if (body !== undefined) {
|
|
15
|
+
const json = JSON.stringify(body);
|
|
16
|
+
process.nextTick(() => {
|
|
17
|
+
req.emit("data", json);
|
|
18
|
+
req.emit("end");
|
|
19
|
+
});
|
|
20
|
+
} else {
|
|
21
|
+
process.nextTick(() => req.emit("end"));
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
return req;
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
describe("ApiExtractor", () => {
|
|
28
|
+
it("extrait un RawEvent GET", async () => {
|
|
29
|
+
const req = mockReq("/events?page=2&type=http");
|
|
30
|
+
|
|
31
|
+
const event = await ApiExtractor.extract(req);
|
|
32
|
+
|
|
33
|
+
expect(event.method).toBe("GET");
|
|
34
|
+
expect(event.path).toBe("/events");
|
|
35
|
+
expect(event.query).toEqual({ page: "2", type: "http" });
|
|
36
|
+
});
|
|
37
|
+
|
|
38
|
+
it("extrait un RawEvent POST avec body JSON", async () => {
|
|
39
|
+
const req = mockReq("/events", "POST", { hello: "world" });
|
|
40
|
+
|
|
41
|
+
const event = await ApiExtractor.extract(req);
|
|
42
|
+
|
|
43
|
+
expect(event.method).toBe("POST");
|
|
44
|
+
expect(event.body).toEqual({ hello: "world" });
|
|
45
|
+
});
|
|
46
|
+
});
|