@linkshell/gateway 0.2.26 → 0.2.28

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/src/tokens.ts CHANGED
@@ -1,4 +1,5 @@
1
1
  import { randomUUID } from "node:crypto";
2
+ import type { GatewayStateStore } from "./state-store.js";
2
3
 
3
4
  const CLEANUP_INTERVAL = 5 * 60_000;
4
5
  const SESSION_TTL = 7 * 24 * 60 * 60_000; // 7 days — prune stale bindings
@@ -15,14 +16,40 @@ export class TokenManager {
15
16
  private sessionToToken = new Map<string, string>();
16
17
  private cleanupTimer: ReturnType<typeof setInterval>;
17
18
 
18
- constructor() {
19
+ constructor(private readonly store?: GatewayStateStore) {
19
20
  this.cleanupTimer = setInterval(() => this.cleanup(), CLEANUP_INTERVAL);
20
21
  }
21
22
 
23
+ async hydrate(): Promise<void> {
24
+ if (!this.store) return;
25
+ try {
26
+ const records = await this.store.loadTokens();
27
+ const now = Date.now();
28
+ for (const record of records) {
29
+ if (now - record.lastUsedAt > SESSION_TTL) {
30
+ void this.store.deleteToken(record.token).catch(() => {});
31
+ continue;
32
+ }
33
+ this.tokens.set(record.token, {
34
+ token: record.token,
35
+ sessionIds: new Set(record.sessionIds),
36
+ createdAt: record.createdAt,
37
+ lastUsedAt: record.lastUsedAt,
38
+ });
39
+ for (const sessionId of record.sessionIds) {
40
+ this.sessionToToken.set(sessionId, record.token);
41
+ }
42
+ }
43
+ } catch (err) {
44
+ process.stderr.write(`[gateway] token store hydrate failed, using memory only: ${err}\n`);
45
+ }
46
+ }
47
+
22
48
  register(deviceToken?: string): string {
23
49
  if (deviceToken && this.tokens.has(deviceToken)) {
24
50
  const record = this.tokens.get(deviceToken)!;
25
51
  record.lastUsedAt = Date.now();
52
+ this.persist(record);
26
53
  return deviceToken;
27
54
  }
28
55
  const token = deviceToken || randomUUID();
@@ -32,6 +59,7 @@ export class TokenManager {
32
59
  createdAt: Date.now(),
33
60
  lastUsedAt: Date.now(),
34
61
  });
62
+ this.persist(this.tokens.get(token)!);
35
63
  return token;
36
64
  }
37
65
 
@@ -41,6 +69,7 @@ export class TokenManager {
41
69
  record.sessionIds.add(sessionId);
42
70
  record.lastUsedAt = Date.now();
43
71
  this.sessionToToken.set(sessionId, token);
72
+ this.persist(record);
44
73
  return true;
45
74
  }
46
75
 
@@ -48,6 +77,7 @@ export class TokenManager {
48
77
  const record = this.tokens.get(token);
49
78
  if (!record) return false;
50
79
  record.lastUsedAt = Date.now();
80
+ this.persist(record);
51
81
  return true;
52
82
  }
53
83
 
@@ -55,6 +85,7 @@ export class TokenManager {
55
85
  const record = this.tokens.get(token);
56
86
  if (!record) return false;
57
87
  record.lastUsedAt = Date.now();
88
+ this.persist(record);
58
89
  return record.sessionIds.has(sessionId);
59
90
  }
60
91
 
@@ -62,6 +93,7 @@ export class TokenManager {
62
93
  const record = this.tokens.get(token);
63
94
  if (!record) return new Set();
64
95
  record.lastUsedAt = Date.now();
96
+ this.persist(record);
65
97
  return record.sessionIds;
66
98
  }
67
99
 
@@ -77,10 +109,22 @@ export class TokenManager {
77
109
  this.sessionToToken.delete(sid);
78
110
  }
79
111
  this.tokens.delete(token);
112
+ void this.store?.deleteToken(token).catch(() => {});
80
113
  }
81
114
  }
82
115
  }
83
116
 
117
+ private persist(record: TokenRecord): void {
118
+ void this.store?.saveToken({
119
+ token: record.token,
120
+ sessionIds: [...record.sessionIds],
121
+ createdAt: record.createdAt,
122
+ lastUsedAt: record.lastUsedAt,
123
+ }).catch((err) => {
124
+ process.stderr.write(`[gateway] token store save failed: ${err}\n`);
125
+ });
126
+ }
127
+
84
128
  destroy(): void {
85
129
  clearInterval(this.cleanupTimer);
86
130
  }