Add temporal versioning for node history tracking (Milestone 1)
Enable time-travel queries and history viewing by creating immutable version records on every node update. Includes database schema changes, store functions, MCP tools, and CLI commands for viewing history, comparing versions, and restoring to previous states.
This commit is contained in:
@@ -1,11 +1,12 @@
|
||||
import { Command } from 'commander';
|
||||
import chalk from 'chalk';
|
||||
import { findNodeByPrefix } from '../../core/store';
|
||||
import { findNodeByPrefix, getNodeAtTime, getCurrentVersion } from '../../core/store';
|
||||
import { getConnections } from '../../core/graph';
|
||||
|
||||
export const showCommand = new Command('show')
|
||||
.argument('<id>', 'Node ID (or prefix)')
|
||||
.option('--format <fmt>', 'Output format: text or json', 'text')
|
||||
.option('--at <timestamp>', 'Show node at a specific point in time (ISO date or timestamp)')
|
||||
.description('Show a node and its connections')
|
||||
.action(async (idRaw: string, opts) => {
|
||||
const node = findNodeByPrefix(idRaw);
|
||||
@@ -14,15 +15,56 @@ export const showCommand = new Command('show')
|
||||
process.exit(1);
|
||||
}
|
||||
|
||||
// If --at is specified, show historical state
|
||||
if (opts.at) {
|
||||
const ts = isNaN(Number(opts.at)) ? Date.parse(opts.at) : Number(opts.at);
|
||||
if (isNaN(ts)) {
|
||||
console.error(chalk.red('Invalid timestamp format. Use ISO date (e.g., 2024-01-01) or Unix timestamp.'));
|
||||
process.exit(1);
|
||||
}
|
||||
|
||||
const historical = getNodeAtTime(node.id, ts);
|
||||
if (!historical) {
|
||||
console.error(chalk.red('No version found for the specified time.'));
|
||||
process.exit(1);
|
||||
}
|
||||
|
||||
if (opts.format === 'json') {
|
||||
console.log(JSON.stringify(historical, null, 2));
|
||||
return;
|
||||
}
|
||||
|
||||
console.log(chalk.dim(`Viewing historical state at: ${new Date(ts).toLocaleString()}`));
|
||||
console.log('');
|
||||
console.log(chalk.bold.cyan(`[${historical.kind}] ${historical.title}`));
|
||||
console.log(`ID: ${historical.id}`);
|
||||
console.log(`Version: v${historical.version}`);
|
||||
if (historical.status) console.log(`Status: ${historical.status}`);
|
||||
if (historical.tags.length) console.log(`Tags: ${historical.tags.join(', ')}`);
|
||||
console.log(`Valid: ${new Date(historical.validFrom).toLocaleString()} - ${historical.validUntil ? new Date(historical.validUntil).toLocaleString() : 'current'}`);
|
||||
if (historical.content) console.log(`\n${historical.content}`);
|
||||
|
||||
// Render structured sections
|
||||
if (historical.metadata?.sections && Array.isArray(historical.metadata.sections)) {
|
||||
for (const sec of historical.metadata.sections) {
|
||||
console.log(`\n${chalk.bold(`-- ${sec.label} --`)}`);
|
||||
if (sec.body) console.log(sec.body);
|
||||
}
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
const conns = getConnections(node.id);
|
||||
const version = getCurrentVersion(node.id);
|
||||
|
||||
if (opts.format === 'json') {
|
||||
console.log(JSON.stringify({ ...node, embedding: undefined, connections: conns }, null, 2));
|
||||
console.log(JSON.stringify({ ...node, embedding: undefined, version, connections: conns }, null, 2));
|
||||
return;
|
||||
}
|
||||
|
||||
console.log(chalk.bold.cyan(`[${node.kind}] ${node.title}`));
|
||||
console.log(`ID: ${node.id}`);
|
||||
console.log(`Version: v${version}`);
|
||||
if (node.status) console.log(`Status: ${node.status}`);
|
||||
if (node.tags.length) console.log(`Tags: ${node.tags.join(', ')}`);
|
||||
console.log(`Created: ${new Date(node.createdAt).toLocaleString()}`);
|
||||
|
||||
Reference in New Issue
Block a user