@timmeck/brain-core 1.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (60) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +144 -0
  3. package/dist/api/server.d.ts +29 -0
  4. package/dist/api/server.js +183 -0
  5. package/dist/api/server.js.map +1 -0
  6. package/dist/cli/colors.d.ts +47 -0
  7. package/dist/cli/colors.js +93 -0
  8. package/dist/cli/colors.js.map +1 -0
  9. package/dist/db/connection.d.ts +2 -0
  10. package/dist/db/connection.js +19 -0
  11. package/dist/db/connection.js.map +1 -0
  12. package/dist/index.d.ts +18 -0
  13. package/dist/index.js +19 -0
  14. package/dist/index.js.map +1 -0
  15. package/dist/ipc/client.d.ts +16 -0
  16. package/dist/ipc/client.js +100 -0
  17. package/dist/ipc/client.js.map +1 -0
  18. package/dist/ipc/protocol.d.ts +8 -0
  19. package/dist/ipc/protocol.js +29 -0
  20. package/dist/ipc/protocol.js.map +1 -0
  21. package/dist/ipc/server.d.ts +22 -0
  22. package/dist/ipc/server.js +156 -0
  23. package/dist/ipc/server.js.map +1 -0
  24. package/dist/mcp/http-server.d.ts +23 -0
  25. package/dist/mcp/http-server.js +126 -0
  26. package/dist/mcp/http-server.js.map +1 -0
  27. package/dist/mcp/server.d.ts +15 -0
  28. package/dist/mcp/server.js +66 -0
  29. package/dist/mcp/server.js.map +1 -0
  30. package/dist/types/ipc.types.d.ts +11 -0
  31. package/dist/types/ipc.types.js +2 -0
  32. package/dist/types/ipc.types.js.map +1 -0
  33. package/dist/utils/events.d.ts +18 -0
  34. package/dist/utils/events.js +27 -0
  35. package/dist/utils/events.js.map +1 -0
  36. package/dist/utils/hash.d.ts +1 -0
  37. package/dist/utils/hash.js +5 -0
  38. package/dist/utils/hash.js.map +1 -0
  39. package/dist/utils/logger.d.ts +16 -0
  40. package/dist/utils/logger.js +43 -0
  41. package/dist/utils/logger.js.map +1 -0
  42. package/dist/utils/paths.d.ts +8 -0
  43. package/dist/utils/paths.js +23 -0
  44. package/dist/utils/paths.js.map +1 -0
  45. package/package.json +59 -0
  46. package/src/api/server.ts +210 -0
  47. package/src/cli/colors.ts +105 -0
  48. package/src/db/connection.ts +22 -0
  49. package/src/index.ts +31 -0
  50. package/src/ipc/client.ts +117 -0
  51. package/src/ipc/protocol.ts +35 -0
  52. package/src/ipc/server.ts +170 -0
  53. package/src/mcp/http-server.ts +148 -0
  54. package/src/mcp/server.ts +84 -0
  55. package/src/types/ipc.types.ts +8 -0
  56. package/src/utils/events.ts +30 -0
  57. package/src/utils/hash.ts +5 -0
  58. package/src/utils/logger.ts +67 -0
  59. package/src/utils/paths.ts +24 -0
  60. package/tsconfig.json +18 -0
