extend the daemon thin-client surface to two more verb families: state get/set/list now routes through `/v1/state`, and remember/recall/forget through `/v1/memory`. same warm-path pattern as 1.25.0 — try the unix socket first, fall back to the cold ws path when the daemon is absent. multi-mesh aware (aggregates on read, requires `--mesh` for writes when ambiguous). also ships an early `claudemesh workspace <verb>` alias surface — bare teaser for the 1.28.0 mesh→workspace public rename. no-arg falls through to launch. Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
54 lines
2.0 KiB
TypeScript
54 lines
2.0 KiB
TypeScript
import { withMesh } from "./connect.js";
|
|
import { tryRecallViaDaemon } from "~/services/bridge/daemon-route.js";
|
|
import { render } from "~/ui/render.js";
|
|
import { bold, clay, dim } from "~/ui/styles.js";
|
|
import { EXIT } from "~/constants/exit-codes.js";
|
|
|
|
export async function recall(
|
|
query: string,
|
|
opts: { mesh?: string; json?: boolean } = {},
|
|
): Promise<number> {
|
|
if (!query) {
|
|
render.err("Usage: claudemesh recall <query>");
|
|
return EXIT.INVALID_ARGS;
|
|
}
|
|
|
|
// Daemon path first.
|
|
const daemonMatches = await tryRecallViaDaemon(query, opts.mesh);
|
|
if (daemonMatches !== null) {
|
|
if (opts.json) { console.log(JSON.stringify(daemonMatches, null, 2)); return EXIT.SUCCESS; }
|
|
if (daemonMatches.length === 0) { render.info(dim("no memories found.")); return EXIT.SUCCESS; }
|
|
render.section(`memories (${daemonMatches.length})`);
|
|
for (const m of daemonMatches) {
|
|
const tags = m.tags.length ? dim(` [${m.tags.map((t) => clay(t)).join(dim(", "))}]`) : "";
|
|
process.stdout.write(` ${bold(m.id.slice(0, 8))}${tags}\n`);
|
|
process.stdout.write(` ${m.content}\n`);
|
|
process.stdout.write(` ${dim(m.rememberedBy + " · " + new Date(m.rememberedAt).toLocaleString())}\n\n`);
|
|
}
|
|
return EXIT.SUCCESS;
|
|
}
|
|
|
|
return await withMesh({ meshSlug: opts.mesh ?? null }, async (client) => {
|
|
const memories = await client.recall(query);
|
|
|
|
if (opts.json) {
|
|
console.log(JSON.stringify(memories, null, 2));
|
|
return EXIT.SUCCESS;
|
|
}
|
|
|
|
if (memories.length === 0) {
|
|
render.info(dim("no memories found."));
|
|
return EXIT.SUCCESS;
|
|
}
|
|
|
|
render.section(`memories (${memories.length})`);
|
|
for (const m of memories) {
|
|
const tags = m.tags.length ? dim(` [${m.tags.map((t) => clay(t)).join(dim(", "))}]`) : "";
|
|
process.stdout.write(` ${bold(m.id.slice(0, 8))}${tags}\n`);
|
|
process.stdout.write(` ${m.content}\n`);
|
|
process.stdout.write(` ${dim(m.rememberedBy + " · " + new Date(m.rememberedAt).toLocaleString())}\n\n`);
|
|
}
|
|
return EXIT.SUCCESS;
|
|
});
|
|
}
|