dandelion-mesh 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.
- package/LICENSE +21 -0
- package/README.md +192 -0
- package/build/main/crypto/CryptoService.d.ts +31 -0
- package/build/main/crypto/CryptoService.js +78 -0
- package/build/main/index.d.ts +11 -0
- package/build/main/index.js +24 -0
- package/build/main/mesh/DandelionMesh.d.ts +115 -0
- package/build/main/mesh/DandelionMesh.js +368 -0
- package/build/main/mesh/types.d.ts +58 -0
- package/build/main/mesh/types.js +3 -0
- package/build/main/raft/RaftNode.d.ts +87 -0
- package/build/main/raft/RaftNode.js +443 -0
- package/build/main/raft/log/InMemoryRaftLog.d.ts +20 -0
- package/build/main/raft/log/InMemoryRaftLog.js +55 -0
- package/build/main/raft/log/LocalStorageRaftLog.d.ts +9 -0
- package/build/main/raft/log/LocalStorageRaftLog.js +16 -0
- package/build/main/raft/log/RaftLog.d.ts +30 -0
- package/build/main/raft/log/RaftLog.js +3 -0
- package/build/main/raft/log/RaftLog.test-util.d.ts +3 -0
- package/build/main/raft/log/RaftLog.test-util.js +82 -0
- package/build/main/raft/log/SessionStorageRaftLog.d.ts +9 -0
- package/build/main/raft/log/SessionStorageRaftLog.js +38 -0
- package/build/main/raft/log/StorageRaftLog.d.ts +28 -0
- package/build/main/raft/log/StorageRaftLog.js +117 -0
- package/build/main/raft/log/StorageRaftLog.test-util.d.ts +3 -0
- package/build/main/raft/log/StorageRaftLog.test-util.js +54 -0
- package/build/main/raft/types.d.ts +53 -0
- package/build/main/raft/types.js +3 -0
- package/build/main/transport/PeerJSTransport.d.ts +46 -0
- package/build/main/transport/PeerJSTransport.js +139 -0
- package/build/main/transport/Transport.d.ts +38 -0
- package/build/main/transport/Transport.js +8 -0
- package/build/module/crypto/CryptoService.d.ts +31 -0
- package/build/module/crypto/CryptoService.js +69 -0
- package/build/module/index.d.ts +11 -0
- package/build/module/index.js +11 -0
- package/build/module/mesh/DandelionMesh.d.ts +115 -0
- package/build/module/mesh/DandelionMesh.js +364 -0
- package/build/module/mesh/types.d.ts +58 -0
- package/build/module/mesh/types.js +2 -0
- package/build/module/raft/RaftNode.d.ts +87 -0
- package/build/module/raft/RaftNode.js +440 -0
- package/build/module/raft/log/InMemoryRaftLog.d.ts +20 -0
- package/build/module/raft/log/InMemoryRaftLog.js +48 -0
- package/build/module/raft/log/LocalStorageRaftLog.d.ts +9 -0
- package/build/module/raft/log/LocalStorageRaftLog.js +12 -0
- package/build/module/raft/log/RaftLog.d.ts +30 -0
- package/build/module/raft/log/RaftLog.js +2 -0
- package/build/module/raft/log/RaftLog.test-util.d.ts +3 -0
- package/build/module/raft/log/RaftLog.test-util.js +78 -0
- package/build/module/raft/log/SessionStorageRaftLog.d.ts +9 -0
- package/build/module/raft/log/SessionStorageRaftLog.js +34 -0
- package/build/module/raft/log/StorageRaftLog.d.ts +28 -0
- package/build/module/raft/log/StorageRaftLog.js +114 -0
- package/build/module/raft/log/StorageRaftLog.test-util.d.ts +3 -0
- package/build/module/raft/log/StorageRaftLog.test-util.js +50 -0
- package/build/module/raft/types.d.ts +53 -0
- package/build/module/raft/types.js +2 -0
- package/build/module/transport/PeerJSTransport.d.ts +46 -0
- package/build/module/transport/PeerJSTransport.js +134 -0
- package/build/module/transport/Transport.d.ts +38 -0
- package/build/module/transport/Transport.js +7 -0
- package/package.json +119 -0
|
@@ -0,0 +1,443 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.RaftNode = void 0;
|
|
4
|
+
/**
|
|
5
|
+
* Core Raft consensus node.
|
|
6
|
+
*
|
|
7
|
+
* This class implements the Raft algorithm for leader election and log
|
|
8
|
+
* replication. It is transport-agnostic — the caller must wire up
|
|
9
|
+
* `sendMessage` to deliver RPCs, and call `handleMessage` when RPCs arrive.
|
|
10
|
+
*
|
|
11
|
+
* Log indices are 1-based per the Raft paper.
|
|
12
|
+
*/
|
|
13
|
+
class RaftNode {
|
|
14
|
+
constructor(id, log, options) {
|
|
15
|
+
var _a, _b, _c;
|
|
16
|
+
this.role = 'follower';
|
|
17
|
+
this.currentTerm = 0;
|
|
18
|
+
this.votedFor = null;
|
|
19
|
+
this.leaderId = null;
|
|
20
|
+
// Volatile state (all servers)
|
|
21
|
+
this.commitIndex = 0;
|
|
22
|
+
this.lastApplied = 0;
|
|
23
|
+
// Volatile state (leader only)
|
|
24
|
+
this.nextIndex = new Map();
|
|
25
|
+
this.matchIndex = new Map();
|
|
26
|
+
// Election / heartbeat timers
|
|
27
|
+
this.electionTimer = null;
|
|
28
|
+
this.heartbeatTimer = null;
|
|
29
|
+
// Cluster membership (peer IDs excluding self)
|
|
30
|
+
this.peers = [];
|
|
31
|
+
// Votes received in current election
|
|
32
|
+
this.votesReceived = new Set();
|
|
33
|
+
/** Called by the node to send an RPC to a peer — must be wired up externally */
|
|
34
|
+
this.sendMessage = () => { };
|
|
35
|
+
this.eventListeners = {
|
|
36
|
+
leaderChanged: new Set(),
|
|
37
|
+
committed: new Set(),
|
|
38
|
+
roleChanged: new Set(),
|
|
39
|
+
};
|
|
40
|
+
this.destroyed = false;
|
|
41
|
+
this.id = id;
|
|
42
|
+
this.log = log;
|
|
43
|
+
this.electionTimeoutMin = (_a = options === null || options === void 0 ? void 0 : options.electionTimeoutMin) !== null && _a !== void 0 ? _a : 2000;
|
|
44
|
+
this.electionTimeoutMax = (_b = options === null || options === void 0 ? void 0 : options.electionTimeoutMax) !== null && _b !== void 0 ? _b : 4000;
|
|
45
|
+
this.heartbeatInterval = (_c = options === null || options === void 0 ? void 0 : options.heartbeatInterval) !== null && _c !== void 0 ? _c : 400;
|
|
46
|
+
// Restore persistent state
|
|
47
|
+
const state = this.log.loadState();
|
|
48
|
+
this.currentTerm = state.currentTerm;
|
|
49
|
+
this.votedFor = state.votedFor;
|
|
50
|
+
}
|
|
51
|
+
// --- Public API ---
|
|
52
|
+
/** Start the node (begin election timer as follower) */
|
|
53
|
+
start(peerIds) {
|
|
54
|
+
this.peers = peerIds.filter((p) => p !== this.id);
|
|
55
|
+
this.resetElectionTimer();
|
|
56
|
+
}
|
|
57
|
+
/** Update the cluster membership dynamically (e.g. when peers join/leave) */
|
|
58
|
+
updatePeers(peerIds) {
|
|
59
|
+
this.peers = peerIds.filter((p) => p !== this.id);
|
|
60
|
+
// Reinitialize leader volatile state for new peers
|
|
61
|
+
if (this.role === 'leader') {
|
|
62
|
+
for (const peer of this.peers) {
|
|
63
|
+
if (!this.nextIndex.has(peer)) {
|
|
64
|
+
this.nextIndex.set(peer, this.log.lastIndex() + 1);
|
|
65
|
+
this.matchIndex.set(peer, 0);
|
|
66
|
+
}
|
|
67
|
+
}
|
|
68
|
+
}
|
|
69
|
+
}
|
|
70
|
+
/** Submit a command to be replicated. Only succeeds on the leader. */
|
|
71
|
+
propose(command) {
|
|
72
|
+
if (this.role !== 'leader') {
|
|
73
|
+
return false;
|
|
74
|
+
}
|
|
75
|
+
const entry = { term: this.currentTerm, command };
|
|
76
|
+
this.log.append([entry]);
|
|
77
|
+
this.persistState();
|
|
78
|
+
// Immediately replicate to followers
|
|
79
|
+
this.sendAppendEntriesToAll();
|
|
80
|
+
// Check if we can commit (single-node cluster)
|
|
81
|
+
this.advanceCommitIndex();
|
|
82
|
+
return true;
|
|
83
|
+
}
|
|
84
|
+
/** Handle an incoming Raft RPC message from a peer */
|
|
85
|
+
handleMessage(fromPeerId, message) {
|
|
86
|
+
if (this.destroyed)
|
|
87
|
+
return;
|
|
88
|
+
// If we see a higher term, step down
|
|
89
|
+
const msgTerm = 'term' in message ? message.term : 0;
|
|
90
|
+
if (msgTerm > this.currentTerm) {
|
|
91
|
+
this.currentTerm = msgTerm;
|
|
92
|
+
this.votedFor = null;
|
|
93
|
+
this.persistState();
|
|
94
|
+
this.becomeFollower();
|
|
95
|
+
}
|
|
96
|
+
switch (message.type) {
|
|
97
|
+
case 'RequestVote':
|
|
98
|
+
this.handleRequestVote(fromPeerId, message);
|
|
99
|
+
break;
|
|
100
|
+
case 'RequestVoteResult':
|
|
101
|
+
this.handleRequestVoteResult(fromPeerId, message);
|
|
102
|
+
break;
|
|
103
|
+
case 'AppendEntries':
|
|
104
|
+
this.handleAppendEntries(fromPeerId, message);
|
|
105
|
+
break;
|
|
106
|
+
case 'AppendEntriesResult':
|
|
107
|
+
this.handleAppendEntriesResult(message);
|
|
108
|
+
break;
|
|
109
|
+
}
|
|
110
|
+
}
|
|
111
|
+
/** Get current role */
|
|
112
|
+
getRole() {
|
|
113
|
+
return this.role;
|
|
114
|
+
}
|
|
115
|
+
/** Get current term */
|
|
116
|
+
getCurrentTerm() {
|
|
117
|
+
return this.currentTerm;
|
|
118
|
+
}
|
|
119
|
+
/** Get the current leader ID (null if unknown) */
|
|
120
|
+
getLeaderId() {
|
|
121
|
+
return this.leaderId;
|
|
122
|
+
}
|
|
123
|
+
/** Get the commit index */
|
|
124
|
+
getCommitIndex() {
|
|
125
|
+
return this.commitIndex;
|
|
126
|
+
}
|
|
127
|
+
/** Check if this node is the leader */
|
|
128
|
+
isLeader() {
|
|
129
|
+
return this.role === 'leader';
|
|
130
|
+
}
|
|
131
|
+
/** Register an event listener */
|
|
132
|
+
on(event, listener) {
|
|
133
|
+
this.eventListeners[event].add(listener);
|
|
134
|
+
}
|
|
135
|
+
/** Remove an event listener */
|
|
136
|
+
off(event, listener) {
|
|
137
|
+
this.eventListeners[event].delete(listener);
|
|
138
|
+
}
|
|
139
|
+
/** Stop all timers and clean up */
|
|
140
|
+
destroy() {
|
|
141
|
+
this.destroyed = true;
|
|
142
|
+
this.clearElectionTimer();
|
|
143
|
+
this.clearHeartbeatTimer();
|
|
144
|
+
}
|
|
145
|
+
// --- Election ---
|
|
146
|
+
becomeFollower() {
|
|
147
|
+
const wasLeader = this.role === 'leader';
|
|
148
|
+
this.role = 'follower';
|
|
149
|
+
this.clearHeartbeatTimer();
|
|
150
|
+
this.resetElectionTimer();
|
|
151
|
+
this.emit('roleChanged', 'follower');
|
|
152
|
+
if (wasLeader) {
|
|
153
|
+
// Leader identity is unknown until we hear from the new leader
|
|
154
|
+
}
|
|
155
|
+
}
|
|
156
|
+
becomeCandidate() {
|
|
157
|
+
this.role = 'candidate';
|
|
158
|
+
this.currentTerm++;
|
|
159
|
+
this.votedFor = this.id;
|
|
160
|
+
this.persistState();
|
|
161
|
+
this.votesReceived.clear();
|
|
162
|
+
this.votesReceived.add(this.id); // vote for self
|
|
163
|
+
this.leaderId = null;
|
|
164
|
+
this.emit('roleChanged', 'candidate');
|
|
165
|
+
this.emit('leaderChanged', null);
|
|
166
|
+
this.resetElectionTimer();
|
|
167
|
+
// Single-node cluster: become leader immediately
|
|
168
|
+
if (this.peers.length === 0) {
|
|
169
|
+
this.becomeLeader();
|
|
170
|
+
return;
|
|
171
|
+
}
|
|
172
|
+
// Send RequestVote to all peers
|
|
173
|
+
const request = {
|
|
174
|
+
type: 'RequestVote',
|
|
175
|
+
term: this.currentTerm,
|
|
176
|
+
candidateId: this.id,
|
|
177
|
+
lastLogIndex: this.log.lastIndex(),
|
|
178
|
+
lastLogTerm: this.log.lastTerm(),
|
|
179
|
+
};
|
|
180
|
+
for (const peer of this.peers) {
|
|
181
|
+
this.sendMessage(peer, request);
|
|
182
|
+
}
|
|
183
|
+
}
|
|
184
|
+
becomeLeader() {
|
|
185
|
+
this.role = 'leader';
|
|
186
|
+
this.leaderId = this.id;
|
|
187
|
+
this.clearElectionTimer();
|
|
188
|
+
this.emit('roleChanged', 'leader');
|
|
189
|
+
this.emit('leaderChanged', this.id);
|
|
190
|
+
// Initialize nextIndex and matchIndex for all peers
|
|
191
|
+
const lastLogIndex = this.log.lastIndex();
|
|
192
|
+
this.nextIndex.clear();
|
|
193
|
+
this.matchIndex.clear();
|
|
194
|
+
for (const peer of this.peers) {
|
|
195
|
+
this.nextIndex.set(peer, lastLogIndex + 1);
|
|
196
|
+
this.matchIndex.set(peer, 0);
|
|
197
|
+
}
|
|
198
|
+
// Send initial heartbeats
|
|
199
|
+
this.sendAppendEntriesToAll();
|
|
200
|
+
// Start heartbeat timer
|
|
201
|
+
this.clearHeartbeatTimer();
|
|
202
|
+
this.heartbeatTimer = setInterval(() => {
|
|
203
|
+
if (this.role === 'leader' && !this.destroyed) {
|
|
204
|
+
this.sendAppendEntriesToAll();
|
|
205
|
+
}
|
|
206
|
+
}, this.heartbeatInterval);
|
|
207
|
+
}
|
|
208
|
+
// --- RequestVote handling ---
|
|
209
|
+
handleRequestVote(_fromPeerId, args) {
|
|
210
|
+
const reply = {
|
|
211
|
+
type: 'RequestVoteResult',
|
|
212
|
+
term: this.currentTerm,
|
|
213
|
+
voteGranted: false,
|
|
214
|
+
};
|
|
215
|
+
// Reject if candidate's term is behind
|
|
216
|
+
if (args.term < this.currentTerm) {
|
|
217
|
+
this.sendMessage(args.candidateId, reply);
|
|
218
|
+
return;
|
|
219
|
+
}
|
|
220
|
+
// Grant vote if we haven't voted for someone else, and candidate's log is up-to-date
|
|
221
|
+
const canVote = this.votedFor === null || this.votedFor === args.candidateId;
|
|
222
|
+
const candidateLogUpToDate = this.isLogUpToDate(args.lastLogTerm, args.lastLogIndex);
|
|
223
|
+
if (canVote && candidateLogUpToDate) {
|
|
224
|
+
this.votedFor = args.candidateId;
|
|
225
|
+
this.persistState();
|
|
226
|
+
reply.voteGranted = true;
|
|
227
|
+
this.resetElectionTimer(); // reset timer on granting vote
|
|
228
|
+
}
|
|
229
|
+
this.sendMessage(args.candidateId, reply);
|
|
230
|
+
}
|
|
231
|
+
handleRequestVoteResult(fromPeerId, result) {
|
|
232
|
+
if (this.role !== 'candidate')
|
|
233
|
+
return;
|
|
234
|
+
if (result.term !== this.currentTerm)
|
|
235
|
+
return;
|
|
236
|
+
if (result.voteGranted) {
|
|
237
|
+
this.votesReceived.add(fromPeerId);
|
|
238
|
+
// Check majority (votesReceived includes self)
|
|
239
|
+
const totalCluster = this.peers.length + 1;
|
|
240
|
+
if (this.votesReceived.size >= Math.floor(totalCluster / 2) + 1) {
|
|
241
|
+
this.becomeLeader();
|
|
242
|
+
}
|
|
243
|
+
}
|
|
244
|
+
}
|
|
245
|
+
/** Check if a candidate's log is at least as up-to-date as ours */
|
|
246
|
+
isLogUpToDate(candidateLastTerm, candidateLastIndex) {
|
|
247
|
+
const myLastTerm = this.log.lastTerm();
|
|
248
|
+
if (candidateLastTerm !== myLastTerm) {
|
|
249
|
+
return candidateLastTerm > myLastTerm;
|
|
250
|
+
}
|
|
251
|
+
return candidateLastIndex >= this.log.lastIndex();
|
|
252
|
+
}
|
|
253
|
+
// --- AppendEntries handling ---
|
|
254
|
+
handleAppendEntries(_fromPeerId, args) {
|
|
255
|
+
const reply = {
|
|
256
|
+
type: 'AppendEntriesResult',
|
|
257
|
+
term: this.currentTerm,
|
|
258
|
+
success: false,
|
|
259
|
+
responderId: this.id,
|
|
260
|
+
matchIndex: 0,
|
|
261
|
+
};
|
|
262
|
+
// Reject if leader's term is behind
|
|
263
|
+
if (args.term < this.currentTerm) {
|
|
264
|
+
this.sendMessage(args.leaderId, reply);
|
|
265
|
+
return;
|
|
266
|
+
}
|
|
267
|
+
// Valid leader — become follower if not already
|
|
268
|
+
if (this.role !== 'follower') {
|
|
269
|
+
this.becomeFollower();
|
|
270
|
+
}
|
|
271
|
+
else {
|
|
272
|
+
this.resetElectionTimer();
|
|
273
|
+
}
|
|
274
|
+
if (this.leaderId !== args.leaderId) {
|
|
275
|
+
this.leaderId = args.leaderId;
|
|
276
|
+
this.emit('leaderChanged', args.leaderId);
|
|
277
|
+
}
|
|
278
|
+
// Check log consistency
|
|
279
|
+
if (args.prevLogIndex > 0) {
|
|
280
|
+
const prevTerm = this.log.getTerm(args.prevLogIndex);
|
|
281
|
+
if (prevTerm === 0 && args.prevLogIndex > this.log.lastIndex()) {
|
|
282
|
+
// We don't have an entry at prevLogIndex
|
|
283
|
+
this.resetElectionTimer();
|
|
284
|
+
this.sendMessage(args.leaderId, reply);
|
|
285
|
+
return;
|
|
286
|
+
}
|
|
287
|
+
if (prevTerm !== args.prevLogTerm) {
|
|
288
|
+
// Conflict: delete this entry and everything after
|
|
289
|
+
this.log.truncateFrom(args.prevLogIndex);
|
|
290
|
+
this.resetElectionTimer();
|
|
291
|
+
this.sendMessage(args.leaderId, reply);
|
|
292
|
+
return;
|
|
293
|
+
}
|
|
294
|
+
}
|
|
295
|
+
// Append new entries (handle conflicts and skip duplicates)
|
|
296
|
+
if (args.entries.length > 0) {
|
|
297
|
+
for (let i = 0; i < args.entries.length; i++) {
|
|
298
|
+
const entryIndex = args.prevLogIndex + 1 + i;
|
|
299
|
+
const existing = this.log.getEntry(entryIndex);
|
|
300
|
+
if (existing) {
|
|
301
|
+
if (existing.term !== args.entries[i].term) {
|
|
302
|
+
// Conflict: truncate from here and append the rest
|
|
303
|
+
this.log.truncateFrom(entryIndex);
|
|
304
|
+
this.log.append(args.entries.slice(i));
|
|
305
|
+
break;
|
|
306
|
+
}
|
|
307
|
+
// Same term — entry already present, skip
|
|
308
|
+
}
|
|
309
|
+
else {
|
|
310
|
+
// No existing entry — append remaining
|
|
311
|
+
this.log.append(args.entries.slice(i));
|
|
312
|
+
break;
|
|
313
|
+
}
|
|
314
|
+
}
|
|
315
|
+
}
|
|
316
|
+
// Update commit index
|
|
317
|
+
if (args.leaderCommit > this.commitIndex) {
|
|
318
|
+
this.commitIndex = Math.min(args.leaderCommit, this.log.lastIndex());
|
|
319
|
+
this.applyCommitted();
|
|
320
|
+
}
|
|
321
|
+
reply.success = true;
|
|
322
|
+
reply.matchIndex = args.prevLogIndex + args.entries.length;
|
|
323
|
+
this.resetElectionTimer();
|
|
324
|
+
this.sendMessage(args.leaderId, reply);
|
|
325
|
+
}
|
|
326
|
+
handleAppendEntriesResult(result) {
|
|
327
|
+
var _a;
|
|
328
|
+
if (this.role !== 'leader')
|
|
329
|
+
return;
|
|
330
|
+
if (result.term !== this.currentTerm)
|
|
331
|
+
return;
|
|
332
|
+
const peer = result.responderId;
|
|
333
|
+
if (result.success) {
|
|
334
|
+
// Update nextIndex and matchIndex
|
|
335
|
+
if (result.matchIndex > 0) {
|
|
336
|
+
this.nextIndex.set(peer, result.matchIndex + 1);
|
|
337
|
+
this.matchIndex.set(peer, result.matchIndex);
|
|
338
|
+
}
|
|
339
|
+
this.advanceCommitIndex();
|
|
340
|
+
}
|
|
341
|
+
else {
|
|
342
|
+
// Decrement nextIndex and retry
|
|
343
|
+
const ni = (_a = this.nextIndex.get(peer)) !== null && _a !== void 0 ? _a : 1;
|
|
344
|
+
this.nextIndex.set(peer, Math.max(1, ni - 1));
|
|
345
|
+
this.sendAppendEntriesTo(peer);
|
|
346
|
+
}
|
|
347
|
+
}
|
|
348
|
+
// --- Log replication ---
|
|
349
|
+
sendAppendEntriesToAll() {
|
|
350
|
+
for (const peer of this.peers) {
|
|
351
|
+
this.sendAppendEntriesTo(peer);
|
|
352
|
+
}
|
|
353
|
+
}
|
|
354
|
+
sendAppendEntriesTo(peer) {
|
|
355
|
+
var _a;
|
|
356
|
+
const ni = (_a = this.nextIndex.get(peer)) !== null && _a !== void 0 ? _a : this.log.lastIndex() + 1;
|
|
357
|
+
const prevLogIndex = ni - 1;
|
|
358
|
+
const prevLogTerm = this.log.getTerm(prevLogIndex);
|
|
359
|
+
const lastIndex = this.log.lastIndex();
|
|
360
|
+
const entries = ni <= lastIndex ? this.log.getEntries(ni, lastIndex) : [];
|
|
361
|
+
const args = {
|
|
362
|
+
type: 'AppendEntries',
|
|
363
|
+
term: this.currentTerm,
|
|
364
|
+
leaderId: this.id,
|
|
365
|
+
prevLogIndex,
|
|
366
|
+
prevLogTerm,
|
|
367
|
+
entries,
|
|
368
|
+
leaderCommit: this.commitIndex,
|
|
369
|
+
};
|
|
370
|
+
this.sendMessage(peer, args);
|
|
371
|
+
}
|
|
372
|
+
/** Advance commitIndex if a majority of matchIndex values allow it */
|
|
373
|
+
advanceCommitIndex() {
|
|
374
|
+
var _a;
|
|
375
|
+
const lastIndex = this.log.lastIndex();
|
|
376
|
+
for (let n = lastIndex; n > this.commitIndex; n--) {
|
|
377
|
+
// Only commit entries from the current term
|
|
378
|
+
if (this.log.getTerm(n) !== this.currentTerm)
|
|
379
|
+
continue;
|
|
380
|
+
// Count replicas (including self)
|
|
381
|
+
let replicaCount = 1; // self
|
|
382
|
+
for (const peer of this.peers) {
|
|
383
|
+
if (((_a = this.matchIndex.get(peer)) !== null && _a !== void 0 ? _a : 0) >= n) {
|
|
384
|
+
replicaCount++;
|
|
385
|
+
}
|
|
386
|
+
}
|
|
387
|
+
const majority = Math.floor((this.peers.length + 1) / 2) + 1;
|
|
388
|
+
if (replicaCount >= majority) {
|
|
389
|
+
this.commitIndex = n;
|
|
390
|
+
this.applyCommitted();
|
|
391
|
+
break;
|
|
392
|
+
}
|
|
393
|
+
}
|
|
394
|
+
}
|
|
395
|
+
/** Apply all committed but not-yet-applied entries */
|
|
396
|
+
applyCommitted() {
|
|
397
|
+
while (this.lastApplied < this.commitIndex) {
|
|
398
|
+
this.lastApplied++;
|
|
399
|
+
const entry = this.log.getEntry(this.lastApplied);
|
|
400
|
+
if (entry) {
|
|
401
|
+
this.emit('committed', entry, this.lastApplied);
|
|
402
|
+
}
|
|
403
|
+
}
|
|
404
|
+
}
|
|
405
|
+
// --- Timer management ---
|
|
406
|
+
resetElectionTimer() {
|
|
407
|
+
this.clearElectionTimer();
|
|
408
|
+
const timeout = this.electionTimeoutMin +
|
|
409
|
+
Math.random() * (this.electionTimeoutMax - this.electionTimeoutMin);
|
|
410
|
+
this.electionTimer = setTimeout(() => {
|
|
411
|
+
if (!this.destroyed) {
|
|
412
|
+
this.becomeCandidate();
|
|
413
|
+
}
|
|
414
|
+
}, timeout);
|
|
415
|
+
}
|
|
416
|
+
clearElectionTimer() {
|
|
417
|
+
if (this.electionTimer !== null) {
|
|
418
|
+
clearTimeout(this.electionTimer);
|
|
419
|
+
this.electionTimer = null;
|
|
420
|
+
}
|
|
421
|
+
}
|
|
422
|
+
clearHeartbeatTimer() {
|
|
423
|
+
if (this.heartbeatTimer !== null) {
|
|
424
|
+
clearInterval(this.heartbeatTimer);
|
|
425
|
+
this.heartbeatTimer = null;
|
|
426
|
+
}
|
|
427
|
+
}
|
|
428
|
+
// --- Persistence ---
|
|
429
|
+
persistState() {
|
|
430
|
+
this.log.saveState({
|
|
431
|
+
currentTerm: this.currentTerm,
|
|
432
|
+
votedFor: this.votedFor,
|
|
433
|
+
});
|
|
434
|
+
}
|
|
435
|
+
// --- Event emission ---
|
|
436
|
+
emit(event, ...args) {
|
|
437
|
+
for (const listener of this.eventListeners[event]) {
|
|
438
|
+
listener(...args);
|
|
439
|
+
}
|
|
440
|
+
}
|
|
441
|
+
}
|
|
442
|
+
exports.RaftNode = RaftNode;
|
|
443
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"RaftNode.js","sourceRoot":"","sources":["../../../src/raft/RaftNode.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
import { LogEntry, RaftPersistentState } from '../types';
|
|
2
|
+
import { RaftLog } from './RaftLog';
|
|
3
|
+
/**
|
|
4
|
+
* In-memory Raft log implementation.
|
|
5
|
+
* All state is lost on page refresh — suitable for ephemeral sessions.
|
|
6
|
+
*/
|
|
7
|
+
export declare class InMemoryRaftLog<T = unknown> implements RaftLog<T> {
|
|
8
|
+
private entries;
|
|
9
|
+
private state;
|
|
10
|
+
loadState(): RaftPersistentState;
|
|
11
|
+
saveState(state: RaftPersistentState): void;
|
|
12
|
+
length(): number;
|
|
13
|
+
getEntry(index: number): LogEntry<T> | undefined;
|
|
14
|
+
getTerm(index: number): number;
|
|
15
|
+
append(entries: LogEntry<T>[]): void;
|
|
16
|
+
truncateFrom(index: number): void;
|
|
17
|
+
getEntries(startIndex: number, endIndex: number): LogEntry<T>[];
|
|
18
|
+
lastIndex(): number;
|
|
19
|
+
lastTerm(): number;
|
|
20
|
+
}
|
|
@@ -0,0 +1,55 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.InMemoryRaftLog = void 0;
|
|
4
|
+
/**
|
|
5
|
+
* In-memory Raft log implementation.
|
|
6
|
+
* All state is lost on page refresh — suitable for ephemeral sessions.
|
|
7
|
+
*/
|
|
8
|
+
class InMemoryRaftLog {
|
|
9
|
+
constructor() {
|
|
10
|
+
this.entries = [];
|
|
11
|
+
this.state = { currentTerm: 0, votedFor: null };
|
|
12
|
+
}
|
|
13
|
+
loadState() {
|
|
14
|
+
return Object.assign({}, this.state);
|
|
15
|
+
}
|
|
16
|
+
saveState(state) {
|
|
17
|
+
this.state = Object.assign({}, state);
|
|
18
|
+
}
|
|
19
|
+
length() {
|
|
20
|
+
return this.entries.length;
|
|
21
|
+
}
|
|
22
|
+
getEntry(index) {
|
|
23
|
+
if (index < 1 || index > this.entries.length) {
|
|
24
|
+
return undefined;
|
|
25
|
+
}
|
|
26
|
+
return this.entries[index - 1];
|
|
27
|
+
}
|
|
28
|
+
getTerm(index) {
|
|
29
|
+
var _a, _b;
|
|
30
|
+
return (_b = (_a = this.getEntry(index)) === null || _a === void 0 ? void 0 : _a.term) !== null && _b !== void 0 ? _b : 0;
|
|
31
|
+
}
|
|
32
|
+
append(entries) {
|
|
33
|
+
this.entries.push(...entries);
|
|
34
|
+
}
|
|
35
|
+
truncateFrom(index) {
|
|
36
|
+
if (index < 1)
|
|
37
|
+
return;
|
|
38
|
+
this.entries.length = index - 1;
|
|
39
|
+
}
|
|
40
|
+
getEntries(startIndex, endIndex) {
|
|
41
|
+
if (startIndex > endIndex || startIndex > this.entries.length)
|
|
42
|
+
return [];
|
|
43
|
+
return this.entries.slice(startIndex - 1, endIndex);
|
|
44
|
+
}
|
|
45
|
+
lastIndex() {
|
|
46
|
+
return this.entries.length;
|
|
47
|
+
}
|
|
48
|
+
lastTerm() {
|
|
49
|
+
if (this.entries.length === 0)
|
|
50
|
+
return 0;
|
|
51
|
+
return this.entries[this.entries.length - 1].term;
|
|
52
|
+
}
|
|
53
|
+
}
|
|
54
|
+
exports.InMemoryRaftLog = InMemoryRaftLog;
|
|
55
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiSW5NZW1vcnlSYWZ0TG9nLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vc3JjL3JhZnQvbG9nL0luTWVtb3J5UmFmdExvZy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFJQTs7O0dBR0c7QUFDSCxNQUFhLGVBQWU7SUFBNUI7UUFDVSxZQUFPLEdBQWtCLEVBQUUsQ0FBQztRQUM1QixVQUFLLEdBQXdCLEVBQUUsV0FBVyxFQUFFLENBQUMsRUFBRSxRQUFRLEVBQUUsSUFBSSxFQUFFLENBQUM7SUErQzFFLENBQUM7SUE3Q0MsU0FBUztRQUNQLHlCQUFZLElBQUksQ0FBQyxLQUFLLEVBQUc7SUFDM0IsQ0FBQztJQUVELFNBQVMsQ0FBQyxLQUEwQjtRQUNsQyxJQUFJLENBQUMsS0FBSyxxQkFBUSxLQUFLLENBQUUsQ0FBQztJQUM1QixDQUFDO0lBRUQsTUFBTTtRQUNKLE9BQU8sSUFBSSxDQUFDLE9BQU8sQ0FBQyxNQUFNLENBQUM7SUFDN0IsQ0FBQztJQUVELFFBQVEsQ0FBQyxLQUFhO1FBQ3BCLElBQUksS0FBSyxHQUFHLENBQUMsSUFBSSxLQUFLLEdBQUcsSUFBSSxDQUFDLE9BQU8sQ0FBQyxNQUFNLEVBQUU7WUFDNUMsT0FBTyxTQUFTLENBQUM7U0FDbEI7UUFDRCxPQUFPLElBQUksQ0FBQyxPQUFPLENBQUMsS0FBSyxHQUFHLENBQUMsQ0FBQyxDQUFDO0lBQ2pDLENBQUM7SUFFRCxPQUFPLENBQUMsS0FBYTs7UUFDbkIsT0FBTyxNQUFBLE1BQUEsSUFBSSxDQUFDLFFBQVEsQ0FBQyxLQUFLLENBQUMsMENBQUUsSUFBSSxtQ0FBSSxDQUFDLENBQUM7SUFDekMsQ0FBQztJQUVELE1BQU0sQ0FBQyxPQUFzQjtRQUMzQixJQUFJLENBQUMsT0FBTyxDQUFDLElBQUksQ0FBQyxHQUFHLE9BQU8sQ0FBQyxDQUFDO0lBQ2hDLENBQUM7SUFFRCxZQUFZLENBQUMsS0FBYTtRQUN4QixJQUFJLEtBQUssR0FBRyxDQUFDO1lBQUUsT0FBTztRQUN0QixJQUFJLENBQUMsT0FBTyxDQUFDLE1BQU0sR0FBRyxLQUFLLEdBQUcsQ0FBQyxDQUFDO0lBQ2xDLENBQUM7SUFFRCxVQUFVLENBQUMsVUFBa0IsRUFBRSxRQUFnQjtRQUM3QyxJQUFJLFVBQVUsR0FBRyxRQUFRLElBQUksVUFBVSxHQUFHLElBQUksQ0FBQyxPQUFPLENBQUMsTUFBTTtZQUFFLE9BQU8sRUFBRSxDQUFDO1FBQ3pFLE9BQU8sSUFBSSxDQUFDLE9BQU8sQ0FBQyxLQUFLLENBQUMsVUFBVSxHQUFHLENBQUMsRUFBRSxRQUFRLENBQUMsQ0FBQztJQUN0RCxDQUFDO0lBRUQsU0FBUztRQUNQLE9BQU8sSUFBSSxDQUFDLE9BQU8sQ0FBQyxNQUFNLENBQUM7SUFDN0IsQ0FBQztJQUVELFFBQVE7UUFDTixJQUFJLElBQUksQ0FBQyxPQUFPLENBQUMsTUFBTSxLQUFLLENBQUM7WUFBRSxPQUFPLENBQUMsQ0FBQztRQUN4QyxPQUFPLElBQUksQ0FBQyxPQUFPLENBQUMsSUFBSSxDQUFDLE9BQU8sQ0FBQyxNQUFNLEdBQUcsQ0FBQyxDQUFDLENBQUMsSUFBSSxDQUFDO0lBQ3BELENBQUM7Q0FDRjtBQWpERCwwQ0FpREMifQ==
|
|
@@ -0,0 +1,9 @@
|
|
|
1
|
+
import { StorageRaftLog } from './StorageRaftLog';
|
|
2
|
+
/**
|
|
3
|
+
* localStorage-backed Raft log implementation.
|
|
4
|
+
* Survives page refreshes — suitable for durable sessions where peers
|
|
5
|
+
* may rejoin after a restart.
|
|
6
|
+
*/
|
|
7
|
+
export declare class LocalStorageRaftLog<T = unknown> extends StorageRaftLog<T> {
|
|
8
|
+
constructor(namespace: string);
|
|
9
|
+
}
|
|
@@ -0,0 +1,16 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.LocalStorageRaftLog = void 0;
|
|
4
|
+
const StorageRaftLog_1 = require("./StorageRaftLog");
|
|
5
|
+
/**
|
|
6
|
+
* localStorage-backed Raft log implementation.
|
|
7
|
+
* Survives page refreshes — suitable for durable sessions where peers
|
|
8
|
+
* may rejoin after a restart.
|
|
9
|
+
*/
|
|
10
|
+
class LocalStorageRaftLog extends StorageRaftLog_1.StorageRaftLog {
|
|
11
|
+
constructor(namespace) {
|
|
12
|
+
super(localStorage, namespace);
|
|
13
|
+
}
|
|
14
|
+
}
|
|
15
|
+
exports.LocalStorageRaftLog = LocalStorageRaftLog;
|
|
16
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiTG9jYWxTdG9yYWdlUmFmdExvZy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3NyYy9yYWZ0L2xvZy9Mb2NhbFN0b3JhZ2VSYWZ0TG9nLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUFBLHFEQUFrRDtBQUVsRDs7OztHQUlHO0FBQ0gsTUFBYSxtQkFBaUMsU0FBUSwrQkFBaUI7SUFDckUsWUFBWSxTQUFpQjtRQUMzQixLQUFLLENBQUMsWUFBWSxFQUFFLFNBQVMsQ0FBQyxDQUFDO0lBQ2pDLENBQUM7Q0FDRjtBQUpELGtEQUlDIn0=
|
|
@@ -0,0 +1,30 @@
|
|
|
1
|
+
import { LogEntry, RaftPersistentState } from '../types';
|
|
2
|
+
/**
|
|
3
|
+
* Abstract log persistence interface for the Raft consensus algorithm.
|
|
4
|
+
* Raft log indices are 1-based.
|
|
5
|
+
*/
|
|
6
|
+
export interface RaftLog<T = unknown> {
|
|
7
|
+
/** Load persistent state (currentTerm, votedFor). Returns defaults if not yet stored. */
|
|
8
|
+
loadState(): RaftPersistentState;
|
|
9
|
+
/** Persist the current term and votedFor */
|
|
10
|
+
saveState(state: RaftPersistentState): void;
|
|
11
|
+
/** Number of entries in the log */
|
|
12
|
+
length(): number;
|
|
13
|
+
/** Get the entry at the given 1-based index, or undefined if out of range */
|
|
14
|
+
getEntry(index: number): LogEntry<T> | undefined;
|
|
15
|
+
/** Get the term of the entry at the given 1-based index, or 0 if out of range */
|
|
16
|
+
getTerm(index: number): number;
|
|
17
|
+
/** Append one or more entries to the end of the log */
|
|
18
|
+
append(entries: LogEntry<T>[]): void;
|
|
19
|
+
/**
|
|
20
|
+
* Delete all entries from the given 1-based index onward (inclusive).
|
|
21
|
+
* Used when resolving log conflicts.
|
|
22
|
+
*/
|
|
23
|
+
truncateFrom(index: number): void;
|
|
24
|
+
/** Get all entries from startIndex to endIndex (both inclusive, 1-based) */
|
|
25
|
+
getEntries(startIndex: number, endIndex: number): LogEntry<T>[];
|
|
26
|
+
/** Get the index of the last entry (0 if log is empty) */
|
|
27
|
+
lastIndex(): number;
|
|
28
|
+
/** Get the term of the last entry (0 if log is empty) */
|
|
29
|
+
lastTerm(): number;
|
|
30
|
+
}
|
|
@@ -0,0 +1,3 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiUmFmdExvZy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3NyYy9yYWZ0L2xvZy9SYWZ0TG9nLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiIifQ==
|