hungry-ghost-hive 0.44.0 → 0.46.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/dist/agents/base-agent.d.ts +1 -0
- package/dist/agents/base-agent.d.ts.map +1 -1
- package/dist/agents/base-agent.js +4 -0
- package/dist/agents/base-agent.js.map +1 -1
- package/dist/agents/intermediate.js +2 -2
- package/dist/agents/intermediate.js.map +1 -1
- package/dist/agents/junior.js +2 -2
- package/dist/agents/junior.js.map +1 -1
- package/dist/agents/qa.d.ts.map +1 -1
- package/dist/agents/qa.js +5 -5
- package/dist/agents/qa.js.map +1 -1
- package/dist/agents/senior.d.ts.map +1 -1
- package/dist/agents/senior.js +5 -5
- package/dist/agents/senior.js.map +1 -1
- package/dist/agents/tech-lead.d.ts.map +1 -1
- package/dist/agents/tech-lead.js +4 -2
- package/dist/agents/tech-lead.js.map +1 -1
- package/dist/cli/commands/assign.d.ts.map +1 -1
- package/dist/cli/commands/assign.js +4 -2
- package/dist/cli/commands/assign.js.map +1 -1
- package/dist/cli/commands/assign.test.js +5 -0
- package/dist/cli/commands/assign.test.js.map +1 -1
- package/dist/cli/commands/cluster.d.ts.map +1 -1
- package/dist/cli/commands/cluster.js +348 -1
- package/dist/cli/commands/cluster.js.map +1 -1
- package/dist/cli/commands/cluster.test.js +313 -9
- package/dist/cli/commands/cluster.test.js.map +1 -1
- package/dist/cli/commands/manager/handoff-recovery.d.ts.map +1 -1
- package/dist/cli/commands/manager/handoff-recovery.js +4 -2
- package/dist/cli/commands/manager/handoff-recovery.js.map +1 -1
- package/dist/cli/commands/manager/index.d.ts.map +1 -1
- package/dist/cli/commands/manager/index.js +16 -12
- package/dist/cli/commands/manager/index.js.map +1 -1
- package/dist/cli/commands/manager/tech-lead-lifecycle.d.ts.map +1 -1
- package/dist/cli/commands/manager/tech-lead-lifecycle.js +4 -2
- package/dist/cli/commands/manager/tech-lead-lifecycle.js.map +1 -1
- package/dist/cli/commands/msg.d.ts.map +1 -1
- package/dist/cli/commands/msg.js +8 -7
- package/dist/cli/commands/msg.js.map +1 -1
- package/dist/cli/commands/my-stories.js +3 -3
- package/dist/cli/commands/my-stories.js.map +1 -1
- package/dist/cli/commands/nuke.d.ts.map +1 -1
- package/dist/cli/commands/nuke.js +18 -7
- package/dist/cli/commands/nuke.js.map +1 -1
- package/dist/cli/commands/nuke.test.js +24 -0
- package/dist/cli/commands/nuke.test.js.map +1 -1
- package/dist/cli/commands/req-spawn.test.d.ts +2 -0
- package/dist/cli/commands/req-spawn.test.d.ts.map +1 -0
- package/dist/cli/commands/req-spawn.test.js +116 -0
- package/dist/cli/commands/req-spawn.test.js.map +1 -0
- package/dist/cli/commands/req.d.ts +1 -1
- package/dist/cli/commands/req.d.ts.map +1 -1
- package/dist/cli/commands/req.js +28 -18
- package/dist/cli/commands/req.js.map +1 -1
- package/dist/cli/commands/stories.js +3 -3
- package/dist/cli/commands/stories.js.map +1 -1
- package/dist/cli/dashboard/panels/agents.d.ts.map +1 -1
- package/dist/cli/dashboard/panels/agents.js +7 -3
- package/dist/cli/dashboard/panels/agents.js.map +1 -1
- package/dist/cluster/cluster-http-server.d.ts +32 -0
- package/dist/cluster/cluster-http-server.d.ts.map +1 -1
- package/dist/cluster/cluster-http-server.js +42 -0
- package/dist/cluster/cluster-http-server.js.map +1 -1
- package/dist/cluster/distributed-runtime-coverage.test.js +9 -0
- package/dist/cluster/distributed-runtime-coverage.test.js.map +1 -1
- package/dist/cluster/distributed-system.test.js +135 -0
- package/dist/cluster/distributed-system.test.js.map +1 -1
- package/dist/cluster/events.d.ts +23 -0
- package/dist/cluster/events.d.ts.map +1 -1
- package/dist/cluster/events.js +74 -0
- package/dist/cluster/events.js.map +1 -1
- package/dist/cluster/heartbeat-manager.d.ts +2 -0
- package/dist/cluster/heartbeat-manager.d.ts.map +1 -1
- package/dist/cluster/heartbeat-manager.js +42 -6
- package/dist/cluster/heartbeat-manager.js.map +1 -1
- package/dist/cluster/membership.test.d.ts +2 -0
- package/dist/cluster/membership.test.d.ts.map +1 -0
- package/dist/cluster/membership.test.js +416 -0
- package/dist/cluster/membership.test.js.map +1 -0
- package/dist/cluster/partition-safety.test.d.ts +2 -0
- package/dist/cluster/partition-safety.test.d.ts.map +1 -0
- package/dist/cluster/partition-safety.test.js +440 -0
- package/dist/cluster/partition-safety.test.js.map +1 -0
- package/dist/cluster/raft-state-machine.d.ts +33 -1
- package/dist/cluster/raft-state-machine.d.ts.map +1 -1
- package/dist/cluster/raft-state-machine.js +65 -3
- package/dist/cluster/raft-state-machine.js.map +1 -1
- package/dist/cluster/raft-store.d.ts +26 -1
- package/dist/cluster/raft-store.d.ts.map +1 -1
- package/dist/cluster/raft-store.js +137 -0
- package/dist/cluster/raft-store.js.map +1 -1
- package/dist/cluster/replication-lag.test.d.ts +2 -0
- package/dist/cluster/replication-lag.test.d.ts.map +1 -0
- package/dist/cluster/replication-lag.test.js +239 -0
- package/dist/cluster/replication-lag.test.js.map +1 -0
- package/dist/cluster/replication.d.ts +2 -2
- package/dist/cluster/replication.d.ts.map +1 -1
- package/dist/cluster/replication.js +1 -1
- package/dist/cluster/replication.js.map +1 -1
- package/dist/cluster/runtime.d.ts +78 -0
- package/dist/cluster/runtime.d.ts.map +1 -1
- package/dist/cluster/runtime.js +400 -13
- package/dist/cluster/runtime.js.map +1 -1
- package/dist/cluster/state-recovery.test.d.ts +2 -0
- package/dist/cluster/state-recovery.test.d.ts.map +1 -0
- package/dist/cluster/state-recovery.test.js +310 -0
- package/dist/cluster/state-recovery.test.js.map +1 -0
- package/dist/cluster/types.d.ts +30 -0
- package/dist/cluster/types.d.ts.map +1 -1
- package/dist/config/schema.d.ts +48 -0
- package/dist/config/schema.d.ts.map +1 -1
- package/dist/config/schema.js +11 -0
- package/dist/config/schema.js.map +1 -1
- package/dist/context-files/generator.d.ts +1 -1
- package/dist/context-files/generator.d.ts.map +1 -1
- package/dist/context-files/generator.js +4 -3
- package/dist/context-files/generator.js.map +1 -1
- package/dist/context-files/generator.test.js +51 -0
- package/dist/context-files/generator.test.js.map +1 -1
- package/dist/context-files/index.test.js +1 -0
- package/dist/context-files/index.test.js.map +1 -1
- package/dist/db/client.d.ts +1 -0
- package/dist/db/client.d.ts.map +1 -1
- package/dist/db/client.js +6 -0
- package/dist/db/client.js.map +1 -1
- package/dist/db/migrations/015-add-story-markdown-path.sql +5 -0
- package/dist/db/queries/stories.d.ts +3 -3
- package/dist/db/queries/stories.d.ts.map +1 -1
- package/dist/db/queries/stories.js +23 -5
- package/dist/db/queries/stories.js.map +1 -1
- package/dist/db/queries/test-helpers.d.ts.map +1 -1
- package/dist/db/queries/test-helpers.js +1 -0
- package/dist/db/queries/test-helpers.js.map +1 -1
- package/dist/git/worktree.d.ts.map +1 -1
- package/dist/git/worktree.js +7 -0
- package/dist/git/worktree.js.map +1 -1
- package/dist/git/worktree.test.js +30 -0
- package/dist/git/worktree.test.js.map +1 -1
- package/dist/orchestrator/orphan-recovery.d.ts +1 -1
- package/dist/orchestrator/orphan-recovery.d.ts.map +1 -1
- package/dist/orchestrator/orphan-recovery.js +4 -4
- package/dist/orchestrator/orphan-recovery.js.map +1 -1
- package/dist/orchestrator/prompt-templates.d.ts +6 -2
- package/dist/orchestrator/prompt-templates.d.ts.map +1 -1
- package/dist/orchestrator/prompt-templates.js +61 -16
- package/dist/orchestrator/prompt-templates.js.map +1 -1
- package/dist/orchestrator/prompt-templates.test.js +214 -0
- package/dist/orchestrator/prompt-templates.test.js.map +1 -1
- package/dist/orchestrator/scheduler.d.ts +1 -0
- package/dist/orchestrator/scheduler.d.ts.map +1 -1
- package/dist/orchestrator/scheduler.js +30 -17
- package/dist/orchestrator/scheduler.js.map +1 -1
- package/dist/orchestrator/scheduler.test.js +98 -6
- package/dist/orchestrator/scheduler.test.js.map +1 -1
- package/dist/tmux/manager.d.ts +7 -6
- package/dist/tmux/manager.d.ts.map +1 -1
- package/dist/tmux/manager.js +29 -13
- package/dist/tmux/manager.js.map +1 -1
- package/dist/utils/instance.d.ts +32 -0
- package/dist/utils/instance.d.ts.map +1 -0
- package/dist/utils/instance.js +82 -0
- package/dist/utils/instance.js.map +1 -0
- package/dist/utils/instance.test.d.ts +2 -0
- package/dist/utils/instance.test.d.ts.map +1 -0
- package/dist/utils/instance.test.js +103 -0
- package/dist/utils/instance.test.js.map +1 -0
- package/dist/utils/paths.d.ts +2 -0
- package/dist/utils/paths.d.ts.map +1 -1
- package/dist/utils/paths.js +2 -0
- package/dist/utils/paths.js.map +1 -1
- package/dist/utils/paths.test.js +6 -0
- package/dist/utils/paths.test.js.map +1 -1
- package/dist/utils/story-markdown.d.ts +16 -0
- package/dist/utils/story-markdown.d.ts.map +1 -0
- package/dist/utils/story-markdown.js +82 -0
- package/dist/utils/story-markdown.js.map +1 -0
- package/dist/utils/story-markdown.test.d.ts +2 -0
- package/dist/utils/story-markdown.test.d.ts.map +1 -0
- package/dist/utils/story-markdown.test.js +143 -0
- package/dist/utils/story-markdown.test.js.map +1 -0
- package/package.json +1 -1
- package/src/agents/base-agent.ts +5 -0
- package/src/agents/intermediate.ts +2 -2
- package/src/agents/junior.ts +2 -2
- package/src/agents/qa.ts +13 -8
- package/src/agents/senior.ts +21 -11
- package/src/agents/tech-lead.ts +24 -12
- package/src/cli/commands/assign.test.ts +5 -0
- package/src/cli/commands/assign.ts +4 -2
- package/src/cli/commands/cluster.test.ts +387 -9
- package/src/cli/commands/cluster.ts +486 -1
- package/src/cli/commands/manager/handoff-recovery.ts +4 -2
- package/src/cli/commands/manager/index.ts +16 -11
- package/src/cli/commands/manager/tech-lead-lifecycle.ts +5 -2
- package/src/cli/commands/msg.ts +8 -7
- package/src/cli/commands/my-stories.ts +22 -13
- package/src/cli/commands/nuke.test.ts +31 -0
- package/src/cli/commands/nuke.ts +18 -7
- package/src/cli/commands/req-spawn.test.ts +153 -0
- package/src/cli/commands/req.ts +40 -23
- package/src/cli/commands/stories.ts +22 -13
- package/src/cli/dashboard/panels/agents.ts +7 -3
- package/src/cluster/cluster-http-server.ts +80 -0
- package/src/cluster/distributed-runtime-coverage.test.ts +9 -0
- package/src/cluster/distributed-system.test.ts +168 -0
- package/src/cluster/events.ts +90 -0
- package/src/cluster/heartbeat-manager.ts +48 -6
- package/src/cluster/membership.test.ts +498 -0
- package/src/cluster/partition-safety.test.ts +523 -0
- package/src/cluster/raft-state-machine.ts +76 -4
- package/src/cluster/raft-store.ts +167 -1
- package/src/cluster/replication-lag.test.ts +284 -0
- package/src/cluster/replication.ts +6 -0
- package/src/cluster/runtime.ts +551 -12
- package/src/cluster/state-recovery.test.ts +420 -0
- package/src/cluster/types.ts +32 -0
- package/src/config/schema.ts +11 -0
- package/src/context-files/generator.test.ts +55 -0
- package/src/context-files/generator.ts +8 -7
- package/src/context-files/index.test.ts +1 -0
- package/src/db/client.ts +7 -0
- package/src/db/migrations/015-add-story-markdown-path.sql +5 -0
- package/src/db/queries/stories.ts +29 -5
- package/src/db/queries/test-helpers.ts +1 -0
- package/src/git/worktree.test.ts +43 -0
- package/src/git/worktree.ts +10 -0
- package/src/orchestrator/orphan-recovery.ts +32 -13
- package/src/orchestrator/prompt-templates.test.ts +267 -0
- package/src/orchestrator/prompt-templates.ts +69 -16
- package/src/orchestrator/scheduler.test.ts +130 -6
- package/src/orchestrator/scheduler.ts +66 -27
- package/src/tmux/manager.ts +42 -13
- package/src/utils/instance.test.ts +129 -0
- package/src/utils/instance.ts +95 -0
- package/src/utils/paths.test.ts +8 -0
- package/src/utils/paths.ts +3 -0
- package/src/utils/story-markdown.test.ts +176 -0
- package/src/utils/story-markdown.ts +94 -0
|
@@ -26,21 +26,27 @@ export class HeartbeatManager {
|
|
|
26
26
|
if (!this.deps.isActive())
|
|
27
27
|
return;
|
|
28
28
|
const { raft } = this.deps;
|
|
29
|
+
const peers = raft.getPeers();
|
|
29
30
|
const heartbeat = {
|
|
30
31
|
term: raft.currentTerm,
|
|
31
32
|
leader_id: this.config.node_id,
|
|
33
|
+
fencing_token: raft.getFencingToken(),
|
|
34
|
+
peers: peers.map(p => ({ id: p.id, url: p.url })),
|
|
32
35
|
};
|
|
33
36
|
raft.appendDurableEntry('heartbeat_sent', {
|
|
34
37
|
term: raft.currentTerm,
|
|
35
38
|
leader_id: this.config.node_id,
|
|
36
|
-
peer_count:
|
|
39
|
+
peer_count: peers.filter(peer => peer.id !== this.config.node_id).length,
|
|
37
40
|
});
|
|
38
|
-
await Promise.all(
|
|
41
|
+
await Promise.all(peers
|
|
39
42
|
.filter(peer => peer.id !== this.config.node_id)
|
|
40
43
|
.map(async (peer) => {
|
|
41
44
|
const response = await this.deps.postJson(peer, '/cluster/v1/election/heartbeat', heartbeat);
|
|
42
|
-
if (response
|
|
43
|
-
|
|
45
|
+
if (response) {
|
|
46
|
+
const remoteTerm = Math.max(response.term, response.fencing_token ?? 0);
|
|
47
|
+
if (remoteTerm > raft.currentTerm) {
|
|
48
|
+
raft.stepDown(remoteTerm, peer.id);
|
|
49
|
+
}
|
|
44
50
|
}
|
|
45
51
|
}));
|
|
46
52
|
}
|
|
@@ -49,8 +55,14 @@ export class HeartbeatManager {
|
|
|
49
55
|
const request = body;
|
|
50
56
|
const term = Number(request.term || 0);
|
|
51
57
|
const leaderId = typeof request.leader_id === 'string' ? request.leader_id : null;
|
|
58
|
+
const fencingToken = Number(request.fencing_token ?? term);
|
|
59
|
+
// Reject heartbeats from stale leaders
|
|
52
60
|
if (term < raft.currentTerm) {
|
|
53
|
-
return { term: raft.currentTerm, success: false };
|
|
61
|
+
return { term: raft.currentTerm, success: false, fencing_token: raft.getFencingToken() };
|
|
62
|
+
}
|
|
63
|
+
// Reject if fencing token doesn't match the heartbeat term
|
|
64
|
+
if (fencingToken < term) {
|
|
65
|
+
return { term: raft.currentTerm, success: false, fencing_token: raft.getFencingToken() };
|
|
54
66
|
}
|
|
55
67
|
const changed = term > raft.currentTerm || leaderId !== raft.leaderId || raft.role !== 'follower';
|
|
56
68
|
if (term > raft.currentTerm) {
|
|
@@ -61,14 +73,38 @@ export class HeartbeatManager {
|
|
|
61
73
|
raft.leaderId = leaderId;
|
|
62
74
|
raft.persistRaftState();
|
|
63
75
|
}
|
|
76
|
+
// Update lease: record that we received a valid heartbeat now
|
|
77
|
+
raft.lastHeartbeatReceivedAt = Date.now();
|
|
64
78
|
raft.resetElectionDeadline();
|
|
79
|
+
// Apply peer list from leader if present
|
|
80
|
+
const requestPeers = request.peers;
|
|
81
|
+
if (Array.isArray(requestPeers)) {
|
|
82
|
+
const parsed = parsePeerList(requestPeers);
|
|
83
|
+
if (parsed.length > 0) {
|
|
84
|
+
raft.setPeers(parsed);
|
|
85
|
+
this.deps.onPeersUpdated?.(parsed);
|
|
86
|
+
}
|
|
87
|
+
}
|
|
65
88
|
if (changed) {
|
|
66
89
|
raft.appendDurableEntry('heartbeat_received', {
|
|
67
90
|
term,
|
|
68
91
|
leader_id: leaderId,
|
|
92
|
+
fencing_token: fencingToken,
|
|
69
93
|
});
|
|
70
94
|
}
|
|
71
|
-
return { term: raft.currentTerm, success: true };
|
|
95
|
+
return { term: raft.currentTerm, success: true, fencing_token: raft.getFencingToken() };
|
|
96
|
+
}
|
|
97
|
+
}
|
|
98
|
+
function parsePeerList(input) {
|
|
99
|
+
const peers = [];
|
|
100
|
+
for (const item of input) {
|
|
101
|
+
if (!item || typeof item !== 'object')
|
|
102
|
+
continue;
|
|
103
|
+
const p = item;
|
|
104
|
+
if (typeof p.id === 'string' && typeof p.url === 'string') {
|
|
105
|
+
peers.push({ id: p.id, url: p.url });
|
|
106
|
+
}
|
|
72
107
|
}
|
|
108
|
+
return peers;
|
|
73
109
|
}
|
|
74
110
|
//# sourceMappingURL=heartbeat-manager.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"heartbeat-manager.js","sourceRoot":"","sources":["../../src/cluster/heartbeat-manager.ts"],"names":[],"mappings":"AAAA,6DAA6D;
|
|
1
|
+
{"version":3,"file":"heartbeat-manager.js","sourceRoot":"","sources":["../../src/cluster/heartbeat-manager.ts"],"names":[],"mappings":"AAAA,6DAA6D;AA0B7D,MAAM,OAAO,gBAAgB;IAIR;IACA;IAJX,cAAc,GAA0B,IAAI,CAAC;IAErD,YACmB,MAAqB,EACrB,IAA0B;QAD1B,WAAM,GAAN,MAAM,CAAe;QACrB,SAAI,GAAJ,IAAI,CAAsB;IAC1C,CAAC;IAEJ,kBAAkB;QAChB,IAAI,CAAC,cAAc,GAAG,WAAW,CAAC,GAAG,EAAE;YACrC,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,OAAO;gBAAE,OAAO;YACjC,IAAI,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,KAAK,QAAQ;gBAAE,OAAO;YAC7C,KAAK,IAAI,CAAC,cAAc,EAAE,CAAC,KAAK,CAAC,KAAK,CAAC,EAAE,CAAC,IAAI,CAAC,IAAI,CAAC,qBAAqB,CAAC,KAAK,CAAC,CAAC,CAAC;QACpF,CAAC,EAAE,IAAI,CAAC,MAAM,CAAC,qBAAqB,CAAC,CAAC;IACxC,CAAC;IAED,iBAAiB;QACf,IAAI,IAAI,CAAC,cAAc,EAAE,CAAC;YACxB,aAAa,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC;YACnC,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC;QAC7B,CAAC;IACH,CAAC;IAED,KAAK,CAAC,cAAc;QAClB,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,QAAQ,EAAE;YAAE,OAAO;QAElC,MAAM,EAAE,IAAI,EAAE,GAAG,IAAI,CAAC,IAAI,CAAC;QAC3B,MAAM,KAAK,GAAG,IAAI,CAAC,QAAQ,EAAE,CAAC;QAE9B,MAAM,SAAS,GAAqB;YAClC,IAAI,EAAE,IAAI,CAAC,WAAW;YACtB,SAAS,EAAE,IAAI,CAAC,MAAM,CAAC,OAAO;YAC9B,aAAa,EAAE,IAAI,CAAC,eAAe,EAAE;YACrC,KAAK,EAAE,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,EAAE,EAAE,EAAE,CAAC,CAAC,EAAE,EAAE,GAAG,EAAE,CAAC,CAAC,GAAG,EAAE,CAAC,CAAC;SAClD,CAAC;QAEF,IAAI,CAAC,kBAAkB,CAAC,gBAAgB,EAAE;YACxC,IAAI,EAAE,IAAI,CAAC,WAAW;YACtB,SAAS,EAAE,IAAI,CAAC,MAAM,CAAC,OAAO;YAC9B,UAAU,EAAE,KAAK,CAAC,MAAM,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,EAAE,KAAK,IAAI,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,MAAM;SACzE,CAAC,CAAC;QAEH,MAAM,OAAO,CAAC,GAAG,CACf,KAAK;aACF,MAAM,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,EAAE,KAAK,IAAI,CAAC,MAAM,CAAC,OAAO,CAAC;aAC/C,GAAG,CAAC,KAAK,EAAC,IAAI,EAAC,EAAE;YAChB,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,IAAI,CAAC,QAAQ,CACvC,IAAI,EACJ,gCAAgC,EAChC,SAAS,CACV,CAAC;YAEF,IAAI,QAAQ,EAAE,CAAC;gBACb,MAAM,UAAU,GAAG,IAAI,CAAC,GAAG,CAAC,QAAQ,CAAC,IAAI,EAAE,QAAQ,CAAC,aAAa,IAAI,CAAC,CAAC,CAAC;gBACxE,IAAI,UAAU,GAAG,IAAI,CAAC,WAAW,EAAE,CAAC;oBAClC,IAAI,CAAC,QAAQ,CAAC,UAAU,EAAE,IAAI,CAAC,EAAE,CAAC,CAAC;gBACrC,CAAC;YACH,CAAC;QACH,CAAC,CAAC,CACL,CAAC;IACJ,CAAC;IAED,eAAe,CAAC,IAAa;QAC3B,MAAM,EAAE,IAAI,EAAE,GAAG,IAAI,CAAC,IAAI,CAAC;QAE3B,MAAM,OAAO,GAAG,IAAiC,CAAC;QAClD,MAAM,IAAI,GAAG,MAAM,CAAC,OAAO,CAAC,IAAI,IAAI,CAAC,CAAC,CAAC;QACvC,MAAM,QAAQ,GAAG,OAAO,OAAO,CAAC,SAAS,KAAK,QAAQ,CAAC,CAAC,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC,CAAC,IAAI,CAAC;QAClF,MAAM,YAAY,GAAG,MAAM,CAAC,OAAO,CAAC,aAAa,IAAI,IAAI,CAAC,CAAC;QAE3D,uCAAuC;QACvC,IAAI,IAAI,GAAG,IAAI,CAAC,WAAW,EAAE,CAAC;YAC5B,OAAO,EAAE,IAAI,EAAE,IAAI,CAAC,WAAW,EAAE,OAAO,EAAE,KAAK,EAAE,aAAa,EAAE,IAAI,CAAC,eAAe,EAAE,EAAE,CAAC;QAC3F,CAAC;QAED,2DAA2D;QAC3D,IAAI,YAAY,GAAG,IAAI,EAAE,CAAC;YACxB,OAAO,EAAE,IAAI,EAAE,IAAI,CAAC,WAAW,EAAE,OAAO,EAAE,KAAK,EAAE,aAAa,EAAE,IAAI,CAAC,eAAe,EAAE,EAAE,CAAC;QAC3F,CAAC;QAED,MAAM,OAAO,GACX,IAAI,GAAG,IAAI,CAAC,WAAW,IAAI,QAAQ,KAAK,IAAI,CAAC,QAAQ,IAAI,IAAI,CAAC,IAAI,KAAK,UAAU,CAAC;QAEpF,IAAI,IAAI,GAAG,IAAI,CAAC,WAAW,EAAE,CAAC;YAC5B,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE,QAAQ,CAAC,CAAC;QAChC,CAAC;aAAM,CAAC;YACN,IAAI,CAAC,IAAI,GAAG,UAAU,CAAC;YACvB,IAAI,CAAC,QAAQ,GAAG,QAAQ,CAAC;YACzB,IAAI,CAAC,gBAAgB,EAAE,CAAC;QAC1B,CAAC;QAED,8DAA8D;QAC9D,IAAI,CAAC,uBAAuB,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QAC1C,IAAI,CAAC,qBAAqB,EAAE,CAAC;QAE7B,yCAAyC;QACzC,MAAM,YAAY,GAAI,OAA+B,CAAC,KAAK,CAAC;QAC5D,IAAI,KAAK,CAAC,OAAO,CAAC,YAAY,CAAC,EAAE,CAAC;YAChC,MAAM,MAAM,GAAG,aAAa,CAAC,YAAY,CAAC,CAAC;YAC3C,IAAI,MAAM,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBACtB,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC;gBACtB,IAAI,CAAC,IAAI,CAAC,cAAc,EAAE,CAAC,MAAM,CAAC,CAAC;YACrC,CAAC;QACH,CAAC;QAED,IAAI,OAAO,EAAE,CAAC;YACZ,IAAI,CAAC,kBAAkB,CAAC,oBAAoB,EAAE;gBAC5C,IAAI;gBACJ,SAAS,EAAE,QAAQ;gBACnB,aAAa,EAAE,YAAY;aAC5B,CAAC,CAAC;QACL,CAAC;QAED,OAAO,EAAE,IAAI,EAAE,IAAI,CAAC,WAAW,EAAE,OAAO,EAAE,IAAI,EAAE,aAAa,EAAE,IAAI,CAAC,eAAe,EAAE,EAAE,CAAC;IAC1F,CAAC;CACF;AAED,SAAS,aAAa,CAAC,KAAgB;IACrC,MAAM,KAAK,GAAwB,EAAE,CAAC;IACtC,KAAK,MAAM,IAAI,IAAI,KAAK,EAAE,CAAC;QACzB,IAAI,CAAC,IAAI,IAAI,OAAO,IAAI,KAAK,QAAQ;YAAE,SAAS;QAChD,MAAM,CAAC,GAAG,IAAuC,CAAC;QAClD,IAAI,OAAO,CAAC,CAAC,EAAE,KAAK,QAAQ,IAAI,OAAO,CAAC,CAAC,GAAG,KAAK,QAAQ,EAAE,CAAC;YAC1D,KAAK,CAAC,IAAI,CAAC,EAAE,EAAE,EAAE,CAAC,CAAC,EAAE,EAAE,GAAG,EAAE,CAAC,CAAC,GAAG,EAAE,CAAC,CAAC;QACvC,CAAC;IACH,CAAC;IACD,OAAO,KAAK,CAAC;AACf,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"membership.test.d.ts","sourceRoot":"","sources":["../../src/cluster/membership.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,416 @@
|
|
|
1
|
+
// Licensed under the Hungry Ghost Hive License. See LICENSE.
|
|
2
|
+
import { mkdirSync, mkdtempSync, rmSync } from 'fs';
|
|
3
|
+
import { createServer as createNetServer } from 'net';
|
|
4
|
+
import { tmpdir } from 'os';
|
|
5
|
+
import { join } from 'path';
|
|
6
|
+
import { afterEach, describe, expect, it } from 'vitest';
|
|
7
|
+
import { ClusterRuntime } from './runtime.js';
|
|
8
|
+
const tempRoots = [];
|
|
9
|
+
const activeRuntimes = [];
|
|
10
|
+
afterEach(async () => {
|
|
11
|
+
for (const runtime of activeRuntimes.splice(0)) {
|
|
12
|
+
try {
|
|
13
|
+
await runtime.stop();
|
|
14
|
+
}
|
|
15
|
+
catch {
|
|
16
|
+
// Best effort shutdown for test cleanup.
|
|
17
|
+
}
|
|
18
|
+
}
|
|
19
|
+
for (const root of tempRoots.splice(0)) {
|
|
20
|
+
rmSync(root, { recursive: true, force: true });
|
|
21
|
+
}
|
|
22
|
+
});
|
|
23
|
+
describe('dynamic membership join', () => {
|
|
24
|
+
it('leader accepts join request and adds peer to cluster', async () => {
|
|
25
|
+
if (!(await canListenOnLocalhost()))
|
|
26
|
+
return;
|
|
27
|
+
const fixture = await startRuntimeFixture({
|
|
28
|
+
node_id: 'leader-join',
|
|
29
|
+
election_timeout_min_ms: 80,
|
|
30
|
+
election_timeout_max_ms: 120,
|
|
31
|
+
heartbeat_interval_ms: 60,
|
|
32
|
+
});
|
|
33
|
+
await waitFor(() => fixture.runtime.getStatus().is_leader, 4000);
|
|
34
|
+
const res = await postJson(fixture.config.public_url, '/cluster/v1/membership/join', {
|
|
35
|
+
node_id: 'new-node',
|
|
36
|
+
url: 'http://127.0.0.1:9999',
|
|
37
|
+
});
|
|
38
|
+
expect(res.success).toBe(true);
|
|
39
|
+
expect(res.leader_id).toBe('leader-join');
|
|
40
|
+
expect(res.peers).toContainEqual({ id: 'new-node', url: 'http://127.0.0.1:9999' });
|
|
41
|
+
const status = fixture.runtime.getStatus();
|
|
42
|
+
expect(status.peers).toContainEqual({ id: 'new-node', url: 'http://127.0.0.1:9999' });
|
|
43
|
+
});
|
|
44
|
+
it('follower redirects join request to leader', async () => {
|
|
45
|
+
if (!(await canListenOnLocalhost()))
|
|
46
|
+
return;
|
|
47
|
+
const fixture = await startRuntimeFixture({
|
|
48
|
+
node_id: 'follower-join',
|
|
49
|
+
election_timeout_min_ms: 5000,
|
|
50
|
+
election_timeout_max_ms: 5000,
|
|
51
|
+
peers: [{ id: 'remote-leader', url: 'http://127.0.0.1:9998' }],
|
|
52
|
+
});
|
|
53
|
+
// Set the node as follower with a known leader
|
|
54
|
+
await postJson(fixture.config.public_url, '/cluster/v1/election/heartbeat', {
|
|
55
|
+
term: 3,
|
|
56
|
+
leader_id: 'remote-leader',
|
|
57
|
+
fencing_token: 3,
|
|
58
|
+
});
|
|
59
|
+
const res = await fetch(`${fixture.config.public_url}/cluster/v1/membership/join`, {
|
|
60
|
+
method: 'POST',
|
|
61
|
+
headers: { 'Content-Type': 'application/json' },
|
|
62
|
+
body: JSON.stringify({ node_id: 'joiner', url: 'http://127.0.0.1:9997' }),
|
|
63
|
+
});
|
|
64
|
+
expect(res.status).toBe(307);
|
|
65
|
+
const body = (await res.json());
|
|
66
|
+
expect(body.success).toBe(false);
|
|
67
|
+
expect(body.leader_id).toBe('remote-leader');
|
|
68
|
+
expect(body.leader_url).toBe('http://127.0.0.1:9998');
|
|
69
|
+
});
|
|
70
|
+
it('rejects join request with missing fields', async () => {
|
|
71
|
+
if (!(await canListenOnLocalhost()))
|
|
72
|
+
return;
|
|
73
|
+
const fixture = await startRuntimeFixture({ node_id: 'leader-join-bad' });
|
|
74
|
+
const res = await fetch(`${fixture.config.public_url}/cluster/v1/membership/join`, {
|
|
75
|
+
method: 'POST',
|
|
76
|
+
headers: { 'Content-Type': 'application/json' },
|
|
77
|
+
body: JSON.stringify({ node_id: 'missing-url' }),
|
|
78
|
+
});
|
|
79
|
+
expect(res.status).toBe(400);
|
|
80
|
+
});
|
|
81
|
+
it('updates url for existing peer on re-join', async () => {
|
|
82
|
+
if (!(await canListenOnLocalhost()))
|
|
83
|
+
return;
|
|
84
|
+
const fixture = await startRuntimeFixture({
|
|
85
|
+
node_id: 'leader-rejoin',
|
|
86
|
+
election_timeout_min_ms: 80,
|
|
87
|
+
election_timeout_max_ms: 120,
|
|
88
|
+
heartbeat_interval_ms: 60,
|
|
89
|
+
});
|
|
90
|
+
await waitFor(() => fixture.runtime.getStatus().is_leader, 4000);
|
|
91
|
+
// First add the peer
|
|
92
|
+
await postJson(fixture.config.public_url, '/cluster/v1/membership/join', {
|
|
93
|
+
node_id: 'existing-peer',
|
|
94
|
+
url: 'http://127.0.0.1:8000',
|
|
95
|
+
});
|
|
96
|
+
// Re-join with different URL
|
|
97
|
+
const res = await postJson(fixture.config.public_url, '/cluster/v1/membership/join', {
|
|
98
|
+
node_id: 'existing-peer',
|
|
99
|
+
url: 'http://127.0.0.1:9000',
|
|
100
|
+
});
|
|
101
|
+
expect(res.success).toBe(true);
|
|
102
|
+
expect(res.peers).toContainEqual({ id: 'existing-peer', url: 'http://127.0.0.1:9000' });
|
|
103
|
+
});
|
|
104
|
+
it('idempotent join with same url returns success', async () => {
|
|
105
|
+
if (!(await canListenOnLocalhost()))
|
|
106
|
+
return;
|
|
107
|
+
const fixture = await startRuntimeFixture({
|
|
108
|
+
node_id: 'leader-idem',
|
|
109
|
+
election_timeout_min_ms: 80,
|
|
110
|
+
election_timeout_max_ms: 120,
|
|
111
|
+
heartbeat_interval_ms: 60,
|
|
112
|
+
});
|
|
113
|
+
await waitFor(() => fixture.runtime.getStatus().is_leader, 4000);
|
|
114
|
+
// Add peer first
|
|
115
|
+
await postJson(fixture.config.public_url, '/cluster/v1/membership/join', {
|
|
116
|
+
node_id: 'peer-x',
|
|
117
|
+
url: 'http://127.0.0.1:7777',
|
|
118
|
+
});
|
|
119
|
+
// Join again with same details — idempotent
|
|
120
|
+
const res = await postJson(fixture.config.public_url, '/cluster/v1/membership/join', {
|
|
121
|
+
node_id: 'peer-x',
|
|
122
|
+
url: 'http://127.0.0.1:7777',
|
|
123
|
+
});
|
|
124
|
+
expect(res.success).toBe(true);
|
|
125
|
+
});
|
|
126
|
+
});
|
|
127
|
+
describe('dynamic membership leave', () => {
|
|
128
|
+
it('leader removes peer on leave request', async () => {
|
|
129
|
+
if (!(await canListenOnLocalhost()))
|
|
130
|
+
return;
|
|
131
|
+
const fixture = await startRuntimeFixture({
|
|
132
|
+
node_id: 'leader-leave',
|
|
133
|
+
election_timeout_min_ms: 80,
|
|
134
|
+
election_timeout_max_ms: 120,
|
|
135
|
+
heartbeat_interval_ms: 60,
|
|
136
|
+
});
|
|
137
|
+
await waitFor(() => fixture.runtime.getStatus().is_leader, 4000);
|
|
138
|
+
// Add peer first, then remove it
|
|
139
|
+
await postJson(fixture.config.public_url, '/cluster/v1/membership/join', {
|
|
140
|
+
node_id: 'departing-node',
|
|
141
|
+
url: 'http://127.0.0.1:8888',
|
|
142
|
+
});
|
|
143
|
+
const res = await postJson(fixture.config.public_url, '/cluster/v1/membership/leave', {
|
|
144
|
+
node_id: 'departing-node',
|
|
145
|
+
});
|
|
146
|
+
expect(res.success).toBe(true);
|
|
147
|
+
expect(res.peers).not.toContainEqual(expect.objectContaining({ id: 'departing-node' }));
|
|
148
|
+
const status = fixture.runtime.getStatus();
|
|
149
|
+
expect(status.peers.find(p => p.id === 'departing-node')).toBeUndefined();
|
|
150
|
+
});
|
|
151
|
+
it('follower rejects leave request', async () => {
|
|
152
|
+
if (!(await canListenOnLocalhost()))
|
|
153
|
+
return;
|
|
154
|
+
const fixture = await startRuntimeFixture({
|
|
155
|
+
node_id: 'follower-leave',
|
|
156
|
+
election_timeout_min_ms: 5000,
|
|
157
|
+
election_timeout_max_ms: 5000,
|
|
158
|
+
});
|
|
159
|
+
const res = await fetch(`${fixture.config.public_url}/cluster/v1/membership/leave`, {
|
|
160
|
+
method: 'POST',
|
|
161
|
+
headers: { 'Content-Type': 'application/json' },
|
|
162
|
+
body: JSON.stringify({ node_id: 'some-node' }),
|
|
163
|
+
});
|
|
164
|
+
expect(res.status).toBe(400);
|
|
165
|
+
const body = (await res.json());
|
|
166
|
+
expect(body.success).toBe(false);
|
|
167
|
+
});
|
|
168
|
+
it('leader cannot remove itself', async () => {
|
|
169
|
+
if (!(await canListenOnLocalhost()))
|
|
170
|
+
return;
|
|
171
|
+
const fixture = await startRuntimeFixture({
|
|
172
|
+
node_id: 'leader-self-leave',
|
|
173
|
+
election_timeout_min_ms: 80,
|
|
174
|
+
election_timeout_max_ms: 120,
|
|
175
|
+
heartbeat_interval_ms: 60,
|
|
176
|
+
});
|
|
177
|
+
await waitFor(() => fixture.runtime.getStatus().is_leader, 4000);
|
|
178
|
+
const res = await fetch(`${fixture.config.public_url}/cluster/v1/membership/leave`, {
|
|
179
|
+
method: 'POST',
|
|
180
|
+
headers: { 'Content-Type': 'application/json' },
|
|
181
|
+
body: JSON.stringify({ node_id: 'leader-self-leave' }),
|
|
182
|
+
});
|
|
183
|
+
expect(res.status).toBe(400);
|
|
184
|
+
});
|
|
185
|
+
it('leave for unknown node is a no-op success', async () => {
|
|
186
|
+
if (!(await canListenOnLocalhost()))
|
|
187
|
+
return;
|
|
188
|
+
const fixture = await startRuntimeFixture({
|
|
189
|
+
node_id: 'leader-unknown-leave',
|
|
190
|
+
election_timeout_min_ms: 80,
|
|
191
|
+
election_timeout_max_ms: 120,
|
|
192
|
+
heartbeat_interval_ms: 60,
|
|
193
|
+
});
|
|
194
|
+
await waitFor(() => fixture.runtime.getStatus().is_leader, 4000);
|
|
195
|
+
const res = await postJson(fixture.config.public_url, '/cluster/v1/membership/leave', {
|
|
196
|
+
node_id: 'ghost-node',
|
|
197
|
+
});
|
|
198
|
+
expect(res.success).toBe(true);
|
|
199
|
+
});
|
|
200
|
+
it('rejects leave request with missing node_id', async () => {
|
|
201
|
+
if (!(await canListenOnLocalhost()))
|
|
202
|
+
return;
|
|
203
|
+
const fixture = await startRuntimeFixture({ node_id: 'leader-leave-bad' });
|
|
204
|
+
const res = await fetch(`${fixture.config.public_url}/cluster/v1/membership/leave`, {
|
|
205
|
+
method: 'POST',
|
|
206
|
+
headers: { 'Content-Type': 'application/json' },
|
|
207
|
+
body: JSON.stringify({}),
|
|
208
|
+
});
|
|
209
|
+
expect(res.status).toBe(400);
|
|
210
|
+
});
|
|
211
|
+
});
|
|
212
|
+
describe('peer list propagation via heartbeat', () => {
|
|
213
|
+
it('leader propagates updated peer list to followers', async () => {
|
|
214
|
+
if (!(await canListenOnLocalhost()))
|
|
215
|
+
return;
|
|
216
|
+
const portLeader = await getFreePort();
|
|
217
|
+
const portFollower = await getFreePort();
|
|
218
|
+
const leaderConfig = await buildConfig({
|
|
219
|
+
node_id: 'leader-prop',
|
|
220
|
+
listen_port: portLeader,
|
|
221
|
+
public_url: `http://127.0.0.1:${portLeader}`,
|
|
222
|
+
peers: [{ id: 'follower-prop', url: `http://127.0.0.1:${portFollower}` }],
|
|
223
|
+
election_timeout_min_ms: 80,
|
|
224
|
+
election_timeout_max_ms: 120,
|
|
225
|
+
heartbeat_interval_ms: 60,
|
|
226
|
+
});
|
|
227
|
+
const followerConfig = await buildConfig({
|
|
228
|
+
node_id: 'follower-prop',
|
|
229
|
+
listen_port: portFollower,
|
|
230
|
+
public_url: `http://127.0.0.1:${portFollower}`,
|
|
231
|
+
peers: [{ id: 'leader-prop', url: `http://127.0.0.1:${portLeader}` }],
|
|
232
|
+
election_timeout_min_ms: 5000,
|
|
233
|
+
election_timeout_max_ms: 5000,
|
|
234
|
+
});
|
|
235
|
+
const leaderFixture = await startRuntimeWithConfig(leaderConfig);
|
|
236
|
+
const followerFixture = await startRuntimeWithConfig(followerConfig);
|
|
237
|
+
// Wait for leader election
|
|
238
|
+
await waitFor(() => leaderFixture.runtime.getStatus().is_leader, 4000);
|
|
239
|
+
// Add a new peer via the leader
|
|
240
|
+
await postJson(leaderFixture.config.public_url, '/cluster/v1/membership/join', {
|
|
241
|
+
node_id: 'new-node-prop',
|
|
242
|
+
url: 'http://127.0.0.1:7777',
|
|
243
|
+
});
|
|
244
|
+
// Wait for heartbeat to propagate peer list to follower
|
|
245
|
+
await waitFor(() => {
|
|
246
|
+
const peers = followerFixture.runtime.getStatus().peers;
|
|
247
|
+
return peers.some(p => p.id === 'new-node-prop');
|
|
248
|
+
}, 4000);
|
|
249
|
+
const followerPeers = followerFixture.runtime.getStatus().peers;
|
|
250
|
+
expect(followerPeers).toContainEqual({ id: 'new-node-prop', url: 'http://127.0.0.1:7777' });
|
|
251
|
+
});
|
|
252
|
+
it('follower applies peer list from heartbeat', async () => {
|
|
253
|
+
if (!(await canListenOnLocalhost()))
|
|
254
|
+
return;
|
|
255
|
+
const fixture = await startRuntimeFixture({
|
|
256
|
+
node_id: 'follower-apply',
|
|
257
|
+
election_timeout_min_ms: 5000,
|
|
258
|
+
election_timeout_max_ms: 5000,
|
|
259
|
+
});
|
|
260
|
+
// Send heartbeat with peer list
|
|
261
|
+
await postJson(fixture.config.public_url, '/cluster/v1/election/heartbeat', {
|
|
262
|
+
term: 5,
|
|
263
|
+
leader_id: 'external-leader',
|
|
264
|
+
fencing_token: 5,
|
|
265
|
+
peers: [
|
|
266
|
+
{ id: 'external-leader', url: 'http://127.0.0.1:6000' },
|
|
267
|
+
{ id: 'follower-apply', url: fixture.config.public_url },
|
|
268
|
+
{ id: 'peer-z', url: 'http://127.0.0.1:6001' },
|
|
269
|
+
],
|
|
270
|
+
});
|
|
271
|
+
const status = fixture.runtime.getStatus();
|
|
272
|
+
expect(status.peers).toHaveLength(3);
|
|
273
|
+
expect(status.peers).toContainEqual({ id: 'peer-z', url: 'http://127.0.0.1:6001' });
|
|
274
|
+
});
|
|
275
|
+
});
|
|
276
|
+
describe('quorum recalculation after membership change', () => {
|
|
277
|
+
it('quorum adjusts after adding a peer', async () => {
|
|
278
|
+
if (!(await canListenOnLocalhost()))
|
|
279
|
+
return;
|
|
280
|
+
// Start as a single node (quorum = 1)
|
|
281
|
+
const fixture = await startRuntimeFixture({
|
|
282
|
+
node_id: 'quorum-node',
|
|
283
|
+
election_timeout_min_ms: 80,
|
|
284
|
+
election_timeout_max_ms: 120,
|
|
285
|
+
heartbeat_interval_ms: 60,
|
|
286
|
+
});
|
|
287
|
+
await waitFor(() => fixture.runtime.getStatus().is_leader, 4000);
|
|
288
|
+
// Single node: quorum = 1
|
|
289
|
+
// Add two peers: 3 nodes total, quorum = 2
|
|
290
|
+
await postJson(fixture.config.public_url, '/cluster/v1/membership/join', {
|
|
291
|
+
node_id: 'peer-1',
|
|
292
|
+
url: 'http://127.0.0.1:9001',
|
|
293
|
+
});
|
|
294
|
+
await postJson(fixture.config.public_url, '/cluster/v1/membership/join', {
|
|
295
|
+
node_id: 'peer-2',
|
|
296
|
+
url: 'http://127.0.0.1:9002',
|
|
297
|
+
});
|
|
298
|
+
const status = fixture.runtime.getStatus();
|
|
299
|
+
expect(status.peers).toHaveLength(2);
|
|
300
|
+
// The node should still be functional with updated peer list
|
|
301
|
+
expect(status.is_leader).toBe(true);
|
|
302
|
+
});
|
|
303
|
+
});
|
|
304
|
+
// --- Test helpers ---
|
|
305
|
+
async function startRuntimeFixture(overrides = {}) {
|
|
306
|
+
const attempts = overrides.listen_port ? 1 : 5;
|
|
307
|
+
let lastError;
|
|
308
|
+
for (let i = 0; i < attempts; i++) {
|
|
309
|
+
const config = await buildConfig(overrides);
|
|
310
|
+
try {
|
|
311
|
+
return await startRuntimeWithConfig(config);
|
|
312
|
+
}
|
|
313
|
+
catch (error) {
|
|
314
|
+
lastError = error;
|
|
315
|
+
const err = error;
|
|
316
|
+
if (!overrides.listen_port && err.code === 'EADDRINUSE') {
|
|
317
|
+
continue;
|
|
318
|
+
}
|
|
319
|
+
throw error;
|
|
320
|
+
}
|
|
321
|
+
}
|
|
322
|
+
throw lastError instanceof Error ? lastError : new Error('Failed to start runtime fixture');
|
|
323
|
+
}
|
|
324
|
+
async function startRuntimeWithConfig(config) {
|
|
325
|
+
const root = mkdtempSync(join(tmpdir(), `hive-membership-${config.node_id}-`));
|
|
326
|
+
const hiveDir = join(root, '.hive');
|
|
327
|
+
mkdirSync(hiveDir, { recursive: true });
|
|
328
|
+
const runtime = new ClusterRuntime(config, { hiveDir });
|
|
329
|
+
try {
|
|
330
|
+
await runtime.start();
|
|
331
|
+
activeRuntimes.push(runtime);
|
|
332
|
+
tempRoots.push(root);
|
|
333
|
+
return { root, hiveDir, config, runtime };
|
|
334
|
+
}
|
|
335
|
+
catch (error) {
|
|
336
|
+
try {
|
|
337
|
+
await runtime.stop();
|
|
338
|
+
}
|
|
339
|
+
catch {
|
|
340
|
+
// Best effort cleanup for partial starts.
|
|
341
|
+
}
|
|
342
|
+
rmSync(root, { recursive: true, force: true });
|
|
343
|
+
throw error;
|
|
344
|
+
}
|
|
345
|
+
}
|
|
346
|
+
async function buildConfig(overrides = {}) {
|
|
347
|
+
const port = overrides.listen_port ?? (await getFreePort());
|
|
348
|
+
const base = {
|
|
349
|
+
enabled: true,
|
|
350
|
+
node_id: 'node-test',
|
|
351
|
+
listen_host: '127.0.0.1',
|
|
352
|
+
listen_port: port,
|
|
353
|
+
public_url: `http://127.0.0.1:${port}`,
|
|
354
|
+
peers: [],
|
|
355
|
+
heartbeat_interval_ms: 100,
|
|
356
|
+
election_timeout_min_ms: 150,
|
|
357
|
+
election_timeout_max_ms: 250,
|
|
358
|
+
sync_interval_ms: 200,
|
|
359
|
+
request_timeout_ms: 600,
|
|
360
|
+
story_similarity_threshold: 0.8,
|
|
361
|
+
};
|
|
362
|
+
return {
|
|
363
|
+
...base,
|
|
364
|
+
...overrides,
|
|
365
|
+
public_url: overrides.public_url || base.public_url,
|
|
366
|
+
peers: overrides.peers || base.peers,
|
|
367
|
+
};
|
|
368
|
+
}
|
|
369
|
+
async function postJson(baseUrl, path, body) {
|
|
370
|
+
const res = await fetch(`${baseUrl}${path}`, {
|
|
371
|
+
method: 'POST',
|
|
372
|
+
headers: { 'Content-Type': 'application/json' },
|
|
373
|
+
body: JSON.stringify(body),
|
|
374
|
+
});
|
|
375
|
+
return (await res.json());
|
|
376
|
+
}
|
|
377
|
+
async function waitFor(predicate, timeoutMs) {
|
|
378
|
+
const start = Date.now();
|
|
379
|
+
while (Date.now() - start < timeoutMs) {
|
|
380
|
+
if (predicate())
|
|
381
|
+
return;
|
|
382
|
+
await new Promise(resolve => setTimeout(resolve, 25));
|
|
383
|
+
}
|
|
384
|
+
throw new Error('Timed out waiting for condition');
|
|
385
|
+
}
|
|
386
|
+
async function getFreePort() {
|
|
387
|
+
return new Promise((resolve, reject) => {
|
|
388
|
+
const server = createNetServer();
|
|
389
|
+
server.once('error', reject);
|
|
390
|
+
server.listen(0, '127.0.0.1', () => {
|
|
391
|
+
const address = server.address();
|
|
392
|
+
if (!address || typeof address === 'string') {
|
|
393
|
+
server.close(() => reject(new Error('Failed to allocate free port')));
|
|
394
|
+
return;
|
|
395
|
+
}
|
|
396
|
+
const port = address.port;
|
|
397
|
+
server.close(err => {
|
|
398
|
+
if (err) {
|
|
399
|
+
reject(err);
|
|
400
|
+
return;
|
|
401
|
+
}
|
|
402
|
+
resolve(port);
|
|
403
|
+
});
|
|
404
|
+
});
|
|
405
|
+
});
|
|
406
|
+
}
|
|
407
|
+
async function canListenOnLocalhost() {
|
|
408
|
+
try {
|
|
409
|
+
await getFreePort();
|
|
410
|
+
return true;
|
|
411
|
+
}
|
|
412
|
+
catch {
|
|
413
|
+
return false;
|
|
414
|
+
}
|
|
415
|
+
}
|
|
416
|
+
//# sourceMappingURL=membership.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"membership.test.js","sourceRoot":"","sources":["../../src/cluster/membership.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"partition-safety.test.d.ts","sourceRoot":"","sources":["../../src/cluster/partition-safety.test.ts"],"names":[],"mappings":""}
|