mirror of
https://github.com/affaan-m/everything-claude-code.git
synced 2026-03-31 06:03:29 +08:00
128 lines
3.5 KiB
JavaScript
128 lines
3.5 KiB
JavaScript
'use strict';
|
|
|
|
const { createClaudeHistoryAdapter } = require('./claude-history');
|
|
const { createDmuxTmuxAdapter } = require('./dmux-tmux');
|
|
|
|
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'
|
|
});
|
|
|
|
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'))
|
|
];
|
|
}
|
|
|
|
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
|
|
};
|
|
}
|
|
|
|
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
|
|
};
|