@chat-adapter/state-memory 4.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.
@@ -0,0 +1,34 @@
1
+ import { StateAdapter, Lock } from 'chat';
2
+
3
+ /**
4
+ * In-memory state adapter for development and testing.
5
+ *
6
+ * WARNING: State is not persisted across restarts.
7
+ * Use RedisStateAdapter for production.
8
+ */
9
+ declare class MemoryStateAdapter implements StateAdapter {
10
+ private subscriptions;
11
+ private locks;
12
+ private cache;
13
+ private connected;
14
+ private connectPromise;
15
+ connect(): Promise<void>;
16
+ disconnect(): Promise<void>;
17
+ subscribe(threadId: string): Promise<void>;
18
+ unsubscribe(threadId: string): Promise<void>;
19
+ isSubscribed(threadId: string): Promise<boolean>;
20
+ listSubscriptions(adapterName?: string): AsyncIterable<string>;
21
+ acquireLock(threadId: string, ttlMs: number): Promise<Lock | null>;
22
+ releaseLock(lock: Lock): Promise<void>;
23
+ extendLock(lock: Lock, ttlMs: number): Promise<boolean>;
24
+ get<T = unknown>(key: string): Promise<T | null>;
25
+ set<T = unknown>(key: string, value: T, ttlMs?: number): Promise<void>;
26
+ delete(key: string): Promise<void>;
27
+ private ensureConnected;
28
+ private cleanExpiredLocks;
29
+ _getSubscriptionCount(): number;
30
+ _getLockCount(): number;
31
+ }
32
+ declare function createMemoryState(): MemoryStateAdapter;
33
+
34
+ export { MemoryStateAdapter, createMemoryState };
package/dist/index.js ADDED
@@ -0,0 +1,146 @@
1
+ // src/index.ts
2
+ var MemoryStateAdapter = class {
3
+ subscriptions = /* @__PURE__ */ new Set();
4
+ locks = /* @__PURE__ */ new Map();
5
+ cache = /* @__PURE__ */ new Map();
6
+ connected = false;
7
+ connectPromise = null;
8
+ async connect() {
9
+ if (this.connected) {
10
+ return;
11
+ }
12
+ if (!this.connectPromise) {
13
+ this.connectPromise = Promise.resolve().then(() => {
14
+ if (process.env.NODE_ENV === "production") {
15
+ console.warn(
16
+ "[chat] MemoryStateAdapter is not recommended for production. Consider using @chat-adapter/state-redis instead."
17
+ );
18
+ }
19
+ this.connected = true;
20
+ });
21
+ }
22
+ await this.connectPromise;
23
+ }
24
+ async disconnect() {
25
+ this.connected = false;
26
+ this.connectPromise = null;
27
+ this.subscriptions.clear();
28
+ this.locks.clear();
29
+ }
30
+ async subscribe(threadId) {
31
+ this.ensureConnected();
32
+ this.subscriptions.add(threadId);
33
+ }
34
+ async unsubscribe(threadId) {
35
+ this.ensureConnected();
36
+ this.subscriptions.delete(threadId);
37
+ }
38
+ async isSubscribed(threadId) {
39
+ this.ensureConnected();
40
+ return this.subscriptions.has(threadId);
41
+ }
42
+ async *listSubscriptions(adapterName) {
43
+ this.ensureConnected();
44
+ for (const threadId of this.subscriptions) {
45
+ if (adapterName) {
46
+ if (threadId.startsWith(`${adapterName}:`)) {
47
+ yield threadId;
48
+ }
49
+ } else {
50
+ yield threadId;
51
+ }
52
+ }
53
+ }
54
+ async acquireLock(threadId, ttlMs) {
55
+ this.ensureConnected();
56
+ this.cleanExpiredLocks();
57
+ const existingLock = this.locks.get(threadId);
58
+ if (existingLock && existingLock.expiresAt > Date.now()) {
59
+ return null;
60
+ }
61
+ const lock = {
62
+ threadId,
63
+ token: generateToken(),
64
+ expiresAt: Date.now() + ttlMs
65
+ };
66
+ this.locks.set(threadId, lock);
67
+ return lock;
68
+ }
69
+ async releaseLock(lock) {
70
+ this.ensureConnected();
71
+ const existingLock = this.locks.get(lock.threadId);
72
+ if (existingLock && existingLock.token === lock.token) {
73
+ this.locks.delete(lock.threadId);
74
+ }
75
+ }
76
+ async extendLock(lock, ttlMs) {
77
+ this.ensureConnected();
78
+ const existingLock = this.locks.get(lock.threadId);
79
+ if (!existingLock || existingLock.token !== lock.token) {
80
+ return false;
81
+ }
82
+ if (existingLock.expiresAt < Date.now()) {
83
+ this.locks.delete(lock.threadId);
84
+ return false;
85
+ }
86
+ existingLock.expiresAt = Date.now() + ttlMs;
87
+ return true;
88
+ }
89
+ async get(key) {
90
+ this.ensureConnected();
91
+ const cached = this.cache.get(key);
92
+ if (!cached) {
93
+ return null;
94
+ }
95
+ if (cached.expiresAt !== null && cached.expiresAt <= Date.now()) {
96
+ this.cache.delete(key);
97
+ return null;
98
+ }
99
+ return cached.value;
100
+ }
101
+ async set(key, value, ttlMs) {
102
+ this.ensureConnected();
103
+ this.cache.set(key, {
104
+ value,
105
+ expiresAt: ttlMs ? Date.now() + ttlMs : null
106
+ });
107
+ }
108
+ async delete(key) {
109
+ this.ensureConnected();
110
+ this.cache.delete(key);
111
+ }
112
+ ensureConnected() {
113
+ if (!this.connected) {
114
+ throw new Error(
115
+ "MemoryStateAdapter is not connected. Call connect() first."
116
+ );
117
+ }
118
+ }
119
+ cleanExpiredLocks() {
120
+ const now = Date.now();
121
+ for (const [threadId, lock] of this.locks) {
122
+ if (lock.expiresAt <= now) {
123
+ this.locks.delete(threadId);
124
+ }
125
+ }
126
+ }
127
+ // For testing purposes
128
+ _getSubscriptionCount() {
129
+ return this.subscriptions.size;
130
+ }
131
+ _getLockCount() {
132
+ this.cleanExpiredLocks();
133
+ return this.locks.size;
134
+ }
135
+ };
136
+ function generateToken() {
137
+ return `mem_${Date.now()}_${Math.random().toString(36).substring(2, 15)}`;
138
+ }
139
+ function createMemoryState() {
140
+ return new MemoryStateAdapter();
141
+ }
142
+ export {
143
+ MemoryStateAdapter,
144
+ createMemoryState
145
+ };
146
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/index.ts"],"sourcesContent":["import type { Lock, StateAdapter } from \"chat\";\n\ninterface MemoryLock extends Lock {\n threadId: string;\n token: string;\n expiresAt: number;\n}\n\ninterface CachedValue<T = unknown> {\n value: T;\n expiresAt: number | null; // null = no expiry\n}\n\n/**\n * In-memory state adapter for development and testing.\n *\n * WARNING: State is not persisted across restarts.\n * Use RedisStateAdapter for production.\n */\nexport class MemoryStateAdapter implements StateAdapter {\n private subscriptions = new Set<string>();\n private locks = new Map<string, MemoryLock>();\n private cache = new Map<string, CachedValue>();\n private connected = false;\n private connectPromise: Promise<void> | null = null;\n\n async connect(): Promise<void> {\n if (this.connected) {\n return;\n }\n\n // Reuse existing connection attempt to avoid race conditions\n if (!this.connectPromise) {\n this.connectPromise = Promise.resolve().then(() => {\n if (process.env.NODE_ENV === \"production\") {\n console.warn(\n \"[chat] MemoryStateAdapter is not recommended for production. \" +\n \"Consider using @chat-adapter/state-redis instead.\",\n );\n }\n this.connected = true;\n });\n }\n\n await this.connectPromise;\n }\n\n async disconnect(): Promise<void> {\n this.connected = false;\n this.connectPromise = null;\n this.subscriptions.clear();\n this.locks.clear();\n }\n\n async subscribe(threadId: string): Promise<void> {\n this.ensureConnected();\n this.subscriptions.add(threadId);\n }\n\n async unsubscribe(threadId: string): Promise<void> {\n this.ensureConnected();\n this.subscriptions.delete(threadId);\n }\n\n async isSubscribed(threadId: string): Promise<boolean> {\n this.ensureConnected();\n return this.subscriptions.has(threadId);\n }\n\n async *listSubscriptions(adapterName?: string): AsyncIterable<string> {\n this.ensureConnected();\n\n for (const threadId of this.subscriptions) {\n if (adapterName) {\n // Thread ID format: \"adapter:channel:thread\"\n if (threadId.startsWith(`${adapterName}:`)) {\n yield threadId;\n }\n } else {\n yield threadId;\n }\n }\n }\n\n async acquireLock(threadId: string, ttlMs: number): Promise<Lock | null> {\n this.ensureConnected();\n this.cleanExpiredLocks();\n\n // Check if already locked\n const existingLock = this.locks.get(threadId);\n if (existingLock && existingLock.expiresAt > Date.now()) {\n return null;\n }\n\n // Create new lock\n const lock: MemoryLock = {\n threadId,\n token: generateToken(),\n expiresAt: Date.now() + ttlMs,\n };\n\n this.locks.set(threadId, lock);\n return lock;\n }\n\n async releaseLock(lock: Lock): Promise<void> {\n this.ensureConnected();\n\n const existingLock = this.locks.get(lock.threadId);\n if (existingLock && existingLock.token === lock.token) {\n this.locks.delete(lock.threadId);\n }\n }\n\n async extendLock(lock: Lock, ttlMs: number): Promise<boolean> {\n this.ensureConnected();\n\n const existingLock = this.locks.get(lock.threadId);\n if (!existingLock || existingLock.token !== lock.token) {\n return false;\n }\n\n if (existingLock.expiresAt < Date.now()) {\n // Lock has already expired\n this.locks.delete(lock.threadId);\n return false;\n }\n\n // Extend the lock\n existingLock.expiresAt = Date.now() + ttlMs;\n return true;\n }\n\n async get<T = unknown>(key: string): Promise<T | null> {\n this.ensureConnected();\n\n const cached = this.cache.get(key);\n if (!cached) {\n return null;\n }\n\n // Check if expired\n if (cached.expiresAt !== null && cached.expiresAt <= Date.now()) {\n this.cache.delete(key);\n return null;\n }\n\n return cached.value as T;\n }\n\n async set<T = unknown>(key: string, value: T, ttlMs?: number): Promise<void> {\n this.ensureConnected();\n\n this.cache.set(key, {\n value,\n expiresAt: ttlMs ? Date.now() + ttlMs : null,\n });\n }\n\n async delete(key: string): Promise<void> {\n this.ensureConnected();\n this.cache.delete(key);\n }\n\n private ensureConnected(): void {\n if (!this.connected) {\n throw new Error(\n \"MemoryStateAdapter is not connected. Call connect() first.\",\n );\n }\n }\n\n private cleanExpiredLocks(): void {\n const now = Date.now();\n for (const [threadId, lock] of this.locks) {\n if (lock.expiresAt <= now) {\n this.locks.delete(threadId);\n }\n }\n }\n\n // For testing purposes\n _getSubscriptionCount(): number {\n return this.subscriptions.size;\n }\n\n _getLockCount(): number {\n this.cleanExpiredLocks();\n return this.locks.size;\n }\n}\n\nfunction generateToken(): string {\n return `mem_${Date.now()}_${Math.random().toString(36).substring(2, 15)}`;\n}\n\nexport function createMemoryState(): MemoryStateAdapter {\n return new MemoryStateAdapter();\n}\n"],"mappings":"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","names":[]}
package/package.json ADDED
@@ -0,0 +1,46 @@
1
+ {
2
+ "name": "@chat-adapter/state-memory",
3
+ "version": "4.0.0",
4
+ "description": "In-memory state adapter for chat (development/testing)",
5
+ "type": "module",
6
+ "main": "./dist/index.js",
7
+ "module": "./dist/index.js",
8
+ "types": "./dist/index.d.ts",
9
+ "exports": {
10
+ ".": {
11
+ "types": "./dist/index.d.ts",
12
+ "import": "./dist/index.js"
13
+ }
14
+ },
15
+ "files": [
16
+ "dist"
17
+ ],
18
+ "dependencies": {
19
+ "chat": "4.0.0"
20
+ },
21
+ "publishConfig": {
22
+ "access": "public"
23
+ },
24
+ "devDependencies": {
25
+ "@types/node": "^22.10.2",
26
+ "tsup": "^8.3.5",
27
+ "typescript": "^5.7.2",
28
+ "vitest": "^2.1.8"
29
+ },
30
+ "keywords": [
31
+ "chat",
32
+ "state",
33
+ "memory",
34
+ "testing"
35
+ ],
36
+ "license": "MIT",
37
+ "scripts": {
38
+ "build": "tsup",
39
+ "dev": "tsup --watch",
40
+ "test": "vitest run",
41
+ "test:watch": "vitest",
42
+ "typecheck": "tsc --noEmit",
43
+ "lint": "biome check src",
44
+ "clean": "rm -rf dist"
45
+ }
46
+ }