feat(workspace): claudemesh me activity + dashboard parity
ships v0.4.0 phase 4. final aggregating verb after this is me search (phase 5). api: GET /v1/me/activity returns topic messages across every mesh the user belongs to in a 24h default window (?since=iso override), excluding messages the caller authored themselves. "what is happening that i missed", capped at 200. cli (1.13.0): claudemesh me activity prints a condensed feed with mesh + topic + sender + relative timestamp + snippet (or [encrypted] for v2 ciphertext). web: /dashboard/activity clusters consecutive messages from the same topic into thread blocks for readability. sidebar gains an activity entry between notifications and invites. Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
This commit is contained in:
@@ -1,6 +1,6 @@
|
||||
{
|
||||
"name": "claudemesh-cli",
|
||||
"version": "1.12.0",
|
||||
"version": "1.13.0",
|
||||
"description": "Peer mesh for Claude Code sessions — CLI + MCP server.",
|
||||
"keywords": [
|
||||
"claude-code",
|
||||
|
||||
@@ -294,6 +294,86 @@ export async function runMeNotifications(
|
||||
);
|
||||
}
|
||||
|
||||
interface WorkspaceActivity {
|
||||
messageId: string;
|
||||
topicId: string;
|
||||
topicName: string;
|
||||
meshId: string;
|
||||
meshSlug: string;
|
||||
meshName: string;
|
||||
senderName: string;
|
||||
senderMemberId: string;
|
||||
snippet: string | null;
|
||||
ciphertext: string | null;
|
||||
bodyVersion: number;
|
||||
createdAt: string;
|
||||
}
|
||||
|
||||
interface WorkspaceActivityResponse {
|
||||
activity: WorkspaceActivity[];
|
||||
totals: { events: number };
|
||||
}
|
||||
|
||||
export interface MeActivityFlags extends MeFlags {
|
||||
since?: string;
|
||||
}
|
||||
|
||||
export async function runMeActivity(flags: MeActivityFlags): Promise<number> {
|
||||
return withRestKey(
|
||||
{
|
||||
meshSlug: flags.mesh ?? null,
|
||||
purpose: "workspace-activity",
|
||||
capabilities: ["read"],
|
||||
},
|
||||
async ({ secret }) => {
|
||||
const params = new URLSearchParams();
|
||||
if (flags.since) params.set("since", flags.since);
|
||||
const path =
|
||||
"/api/v1/me/activity" +
|
||||
(params.toString() ? `?${params.toString()}` : "");
|
||||
const ws = await request<WorkspaceActivityResponse>({
|
||||
path,
|
||||
token: secret,
|
||||
});
|
||||
|
||||
if (flags.json) {
|
||||
console.log(JSON.stringify(ws, null, 2));
|
||||
return EXIT.SUCCESS;
|
||||
}
|
||||
|
||||
render.section(
|
||||
`${clay("activity")} — ${ws.totals.events} ${dim(
|
||||
flags.since ? `since ${flags.since}` : "in the last 24h",
|
||||
)}`,
|
||||
);
|
||||
|
||||
if (ws.activity.length === 0) {
|
||||
process.stdout.write(dim(" quiet — no activity in window\n"));
|
||||
return EXIT.SUCCESS;
|
||||
}
|
||||
|
||||
const slugWidth = Math.max(
|
||||
...ws.activity.map((a) => a.meshSlug.length),
|
||||
6,
|
||||
);
|
||||
|
||||
for (const a of ws.activity) {
|
||||
const slug = dim(a.meshSlug.padEnd(slugWidth));
|
||||
const topic = cyan(`#${a.topicName}`);
|
||||
const sender = a.senderName ?? "?";
|
||||
const ago = formatRelativeTime(a.createdAt);
|
||||
const snippet =
|
||||
a.snippet ?? (a.ciphertext ? dim("[encrypted]") : dim("[empty]"));
|
||||
process.stdout.write(
|
||||
` ${slug} ${topic} ${dim(sender + " ·")} ${dim(ago)}\n` +
|
||||
` ${snippet.length > 200 ? snippet.slice(0, 200) + "…" : snippet}\n`,
|
||||
);
|
||||
}
|
||||
return EXIT.SUCCESS;
|
||||
},
|
||||
);
|
||||
}
|
||||
|
||||
function formatRelativeTime(iso: string): string {
|
||||
const then = new Date(iso).getTime();
|
||||
const now = Date.now();
|
||||
|
||||
@@ -126,6 +126,7 @@ Topic (conversation scope, v0.2.0)
|
||||
claudemesh me cross-mesh workspace overview (v0.4.0)
|
||||
claudemesh me topics cross-mesh topic list [--unread]
|
||||
claudemesh me notifications cross-mesh @-mentions [--all] [--since=ISO]
|
||||
claudemesh me activity cross-mesh recent messages [--since=ISO]
|
||||
claudemesh member list mesh roster with online state [--online]
|
||||
claudemesh notification list recent @-mentions of you [--since <ISO>]
|
||||
|
||||
@@ -697,6 +698,14 @@ async function main(): Promise<void> {
|
||||
since: flags.since as string | undefined,
|
||||
}),
|
||||
);
|
||||
} else if (sub === "activity") {
|
||||
const { runMeActivity } = await import("~/commands/me.js");
|
||||
process.exit(
|
||||
await runMeActivity({
|
||||
...f,
|
||||
since: flags.since as string | undefined,
|
||||
}),
|
||||
);
|
||||
} else {
|
||||
console.error(
|
||||
"Usage: claudemesh me (cross-mesh overview)\n" +
|
||||
@@ -704,7 +713,9 @@ async function main(): Promise<void> {
|
||||
" claudemesh me topics --unread (only unread topics)\n" +
|
||||
" claudemesh me notifications (unread @-mentions, last 7d)\n" +
|
||||
" claudemesh me notifications --all (include already-read)\n" +
|
||||
" claudemesh me notifications --since=ISO (custom window)",
|
||||
" claudemesh me notifications --since=ISO (custom window)\n" +
|
||||
" claudemesh me activity (recent messages, last 24h)\n" +
|
||||
" claudemesh me activity --since=ISO (custom window)",
|
||||
);
|
||||
process.exit(EXIT.INVALID_ARGS);
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user