mirror of
https://github.com/affaan-m/everything-claude-code.git
synced 2026-06-11 02:33:10 +08:00
Adds the third session adapter (after dmux-tmux and claude-history), normalizing Codex rollout sessions into the harness-neutral ecc.session.v1 snapshot. Reads ~/.codex/sessions rollout JSONL, derives objective (skipping the AGENTS.md preamble + leading message UUID), model, originator, worktree cwd, and best-effort git branch. This is step 1 of ECC-2.0-SESSION-ADAPTER-DISCOVERY (move the abstraction beyond tmux + Claude-history) and supports the wrap/adapt control-pane strategy: ECC reads sessions from any harness rather than owning one UX. - scripts/lib/session-adapters/codex-worktree.js: adapter + rollout parser - canonical-session.js: normalizeCodexWorktreeSession - registry.js: register adapter, codex/codex-worktree target types - tests/lib/session-adapters-codex.test.js: 4 tests (unit + registry routing)
139 lines
3.9 KiB
JavaScript
139 lines
3.9 KiB
JavaScript
'use strict';
|
|
|
|
const { createClaudeHistoryAdapter } = require('./claude-history');
|
|
const { createDmuxTmuxAdapter } = require('./dmux-tmux');
|
|
const { createCodexWorktreeAdapter } = require('./codex-worktree');
|
|
|
|
const TARGET_TYPE_TO_ADAPTER_ID = Object.freeze({
|
|
plan: 'dmux-tmux',
|
|
session: 'dmux-tmux',
|
|
'claude-history': 'claude-history',
|
|
'claude-alias': 'claude-history',
|
|
'session-file': 'claude-history',
|
|
'codex-worktree': 'codex-worktree',
|
|
codex: 'codex-worktree'
|
|
});
|
|
|
|
function buildDefaultAdapterOptions(options, adapterId) {
|
|
const sharedOptions = {
|
|
loadStateStoreImpl: options.loadStateStoreImpl,
|
|
persistSnapshots: options.persistSnapshots,
|
|
recordingDir: options.recordingDir,
|
|
stateStore: options.stateStore
|
|
};
|
|
|
|
return {
|
|
...sharedOptions,
|
|
...(options.adapterOptions && options.adapterOptions[adapterId]
|
|
? options.adapterOptions[adapterId]
|
|
: {})
|
|
};
|
|
}
|
|
|
|
function createDefaultAdapters(options = {}) {
|
|
return [
|
|
createClaudeHistoryAdapter(buildDefaultAdapterOptions(options, 'claude-history')),
|
|
createDmuxTmuxAdapter(buildDefaultAdapterOptions(options, 'dmux-tmux')),
|
|
createCodexWorktreeAdapter(buildDefaultAdapterOptions(options, 'codex-worktree'))
|
|
];
|
|
}
|
|
|
|
function coerceTargetValue(value) {
|
|
if (typeof value !== 'string' || value.trim().length === 0) {
|
|
throw new Error('Structured session targets require a non-empty string value');
|
|
}
|
|
|
|
return value.trim();
|
|
}
|
|
|
|
function normalizeStructuredTarget(target, context = {}) {
|
|
if (!target || typeof target !== 'object' || Array.isArray(target)) {
|
|
return {
|
|
target,
|
|
context: { ...context }
|
|
};
|
|
}
|
|
|
|
const value = coerceTargetValue(target.value);
|
|
const type = typeof target.type === 'string' ? target.type.trim() : '';
|
|
if (type.length === 0) {
|
|
throw new Error('Structured session targets require a non-empty type');
|
|
}
|
|
|
|
const adapterId = target.adapterId || TARGET_TYPE_TO_ADAPTER_ID[type] || context.adapterId || null;
|
|
const nextContext = {
|
|
...context,
|
|
adapterId
|
|
};
|
|
|
|
if (type === 'claude-history' || type === 'claude-alias') {
|
|
return {
|
|
target: `claude:${value}`,
|
|
context: nextContext
|
|
};
|
|
}
|
|
|
|
if (type === 'codex-worktree' || type === 'codex') {
|
|
return {
|
|
target: `codex:${value}`,
|
|
context: nextContext
|
|
};
|
|
}
|
|
|
|
return {
|
|
target: value,
|
|
context: nextContext
|
|
};
|
|
}
|
|
|
|
function createAdapterRegistry(options = {}) {
|
|
const adapters = options.adapters || createDefaultAdapters(options);
|
|
|
|
return {
|
|
adapters,
|
|
getAdapter(id) {
|
|
const adapter = adapters.find(candidate => candidate.id === id);
|
|
if (!adapter) {
|
|
throw new Error(`Unknown session adapter: ${id}`);
|
|
}
|
|
|
|
return adapter;
|
|
},
|
|
listAdapters() {
|
|
return adapters.map(adapter => ({
|
|
id: adapter.id,
|
|
description: adapter.description || '',
|
|
targetTypes: Array.isArray(adapter.targetTypes) ? [...adapter.targetTypes] : []
|
|
}));
|
|
},
|
|
select(target, context = {}) {
|
|
const normalized = normalizeStructuredTarget(target, context);
|
|
const adapter = normalized.context.adapterId
|
|
? this.getAdapter(normalized.context.adapterId)
|
|
: adapters.find(candidate => candidate.canOpen(normalized.target, normalized.context));
|
|
if (!adapter) {
|
|
throw new Error(`No session adapter matched target: ${target}`);
|
|
}
|
|
|
|
return adapter;
|
|
},
|
|
open(target, context = {}) {
|
|
const normalized = normalizeStructuredTarget(target, context);
|
|
const adapter = this.select(normalized.target, normalized.context);
|
|
return adapter.open(normalized.target, normalized.context);
|
|
}
|
|
};
|
|
}
|
|
|
|
function inspectSessionTarget(target, options = {}) {
|
|
const registry = createAdapterRegistry(options);
|
|
return registry.open(target, options).getSnapshot();
|
|
}
|
|
|
|
module.exports = {
|
|
createAdapterRegistry,
|
|
createDefaultAdapters,
|
|
inspectSessionTarget,
|
|
normalizeStructuredTarget
|
|
};
|