dolphin-server-modules 1.1.1 → 1.2.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.
@@ -19,6 +19,7 @@
19
19
  - [११. स्केलिङ र पर्फर्मेन्स (Scaling & Performance)](#११-स्केलिङ-र-पर्फर्मेन्स-scaling--performance)
20
20
  - [१२. टेस्टिङ र डेभप्स (Testing & DevOps)](#१२-टेस्टिङ-र-डेभप्स-testing--devops)
21
21
  - [१३. भविष्य र योगदान (Future Roadmap)](#१३-भविष्य-र-योगदान-future-roadmap)
22
+ - [१४. रियलटाइम र IoT मास्टरक्लास (Realtime & IoT Masterclass) [NEW]](#१४-रियलटाइम-र-iot-मास्टरक्लास-realtime--iot-masterclass-new)
22
23
 
23
24
  ---
24
25
 
@@ -508,7 +509,8 @@ CMD ["npm", "start"]
508
509
 
509
510
  Dolphin अझै विकसित हुँदैछ। हाम्रो आगामी योजनाहरू:
510
511
  - **Dolphin CLI**: एउटा कमान्डले प्रोजेक्ट सेटअप गर्ने।
511
- - **WebSocket Support**: रियल-टाइम च्याटको लागि।
512
+ - **Dolphin CLI**: एउटा कमान्डले प्रोजेक्ट सेटअप गर्ने।
513
+ - **Realtime & IoT Integration**: उच्च क्षमताको डाटा इन्जेसनको लागि। [DONE]
512
514
  - **Native SQL Adapters**: PostgreSQL र MySQL का लागि विशेष एडाप्टरहरू।
513
515
 
514
516
  ### योगदान कसरी गर्ने?
@@ -516,6 +518,50 @@ Dolphin अझै विकसित हुँदैछ। हाम्रो
516
518
 
517
519
  ---
518
520
 
521
+ ## १४. रियलटाइम र IoT मास्टरक्लास (Realtime & IoT Masterclass) [NEW]
522
+
523
+ आधुनिक एप्लिकेसनहरूलाई केवल "Request-Response" मात्र पुग्दैन। तिनीहरूलाई "Real-time" डाटा चाहिन्छ। Dolphin को Realtime मोड्युलले यसलाई सम्भव बनाउँछ।
524
+
525
+ ### १४.१ रियलटाइम कोर (Realtime Core) के हो?
526
+ यो एउटा इन्टरनल "Event Bus" हो जसले मेसेजहरूलाई एक ठाउँबाट अर्को ठाउँमा तुरुन्तै पुर्‍याउँछ। यसले MQTT जस्तो "Topic-based" सिस्टम प्रयोग गर्छ।
527
+
528
+ ### १४.२ मुख्य अवधारणाहरू (Key Concepts)
529
+ १. **TopicTrie**: मेसेजहरू कुन टपिकमा जाने भनेर छिटो पत्ता लगाउने इन्जिन।
530
+ २. **Binary Codec**: डाटालाई सानो बनाउन बाइनरी फर्म्याटमा लैजाने सफ्टवेयर।
531
+ ३. **Plugins**: विभिन्न प्रोटोकलहरू (HL7, Modbus) सपोर्ट गर्न।
532
+
533
+ ### १४.३ कोड उदाहरण: बेसिक पब-सब (Pub/Sub)
534
+ ```typescript
535
+ import { RealtimeCore } from 'dolphin-server-modules/realtime';
536
+
537
+ const rt = new RealtimeCore({
538
+ maxMessageSize: 512 * 1024 // ५१२ KB म्याक्स साइज
539
+ });
540
+
541
+ // १. सब्सक्राइब (Subscribe) गर्ने
542
+ rt.subscribe('sensors/temperature/+', (ctx) => {
543
+ console.log(`नयाँ डाटा आयो: ${ctx.payload.value}°C`);
544
+ });
545
+
546
+ // २. पब्लिस (Publish) गर्ने
547
+ rt.publish('sensors/temperature/room1', { value: 22.5 });
548
+ ```
549
+
550
+ ### १४.४ वाइल्डकार्डको शक्ति (Power of Wildcards)
551
+ - `sensors/+`: `sensors/temp` र `sensors/hum` दुवै म्याच गर्छ।
552
+ - `sensors/#`: `sensors/a/b/c` जति पनि गहिराइ (depth) सम्म म्याच गर्छ।
553
+
554
+ ### १४.५ रेडिस स्केलिङ (Redis Scaling)
555
+ यदि तपाईँको धेरै वटा सर्भरहरू छन् भने, तिनीहरूलाई रेडिस मार्फत जोड्न सक्नुहुन्छ:
556
+ ```typescript
557
+ const rt = new RealtimeCore({
558
+ redisUrl: 'redis://localhost:6379'
559
+ });
560
+ ```
561
+ यसो गर्दा एउटा सर्भरबाट पठाएको मेसेज अर्को सर्भरमा बस्ने युजरले पनि तुरुन्तै पाउँछ।
562
+
563
+ ---
564
+
519
565
  ## निष्कर्ष (Conclusion)
520
566
 
521
567
  बधाई छ! तपाईँले Dolphin Framework को **Master Guide** को अन्त्य सम्म पढ्नुभयो। अब तपाईँ कुनै पनि जटिल ब्याकइन्ड सिस्टम Dolphin प्रयोग गरेर बनाउन पूर्ण सक्षम हुनुहुन्छ।
package/README.md CHANGED
@@ -115,13 +115,19 @@ Seamlessly switch between databases with the Adapter pattern.
115
115
  ### ✅ 5. Zod-Powered Validation (`/middleware/zod`)
116
116
  Validate payloads and params with 100% type inference.
117
117
 
118
+ ### 🌐 6. Realtime & IoT Core (`/realtime`) [NEW]
119
+ High-performance pub/sub with MQTT-style matching.
120
+ - **TopicTrie**: $O(1)$ pattern matching for `+` and `#`.
121
+ - **Binary Codec**: Ultra-lightweight ingestion for IoT devices.
122
+ - **Redis Scaling**: Seamlessly scale across multiple instances.
123
+
118
124
  ---
119
125
 
120
126
  ## 🗺️ Roadmap & Future Vision
121
127
  1. **`defineModel` Engine**: Define a schema once, auto-generate CRUD, validation, and types.
122
- 2. **Plugin System**: A robust "hook" based system for extending the framework core.
123
- 3. **IoT & WebSocket Support**: Dedicated ingestion layer for high-throughput data.
124
- 4. **CLI Presets**: `npx dolphin init` for instant project scaffolding.
128
+ 2. **Plugin System**: A robust "hook" based system for extending the framework core. [DONE]
129
+ 3. **CLI Presets**: `npx dolphin init` for instant project scaffolding.
130
+ 4. **Adaptive Load Balancing**: Native cluster support for multi-core CPUs.
125
131
 
126
132
  ---
127
133
 
@@ -180,7 +180,27 @@ app.post('/register', validate(registerSchema), (ctx) => {
180
180
 
181
181
  ---
182
182
 
183
- ## ११. अन्तिममा (Conclusion)
183
+ ## ११. रियलटाइम र IoT (Realtime & IoT Core) [NEW]
184
+ Dolphin ले अब उच्च क्षमताको रियलटाइम कम्युनिकेसन सपोर्ट गर्छ।
185
+
186
+ ```typescript
187
+ import { RealtimeCore, JSONPlugin } from 'dolphin-server-modules/realtime';
188
+
189
+ const rt = new RealtimeCore();
190
+ rt.use(JSONPlugin);
191
+
192
+ // टपिकहरूमा सब्सक्राइब (Subscribe) गर्नुहोस्
193
+ rt.subscribe('sensors/+', (ctx) => {
194
+ console.log(`टपिक: ${ctx.topic}, डाटा:`, ctx.payload);
195
+ });
196
+
197
+ // पब्लिस (Publish) गर्नुहोस्
198
+ rt.publish('sensors/temp', { value: 24.5 });
199
+ ```
200
+
201
+ ---
202
+
203
+ ## १२. अन्तिममा (Conclusion)
184
204
 
185
205
  Dolphin Framework निकै छिटो र सजिलो छ। यसले तपाईँको ब्याकइन्ड डेभलपमेन्टको अनुभवलाई नयाँ उचाइमा पुर्‍याउँछ।
186
206
 
package/dist/index.d.ts CHANGED
@@ -3,3 +3,4 @@ export * from './controller/controller';
3
3
  export { createCRUD, BaseDocument, QueryFilter, PaginationOptions, DatabaseAdapter as CrudDatabaseAdapter } from './curd/crud';
4
4
  export * from './server/server';
5
5
  export * from './router/router';
6
+ export * from './realtime/index';
package/dist/index.js CHANGED
@@ -25,4 +25,6 @@ Object.defineProperty(exports, "createCRUD", { enumerable: true, get: function (
25
25
  // Re-export Server & Router
26
26
  __exportStar(require("./server/server"), exports);
27
27
  __exportStar(require("./router/router"), exports);
28
+ // Re-export Realtime
29
+ __exportStar(require("./realtime/index"), exports);
28
30
  //# sourceMappingURL=index.js.map
package/dist/index.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"file":"index.js","sourceRoot":"","sources":["../index.ts"],"names":[],"mappings":";;;;;;;;;;;;;;;;;AAAA,iBAAiB;AACjB,8CAA4B;AAC5B,uBAAuB;AACvB,0DAAwC;AACxC,8DAA8D;AAC9D,oCAMqB;AALnB,kGAAA,UAAU,OAAA;AAMZ,4BAA4B;AAC5B,kDAAgC;AAChC,kDAAgC"}
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../index.ts"],"names":[],"mappings":";;;;;;;;;;;;;;;;;AAAA,iBAAiB;AACjB,8CAA4B;AAC5B,uBAAuB;AACvB,0DAAwC;AACxC,8DAA8D;AAC9D,oCAMqB;AALnB,kGAAA,UAAU,OAAA;AAMZ,4BAA4B;AAC5B,kDAAgC;AAChC,kDAAgC;AAEhC,qBAAqB;AACrB,mDAAiC"}
@@ -0,0 +1,18 @@
1
+ /**
2
+ * Lightweight binary/JSON codec for Dolphin Realtime.
3
+ * Optimized for small payloads and cross-platform compatibility.
4
+ */
5
+ export declare function getSize(data: any): number;
6
+ /**
7
+ * Encode data to Buffer.
8
+ * Types:
9
+ * 1: Int32
10
+ * 2: String
11
+ * 3: JSON
12
+ * Default: Buffer as is
13
+ */
14
+ export declare function encode(data: any): Buffer;
15
+ /**
16
+ * Decode Buffer to data.
17
+ */
18
+ export declare function decode(buf: Buffer): any;
@@ -0,0 +1,65 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.getSize = getSize;
4
+ exports.encode = encode;
5
+ exports.decode = decode;
6
+ /**
7
+ * Lightweight binary/JSON codec for Dolphin Realtime.
8
+ * Optimized for small payloads and cross-platform compatibility.
9
+ */
10
+ function getSize(data) {
11
+ if (Buffer.isBuffer(data))
12
+ return data.length;
13
+ if (typeof data === 'string')
14
+ return Buffer.byteLength(data);
15
+ return Buffer.byteLength(JSON.stringify(data));
16
+ }
17
+ /**
18
+ * Encode data to Buffer.
19
+ * Types:
20
+ * 1: Int32
21
+ * 2: String
22
+ * 3: JSON
23
+ * Default: Buffer as is
24
+ */
25
+ function encode(data) {
26
+ if (Buffer.isBuffer(data))
27
+ return data;
28
+ if (typeof data === 'number') {
29
+ const b = Buffer.allocUnsafe(5);
30
+ b[0] = 1;
31
+ b.writeInt32BE(data, 1);
32
+ return b;
33
+ }
34
+ if (typeof data === 'string') {
35
+ const str = Buffer.from(data);
36
+ const len = str.length;
37
+ // Simple header: type(1 byte) + len(2 bytes)
38
+ const b = Buffer.allocUnsafe(3);
39
+ b[0] = 2;
40
+ b.writeUInt16BE(len, 1);
41
+ return Buffer.concat([b, str]);
42
+ }
43
+ const json = Buffer.from(JSON.stringify(data));
44
+ const b = Buffer.allocUnsafe(1);
45
+ b[0] = 3;
46
+ return Buffer.concat([b, json]);
47
+ }
48
+ /**
49
+ * Decode Buffer to data.
50
+ */
51
+ function decode(buf) {
52
+ if (!buf || buf.length === 0)
53
+ return null;
54
+ const t = buf[0];
55
+ if (t === 1)
56
+ return buf.readInt32BE(1);
57
+ if (t === 2) {
58
+ const len = buf.readUInt16BE(1);
59
+ return buf.slice(3, 3 + len).toString();
60
+ }
61
+ if (t === 3)
62
+ return JSON.parse(buf.slice(1).toString());
63
+ return buf;
64
+ }
65
+ //# sourceMappingURL=codec.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"codec.js","sourceRoot":"","sources":["../../realtime/codec.ts"],"names":[],"mappings":";;AAIA,0BAIC;AAUD,wBAwBC;AAKD,wBAUC;AAzDD;;;GAGG;AACH,SAAgB,OAAO,CAAC,IAAS;IAC/B,IAAI,MAAM,CAAC,QAAQ,CAAC,IAAI,CAAC;QAAE,OAAO,IAAI,CAAC,MAAM,CAAC;IAC9C,IAAI,OAAO,IAAI,KAAK,QAAQ;QAAE,OAAO,MAAM,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC;IAC7D,OAAO,MAAM,CAAC,UAAU,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,CAAC;AACjD,CAAC;AAED;;;;;;;GAOG;AACH,SAAgB,MAAM,CAAC,IAAS;IAC9B,IAAI,MAAM,CAAC,QAAQ,CAAC,IAAI,CAAC;QAAE,OAAO,IAAI,CAAC;IAEvC,IAAI,OAAO,IAAI,KAAK,QAAQ,EAAE,CAAC;QAC7B,MAAM,CAAC,GAAG,MAAM,CAAC,WAAW,CAAC,CAAC,CAAC,CAAC;QAChC,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC;QACT,CAAC,CAAC,YAAY,CAAC,IAAI,EAAE,CAAC,CAAC,CAAC;QACxB,OAAO,CAAC,CAAC;IACX,CAAC;IAED,IAAI,OAAO,IAAI,KAAK,QAAQ,EAAE,CAAC;QAC7B,MAAM,GAAG,GAAG,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAC9B,MAAM,GAAG,GAAG,GAAG,CAAC,MAAM,CAAC;QACvB,6CAA6C;QAC7C,MAAM,CAAC,GAAG,MAAM,CAAC,WAAW,CAAC,CAAC,CAAC,CAAC;QAChC,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC;QACT,CAAC,CAAC,aAAa,CAAC,GAAG,EAAE,CAAC,CAAC,CAAC;QACxB,OAAO,MAAM,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,GAAG,CAAC,CAAC,CAAC;IACjC,CAAC;IAED,MAAM,IAAI,GAAG,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,CAAC;IAC/C,MAAM,CAAC,GAAG,MAAM,CAAC,WAAW,CAAC,CAAC,CAAC,CAAC;IAChC,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC;IACT,OAAO,MAAM,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,IAAI,CAAC,CAAC,CAAC;AAClC,CAAC;AAED;;GAEG;AACH,SAAgB,MAAM,CAAC,GAAW;IAChC,IAAI,CAAC,GAAG,IAAI,GAAG,CAAC,MAAM,KAAK,CAAC;QAAE,OAAO,IAAI,CAAC;IAC1C,MAAM,CAAC,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC;IACjB,IAAI,CAAC,KAAK,CAAC;QAAE,OAAO,GAAG,CAAC,WAAW,CAAC,CAAC,CAAC,CAAC;IACvC,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC;QACZ,MAAM,GAAG,GAAG,GAAG,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC;QAChC,OAAO,GAAG,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,GAAG,GAAG,CAAC,CAAC,QAAQ,EAAE,CAAC;IAC1C,CAAC;IACD,IAAI,CAAC,KAAK,CAAC;QAAE,OAAO,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,QAAQ,EAAE,CAAC,CAAC;IACxD,OAAO,GAAG,CAAC;AACb,CAAC"}
@@ -0,0 +1,63 @@
1
+ import { EventEmitter } from 'events';
2
+ import { RealtimePlugin } from './plugins';
3
+ /**
4
+ * RealtimeCore - High performance unified pub/sub bus for Dolphin.
5
+ * Supports:
6
+ * - Local event emitter (in-process)
7
+ * - Distributed bus via Redis (optional)
8
+ * - Retained messages with TTL
9
+ * - Device/Client tracking
10
+ * - Plugin-based protocol handling
11
+ */
12
+ export declare class RealtimeCore extends EventEmitter {
13
+ private config;
14
+ private trie;
15
+ private retained;
16
+ private devices;
17
+ private plugins;
18
+ private pending;
19
+ private msgId;
20
+ private redisPub?;
21
+ private redisSub?;
22
+ constructor(config?: {
23
+ maxMessageSize?: number;
24
+ redisUrl?: string;
25
+ acl?: {
26
+ canSubscribe: (deviceId: string, topic: string) => boolean;
27
+ canPublish: (deviceId: string, topic: string) => boolean;
28
+ };
29
+ });
30
+ /**
31
+ * Initialize Redis for distributed pub/sub.
32
+ */
33
+ private initRedis;
34
+ /**
35
+ * Subscribe to a topic pattern.
36
+ */
37
+ subscribe(topic: string, fn: (data: any) => void, deviceId?: string): void;
38
+ /**
39
+ * Publish a message to a topic.
40
+ */
41
+ publish(topic: string, payload: any, opts?: {
42
+ retain?: boolean;
43
+ ttl?: number;
44
+ }, deviceId?: string): void;
45
+ private publishInternal;
46
+ /**
47
+ * Handle raw data from a socket.
48
+ */
49
+ handle(raw: Buffer, socket?: any, deviceId?: string): Promise<void>;
50
+ /**
51
+ * Register a plugin.
52
+ */
53
+ use(plugin: RealtimePlugin): void;
54
+ /**
55
+ * Register a device/client.
56
+ */
57
+ register(deviceId: string, socket?: any): void;
58
+ /**
59
+ * Heartbeat for a device.
60
+ */
61
+ touch(deviceId: string): void;
62
+ private startCleanup;
63
+ }
@@ -0,0 +1,200 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
14
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
15
+ }) : function(o, v) {
16
+ o["default"] = v;
17
+ });
18
+ var __importStar = (this && this.__importStar) || (function () {
19
+ var ownKeys = function(o) {
20
+ ownKeys = Object.getOwnPropertyNames || function (o) {
21
+ var ar = [];
22
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
23
+ return ar;
24
+ };
25
+ return ownKeys(o);
26
+ };
27
+ return function (mod) {
28
+ if (mod && mod.__esModule) return mod;
29
+ var result = {};
30
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
31
+ __setModuleDefault(result, mod);
32
+ return result;
33
+ };
34
+ })();
35
+ Object.defineProperty(exports, "__esModule", { value: true });
36
+ exports.RealtimeCore = void 0;
37
+ const events_1 = require("events");
38
+ const trie_1 = require("./trie");
39
+ const codec_1 = require("./codec");
40
+ /**
41
+ * RealtimeCore - High performance unified pub/sub bus for Dolphin.
42
+ * Supports:
43
+ * - Local event emitter (in-process)
44
+ * - Distributed bus via Redis (optional)
45
+ * - Retained messages with TTL
46
+ * - Device/Client tracking
47
+ * - Plugin-based protocol handling
48
+ */
49
+ class RealtimeCore extends events_1.EventEmitter {
50
+ config;
51
+ trie = new trie_1.TopicTrie();
52
+ retained = new Map();
53
+ devices = new Map();
54
+ plugins = new Map();
55
+ pending = new Map();
56
+ msgId = 0;
57
+ redisPub;
58
+ redisSub;
59
+ constructor(config = {}) {
60
+ super();
61
+ this.config = config;
62
+ if (config.redisUrl) {
63
+ this.initRedis(config.redisUrl);
64
+ }
65
+ this.startCleanup();
66
+ }
67
+ /**
68
+ * Initialize Redis for distributed pub/sub.
69
+ */
70
+ async initRedis(url) {
71
+ try {
72
+ // @ts-ignore
73
+ const Redis = (await Promise.resolve().then(() => __importStar(require('ioredis')))).default;
74
+ this.redisPub = new Redis(url);
75
+ this.redisSub = new Redis(url);
76
+ this.redisSub.subscribe('dolphin-rt');
77
+ this.redisSub.on('message', (_, msg) => {
78
+ const { topic, payload } = JSON.parse(msg);
79
+ this.publishInternal(topic, payload, { skipRedis: true });
80
+ });
81
+ }
82
+ catch (err) {
83
+ console.warn('Redis initialization failed (ioredis not found or connection error):', err);
84
+ }
85
+ }
86
+ /**
87
+ * Subscribe to a topic pattern.
88
+ */
89
+ subscribe(topic, fn, deviceId) {
90
+ if (deviceId && this.config.acl && !this.config.acl.canSubscribe(deviceId, topic)) {
91
+ throw new Error('ACL deny');
92
+ }
93
+ this.trie.add(topic, fn);
94
+ // Replay retained messages
95
+ for (const [t, data] of this.retained.entries()) {
96
+ // Small hack: if we match the new subscription, replay
97
+ // We can improve this by matching the pattern against the topic
98
+ // For now, only exact match for simplicity or improve TopicTrie
99
+ if (t === topic)
100
+ fn(data.payload);
101
+ }
102
+ }
103
+ /**
104
+ * Publish a message to a topic.
105
+ */
106
+ publish(topic, payload, opts = {}, deviceId) {
107
+ if ((0, codec_1.getSize)(payload) > (this.config.maxMessageSize || 256 * 1024)) {
108
+ throw new Error('Payload too large');
109
+ }
110
+ if (deviceId && this.config.acl && !this.config.acl.canPublish(deviceId, topic)) {
111
+ throw new Error('ACL deny');
112
+ }
113
+ this.publishInternal(topic, payload, opts);
114
+ }
115
+ publishInternal(topic, payload, opts = {}) {
116
+ if (opts.retain) {
117
+ this.retained.set(topic, { payload, ts: Date.now(), ttl: opts.ttl || 0 });
118
+ }
119
+ // Match and emit locally
120
+ this.trie.match(topic, (fn) => fn({ topic, payload }));
121
+ this.emit('message', { topic, payload });
122
+ // Publish to Redis if available
123
+ if (this.redisPub && !opts.skipRedis) {
124
+ this.redisPub.publish('dolphin-rt', JSON.stringify({ topic, payload }));
125
+ }
126
+ }
127
+ /**
128
+ * Handle raw data from a socket.
129
+ */
130
+ async handle(raw, socket, deviceId) {
131
+ if (raw.length > (this.config.maxMessageSize || 256 * 1024))
132
+ return;
133
+ // Create initial context
134
+ const ctx = {
135
+ type: 'raw',
136
+ raw,
137
+ socket,
138
+ deviceId,
139
+ ts: Date.now(),
140
+ publish: this.publish.bind(this)
141
+ };
142
+ // Plugin matching and decoding
143
+ for (const p of this.plugins.values()) {
144
+ if (p.match(ctx)) {
145
+ if (p.decode)
146
+ ctx.payload = p.decode(raw);
147
+ p.onMessage?.(ctx);
148
+ // If plugin handled it, we might want to stop or continue.
149
+ // For now, let's continue to allow multiple plugins or default handling.
150
+ }
151
+ }
152
+ // Default handling for 'pub' style packets if no plugin decoded it or just generic
153
+ try {
154
+ const decoded = (0, codec_1.decode)(raw);
155
+ if (decoded && typeof decoded === 'object' && decoded.type === 'pub') {
156
+ this.publish(decoded.topic, decoded.payload, {}, deviceId);
157
+ }
158
+ }
159
+ catch {
160
+ // Ignore decode errors for raw data
161
+ }
162
+ }
163
+ /**
164
+ * Register a plugin.
165
+ */
166
+ use(plugin) {
167
+ this.plugins.set(plugin.name, plugin);
168
+ }
169
+ /**
170
+ * Register a device/client.
171
+ */
172
+ register(deviceId, socket) {
173
+ this.devices.set(deviceId, { lastSeen: Date.now(), socket });
174
+ }
175
+ /**
176
+ * Heartbeat for a device.
177
+ */
178
+ touch(deviceId) {
179
+ const d = this.devices.get(deviceId);
180
+ if (d)
181
+ d.lastSeen = Date.now();
182
+ }
183
+ startCleanup() {
184
+ setInterval(() => {
185
+ const now = Date.now();
186
+ // Clean retained messages
187
+ for (const [k, v] of this.retained) {
188
+ if (v.ttl && now - v.ts > v.ttl)
189
+ this.retained.delete(k);
190
+ }
191
+ // Clean inactive devices (1 minute timeout)
192
+ for (const [id, d] of this.devices) {
193
+ if (now - d.lastSeen > 60000)
194
+ this.devices.delete(id);
195
+ }
196
+ }, 5000);
197
+ }
198
+ }
199
+ exports.RealtimeCore = RealtimeCore;
200
+ //# sourceMappingURL=core.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"core.js","sourceRoot":"","sources":["../../realtime/core.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,4 @@
1
+ export * from './core';
2
+ export * from './trie';
3
+ export * from './codec';
4
+ export * from './plugins';
@@ -0,0 +1,21 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __exportStar = (this && this.__exportStar) || function(m, exports) {
14
+ for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
15
+ };
16
+ Object.defineProperty(exports, "__esModule", { value: true });
17
+ __exportStar(require("./core"), exports);
18
+ __exportStar(require("./trie"), exports);
19
+ __exportStar(require("./codec"), exports);
20
+ __exportStar(require("./plugins"), exports);
21
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../realtime/index.ts"],"names":[],"mappings":";;;;;;;;;;;;;;;;AAAA,yCAAuB;AACvB,yCAAuB;AACvB,0CAAwB;AACxB,4CAA0B"}
@@ -0,0 +1,34 @@
1
+ /**
2
+ * Realtime Plugins for Dolphin.
3
+ * Allows handling custom protocols (e.g. Modbus, HL7) seamlessly.
4
+ */
5
+ export type RealtimeContext = {
6
+ id?: number;
7
+ type: string;
8
+ topic?: string;
9
+ payload?: any;
10
+ raw?: Buffer;
11
+ socket?: any;
12
+ deviceId?: string;
13
+ ts: number;
14
+ publish: (topic: string, payload: any, opts?: any) => void;
15
+ };
16
+ export type RealtimePlugin = {
17
+ name: string;
18
+ match: (ctx: RealtimeContext) => boolean;
19
+ decode?: (buf: Buffer) => any;
20
+ encode?: (data: any) => Buffer;
21
+ onMessage?: (ctx: RealtimeContext) => void;
22
+ };
23
+ /**
24
+ * Sample HL7 Plugin
25
+ */
26
+ export declare const HL7Plugin: RealtimePlugin;
27
+ /**
28
+ * Sample Modbus Plugin
29
+ */
30
+ export declare const ModbusPlugin: RealtimePlugin;
31
+ /**
32
+ * Standard JSON Plugin for Web
33
+ */
34
+ export declare const JSONPlugin: RealtimePlugin;
@@ -0,0 +1,47 @@
1
+ "use strict";
2
+ /**
3
+ * Realtime Plugins for Dolphin.
4
+ * Allows handling custom protocols (e.g. Modbus, HL7) seamlessly.
5
+ */
6
+ Object.defineProperty(exports, "__esModule", { value: true });
7
+ exports.JSONPlugin = exports.ModbusPlugin = exports.HL7Plugin = void 0;
8
+ /**
9
+ * Sample HL7 Plugin
10
+ */
11
+ exports.HL7Plugin = {
12
+ name: 'hl7',
13
+ match: (ctx) => ctx.raw?.includes?.(0x0b) ?? false,
14
+ decode: (buf) => ({ msg: buf.toString().split('\r') })
15
+ };
16
+ /**
17
+ * Sample Modbus Plugin
18
+ */
19
+ exports.ModbusPlugin = {
20
+ name: 'modbus',
21
+ match: (ctx) => ctx.raw?.length === 8,
22
+ decode: (buf) => ({
23
+ addr: buf[0],
24
+ func: buf[1],
25
+ value: buf.readUInt16BE(2)
26
+ })
27
+ };
28
+ /**
29
+ * Standard JSON Plugin for Web
30
+ */
31
+ exports.JSONPlugin = {
32
+ name: 'json',
33
+ match: (ctx) => {
34
+ try {
35
+ if (ctx.raw) {
36
+ JSON.parse(ctx.raw.toString());
37
+ return true;
38
+ }
39
+ }
40
+ catch {
41
+ return false;
42
+ }
43
+ return false;
44
+ },
45
+ decode: (buf) => JSON.parse(buf.toString())
46
+ };
47
+ //# sourceMappingURL=plugins.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"plugins.js","sourceRoot":"","sources":["../../realtime/plugins.ts"],"names":[],"mappings":";AAAA;;;GAGG;;;AAsBH;;GAEG;AACU,QAAA,SAAS,GAAmB;IACvC,IAAI,EAAE,KAAK;IACX,KAAK,EAAE,CAAC,GAAG,EAAE,EAAE,CAAC,GAAG,CAAC,GAAG,EAAE,QAAQ,EAAE,CAAC,IAAI,CAAC,IAAI,KAAK;IAClD,MAAM,EAAE,CAAC,GAAG,EAAE,EAAE,CAAC,CAAC,EAAE,GAAG,EAAE,GAAG,CAAC,QAAQ,EAAE,CAAC,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC;CACvD,CAAC;AAEF;;GAEG;AACU,QAAA,YAAY,GAAmB;IAC1C,IAAI,EAAE,QAAQ;IACd,KAAK,EAAE,CAAC,GAAG,EAAE,EAAE,CAAC,GAAG,CAAC,GAAG,EAAE,MAAM,KAAK,CAAC;IACrC,MAAM,EAAE,CAAC,GAAG,EAAE,EAAE,CAAC,CAAC;QAChB,IAAI,EAAE,GAAG,CAAC,CAAC,CAAC;QACZ,IAAI,EAAE,GAAG,CAAC,CAAC,CAAC;QACZ,KAAK,EAAE,GAAG,CAAC,YAAY,CAAC,CAAC,CAAC;KAC3B,CAAC;CACH,CAAC;AAEF;;GAEG;AACU,QAAA,UAAU,GAAmB;IACxC,IAAI,EAAE,MAAM;IACZ,KAAK,EAAE,CAAC,GAAG,EAAE,EAAE;QACb,IAAI,CAAC;YACH,IAAI,GAAG,CAAC,GAAG,EAAE,CAAC;gBACZ,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,GAAG,CAAC,QAAQ,EAAE,CAAC,CAAC;gBAC/B,OAAO,IAAI,CAAC;YACd,CAAC;QACH,CAAC;QAAC,MAAM,CAAC;YACP,OAAO,KAAK,CAAC;QACf,CAAC;QACD,OAAO,KAAK,CAAC;IACf,CAAC;IACD,MAAM,EAAE,CAAC,GAAG,EAAE,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,QAAQ,EAAE,CAAC;CAC5C,CAAC"}
@@ -0,0 +1,22 @@
1
+ /**
2
+ * TopicTrie for high-performance MQTT-style topic matching.
3
+ * Supports:
4
+ * - Simple topics: "sensors/temp"
5
+ * - Single level wildcard: "sensors/+" (matches "sensors/temp", "sensors/hum")
6
+ * - Multi level wildcard: "sensors/#" (matches "sensors/temp", "sensors/room1/temp")
7
+ */
8
+ export declare class TopicTrie {
9
+ private root;
10
+ /**
11
+ * Add a subscriber function to a topic pattern.
12
+ */
13
+ add(topic: string, fn: Function): void;
14
+ /**
15
+ * Remove a subscriber function from a topic pattern.
16
+ */
17
+ remove(topic: string, fn: Function): void;
18
+ /**
19
+ * Match a topic and execute callback for each matching subscriber.
20
+ */
21
+ match(topic: string, cb: (fn: Function) => void): void;
22
+ }
@@ -0,0 +1,76 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.TopicTrie = void 0;
4
+ /**
5
+ * TopicTrie for high-performance MQTT-style topic matching.
6
+ * Supports:
7
+ * - Simple topics: "sensors/temp"
8
+ * - Single level wildcard: "sensors/+" (matches "sensors/temp", "sensors/hum")
9
+ * - Multi level wildcard: "sensors/#" (matches "sensors/temp", "sensors/room1/temp")
10
+ */
11
+ class TopicTrie {
12
+ root = {};
13
+ /**
14
+ * Add a subscriber function to a topic pattern.
15
+ */
16
+ add(topic, fn) {
17
+ const parts = topic.split('/');
18
+ let node = this.root;
19
+ for (const p of parts) {
20
+ if (!node[p])
21
+ node[p] = {};
22
+ node = node[p];
23
+ }
24
+ if (!node._)
25
+ node._ = [];
26
+ node._.push(fn);
27
+ }
28
+ /**
29
+ * Remove a subscriber function from a topic pattern.
30
+ */
31
+ remove(topic, fn) {
32
+ const parts = topic.split('/');
33
+ let node = this.root;
34
+ for (const p of parts) {
35
+ if (!node[p])
36
+ return;
37
+ node = node[p];
38
+ }
39
+ if (node._) {
40
+ node._ = node._.filter((f) => f !== fn);
41
+ if (node._.length === 0)
42
+ delete node._;
43
+ }
44
+ }
45
+ /**
46
+ * Match a topic and execute callback for each matching subscriber.
47
+ */
48
+ match(topic, cb) {
49
+ const parts = topic.split('/');
50
+ const walk = (node, i) => {
51
+ if (!node)
52
+ return;
53
+ // Exact match level
54
+ if (i === parts.length) {
55
+ if (node._)
56
+ node._.forEach(cb);
57
+ // Also check if '#' exists at this level (e.g. pattern 'a/#' matches 'a')
58
+ if (node['#'] && node['#']._)
59
+ node['#']._.forEach(cb);
60
+ return;
61
+ }
62
+ // 1. Direct match
63
+ if (node[parts[i]])
64
+ walk(node[parts[i]], i + 1);
65
+ // 2. Single level wildcard '+'
66
+ if (node['+'])
67
+ walk(node['+'], i + 1);
68
+ // 3. Multi level wildcard '#'
69
+ if (node['#'] && node['#']._)
70
+ node['#']._.forEach(cb);
71
+ };
72
+ walk(this.root, 0);
73
+ }
74
+ }
75
+ exports.TopicTrie = TopicTrie;
76
+ //# sourceMappingURL=trie.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"trie.js","sourceRoot":"","sources":["../../realtime/trie.ts"],"names":[],"mappings":";;;AAAA;;;;;;GAMG;AACH,MAAa,SAAS;IACZ,IAAI,GAAQ,EAAE,CAAC;IAEvB;;OAEG;IACH,GAAG,CAAC,KAAa,EAAE,EAAY;QAC7B,MAAM,KAAK,GAAG,KAAK,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;QAC/B,IAAI,IAAI,GAAG,IAAI,CAAC,IAAI,CAAC;QAErB,KAAK,MAAM,CAAC,IAAI,KAAK,EAAE,CAAC;YACtB,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC;gBAAE,IAAI,CAAC,CAAC,CAAC,GAAG,EAAE,CAAC;YAC3B,IAAI,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC;QACjB,CAAC;QAED,IAAI,CAAC,IAAI,CAAC,CAAC;YAAE,IAAI,CAAC,CAAC,GAAG,EAAE,CAAC;QACzB,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;IAClB,CAAC;IAED;;OAEG;IACH,MAAM,CAAC,KAAa,EAAE,EAAY;QAChC,MAAM,KAAK,GAAG,KAAK,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;QAC/B,IAAI,IAAI,GAAG,IAAI,CAAC,IAAI,CAAC;QAErB,KAAK,MAAM,CAAC,IAAI,KAAK,EAAE,CAAC;YACtB,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC;gBAAE,OAAO;YACrB,IAAI,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC;QACjB,CAAC;QAED,IAAI,IAAI,CAAC,CAAC,EAAE,CAAC;YACX,IAAI,CAAC,CAAC,GAAG,IAAI,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAW,EAAE,EAAE,CAAC,CAAC,KAAK,EAAE,CAAC,CAAC;YAClD,IAAI,IAAI,CAAC,CAAC,CAAC,MAAM,KAAK,CAAC;gBAAE,OAAO,IAAI,CAAC,CAAC,CAAC;QACzC,CAAC;IACH,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,KAAa,EAAE,EAA0B;QAC7C,MAAM,KAAK,GAAG,KAAK,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;QAE/B,MAAM,IAAI,GAAG,CAAC,IAAS,EAAE,CAAS,EAAE,EAAE;YACpC,IAAI,CAAC,IAAI;gBAAE,OAAO;YAElB,oBAAoB;YACpB,IAAI,CAAC,KAAK,KAAK,CAAC,MAAM,EAAE,CAAC;gBACvB,IAAI,IAAI,CAAC,CAAC;oBAAE,IAAI,CAAC,CAAC,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC;gBAC/B,0EAA0E;gBAC1E,IAAI,IAAI,CAAC,GAAG,CAAC,IAAI,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC;oBAAE,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC;gBACtD,OAAO;YACT,CAAC;YAED,kBAAkB;YAClB,IAAI,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;gBAAE,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC;YAEhD,+BAA+B;YAC/B,IAAI,IAAI,CAAC,GAAG,CAAC;gBAAE,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC;YAEtC,8BAA8B;YAC9B,IAAI,IAAI,CAAC,GAAG,CAAC,IAAI,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC;gBAAE,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC;QACxD,CAAC,CAAC;QAEF,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC,CAAC,CAAC;IACrB,CAAC;CACF;AAlED,8BAkEC"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "dolphin-server-modules",
3
- "version": "1.1.1",
3
+ "version": "1.2.0",
4
4
  "homepage": "https://github.com/Phuyalshankar/dolphin-server-modules#readme",
5
5
  "description": "Core utility modules for Auth, CRUD, and Controllers",
6
6
  "main": "dist/index.js",
@@ -21,7 +21,12 @@
21
21
  "./middleware/zod": "./dist/middleware/zod.js",
22
22
  "./adapters/mongoose": "./dist/adapters/mongoose/index.js",
23
23
  "./server": "./dist/server/server.js",
24
- "./router": "./dist/router/router.js"
24
+ "./router": "./dist/router/router.js",
25
+ "./realtime": "./dist/realtime/index.js",
26
+ "./realtime/core": "./dist/realtime/core.js",
27
+ "./realtime/trie": "./dist/realtime/trie.js",
28
+ "./realtime/codec": "./dist/realtime/codec.js",
29
+ "./realtime/plugins": "./dist/realtime/plugins.js"
25
30
  },
26
31
  "scripts": {
27
32
  "build": "tsc",
@@ -33,12 +38,15 @@
33
38
  "server",
34
39
  "auth",
35
40
  "crud",
36
- "controller"
41
+ "controller",
42
+ "realtime",
43
+ "iot"
37
44
  ],
38
45
  "author": "",
39
46
  "license": "ISC",
40
47
  "dependencies": {
41
48
  "argon2": "^0.40.1",
49
+ "ioredis": "^15.15.5",
42
50
  "zod": "^4.3.6"
43
51
  },
44
52
  "devDependencies": {