Files
claude-mem/tests/sdk/parse-summary.test.ts
T
Ousama Ben Younes 93a30c5c8f fix: skip parseSummary false positives with no sub-tags (#1360)
When an observation response accidentally contains a <summary> tag with
plain text (no <request>/<investigated>/etc. sub-tags), parseSummary was
creating empty SESSION SUMMARY records with all fields as empty strings.

Add an all-null guard AFTER field extraction: if none of the 5 sub-tags
matched, the <summary> match is a false positive and we return null.

This is distinct from the commented-out validation above (which rejected
summaries with SOME missing fields). We only reject when ALL are absent —
real partial summaries are still saved per the maintainer's explicit note.

Closes #1360

Co-Authored-By: Claude <noreply@anthropic.com>
2026-04-01 06:51:33 +00:00

54 lines
2.2 KiB
TypeScript

/**
* Tests for parseSummary (fix for #1360)
*
* Validates that false-positive summary matches (no sub-tags) are rejected
* while real summaries — even with some missing fields — are still saved.
*/
import { describe, it, expect } from 'bun:test';
import { parseSummary } from '../../src/sdk/parser.js';
describe('parseSummary', () => {
it('returns null when no <summary> tag present', () => {
expect(parseSummary('<observation><title>foo</title></observation>')).toBeNull();
});
it('returns null when <summary> has no sub-tags (false positive — fix for #1360)', () => {
// This is the bug: observation response accidentally contains <summary>some text</summary>
expect(parseSummary('<observation>done <summary>some content here</summary></observation>')).toBeNull();
});
it('returns null for bare <summary> with only plain text, no sub-tags', () => {
expect(parseSummary('<summary>This session was productive.</summary>')).toBeNull();
});
it('returns summary when at least one sub-tag is present (respects maintainer note)', () => {
const text = `<summary><request>Fix the bug</request></summary>`;
const result = parseSummary(text);
expect(result).not.toBeNull();
expect(result?.request).toBe('Fix the bug');
expect(result?.investigated).toBeNull();
expect(result?.learned).toBeNull();
});
it('returns full summary when all fields are present', () => {
const text = `<summary>
<request>Fix login bug</request>
<investigated>Auth flow and JWT expiry</investigated>
<learned>Token was expiring too soon</learned>
<completed>Extended token TTL to 24h</completed>
<next_steps>Monitor error rates</next_steps>
</summary>`;
const result = parseSummary(text);
expect(result).not.toBeNull();
expect(result?.request).toBe('Fix login bug');
expect(result?.investigated).toBe('Auth flow and JWT expiry');
expect(result?.learned).toBe('Token was expiring too soon');
expect(result?.completed).toBe('Extended token TTL to 24h');
expect(result?.next_steps).toBe('Monitor error rates');
});
it('returns null when skip_summary tag is present', () => {
expect(parseSummary('<skip_summary reason="no work done"/>')).toBeNull();
});
});