2b6170313b
Единица разбора — спан: реальный промпт владельца + вся активность ассистента
до следующего реального промпта. Системные ходы (гейт-фидбек, загрузка навыка)
приклеиваются к спану, не считаются отдельными. Разбор отложенный: закрытые
спаны разбираются один раз (курсор в флажке сессии); reconcile и аудитор
получают ПОЛНЫЙ склеенный спан (промпт + все ответы + все действия).
- Слой 1: снят обрез вывода действий (полная картина), защита структурных меток.
- Граница спана — событие UserPromptSubmit (prompt-hook метит realPromptTurns),
фолбэк по sysLabel; выключение через mode:closing (финальный спан добивает Stop).
- Калибровка скрытых вопросов: страж-ноп (не мутировать при неизменном тексте) +
кап показа родословной (~~первая~~ → текущая, данные целы).
- Шаги — по спанам («Ход (промпт) N [вобрал ходы X-Y]»); «висит N промптов».
- Новый модуль secretary-span.mjs (computeSpans/spansToDistill/recordRealPrompt/
parseTurnBlock/assembleSpan).
Свод секретаря зелёный (138 тестов), живой прогон на реальной модели подтвердил:
Шаги по спанам, гейт-шум не плодит скрытые вопросы, находки выживают по одному раз.
Спека/план: docs/superpowers/{specs,plans}/2026-06-23-secretary-span-redesign*.
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
89 lines
4.8 KiB
JavaScript
89 lines
4.8 KiB
JavaScript
import { describe, it, expect } from 'vitest';
|
||
import { parseLastExchange } from './secretary-transcript.mjs';
|
||
|
||
describe('parseLastExchange', () => {
|
||
it('тащит последний user + assistant + действия', () => {
|
||
const t = [
|
||
JSON.stringify({ message: { role: 'user', content: 'старое' } }),
|
||
JSON.stringify({ message: { role: 'user', content: 'привет' } }),
|
||
JSON.stringify({ message: { role: 'assistant', content: [
|
||
{ type: 'text', text: 'ответ' },
|
||
{ type: 'tool_use', name: 'Read', input: { f: 'x' } },
|
||
] } }),
|
||
].join('\n');
|
||
const ex = parseLastExchange(t);
|
||
expect(ex.user).toBe('привет');
|
||
expect(ex.assistant).toBe('ответ');
|
||
expect(ex.actions).toEqual([{ tool: 'Read', input: '{"f":"x"}' }]);
|
||
});
|
||
it('строковый content тоже понимает; битые строки пропускает', () => {
|
||
const t = ['не-json', JSON.stringify({ message: { role: 'user', content: 'у' } }),
|
||
JSON.stringify({ message: { role: 'assistant', content: 'а' } })].join('\n');
|
||
const ex = parseLastExchange(t);
|
||
expect(ex.user).toBe('у');
|
||
expect(ex.assistant).toBe('а');
|
||
expect(ex.actions).toEqual([]);
|
||
});
|
||
it('пропускает tool_result (role:user) — берёт настоящий промпт + все действия', () => {
|
||
const t = [
|
||
JSON.stringify({ message: { role: 'user', content: 'настоящий вопрос' } }),
|
||
JSON.stringify({ message: { role: 'assistant', content: [
|
||
{ type: 'text', text: 'думаю' }, { type: 'tool_use', name: 'Read', input: { f: 'a' } }] } }),
|
||
JSON.stringify({ message: { role: 'user', content: [{ type: 'tool_result', content: 'результат' }] } }),
|
||
JSON.stringify({ message: { role: 'assistant', content: [{ type: 'text', text: 'готово' }] } }),
|
||
].join('\n');
|
||
const ex = parseLastExchange(t);
|
||
expect(ex.user).toBe('настоящий вопрос');
|
||
expect(ex.assistant).toContain('думаю');
|
||
expect(ex.assistant).toContain('готово');
|
||
expect(ex.actions).toEqual([{ tool: 'Read', input: '{"f":"a"}' }]);
|
||
});
|
||
});
|
||
|
||
describe('parseLastExchange — захват выдачи инструмента (tool_result по tool_use_id)', () => {
|
||
it('привязывает результат к действию по совпадающему id', () => {
|
||
const t = [
|
||
JSON.stringify({ message: { role: 'user', content: 'вопрос' } }),
|
||
JSON.stringify({ message: { role: 'assistant', content: [
|
||
{ type: 'tool_use', id: 'tu_1', name: 'Read', input: { f: 'a' } }] } }),
|
||
JSON.stringify({ message: { role: 'user', content: [
|
||
{ type: 'tool_result', tool_use_id: 'tu_1', content: 'СОДЕРЖИМОЕ ФАЙЛА' }] } }),
|
||
].join('\n');
|
||
const ex = parseLastExchange(t);
|
||
expect(ex.actions).toEqual([{ tool: 'Read', input: '{"f":"a"}', result: 'СОДЕРЖИМОЕ ФАЙЛА' }]);
|
||
});
|
||
it('результат из массива text-блоков склеивается', () => {
|
||
const t = [
|
||
JSON.stringify({ message: { role: 'user', content: 'в' } }),
|
||
JSON.stringify({ message: { role: 'assistant', content: [
|
||
{ type: 'tool_use', id: 'tu_9', name: 'Bash', input: {} }] } }),
|
||
JSON.stringify({ message: { role: 'user', content: [
|
||
{ type: 'tool_result', tool_use_id: 'tu_9', content: [{ type: 'text', text: 'строка вывода' }] }] } }),
|
||
].join('\n');
|
||
const ex = parseLastExchange(t);
|
||
expect(ex.actions[0].result).toBe('строка вывода');
|
||
});
|
||
it('длинный результат НЕ обрезается (полная картина для секретаря)', () => {
|
||
const big = 'x'.repeat(5000);
|
||
const t = [
|
||
JSON.stringify({ message: { role: 'user', content: 'в' } }),
|
||
JSON.stringify({ message: { role: 'assistant', content: [
|
||
{ type: 'tool_use', id: 'tu_2', name: 'Read', input: {} }] } }),
|
||
JSON.stringify({ message: { role: 'user', content: [
|
||
{ type: 'tool_result', tool_use_id: 'tu_2', content: big }] } }),
|
||
].join('\n');
|
||
const ex = parseLastExchange(t);
|
||
expect(ex.actions[0].result).toBe(big); // целиком
|
||
expect(ex.actions[0].result.endsWith('…')).toBe(false);
|
||
});
|
||
it('без совпадающего id результат не привязывается — старая форма {tool,input} цела', () => {
|
||
const t = [
|
||
JSON.stringify({ message: { role: 'user', content: 'в' } }),
|
||
JSON.stringify({ message: { role: 'assistant', content: [
|
||
{ type: 'tool_use', id: 'tu_3', name: 'Read', input: { f: 'z' } }] } }),
|
||
].join('\n');
|
||
const ex = parseLastExchange(t);
|
||
expect(ex.actions).toEqual([{ tool: 'Read', input: '{"f":"z"}' }]);
|
||
});
|
||
});
|