@adhdev/daemon-core 0.9.82-rc.6 → 0.9.82-rc.61
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/boot/daemon-lifecycle.d.ts +2 -0
- package/dist/commands/router.d.ts +24 -0
- package/dist/config/mesh-config.d.ts +66 -1
- package/dist/git/git-commands.d.ts +1 -0
- package/dist/git/git-status.d.ts +5 -0
- package/dist/git/git-types.d.ts +10 -0
- package/dist/index.d.ts +13 -6
- package/dist/index.js +3522 -593
- package/dist/index.js.map +1 -1
- package/dist/index.mjs +3496 -587
- package/dist/index.mjs.map +1 -1
- package/dist/mesh/mesh-active-work.d.ts +48 -0
- package/dist/mesh/mesh-events.d.ts +17 -5
- package/dist/mesh/mesh-fast-forward.d.ts +39 -0
- package/dist/mesh/mesh-host-ownership.d.ts +9 -0
- package/dist/mesh/mesh-ledger.d.ts +38 -1
- package/dist/mesh/mesh-work-queue.d.ts +23 -5
- package/dist/mesh/refine-config.d.ts +119 -0
- package/dist/providers/chat-message-normalization.d.ts +1 -0
- package/dist/providers/cli-provider-instance.d.ts +1 -0
- package/dist/repo-mesh-types.d.ts +160 -0
- package/package.json +1 -1
- package/src/boot/daemon-lifecycle.ts +4 -0
- package/src/cli-adapters/provider-cli-runtime.ts +3 -1
- package/src/commands/router.ts +2178 -419
- package/src/config/mesh-config.ts +244 -1
- package/src/git/git-commands.ts +3 -3
- package/src/git/git-status.ts +97 -6
- package/src/git/git-summary.ts +3 -0
- package/src/git/git-types.ts +11 -0
- package/src/index.ts +39 -5
- package/src/mesh/coordinator-prompt.ts +4 -2
- package/src/mesh/mesh-active-work.ts +205 -0
- package/src/mesh/mesh-events.ts +210 -38
- package/src/mesh/mesh-fast-forward.ts +430 -0
- package/src/mesh/mesh-host-ownership.ts +73 -0
- package/src/mesh/mesh-ledger.ts +137 -0
- package/src/mesh/mesh-work-queue.ts +202 -122
- package/src/mesh/refine-config.ts +306 -0
- package/src/providers/chat-message-normalization.ts +3 -1
- package/src/providers/cli-provider-instance.ts +66 -1
- package/src/repo-mesh-types.ts +174 -0
|
@@ -8,7 +8,7 @@
|
|
|
8
8
|
|
|
9
9
|
import { existsSync, readFileSync, writeFileSync } from 'fs';
|
|
10
10
|
import { join } from 'path';
|
|
11
|
-
import { randomUUID } from 'crypto';
|
|
11
|
+
import { createHash, randomBytes, randomUUID } from 'crypto';
|
|
12
12
|
import { getConfigDir } from './config.js';
|
|
13
13
|
import type {
|
|
14
14
|
LocalMeshConfig,
|
|
@@ -18,8 +18,11 @@ import type {
|
|
|
18
18
|
RepoMeshNodePolicy,
|
|
19
19
|
RepoMeshNodeCapabilities,
|
|
20
20
|
RepoMeshCoordinatorConfig,
|
|
21
|
+
RepoMeshHostMetadata,
|
|
22
|
+
RepoMeshDaemonRole,
|
|
21
23
|
} from '../repo-mesh-types.js';
|
|
22
24
|
import { DEFAULT_MESH_POLICY } from '../repo-mesh-types.js';
|
|
25
|
+
import { createDefaultMeshHostMetadata } from '../mesh/mesh-host-ownership.js';
|
|
23
26
|
|
|
24
27
|
// ─── Persistence ────────────────────────────────
|
|
25
28
|
|
|
@@ -112,6 +115,7 @@ export interface CreateMeshOptions {
|
|
|
112
115
|
defaultBranch?: string;
|
|
113
116
|
policy?: Partial<RepoMeshPolicy>;
|
|
114
117
|
coordinator?: RepoMeshCoordinatorConfig;
|
|
118
|
+
meshHost?: RepoMeshHostMetadata;
|
|
115
119
|
}
|
|
116
120
|
|
|
117
121
|
export function createMesh(opts: CreateMeshOptions): LocalMeshEntry {
|
|
@@ -133,6 +137,7 @@ export function createMesh(opts: CreateMeshOptions): LocalMeshEntry {
|
|
|
133
137
|
defaultBranch: opts.defaultBranch,
|
|
134
138
|
policy: mergeMeshPolicy(undefined, opts.policy),
|
|
135
139
|
coordinator: opts.coordinator || {},
|
|
140
|
+
meshHost: opts.meshHost || createDefaultMeshHostMetadata(),
|
|
136
141
|
nodes: [],
|
|
137
142
|
createdAt: now,
|
|
138
143
|
updatedAt: now,
|
|
@@ -148,6 +153,7 @@ export interface UpdateMeshOptions {
|
|
|
148
153
|
defaultBranch?: string;
|
|
149
154
|
policy?: Partial<RepoMeshPolicy>;
|
|
150
155
|
coordinator?: RepoMeshCoordinatorConfig;
|
|
156
|
+
meshHost?: RepoMeshHostMetadata;
|
|
151
157
|
}
|
|
152
158
|
|
|
153
159
|
export function updateMesh(meshId: string, opts: UpdateMeshOptions): LocalMeshEntry | undefined {
|
|
@@ -159,6 +165,7 @@ export function updateMesh(meshId: string, opts: UpdateMeshOptions): LocalMeshEn
|
|
|
159
165
|
if (opts.defaultBranch !== undefined) mesh.defaultBranch = opts.defaultBranch;
|
|
160
166
|
if (opts.policy) mesh.policy = mergeMeshPolicy(mesh.policy, opts.policy);
|
|
161
167
|
if (opts.coordinator) mesh.coordinator = opts.coordinator;
|
|
168
|
+
if (opts.meshHost) mesh.meshHost = opts.meshHost;
|
|
162
169
|
mesh.updatedAt = new Date().toISOString();
|
|
163
170
|
|
|
164
171
|
saveMeshConfig(config);
|
|
@@ -174,6 +181,240 @@ export function deleteMesh(meshId: string): boolean {
|
|
|
174
181
|
return true;
|
|
175
182
|
}
|
|
176
183
|
|
|
184
|
+
function normalizeManualHostAddress(hostAddress: string): string {
|
|
185
|
+
const normalized = hostAddress.trim().replace(/\/+$/, '');
|
|
186
|
+
if (!normalized) throw new Error('hostAddress required');
|
|
187
|
+
let parsed: URL;
|
|
188
|
+
try {
|
|
189
|
+
parsed = new URL(normalized);
|
|
190
|
+
} catch {
|
|
191
|
+
throw new Error('hostAddress must be a valid http(s) or ws(s) URL');
|
|
192
|
+
}
|
|
193
|
+
if (!['http:', 'https:', 'ws:', 'wss:'].includes(parsed.protocol)) {
|
|
194
|
+
throw new Error('hostAddress must use http, https, ws, or wss');
|
|
195
|
+
}
|
|
196
|
+
return normalized;
|
|
197
|
+
}
|
|
198
|
+
|
|
199
|
+
export function tokenIdForManualPairing(token: string): string {
|
|
200
|
+
return `tok_${createHash('sha256').update(token).digest('hex').slice(0, 16)}`;
|
|
201
|
+
}
|
|
202
|
+
|
|
203
|
+
function normalizeTokenExpiry(value: unknown): string | undefined {
|
|
204
|
+
if (typeof value !== 'string' || !value.trim()) return undefined;
|
|
205
|
+
const date = new Date(value);
|
|
206
|
+
if (Number.isNaN(date.getTime())) throw new Error('expiresAt must be a valid ISO date');
|
|
207
|
+
return date.toISOString();
|
|
208
|
+
}
|
|
209
|
+
|
|
210
|
+
function assertPairingTokenValid(pairing: RepoMeshHostMetadata['pairing'], rawToken: string, nowIso: string): { ok: true; tokenId: string } | { ok: false; reason: string; expectedTokenId?: string; presentedTokenId?: string } {
|
|
211
|
+
const token = rawToken.trim();
|
|
212
|
+
if (!token) return { ok: false, reason: 'token required' };
|
|
213
|
+
const presentedTokenId = tokenIdForManualPairing(token);
|
|
214
|
+
const expectedTokenId = pairing?.tokenId;
|
|
215
|
+
if (!expectedTokenId || pairing?.status === 'not_configured' || pairing?.status === 'revoked') {
|
|
216
|
+
return { ok: false, reason: 'host pairing token is not configured', presentedTokenId };
|
|
217
|
+
}
|
|
218
|
+
if (pairing.expiresAt && new Date(pairing.expiresAt).getTime() <= new Date(nowIso).getTime()) {
|
|
219
|
+
return { ok: false, reason: 'host pairing token expired', expectedTokenId, presentedTokenId };
|
|
220
|
+
}
|
|
221
|
+
if (presentedTokenId !== expectedTokenId) {
|
|
222
|
+
return { ok: false, reason: 'invalid pairing token', expectedTokenId, presentedTokenId };
|
|
223
|
+
}
|
|
224
|
+
return { ok: true, tokenId: presentedTokenId };
|
|
225
|
+
}
|
|
226
|
+
|
|
227
|
+
export interface ConfigureMeshHostPairingOptions {
|
|
228
|
+
hostAddress: string;
|
|
229
|
+
token: string;
|
|
230
|
+
now?: string;
|
|
231
|
+
}
|
|
232
|
+
|
|
233
|
+
export function configureMeshHostPairing(
|
|
234
|
+
meshId: string,
|
|
235
|
+
opts: ConfigureMeshHostPairingOptions,
|
|
236
|
+
): { mesh: LocalMeshEntry; meshHost: RepoMeshHostMetadata; hostAddress: string } | undefined {
|
|
237
|
+
const hostAddress = normalizeManualHostAddress(opts.hostAddress);
|
|
238
|
+
const token = opts.token.trim();
|
|
239
|
+
if (!token) throw new Error('token required');
|
|
240
|
+
|
|
241
|
+
const config = loadMeshConfig();
|
|
242
|
+
const mesh = config.meshes.find(m => m.id === meshId);
|
|
243
|
+
if (!mesh) return undefined;
|
|
244
|
+
|
|
245
|
+
const now = opts.now || new Date().toISOString();
|
|
246
|
+
const previous = mesh.meshHost || createDefaultMeshHostMetadata();
|
|
247
|
+
const meshHost: RepoMeshHostMetadata = {
|
|
248
|
+
...previous,
|
|
249
|
+
role: 'member',
|
|
250
|
+
hostAddress,
|
|
251
|
+
pairing: {
|
|
252
|
+
status: 'pairing',
|
|
253
|
+
tokenId: tokenIdForManualPairing(token),
|
|
254
|
+
lastPairedAt: now,
|
|
255
|
+
},
|
|
256
|
+
};
|
|
257
|
+
|
|
258
|
+
mesh.meshHost = meshHost;
|
|
259
|
+
mesh.updatedAt = now;
|
|
260
|
+
saveMeshConfig(config);
|
|
261
|
+
return { mesh, meshHost, hostAddress };
|
|
262
|
+
}
|
|
263
|
+
|
|
264
|
+
export interface CreateMeshHostPairingTokenOptions {
|
|
265
|
+
token?: string;
|
|
266
|
+
expiresAt?: string;
|
|
267
|
+
now?: string;
|
|
268
|
+
}
|
|
269
|
+
|
|
270
|
+
export function createMeshHostPairingToken(
|
|
271
|
+
meshId: string,
|
|
272
|
+
opts: CreateMeshHostPairingTokenOptions = {},
|
|
273
|
+
): { mesh: LocalMeshEntry; meshHost: RepoMeshHostMetadata; token: string; tokenId: string; expiresAt?: string } | undefined {
|
|
274
|
+
const config = loadMeshConfig();
|
|
275
|
+
const mesh = config.meshes.find(m => m.id === meshId);
|
|
276
|
+
if (!mesh) return undefined;
|
|
277
|
+
const now = opts.now || new Date().toISOString();
|
|
278
|
+
const token = (opts.token || `mhj_${randomBytes(24).toString('base64url')}`).trim();
|
|
279
|
+
if (!token) throw new Error('token required');
|
|
280
|
+
const tokenId = tokenIdForManualPairing(token);
|
|
281
|
+
const expiresAt = normalizeTokenExpiry(opts.expiresAt);
|
|
282
|
+
const previous = mesh.meshHost || createDefaultMeshHostMetadata();
|
|
283
|
+
if (previous.role === 'member') {
|
|
284
|
+
throw new Error('Mesh Host daemon required to create host pairing tokens; member daemons cannot mint host join tokens.');
|
|
285
|
+
}
|
|
286
|
+
const meshHost: RepoMeshHostMetadata = {
|
|
287
|
+
...previous,
|
|
288
|
+
role: 'host',
|
|
289
|
+
pairing: {
|
|
290
|
+
status: 'pairing',
|
|
291
|
+
tokenId,
|
|
292
|
+
lastPairedAt: now,
|
|
293
|
+
...(expiresAt ? { expiresAt } : {}),
|
|
294
|
+
},
|
|
295
|
+
};
|
|
296
|
+
mesh.meshHost = meshHost;
|
|
297
|
+
mesh.updatedAt = now;
|
|
298
|
+
saveMeshConfig(config);
|
|
299
|
+
return { mesh, meshHost, token, tokenId, ...(expiresAt ? { expiresAt } : {}) };
|
|
300
|
+
}
|
|
301
|
+
|
|
302
|
+
export interface MeshHostJoinMemberNodeInput {
|
|
303
|
+
id?: string;
|
|
304
|
+
workspace: string;
|
|
305
|
+
repoRoot?: string;
|
|
306
|
+
daemonId?: string;
|
|
307
|
+
machineId?: string;
|
|
308
|
+
userOverrides?: Partial<RepoMeshNodeCapabilities>;
|
|
309
|
+
policy?: RepoMeshNodePolicy;
|
|
310
|
+
role?: RepoMeshDaemonRole;
|
|
311
|
+
}
|
|
312
|
+
|
|
313
|
+
export interface ApplyMeshHostJoinOptions {
|
|
314
|
+
token: string;
|
|
315
|
+
memberNode: MeshHostJoinMemberNodeInput;
|
|
316
|
+
memberMeshId?: string;
|
|
317
|
+
now?: string;
|
|
318
|
+
}
|
|
319
|
+
|
|
320
|
+
export function applyMeshHostJoinRequest(
|
|
321
|
+
meshId: string,
|
|
322
|
+
opts: ApplyMeshHostJoinOptions,
|
|
323
|
+
): { accepted: true; mesh: LocalMeshEntry; meshHost: RepoMeshHostMetadata; node: LocalMeshNodeEntry; tokenId: string } | { accepted: false; mesh?: LocalMeshEntry; meshHost?: RepoMeshHostMetadata; tokenId?: string; reason: string } | undefined {
|
|
324
|
+
const config = loadMeshConfig();
|
|
325
|
+
const mesh = config.meshes.find(m => m.id === meshId);
|
|
326
|
+
if (!mesh) return undefined;
|
|
327
|
+
const now = opts.now || new Date().toISOString();
|
|
328
|
+
const previous = mesh.meshHost || createDefaultMeshHostMetadata();
|
|
329
|
+
if (previous.role === 'member') {
|
|
330
|
+
return { accepted: false, mesh, meshHost: previous, reason: 'Mesh Host daemon required to accept join requests' };
|
|
331
|
+
}
|
|
332
|
+
const meshHost: RepoMeshHostMetadata = { ...previous, role: 'host' };
|
|
333
|
+
const validation = assertPairingTokenValid(meshHost.pairing, opts.token, now);
|
|
334
|
+
if (!validation.ok) {
|
|
335
|
+
mesh.meshHost = {
|
|
336
|
+
...meshHost,
|
|
337
|
+
pairing: {
|
|
338
|
+
...(meshHost.pairing || { status: 'not_configured' as const }),
|
|
339
|
+
status: 'rejected',
|
|
340
|
+
lastRejectedAt: now,
|
|
341
|
+
},
|
|
342
|
+
};
|
|
343
|
+
mesh.updatedAt = now;
|
|
344
|
+
saveMeshConfig(config);
|
|
345
|
+
return { accepted: false, mesh, meshHost: mesh.meshHost, tokenId: validation.presentedTokenId, reason: validation.reason };
|
|
346
|
+
}
|
|
347
|
+
|
|
348
|
+
const workspace = opts.memberNode.workspace.trim();
|
|
349
|
+
if (!workspace) throw new Error('memberNode.workspace required');
|
|
350
|
+
const memberId = opts.memberNode.id?.trim();
|
|
351
|
+
let node = mesh.nodes.find(n => (memberId && n.id === memberId) || n.workspace === workspace);
|
|
352
|
+
if (node) {
|
|
353
|
+
node.workspace = workspace;
|
|
354
|
+
node.repoRoot = opts.memberNode.repoRoot;
|
|
355
|
+
node.daemonId = opts.memberNode.daemonId;
|
|
356
|
+
node.machineId = opts.memberNode.machineId;
|
|
357
|
+
node.userOverrides = opts.memberNode.userOverrides || node.userOverrides || {};
|
|
358
|
+
node.policy = { ...(node.policy || {}), ...(opts.memberNode.policy || {}) };
|
|
359
|
+
node.role = 'member';
|
|
360
|
+
} else {
|
|
361
|
+
if (mesh.nodes.length >= 10) throw new Error('Maximum 10 nodes per mesh');
|
|
362
|
+
node = {
|
|
363
|
+
id: memberId || `node_${randomUUID().replace(/-/g, '')}`,
|
|
364
|
+
workspace,
|
|
365
|
+
repoRoot: opts.memberNode.repoRoot,
|
|
366
|
+
daemonId: opts.memberNode.daemonId,
|
|
367
|
+
machineId: opts.memberNode.machineId,
|
|
368
|
+
userOverrides: opts.memberNode.userOverrides || {},
|
|
369
|
+
policy: opts.memberNode.policy || {},
|
|
370
|
+
role: 'member',
|
|
371
|
+
};
|
|
372
|
+
mesh.nodes.push(node);
|
|
373
|
+
}
|
|
374
|
+
mesh.meshHost = {
|
|
375
|
+
...meshHost,
|
|
376
|
+
pairing: {
|
|
377
|
+
...(meshHost.pairing || {}),
|
|
378
|
+
status: 'paired',
|
|
379
|
+
tokenId: validation.tokenId,
|
|
380
|
+
joinedAt: now,
|
|
381
|
+
lastPairedAt: meshHost.pairing?.lastPairedAt || now,
|
|
382
|
+
...(meshHost.pairing?.expiresAt ? { expiresAt: meshHost.pairing.expiresAt } : {}),
|
|
383
|
+
},
|
|
384
|
+
};
|
|
385
|
+
mesh.updatedAt = now;
|
|
386
|
+
saveMeshConfig(config);
|
|
387
|
+
return { accepted: true, mesh, meshHost: mesh.meshHost, node, tokenId: validation.tokenId };
|
|
388
|
+
}
|
|
389
|
+
|
|
390
|
+
export function markMeshHostPairingJoined(
|
|
391
|
+
meshId: string,
|
|
392
|
+
opts: { hostDaemonId?: string; hostNodeId?: string; joinedAt?: string; token?: string; tokenId?: string },
|
|
393
|
+
): { mesh: LocalMeshEntry; meshHost: RepoMeshHostMetadata } | undefined {
|
|
394
|
+
const config = loadMeshConfig();
|
|
395
|
+
const mesh = config.meshes.find(m => m.id === meshId);
|
|
396
|
+
if (!mesh) return undefined;
|
|
397
|
+
const now = opts.joinedAt || new Date().toISOString();
|
|
398
|
+
const previous = mesh.meshHost || createDefaultMeshHostMetadata();
|
|
399
|
+
const tokenId = opts.tokenId || (opts.token ? tokenIdForManualPairing(opts.token) : previous.pairing?.tokenId);
|
|
400
|
+
mesh.meshHost = {
|
|
401
|
+
...previous,
|
|
402
|
+
role: 'member',
|
|
403
|
+
...(opts.hostDaemonId ? { hostDaemonId: opts.hostDaemonId } : {}),
|
|
404
|
+
...(opts.hostNodeId ? { hostNodeId: opts.hostNodeId } : {}),
|
|
405
|
+
pairing: {
|
|
406
|
+
...(previous.pairing || {}),
|
|
407
|
+
status: 'paired',
|
|
408
|
+
...(tokenId ? { tokenId } : {}),
|
|
409
|
+
joinedAt: now,
|
|
410
|
+
lastPairedAt: previous.pairing?.lastPairedAt || now,
|
|
411
|
+
},
|
|
412
|
+
};
|
|
413
|
+
mesh.updatedAt = now;
|
|
414
|
+
saveMeshConfig(config);
|
|
415
|
+
return { mesh, meshHost: mesh.meshHost };
|
|
416
|
+
}
|
|
417
|
+
|
|
177
418
|
// ─── Node Operations ────────────────────────────
|
|
178
419
|
|
|
179
420
|
export interface AddNodeOptions {
|
|
@@ -186,6 +427,7 @@ export interface AddNodeOptions {
|
|
|
186
427
|
isLocalWorktree?: boolean;
|
|
187
428
|
worktreeBranch?: string;
|
|
188
429
|
clonedFromNodeId?: string;
|
|
430
|
+
role?: RepoMeshDaemonRole;
|
|
189
431
|
}
|
|
190
432
|
|
|
191
433
|
export function addNode(meshId: string, opts: AddNodeOptions): LocalMeshNodeEntry | undefined {
|
|
@@ -213,6 +455,7 @@ export function addNode(meshId: string, opts: AddNodeOptions): LocalMeshNodeEntr
|
|
|
213
455
|
isLocalWorktree: opts.isLocalWorktree,
|
|
214
456
|
worktreeBranch: opts.worktreeBranch,
|
|
215
457
|
clonedFromNodeId: opts.clonedFromNodeId,
|
|
458
|
+
role: opts.role,
|
|
216
459
|
};
|
|
217
460
|
|
|
218
461
|
mesh.nodes.push(node);
|
package/src/git/git-commands.ts
CHANGED
|
@@ -62,7 +62,7 @@ export interface GitPushResult extends GitRepoIdentity {
|
|
|
62
62
|
}
|
|
63
63
|
|
|
64
64
|
export interface GitCommandServices {
|
|
65
|
-
getStatus?: (params: { workspace: string }) => Promise<GitRepoStatus> | GitRepoStatus;
|
|
65
|
+
getStatus?: (params: { workspace: string; refreshUpstream?: boolean }) => Promise<GitRepoStatus> | GitRepoStatus;
|
|
66
66
|
getDiffSummary?: (params: { workspace: string; staged?: boolean }) => Promise<GitDiffSummary> | GitDiffSummary;
|
|
67
67
|
getDiffFile?: (params: { workspace: string; path: string; staged?: boolean }) => Promise<GitFileDiff> | GitFileDiff;
|
|
68
68
|
createSnapshot?: (params: {
|
|
@@ -171,7 +171,7 @@ const defaultSnapshotStore = createGitSnapshotStore({
|
|
|
171
171
|
|
|
172
172
|
export function createDefaultGitCommandServices(): GitCommandServices {
|
|
173
173
|
return {
|
|
174
|
-
getStatus: ({ workspace }) => getGitRepoStatus(workspace),
|
|
174
|
+
getStatus: ({ workspace, refreshUpstream }) => getGitRepoStatus(workspace, { refreshUpstream }),
|
|
175
175
|
getDiffSummary: ({ workspace }) => getGitDiffSummary(workspace),
|
|
176
176
|
getDiffFile: ({ workspace, path: filePath }) => getGitFileDiff(workspace, filePath),
|
|
177
177
|
createSnapshot: ({ workspace, reason, sessionId, turnId }) => defaultSnapshotStore.create({
|
|
@@ -290,7 +290,7 @@ export async function handleGitCommand(
|
|
|
290
290
|
switch (command) {
|
|
291
291
|
case 'git_status': {
|
|
292
292
|
if (!services.getStatus) return serviceNotImplemented(command);
|
|
293
|
-
const status = await runService(() => services.getStatus!({ workspace }));
|
|
293
|
+
const status = await runService(() => services.getStatus!({ workspace, refreshUpstream: optionalBoolean(args?.refreshUpstream) }));
|
|
294
294
|
return 'success' in status ? status : { success: true, status };
|
|
295
295
|
}
|
|
296
296
|
|
package/src/git/git-status.ts
CHANGED
|
@@ -1,12 +1,25 @@
|
|
|
1
|
-
import type { GitRepoStatus, GitSubmoduleStatus } from './git-types.js';
|
|
1
|
+
import type { GitRepoStatus, GitSubmoduleStatus, GitUpstreamFreshness } from './git-types.js';
|
|
2
2
|
import { GitCommandError, resolveGitRepository, runGit } from './git-executor.js';
|
|
3
3
|
|
|
4
|
+
type ResolvedGitRepo = { workspace: string; repoRoot: string | null; isGitRepo: boolean };
|
|
5
|
+
|
|
4
6
|
export interface GitStatusOptions {
|
|
5
7
|
timeoutMs?: number;
|
|
6
8
|
/** When true, include submodule status in the result. Defaults to true. */
|
|
7
9
|
includeSubmodules?: boolean;
|
|
8
10
|
/** Optional filter to exclude specific submodule paths from status */
|
|
9
11
|
submoduleIgnorePaths?: string[];
|
|
12
|
+
/**
|
|
13
|
+
* When true, refresh the tracked remote before trusting ahead/behind.
|
|
14
|
+
* Callers should opt into this only for convergence-critical surfaces.
|
|
15
|
+
*/
|
|
16
|
+
refreshUpstream?: boolean;
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
interface GitUpstreamProbe {
|
|
20
|
+
upstreamStatus: GitUpstreamFreshness;
|
|
21
|
+
upstreamFetchedAt?: number;
|
|
22
|
+
upstreamFetchError?: string;
|
|
10
23
|
}
|
|
11
24
|
|
|
12
25
|
export async function getGitRepoStatus(
|
|
@@ -18,8 +31,16 @@ export async function getGitRepoStatus(
|
|
|
18
31
|
|
|
19
32
|
try {
|
|
20
33
|
const repo = await resolveGitRepository(workspace, options);
|
|
21
|
-
|
|
22
|
-
|
|
34
|
+
let parsed = await readPorcelainStatus(repo, options);
|
|
35
|
+
let upstreamProbe: GitUpstreamProbe = getInitialUpstreamProbe(parsed);
|
|
36
|
+
|
|
37
|
+
if (options.refreshUpstream) {
|
|
38
|
+
upstreamProbe = await refreshTrackedUpstream(repo, parsed, options);
|
|
39
|
+
if (upstreamProbe.upstreamStatus === 'fresh') {
|
|
40
|
+
parsed = await readPorcelainStatus(repo, options);
|
|
41
|
+
}
|
|
42
|
+
}
|
|
43
|
+
|
|
23
44
|
const head = await readHead(repo, options);
|
|
24
45
|
const stashCount = await readStashCount(repo, options);
|
|
25
46
|
|
|
@@ -36,6 +57,9 @@ export async function getGitRepoStatus(
|
|
|
36
57
|
headCommit: head.commit,
|
|
37
58
|
headMessage: head.message,
|
|
38
59
|
upstream: parsed.upstream,
|
|
60
|
+
upstreamStatus: parsed.upstream ? upstreamProbe.upstreamStatus : 'no_upstream',
|
|
61
|
+
upstreamFetchedAt: upstreamProbe.upstreamFetchedAt,
|
|
62
|
+
upstreamFetchError: upstreamProbe.upstreamFetchError,
|
|
39
63
|
ahead: parsed.ahead,
|
|
40
64
|
behind: parsed.behind,
|
|
41
65
|
staged: parsed.staged,
|
|
@@ -74,6 +98,72 @@ interface ParsedPorcelainStatus {
|
|
|
74
98
|
conflictFiles: string[];
|
|
75
99
|
}
|
|
76
100
|
|
|
101
|
+
async function readPorcelainStatus(repo: ResolvedGitRepo, options: GitStatusOptions): Promise<ParsedPorcelainStatus> {
|
|
102
|
+
const statusOutput = await runGit(repo, ['status', '--porcelain=v2', '--branch'], options);
|
|
103
|
+
return parsePorcelainV2Status(statusOutput.stdout);
|
|
104
|
+
}
|
|
105
|
+
|
|
106
|
+
function getInitialUpstreamProbe(parsed: ParsedPorcelainStatus): GitUpstreamProbe {
|
|
107
|
+
return {
|
|
108
|
+
upstreamStatus: parsed.upstream ? 'unchecked' : 'no_upstream',
|
|
109
|
+
};
|
|
110
|
+
}
|
|
111
|
+
|
|
112
|
+
async function refreshTrackedUpstream(
|
|
113
|
+
repo: ResolvedGitRepo,
|
|
114
|
+
parsed: ParsedPorcelainStatus,
|
|
115
|
+
options: GitStatusOptions,
|
|
116
|
+
): Promise<GitUpstreamProbe> {
|
|
117
|
+
if (!parsed.upstream || !parsed.branch) {
|
|
118
|
+
return { upstreamStatus: 'no_upstream' };
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
const remoteName = (await readBranchRemote(repo, parsed.branch, options)) ?? inferRemoteName(parsed.upstream);
|
|
122
|
+
if (!remoteName) {
|
|
123
|
+
return {
|
|
124
|
+
upstreamStatus: 'stale',
|
|
125
|
+
upstreamFetchError: `Unable to resolve remote for upstream '${parsed.upstream}'`,
|
|
126
|
+
};
|
|
127
|
+
}
|
|
128
|
+
|
|
129
|
+
try {
|
|
130
|
+
await runGit(repo, ['fetch', '--quiet', '--prune', '--no-tags', remoteName], options);
|
|
131
|
+
return {
|
|
132
|
+
upstreamStatus: 'fresh',
|
|
133
|
+
upstreamFetchedAt: Date.now(),
|
|
134
|
+
};
|
|
135
|
+
} catch (error) {
|
|
136
|
+
return {
|
|
137
|
+
upstreamStatus: 'stale',
|
|
138
|
+
upstreamFetchError: formatGitError(error),
|
|
139
|
+
};
|
|
140
|
+
}
|
|
141
|
+
}
|
|
142
|
+
|
|
143
|
+
async function readBranchRemote(repo: ResolvedGitRepo, branch: string, options: GitStatusOptions): Promise<string | null> {
|
|
144
|
+
try {
|
|
145
|
+
const result = await runGit(repo, ['config', '--get', `branch.${branch}.remote`], options);
|
|
146
|
+
return result.stdout.trim() || null;
|
|
147
|
+
} catch {
|
|
148
|
+
return null;
|
|
149
|
+
}
|
|
150
|
+
}
|
|
151
|
+
|
|
152
|
+
function inferRemoteName(upstream: string): string | null {
|
|
153
|
+
const [remoteName] = upstream.split('/');
|
|
154
|
+
return remoteName?.trim() || null;
|
|
155
|
+
}
|
|
156
|
+
|
|
157
|
+
function formatGitError(error: unknown): string {
|
|
158
|
+
if (error instanceof GitCommandError) {
|
|
159
|
+
return error.stderr || error.message;
|
|
160
|
+
}
|
|
161
|
+
if (error instanceof Error) {
|
|
162
|
+
return error.message;
|
|
163
|
+
}
|
|
164
|
+
return String(error);
|
|
165
|
+
}
|
|
166
|
+
|
|
77
167
|
export function parsePorcelainV2Status(output: string): ParsedPorcelainStatus {
|
|
78
168
|
const parsed: ParsedPorcelainStatus = {
|
|
79
169
|
branch: null,
|
|
@@ -145,7 +235,7 @@ export function parsePorcelainV2Status(output: string): ParsedPorcelainStatus {
|
|
|
145
235
|
}
|
|
146
236
|
|
|
147
237
|
async function readHead(
|
|
148
|
-
repo:
|
|
238
|
+
repo: ResolvedGitRepo,
|
|
149
239
|
options: GitStatusOptions,
|
|
150
240
|
): Promise<{ commit: string | null; message: string | null }> {
|
|
151
241
|
try {
|
|
@@ -163,7 +253,7 @@ async function readHead(
|
|
|
163
253
|
}
|
|
164
254
|
|
|
165
255
|
async function readStashCount(
|
|
166
|
-
repo:
|
|
256
|
+
repo: ResolvedGitRepo,
|
|
167
257
|
options: GitStatusOptions,
|
|
168
258
|
): Promise<number> {
|
|
169
259
|
try {
|
|
@@ -187,6 +277,7 @@ function emptyStatus(workspace: string, lastCheckedAt: number, error: GitCommand
|
|
|
187
277
|
headCommit: null,
|
|
188
278
|
headMessage: null,
|
|
189
279
|
upstream: null,
|
|
280
|
+
upstreamStatus: 'unavailable',
|
|
190
281
|
ahead: 0,
|
|
191
282
|
behind: 0,
|
|
192
283
|
staged: 0,
|
|
@@ -206,7 +297,7 @@ function emptyStatus(workspace: string, lastCheckedAt: number, error: GitCommand
|
|
|
206
297
|
// ─── Submodule Status ───────────────────────────
|
|
207
298
|
|
|
208
299
|
async function getSubmoduleStatuses(
|
|
209
|
-
repo:
|
|
300
|
+
repo: ResolvedGitRepo,
|
|
210
301
|
options: GitStatusOptions,
|
|
211
302
|
): Promise<GitSubmoduleStatus[]> {
|
|
212
303
|
if (!repo.repoRoot) return [];
|
package/src/git/git-summary.ts
CHANGED
|
@@ -22,6 +22,9 @@ export function createGitCompactSummary(status: GitRepoStatus, diffSummary?: Git
|
|
|
22
22
|
isGitRepo: status.isGitRepo,
|
|
23
23
|
repoRoot: status.repoRoot,
|
|
24
24
|
branch: status.branch,
|
|
25
|
+
upstreamStatus: status.upstreamStatus,
|
|
26
|
+
upstreamFetchedAt: status.upstreamFetchedAt,
|
|
27
|
+
upstreamFetchError: status.upstreamFetchError,
|
|
25
28
|
dirty:
|
|
26
29
|
status.staged > 0 ||
|
|
27
30
|
status.modified > 0 ||
|
package/src/git/git-types.ts
CHANGED
|
@@ -40,11 +40,19 @@ export interface GitSubmoduleStatus {
|
|
|
40
40
|
error?: string;
|
|
41
41
|
}
|
|
42
42
|
|
|
43
|
+
export type GitUpstreamFreshness = 'fresh' | 'unchecked' | 'stale' | 'no_upstream' | 'unavailable';
|
|
44
|
+
|
|
43
45
|
export interface GitRepoStatus extends GitRepoIdentity {
|
|
44
46
|
branch: string | null;
|
|
45
47
|
headCommit: string | null;
|
|
46
48
|
headMessage: string | null;
|
|
47
49
|
upstream: string | null;
|
|
50
|
+
/** Whether ahead/behind was verified against a freshly fetched upstream ref. */
|
|
51
|
+
upstreamStatus: GitUpstreamFreshness;
|
|
52
|
+
/** Timestamp for the fetch that refreshed upstream refs when upstreamStatus === 'fresh'. */
|
|
53
|
+
upstreamFetchedAt?: number;
|
|
54
|
+
/** Error from the last refresh attempt when upstreamStatus === 'stale'. */
|
|
55
|
+
upstreamFetchError?: string;
|
|
48
56
|
ahead: number;
|
|
49
57
|
behind: number;
|
|
50
58
|
staged: number;
|
|
@@ -134,6 +142,9 @@ export interface GitCompactSummary {
|
|
|
134
142
|
isGitRepo: boolean;
|
|
135
143
|
repoRoot: string | null;
|
|
136
144
|
branch: string | null;
|
|
145
|
+
upstreamStatus: GitUpstreamFreshness;
|
|
146
|
+
upstreamFetchedAt?: number;
|
|
147
|
+
upstreamFetchError?: string;
|
|
137
148
|
dirty: boolean;
|
|
138
149
|
changedFiles: number;
|
|
139
150
|
ahead: number;
|
package/src/index.ts
CHANGED
|
@@ -88,6 +88,10 @@ export type {
|
|
|
88
88
|
// ── Repo Mesh Types (cross-package) ──
|
|
89
89
|
export type {
|
|
90
90
|
RepoMesh,
|
|
91
|
+
RepoMeshDaemonRole,
|
|
92
|
+
RepoMeshHostMetadata,
|
|
93
|
+
RepoMeshHostPairingMetadata,
|
|
94
|
+
RepoMeshHostStatus,
|
|
91
95
|
RepoMeshNode,
|
|
92
96
|
RepoMeshNodeHealth,
|
|
93
97
|
RepoMeshPolicy,
|
|
@@ -103,6 +107,14 @@ export type {
|
|
|
103
107
|
LocalMeshNodeEntry,
|
|
104
108
|
RepoMeshStatus,
|
|
105
109
|
RepoMeshNodeStatus,
|
|
110
|
+
RepoMeshSessionStatus,
|
|
111
|
+
RepoMeshQueueTask,
|
|
112
|
+
RepoMeshQueueTaskStatus,
|
|
113
|
+
RepoMeshQueueSummary,
|
|
114
|
+
RepoMeshQueueStatus,
|
|
115
|
+
RepoMeshLedgerEntryStatus,
|
|
116
|
+
RepoMeshLedgerSummaryStatus,
|
|
117
|
+
RepoMeshLedgerStatus,
|
|
106
118
|
} from './repo-mesh-types.js';
|
|
107
119
|
export { DEFAULT_MESH_POLICY } from './repo-mesh-types.js';
|
|
108
120
|
|
|
@@ -147,18 +159,40 @@ export type { CreateMeshOptions, UpdateMeshOptions, AddNodeOptions } from './con
|
|
|
147
159
|
// ── Mesh Coordinator ──
|
|
148
160
|
export { buildCoordinatorSystemPrompt } from './mesh/coordinator-prompt.js';
|
|
149
161
|
export type { CoordinatorPromptContext } from './mesh/coordinator-prompt.js';
|
|
162
|
+
export {
|
|
163
|
+
MESH_REFINE_CONFIG_LOCATIONS,
|
|
164
|
+
MESH_REFINE_CONFIG_SCHEMA,
|
|
165
|
+
loadMeshRefineConfig,
|
|
166
|
+
resolveMeshRefineValidationPlan,
|
|
167
|
+
suggestMeshRefineConfig,
|
|
168
|
+
validateMeshRefineConfig,
|
|
169
|
+
} from './mesh/refine-config.js';
|
|
170
|
+
export type {
|
|
171
|
+
MeshRefineValidationCategory,
|
|
172
|
+
MeshRefineValidationCommandPlan,
|
|
173
|
+
MeshRefineValidationPlan,
|
|
174
|
+
RepoMeshRefineConfig,
|
|
175
|
+
RepoMeshRefineValidationCommandConfig,
|
|
176
|
+
} from './mesh/refine-config.js';
|
|
150
177
|
export { syncMeshes } from './mesh/mesh-sync.js';
|
|
151
178
|
export type { MeshSyncTransport, MeshSyncResult, RemoteMeshRecord } from './mesh/mesh-sync.js';
|
|
152
179
|
|
|
153
180
|
// ── Mesh Task Ledger ──
|
|
154
|
-
export { appendLedgerEntry, appendRemoteLedgerEntries, readLedgerEntries, readLedgerSlice, getLedgerSummary, getLedgerDir, getSessionRecoveryContext, MAX_LEDGER_SLICE_LIMIT } from './mesh/mesh-ledger.js';
|
|
155
|
-
export type { AppendRemoteLedgerResult, MeshLedgerEntry, MeshLedgerKind, MeshLedgerSlice, MeshLedgerSummary, ReadLedgerOptions, ReadLedgerSliceOptions, SessionRecoveryContext } from './mesh/mesh-ledger.js';
|
|
181
|
+
export { appendLedgerEntry, appendRemoteLedgerEntries, buildTaskCompletionEvidence, normalizeMeshWorkerResult, readLedgerEntries, readLedgerSlice, getLedgerSummary, getLedgerDir, getSessionRecoveryContext, MAX_LEDGER_SLICE_LIMIT } from './mesh/mesh-ledger.js';
|
|
182
|
+
export type { AppendRemoteLedgerResult, MeshLedgerEntry, MeshLedgerKind, MeshLedgerSlice, MeshLedgerSummary, ReadLedgerOptions, ReadLedgerSliceOptions, SessionRecoveryContext, MeshTaskCompletionEvidence, MeshWorkerResultArtifact, MeshProcessArtifact, MeshValidationResultArtifact } from './mesh/mesh-ledger.js';
|
|
183
|
+
export { fastForwardMeshNode } from './mesh/mesh-fast-forward.js';
|
|
184
|
+
export type { MeshFastForwardNodeArgs, MeshFastForwardPlannedStep, MeshFastForwardResult } from './mesh/mesh-fast-forward.js';
|
|
156
185
|
export { buildMeshLedgerReconciliationEvidence, buildMeshLedgerReplicaEvidence } from './mesh/mesh-ledger-reconciliation.js';
|
|
157
186
|
export type { MeshLedgerReconciliationEvidence, MeshLedgerReplicaEvidence, MeshLedgerReplicaStatus } from './mesh/mesh-ledger-reconciliation.js';
|
|
158
187
|
|
|
159
188
|
// ── Mesh Work Queue (GUPP) ──
|
|
160
|
-
export { enqueueTask, getQueue, claimNextTask, updateTaskStatus, updateSessionTaskStatus, cancelTask, requeueTask, getMeshQueueStats } from './mesh/mesh-work-queue.js';
|
|
161
|
-
export type { MeshWorkQueueEntry, MeshTaskStatus, MeshWorkQueueStats } from './mesh/mesh-work-queue.js';
|
|
189
|
+
export { enqueueTask, getQueue, claimNextTask, updateTaskStatus, updateSessionTaskStatus, cancelTask, requeueTask, getMeshQueueStats, normalizeMeshTaskMode, validateMeshTaskModeRequest } from './mesh/mesh-work-queue.js';
|
|
190
|
+
export type { MeshWorkQueueEntry, MeshTaskStatus, MeshTaskMode, MeshWorkQueueStats, MeshQueueMutationOptions, MeshTaskModeValidationResult } from './mesh/mesh-work-queue.js';
|
|
191
|
+
export { buildMeshActiveWork, buildMeshActiveWorkSummary } from './mesh/mesh-active-work.js';
|
|
192
|
+
export type { MeshActiveWorkRecord, MeshActiveWorkStatus, MeshActiveWorkSummary, MeshActiveWorkSource } from './mesh/mesh-active-work.js';
|
|
193
|
+
|
|
194
|
+
// ── Mesh Host Ownership ──
|
|
195
|
+
export { buildMeshHostRequiredFailure, createDefaultMeshHostMetadata, isMeshHostOwner, normalizeMeshDaemonRole, requireMeshHostQueueOwner, resolveMeshHostStatus } from './mesh/mesh-host-ownership.js';
|
|
162
196
|
|
|
163
197
|
// ── Mesh Visualization ──
|
|
164
198
|
// buildMeshGraph and MeshGraph types moved to @adhdev/web-core to avoid
|
|
@@ -168,7 +202,7 @@ export type { MeshWorkQueueEntry, MeshTaskStatus, MeshWorkQueueStats } from './m
|
|
|
168
202
|
// export type { MeshGraph, MeshGraphNode, MeshGraphEdge, MeshGraphNodeType, MeshGraphEdgeType } from './mesh/mesh-visualization.js';
|
|
169
203
|
|
|
170
204
|
// ── Mesh Events ──
|
|
171
|
-
export { triggerMeshQueue, drainPendingMeshCoordinatorEvents, getPendingMeshCoordinatorEvents, clearPendingMeshCoordinatorEvents } from './mesh/mesh-events.js';
|
|
205
|
+
export { triggerMeshQueue, drainPendingMeshCoordinatorEvents, getPendingMeshCoordinatorEvents, clearPendingMeshCoordinatorEvents, queuePendingMeshCoordinatorEvent } from './mesh/mesh-events.js';
|
|
172
206
|
export type { PendingMeshCoordinatorEvent } from './mesh/mesh-events.js';
|
|
173
207
|
|
|
174
208
|
// ── Mesh P2P Relay Failure Classification ──
|
|
@@ -140,6 +140,7 @@ const TOOLS_SECTION = `## Available Tools
|
|
|
140
140
|
| \`mesh_read_debug\` | Collect a daemon-side chat/parser debug bundle for a session |
|
|
141
141
|
| \`mesh_task_history\` | Read the task ledger — dispatches, completions, failures. Use to understand what has been done before deciding next steps |
|
|
142
142
|
| \`mesh_git_status\` | Check git status on a specific node |
|
|
143
|
+
| \`mesh_fast_forward_node\` | Safely dry-run or explicitly execute an obvious clean fast-forward without launching an agent session |
|
|
143
144
|
| \`mesh_checkpoint\` | Create a git checkpoint on a node |
|
|
144
145
|
| \`mesh_approve\` | Approve/reject a pending agent action |
|
|
145
146
|
| \`mesh_clone_node\` | Create a worktree node for isolated parallel branch work |
|
|
@@ -164,7 +165,7 @@ const WORKFLOW_SECTION = `## Orchestration Workflow
|
|
|
164
165
|
4. **Monitor** — Prefer event-driven completion/status notifications. Do **not** poll \`mesh_read_chat\` repeatedly. Use \`mesh_view_queue\` to see the status of all pending, assigned, completed, and failed tasks. Do not call \`mesh_read_chat\` again within a few seconds for the same generating session. Use at most one compact \`mesh_read_chat\` check after a completion/approval signal. Handle approvals via \`mesh_approve\`.
|
|
165
166
|
5. **Verify** — When a task reports completion or git work is visible, call \`mesh_git_status\` to verify changes were made.
|
|
166
167
|
6. **Checkpoint** — Call \`mesh_checkpoint\` to save the work.
|
|
167
|
-
7. **Converge branches** — Before marking any task complete, classify every touched node/branch into exactly one final state: \`merged_to_main\`, \`pushed_feature_branch_needs_merge\`, \`blocked_review\`, \`cleanup_candidate\`, or \`not_mergeable\`. Use \`mesh_status\` branchConvergenceSummary and \`mesh_refine_node\` for clean worktree branches when safe. A task that remains on a non-main branch is not fully complete unless the final report names the follow-up state and next step.
|
|
168
|
+
7. **Converge branches** — Before marking any task complete, classify every touched node/branch into exactly one final state: \`merged_to_main\`, \`pushed_feature_branch_needs_merge\`, \`blocked_review\`, \`cleanup_candidate\`, or \`not_mergeable\`. Use \`mesh_status\` branchConvergenceSummary. For obvious clean branch catch-up (ahead 0, behind > 0, upstream fresh, no dirty/stash/submodule issues), use \`mesh_fast_forward_node\` dry-run first and execute only when explicitly safe/approved; this avoids consuming an agent session. Use \`mesh_refine_node\` for clean worktree branches when safe. A task that remains on a non-main branch is not fully complete unless the final report names the follow-up state and next step.
|
|
168
169
|
8. **Clean up** — Remove worktree nodes via \`mesh_remove_node\` after their work is merged or no longer needed.
|
|
169
170
|
9. **Report** — Summarize what was done, what changed, any issues, and the branch convergence state.
|
|
170
171
|
|
|
@@ -201,6 +202,7 @@ function buildRulesSection(coordinatorCliType?: string): string {
|
|
|
201
202
|
- **Respect node capabilities.** Don't send build tasks to read-only nodes. Don't push from nodes that aren't allowed to.
|
|
202
203
|
- **Never fabricate tool results.** Always call the actual tool; never pretend you did.
|
|
203
204
|
- **Clean up worktree nodes.** After a worktree task completes and its changes are merged or checkpointed, call \`mesh_remove_node\` to free resources.
|
|
204
|
-
- **Do not strand completed branches.** A checkpointed or clean feature/worktree branch is not done by itself. Merge/refine it to the mesh default branch, or explicitly report one of \`pushed_feature_branch_needs_merge\`, \`blocked_review\`, \`cleanup_candidate\`, or \`not_mergeable\` with the next action.
|
|
205
|
+
- **Do not strand completed branches.** A checkpointed or clean feature/worktree branch is not done by itself. Merge/refine it to the mesh default branch, fast-forward obvious clean behind-only branches with \`mesh_fast_forward_node\`, or explicitly report one of \`pushed_feature_branch_needs_merge\`, \`blocked_review\`, \`cleanup_candidate\`, or \`not_mergeable\` with the next action.
|
|
206
|
+
- **Keep Refinery validation project-configurable.** \`mesh_refine_node\` must execute validation from repo mesh/refine config (for example \`.adhdev/refine.{json,yaml,yml}\`, \`.adhdev/repo-mesh-refine.*\`, or \`repo-mesh.refine.*\`). Heuristics are suggestions/scaffolding only, not the execution path.
|
|
205
207
|
- **Name worktree branches meaningfully.** Use descriptive names like \`feat/auth-refactor\` or \`fix/build-123\`.${coordinatorNote}`;
|
|
206
208
|
}
|