@@ -0,0 +1,35 @@
1
+ import { Buffer } from 'node:buffer';
2
+ import type { IpcMessage } from '../types/ipc.types.js';
3
+
4
+ export function encodeMessage(msg: IpcMessage): Buffer {
5
+ const json = JSON.stringify(msg);
6
+ const payload = Buffer.from(json, 'utf8');
7
+ const frame = Buffer.alloc(4 + payload.length);
8
+ frame.writeUInt32BE(payload.length, 0);
9
+ payload.copy(frame, 4);
10
+ return frame;
11
+ }
12
+
13
+ export class MessageDecoder {
14
+ private buffer = Buffer.alloc(0);
15
+
16
+ feed(chunk: Buffer): IpcMessage[] {
17
+ this.buffer = Buffer.concat([this.buffer, chunk]);
18
+ const messages: IpcMessage[] = [];
19
+
20
+ while (this.buffer.length >= 4) {
21
+ const length = this.buffer.readUInt32BE(0);
22
+ if (this.buffer.length < 4 + length) break;
23
+
24
+ const json = this.buffer.subarray(4, 4 + length).toString('utf8');
25
+ this.buffer = this.buffer.subarray(4 + length);
26
+ messages.push(JSON.parse(json) as IpcMessage);
27
+ }
28
+
29
+ return messages;
30
+ }
31
+
32
+ reset(): void {
33
+ this.buffer = Buffer.alloc(0);
34
+ }
35
+ }
@@ -0,0 +1,170 @@
1
+ import net from 'node:net';
2
+ import fs from 'node:fs';
3
+ import { randomUUID } from 'node:crypto';
4
+ import { getLogger } from '../utils/logger.js';
5
+ import type { IpcMessage } from '../types/ipc.types.js';
6
+ import { encodeMessage, MessageDecoder } from './protocol.js';
7
+
8
+ export interface IpcRouter {
9
+ handle(method: string, params: unknown): unknown;
10
+ listMethods(): string[];
11
+ }
12
+
13
+ export class IpcServer {
14
+ private server: net.Server | null = null;
15
+ private clients = new Map<string, net.Socket>();
16
+ private logger = getLogger();
17
+
18
+ constructor(
19
+ private router: IpcRouter,
20
+ private pipeName: string,
21
+ private daemonName: string = 'brain',
22
+ ) {}
23
+
24
+ start(): void {
25
+ this.createServer();
26
+ this.listen();
27
+ }
28
+
29
+ private createServer(): void {
30
+ this.server = net.createServer((socket) => {
31
+ const clientId = randomUUID();
32
+ this.clients.set(clientId, socket);
33
+ const decoder = new MessageDecoder();
34
+
35
+ this.logger.info(`IPC client connected: ${clientId}`);
36
+
37
+ socket.on('data', (chunk) => {
38
+ const messages = decoder.feed(chunk);
39
+ for (const msg of messages) {
40
+ this.handleMessage(clientId, msg, socket);
41
+ }
42
+ });
43
+
44
+ socket.on('close', () => {
45
+ this.logger.info(`IPC client disconnected: ${clientId}`);
46
+ this.clients.delete(clientId);
47
+ });
48
+
49
+ socket.on('error', (err) => {
50
+ this.logger.error(`IPC client ${clientId} error:`, err);
51
+ this.clients.delete(clientId);
52
+ });
53
+ });
54
+ }
55
+
56
+ private listen(retried = false): void {
57
+ if (!this.server) return;
58
+
59
+ this.server.on('error', (err: NodeJS.ErrnoException) => {
60
+ if (err.code === 'EADDRINUSE' && !retried) {
61
+ this.logger.warn(`IPC pipe in use, attempting to recover stale pipe: ${this.pipeName}`);
62
+ this.recoverStalePipe();
63
+ } else {
64
+ this.logger.error('IPC server error:', err);
65
+ }
66
+ });
67
+
68
+ this.server.listen(this.pipeName, () => {
69
+ this.logger.info(`IPC server listening on ${this.pipeName}`);
70
+ });
71
+ }
72
+
73
+ private recoverStalePipe(): void {
74
+ const probe = net.createConnection(this.pipeName);
75
+
76
+ probe.on('connect', () => {
77
+ probe.destroy();
78
+ this.logger.error(`IPC pipe is held by another running daemon. Stop it first with: ${this.daemonName} stop`);
79
+ });
80
+
81
+ probe.on('error', () => {
82
+ probe.destroy();
83
+ this.logger.info('Stale IPC pipe detected, reclaiming...');
84
+
85
+ if (process.platform !== 'win32') {
86
+ try { fs.unlinkSync(this.pipeName); } catch { /* ignore */ }
87
+ }
88
+
89
+ this.createServer();
90
+ this.server!.on('error', (err) => {
91
+ this.logger.error('IPC server error after recovery:', err);
92
+ });
93
+ this.server!.listen(this.pipeName, () => {
94
+ this.logger.info(`IPC server recovered and listening on ${this.pipeName}`);
95
+ });
96
+ });
97
+
98
+ probe.setTimeout(2000, () => {
99
+ probe.destroy();
100
+ this.logger.warn('IPC pipe probe timed out, treating as stale');
101
+ if (process.platform !== 'win32') {
102
+ try { fs.unlinkSync(this.pipeName); } catch { /* ignore */ }
103
+ }
104
+ this.createServer();
105
+ this.server!.on('error', (err) => {
106
+ this.logger.error('IPC server error after timeout recovery:', err);
107
+ });
108
+ this.server!.listen(this.pipeName, () => {
109
+ this.logger.info(`IPC server recovered (timeout) and listening on ${this.pipeName}`);
110
+ });
111
+ });
112
+ }
113
+
114
+ private handleMessage(_clientId: string, msg: IpcMessage, socket: net.Socket): void {
115
+ if (msg.type !== 'request' || !msg.method) return;
116
+
117
+ try {
118
+ const result = this.router.handle(msg.method, msg.params);
119
+ const response: IpcMessage = {
120
+ id: msg.id,
121
+ type: 'response',
122
+ result,
123
+ };
124
+ socket.write(encodeMessage(response));
125
+ } catch (err) {
126
+ const response: IpcMessage = {
127
+ id: msg.id,
128
+ type: 'response',
129
+ error: { code: -1, message: err instanceof Error ? err.message : String(err) },
130
+ };
131
+ socket.write(encodeMessage(response));
132
+ }
133
+ }
134
+
135
+ notify(clientId: string | null, notification: Omit<IpcMessage, 'id' | 'type'>): void {
136
+ const msg: IpcMessage = {
137
+ id: randomUUID(),
138
+ type: 'notification',
139
+ ...notification,
140
+ };
141
+ const encoded = encodeMessage(msg);
142
+
143
+ if (clientId) {
144
+ const socket = this.clients.get(clientId);
145
+ if (socket && !socket.destroyed) {
146
+ socket.write(encoded);
147
+ }
148
+ } else {
149
+ for (const socket of this.clients.values()) {
150
+ if (!socket.destroyed) {
151
+ socket.write(encoded);
152
+ }
153
+ }
154
+ }
155
+ }
156
+
157
+ getClientCount(): number {
158
+ return this.clients.size;
159
+ }
160
+
161
+ stop(): void {
162
+ for (const socket of this.clients.values()) {
163
+ socket.destroy();
164
+ }
165
+ this.clients.clear();
166
+ this.server?.close();
167
+ this.server = null;
168
+ this.logger.info('IPC server stopped');
169
+ }
170
+ }
@@ -0,0 +1,148 @@
1
+ import http from 'node:http';
2
+ import { randomUUID } from 'node:crypto';
3
+ import { McpServer } from '@modelcontextprotocol/sdk/server/mcp.js';
4
+ import { SSEServerTransport } from '@modelcontextprotocol/sdk/server/sse.js';
5
+ import { getLogger } from '../utils/logger.js';
6
+ import type { IpcRouter } from '../ipc/server.js';
7
+
8
+ export interface McpHttpServerOptions {
9
+ /** MCP server name */
10
+ name: string;
11
+ /** MCP server version */
12
+ version: string;
13
+ }
14
+
15
+ export class McpHttpServer {
16
+ private server: http.Server | null = null;
17
+ private transports = new Map<string, SSEServerTransport>();
18
+ private logger = getLogger();
19
+
20
+ constructor(
21
+ private port: number,
22
+ private router: IpcRouter,
23
+ private meta: McpHttpServerOptions,
24
+ private registerToolsDirect: (server: McpServer, router: IpcRouter) => void,
25
+ ) {}
26
+
27
+ start(): void {
28
+ this.server = http.createServer((req, res) => {
29
+ res.setHeader('Access-Control-Allow-Origin', '*');
30
+ res.setHeader('Access-Control-Allow-Methods', 'GET, POST, DELETE, OPTIONS');
31
+ res.setHeader('Access-Control-Allow-Headers', 'Content-Type');
32
+
33
+ if (req.method === 'OPTIONS') {
34
+ res.writeHead(204);
35
+ res.end();
36
+ return;
37
+ }
38
+
39
+ const url = new URL(req.url ?? '/', `http://localhost:${this.port}`);
40
+
41
+ if (url.pathname === '/sse' && req.method === 'GET') {
42
+ this.handleSSE(res);
43
+ return;
44
+ }
45
+
46
+ if (url.pathname === '/messages' && req.method === 'POST') {
47
+ this.handleMessage(req, res, url);
48
+ return;
49
+ }
50
+
51
+ if (url.pathname === '/' && req.method === 'GET') {
52
+ res.writeHead(200, { 'Content-Type': 'application/json' });
53
+ res.end(JSON.stringify({
54
+ name: this.meta.name,
55
+ version: this.meta.version,
56
+ protocol: 'MCP',
57
+ transport: 'sse',
58
+ endpoints: {
59
+ sse: '/sse',
60
+ messages: '/messages',
61
+ },
62
+ clients: this.transports.size,
63
+ }));
64
+ return;
65
+ }
66
+
67
+ res.writeHead(404, { 'Content-Type': 'text/plain' });
68
+ res.end('Not Found');
69
+ });
70
+
71
+ this.server.listen(this.port, () => {
72
+ this.logger.info(`MCP HTTP server (SSE) started on http://localhost:${this.port}`);
73
+ });
74
+ }
75
+
76
+ stop(): void {
77
+ for (const transport of this.transports.values()) {
78
+ try { transport.close?.(); } catch { /* ignore */ }
79
+ }
80
+ this.transports.clear();
81
+ this.server?.close();
82
+ this.server = null;
83
+ this.logger.info('MCP HTTP server stopped');
84
+ }
85
+
86
+ getClientCount(): number {
87
+ return this.transports.size;
88
+ }
89
+
90
+ private async handleSSE(res: http.ServerResponse): Promise<void> {
91
+ try {
92
+ const transport = new SSEServerTransport('/messages', res);
93
+ const sessionId = transport.sessionId ?? randomUUID();
94
+ this.transports.set(sessionId, transport);
95
+
96
+ const server = new McpServer({
97
+ name: this.meta.name,
98
+ version: this.meta.version,
99
+ });
100
+
101
+ this.registerToolsDirect(server, this.router);
102
+
103
+ res.on('close', () => {
104
+ this.transports.delete(sessionId);
105
+ this.logger.debug(`MCP SSE client disconnected: ${sessionId}`);
106
+ });
107
+
108
+ await server.connect(transport);
109
+ this.logger.info(`MCP SSE client connected: ${sessionId}`);
110
+ } catch (err) {
111
+ this.logger.error('MCP SSE connection error:', err);
112
+ if (!res.headersSent) {
113
+ res.writeHead(500, { 'Content-Type': 'text/plain' });
114
+ res.end('Internal Server Error');
115
+ }
116
+ }
117
+ }
118
+
119
+ private async handleMessage(
120
+ req: http.IncomingMessage,
121
+ res: http.ServerResponse,
122
+ url: URL,
123
+ ): Promise<void> {
124
+ try {
125
+ const sessionId = url.searchParams.get('sessionId');
126
+ if (!sessionId) {
127
+ res.writeHead(400, { 'Content-Type': 'text/plain' });
128
+ res.end('Missing sessionId parameter');
129
+ return;
130
+ }
131
+
132
+ const transport = this.transports.get(sessionId);
133
+ if (!transport) {
134
+ res.writeHead(404, { 'Content-Type': 'text/plain' });
135
+ res.end('Session not found. Connect to /sse first.');
136
+ return;
137
+ }
138
+
139
+ await transport.handlePostMessage(req, res);
140
+ } catch (err) {
141
+ this.logger.error('MCP message error:', err);
142
+ if (!res.headersSent) {
143
+ res.writeHead(500, { 'Content-Type': 'text/plain' });
144
+ res.end('Internal Server Error');
145
+ }
146
+ }
147
+ }
148
+ }
@@ -0,0 +1,84 @@
1
+ import { McpServer } from '@modelcontextprotocol/sdk/server/mcp.js';
2
+ import { StdioServerTransport } from '@modelcontextprotocol/sdk/server/stdio.js';
3
+ import { spawn } from 'node:child_process';
4
+ import path from 'node:path';
5
+ import { IpcClient } from '../ipc/client.js';
6
+ import { getPipeName } from '../utils/paths.js';
7
+
8
+ export interface McpServerOptions {
9
+ /** MCP server name (e.g. 'brain', 'trading-brain') */
10
+ name: string;
11
+ /** MCP server version */
12
+ version: string;
13
+ /** Pipe name for IPC connection */
14
+ pipeName?: string;
15
+ /** Path to the CLI entry point for auto-starting the daemon */
16
+ entryPoint: string;
17
+ /** Register MCP tools on the server */
18
+ registerTools: (server: McpServer, ipc: IpcClient) => void;
19
+ }
20
+
21
+ function spawnDaemon(opts: McpServerOptions): void {
22
+ const child = spawn(process.execPath, [
23
+ '--import', 'tsx',
24
+ opts.entryPoint, 'daemon',
25
+ ], {
26
+ detached: true,
27
+ stdio: 'ignore',
28
+ cwd: path.resolve(path.dirname(opts.entryPoint), '..'),
29
+ });
30
+ child.unref();
31
+ process.stderr.write(`${opts.name}: Auto-started daemon (PID: ${child.pid})\n`);
32
+ }
33
+
34
+ async function connectWithRetry(ipc: IpcClient, retries: number, delayMs: number): Promise<void> {
35
+ for (let i = 0; i < retries; i++) {
36
+ try {
37
+ await ipc.connect();
38
+ return;
39
+ } catch {
40
+ if (i < retries - 1) {
41
+ await new Promise(r => setTimeout(r, delayMs));
42
+ }
43
+ }
44
+ }
45
+ throw new Error('Could not connect to daemon after retries');
46
+ }
47
+
48
+ export async function startMcpServer(opts: McpServerOptions): Promise<void> {
49
+ const server = new McpServer({
50
+ name: opts.name,
51
+ version: opts.version,
52
+ });
53
+
54
+ const pipeName = opts.pipeName ?? getPipeName(opts.name);
55
+ const ipc = new IpcClient(pipeName);
56
+
57
+ try {
58
+ await ipc.connect();
59
+ } catch {
60
+ process.stderr.write(`${opts.name}: Daemon not running, starting automatically...\n`);
61
+ spawnDaemon(opts);
62
+ try {
63
+ await connectWithRetry(ipc, 10, 500);
64
+ } catch {
65
+ process.stderr.write(`${opts.name}: Could not connect to daemon after auto-start. Check logs.\n`);
66
+ process.exit(1);
67
+ }
68
+ }
69
+
70
+ opts.registerTools(server, ipc);
71
+
72
+ const transport = new StdioServerTransport();
73
+ await server.connect(transport);
74
+
75
+ process.on('SIGINT', () => {
76
+ ipc.disconnect();
77
+ process.exit(0);
78
+ });
79
+
80
+ process.on('SIGTERM', () => {
81
+ ipc.disconnect();
82
+ process.exit(0);
83
+ });
84
+ }
@@ -0,0 +1,8 @@
1
+ export interface IpcMessage {
2
+ id: string;
3
+ type: 'request' | 'response' | 'notification';
4
+ method?: string;
5
+ params?: unknown;
6
+ result?: unknown;
7
+ error?: { code: number; message: string };
8
+ }
@@ -0,0 +1,30 @@
1
+ import { EventEmitter } from 'node:events';
2
+
3
+ /**
4
+ * Generic typed event bus. Each brain defines its own events interface:
5
+ *
6
+ * ```ts
7
+ * interface MyEvents {
8
+ * 'trade:recorded': { tradeId: number; win: boolean };
9
+ * 'rule:learned': { ruleId: number; pattern: string };
10
+ * }
11
+ * const bus = new TypedEventBus<MyEvents>();
12
+ * ```
13
+ */
14
+ export class TypedEventBus<T extends Record<string, unknown>> extends EventEmitter {
15
+ emit<K extends keyof T & string>(event: K, data: T[K]): boolean {
16
+ return super.emit(event, data);
17
+ }
18
+
19
+ on<K extends keyof T & string>(event: K, listener: (data: T[K]) => void): this {
20
+ return super.on(event, listener);
21
+ }
22
+
23
+ once<K extends keyof T & string>(event: K, listener: (data: T[K]) => void): this {
24
+ return super.once(event, listener);
25
+ }
26
+
27
+ off<K extends keyof T & string>(event: K, listener: (data: T[K]) => void): this {
28
+ return super.off(event, listener);
29
+ }
30
+ }
@@ -0,0 +1,5 @@
1
+ import { createHash } from 'node:crypto';
2
+
3
+ export function sha256(input: string): string {
4
+ return createHash('sha256').update(input).digest('hex');
5
+ }
@@ -0,0 +1,67 @@
1
+ import winston from 'winston';
2
+ import path from 'node:path';
3
+
4
+ const { combine, timestamp, printf, colorize } = winston.format;
5
+
6
+ const logFormat = printf(({ level, message, timestamp, ...meta }) => {
7
+ const metaStr = Object.keys(meta).length ? ` ${JSON.stringify(meta)}` : '';
8
+ return `${timestamp} [${level}]${metaStr} ${message}`;
9
+ });
10
+
11
+ let loggerInstance: winston.Logger | null = null;
12
+
13
+ export interface LoggerOptions {
14
+ level?: string;
15
+ file?: string;
16
+ maxSize?: number;
17
+ maxFiles?: number;
18
+ /** Environment variable name for log level override (e.g. 'BRAIN_LOG_LEVEL') */
19
+ envVar?: string;
20
+ /** Default log filename when no file is specified (e.g. 'brain.log') */
21
+ defaultFilename?: string;
22
+ /** Data directory to place the log file in */
23
+ dataDir?: string;
24
+ }
25
+
26
+ export function createLogger(opts?: LoggerOptions): winston.Logger {
27
+ if (loggerInstance) return loggerInstance;
28
+
29
+ const envVar = opts?.envVar ?? 'BRAIN_LOG_LEVEL';
30
+ const defaultFilename = opts?.defaultFilename ?? 'brain.log';
31
+
32
+ const level = opts?.level ?? process.env[envVar] ?? 'info';
33
+ const logFile = opts?.file ?? (opts?.dataDir ? path.join(opts.dataDir, defaultFilename) : defaultFilename);
34
+ const maxSize = opts?.maxSize ?? 10 * 1024 * 1024; // 10MB
35
+ const maxFiles = opts?.maxFiles ?? 3;
36
+
37
+ const transports: winston.transport[] = [
38
+ new winston.transports.File({
39
+ filename: logFile,
40
+ maxsize: maxSize,
41
+ maxFiles,
42
+ format: combine(timestamp(), logFormat),
43
+ }),
44
+ ];
45
+
46
+ if (process.env['NODE_ENV'] !== 'production') {
47
+ transports.push(
48
+ new winston.transports.Console({
49
+ format: combine(colorize(), timestamp(), logFormat),
50
+ })
51
+ );
52
+ }
53
+
54
+ loggerInstance = winston.createLogger({ level, transports });
55
+ return loggerInstance;
56
+ }
57
+
58
+ export function getLogger(): winston.Logger {
59
+ if (!loggerInstance) {
60
+ return createLogger();
61
+ }
62
+ return loggerInstance;
63
+ }
64
+
65
+ export function resetLogger(): void {
66
+ loggerInstance = null;
67
+ }
@@ -0,0 +1,24 @@
1
+ import path from 'node:path';
2
+ import os from 'node:os';
3
+
4
+ export function normalizePath(filePath: string): string {
5
+ return filePath.replace(/\\/g, '/');
6
+ }
7
+
8
+ /**
9
+ * Get the data directory for a brain instance.
10
+ * @param envVar - Environment variable name (e.g. 'BRAIN_DATA_DIR')
11
+ * @param defaultDir - Default directory name in home (e.g. '.brain')
12
+ */
13
+ export function getDataDir(envVar: string = 'BRAIN_DATA_DIR', defaultDir: string = '.brain'): string {
14
+ const envDirValue = process.env[envVar];
15
+ if (envDirValue) return path.resolve(envDirValue);
16
+ return path.join(os.homedir(), defaultDir);
17
+ }
18
+
19
+ export function getPipeName(name: string = 'brain'): string {
20
+ if (process.platform === 'win32') {
21
+ return `\\\\.\\pipe\\${name}`;
22
+ }
23
+ return path.join(os.tmpdir(), `${name}.sock`);
24
+ }
package/tsconfig.json ADDED
@@ -0,0 +1,18 @@
1
+ {
2
+ "compilerOptions": {
3
+ "target": "ES2022",
4
+ "module": "NodeNext",
5
+ "moduleResolution": "NodeNext",
6
+ "strict": true,
7
+ "esModuleInterop": true,
8
+ "skipLibCheck": true,
9
+ "forceConsistentCasingInFileNames": true,
10
+ "outDir": "dist",
11
+ "rootDir": "src",
12
+ "declaration": true,
13
+ "sourceMap": true,
14
+ "resolveJsonModule": true
15
+ },
16
+ "include": ["src/**/*"],
17
+ "exclude": ["node_modules", "dist"]
18
+ }