Files
repomix-mirror/tests/core/file/fileRead.test.ts
Kazuki Yamada 3ff49306e1 refactor(file): Simplify cheap pre-screen down to NULL probe + BOM exemption
intent(simpler): 前 commit の cheap pre-screen は `isbinaryfile@5.0.2` の `isBinaryCheck` のうち valid UTF-8 でも binary 判定する 3 規則 (PDF magic / NULL / suspicious 制御バイト比率 >10%) を mirror していたが、`TextDecoder('utf-8', { fatal: true })` を `isBinaryFile` の前に動かしている時点で valid UTF-8 buffer は protobuf detector に渡らない。pathological case 回避という主目的は TextDecoder の reorder だけで達成しており、PDF magic と suspicious-byte ratio の mirror は (1) 実害ほぼゼロのエッジケースを救うだけで (2) `isbinaryfile` 内部実装への coupling を抱える、という割に合わない構成だった。

fix(simplify): cheap pre-screen を NULL-byte probe + BOM exemption の最小構成に縮小。

decision(keep-null-probe): NULL byte だけは独立した正当な理由で残す — `U+0000` は **XML 1.0 で不正な文字** で、本ツールの主出力フォーマット (XML) の正当性を破壊する。`TextDecoder` は `0x00` を valid UTF-8 (U+0000) として通すので、ここで弾かないと NULL を含む buffer が text として pack され、downstream の XML parser が落ちる。これは `isbinaryfile` の rule mirror ではなく、repomix 自身の出力 robustness 要件。

decision(drop-pdf-magic): PDF は `is-binary-path` の `.pdf` 拡張子で先に弾かれる。拡張子なしの ASCII-only PDF stub は実例ほぼゼロ (本物の PDF は cross-ref とバイナリストリームを内包し UTF-8 decode で失敗する経路を通る)。守る価値が低い。

decision(drop-suspicious-ratio): 純粋な C0 制御バイト高比率の valid UTF-8 buffer は実プロジェクトに存在しない。`isbinaryfile` の UTF-8 lookahead を完全 mirror する保守コスト (DEL boundary 等のドリフトリスク) > 効用。

constraint(coupling-minimal): NULL byte は universal な binary signal で、`isbinaryfile` の rule に縛られない。同 BOM exemption も標準 BOM の規格に準拠したもので upstream ドリフトの影響を受けない。

test(cleanup): 関連 regression test 3 件を削除 (PDF magic / suspicious ratio / DEL boundary)。これらは削除した規則の挙動を保証するもので、現実装ではすべて意図的に「text として pack」する。残るのは UTF-8 multi-byte / UTF-8 BOM+NULL / UTF-16 LE BOM の 3 件で、いずれも本 PR が回避したい pathological / regression を直接守る。

bench(no-regression, M-series Mac, hyperfine --warmup 1 --runs 5):
- `node bin/repomix.cjs --quiet`: 399ms → 418ms (誤差範囲、JS の手書き 512-byte loop が消えた分の差は noise floor)
- 出力差分: 既存の base に対して 0 ファイル削除、Korean md が +1 (silent drop 解消、変更なし)
- fileRead.ts: 175 → 159 行 (-16 行)。pre-screen 関連で実質 ~35 行削減。

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-05 23:11:29 +09:00

198 lines
8.1 KiB
TypeScript

import * as fs from 'node:fs/promises';
import os from 'node:os';
import path from 'node:path';
import { afterEach, beforeEach, describe, expect, test } from 'vitest';
import { readRawFile } from '../../../src/core/file/fileRead.js';
describe('readRawFile', () => {
let testDir: string;
beforeEach(async () => {
testDir = await fs.mkdtemp(path.join(os.tmpdir(), 'repomix-fileRead-'));
});
afterEach(async () => {
await fs.rm(testDir, { recursive: true, force: true });
});
test('should read normal text file successfully', async () => {
const filePath = path.join(testDir, 'normal.txt');
const content = 'Hello World';
await fs.writeFile(filePath, content, 'utf-8');
const result = await readRawFile(filePath, 1024);
expect(result.content).toBe(content);
expect(result.skippedReason).toBeUndefined();
});
test('should read file with low jschardet confidence (Issue #869)', async () => {
// This tests that files with low confidence scores from jschardet
// are NOT skipped if they contain valid UTF-8 content
const filePath = path.join(testDir, 'server.py');
const content = `import json
import time
import uuid
def hello():
print("Hello, World!")
`;
await fs.writeFile(filePath, content, 'utf-8');
const result = await readRawFile(filePath, 1024 * 1024);
expect(result.content).toBe(content);
expect(result.skippedReason).toBeUndefined();
});
test('should read HTML file with Thymeleaf syntax (Issue #847)', async () => {
// This tests that HTML files with special syntax like Thymeleaf (~{})
// are NOT skipped even if jschardet returns low confidence
const filePath = path.join(testDir, 'thymeleaf.html');
const content = '<html lang="en" xmlns:th="http://www.thymeleaf.org" layout:decorate="~{layouts/default}"></html>';
await fs.writeFile(filePath, content, 'utf-8');
const result = await readRawFile(filePath, 1024);
expect(result.content).toBe(content);
expect(result.skippedReason).toBeUndefined();
});
test('should read empty file successfully', async () => {
// Empty files should not be skipped (jschardet may return 0 confidence for empty files)
const filePath = path.join(testDir, '__init__.py');
await fs.writeFile(filePath, '', 'utf-8');
const result = await readRawFile(filePath, 1024);
expect(result.content).toBe('');
expect(result.skippedReason).toBeUndefined();
});
test('should read file containing legitimate U+FFFD character', async () => {
// This tests that files with intentional U+FFFD characters in the source
// are NOT skipped (TextDecoder can decode them successfully)
const filePath = path.join(testDir, 'with-replacement-char.txt');
// U+FFFD is a valid Unicode character that can appear in source files
const content = 'Some text with replacement char: \uFFFD and more text';
await fs.writeFile(filePath, content, 'utf-8');
const result = await readRawFile(filePath, 1024);
expect(result.content).toBe(content);
expect(result.skippedReason).toBeUndefined();
});
test('should skip file with actual decode errors (U+FFFD)', async () => {
const filePath = path.join(testDir, 'invalid.txt');
// Create a file with a UTF-8 BOM followed by valid text and invalid UTF-8 sequences
// The BOM forces UTF-8 detection, and the invalid sequence will produce U+FFFD
const utf8Bom = Buffer.from([0xef, 0xbb, 0xbf]); // UTF-8 BOM
const validText = 'Hello World\n'.repeat(50);
// Invalid UTF-8: 0x80 is a continuation byte without a leading byte
const invalidSequence = Buffer.from([0x80, 0x81, 0x82]);
const buffer = Buffer.concat([utf8Bom, Buffer.from(validText), invalidSequence, Buffer.from(validText)]);
await fs.writeFile(filePath, buffer);
const result = await readRawFile(filePath, 1024 * 1024);
expect(result.content).toBeNull();
expect(result.skippedReason).toBe('encoding-error');
});
test('should skip file if it exceeds size limit', async () => {
const filePath = path.join(testDir, 'large.txt');
const content = 'x'.repeat(1000);
await fs.writeFile(filePath, content, 'utf-8');
const result = await readRawFile(filePath, 100);
expect(result.content).toBeNull();
expect(result.skippedReason).toBe('size-limit');
});
test('should skip binary file by extension', async () => {
const filePath = path.join(testDir, 'test.jpg');
const binaryData = Buffer.from([0xff, 0xd8, 0xff, 0xe0]);
await fs.writeFile(filePath, binaryData);
const result = await readRawFile(filePath, 1024);
expect(result.content).toBeNull();
expect(result.skippedReason).toBe('binary-extension');
});
test('should skip binary content in text extension file', async () => {
const filePath = path.join(testDir, 'binary.txt');
// Create file with binary content (null bytes and control characters)
const binaryData = Buffer.alloc(256);
for (let i = 0; i < 256; i++) {
binaryData[i] = i;
}
await fs.writeFile(filePath, binaryData);
const result = await readRawFile(filePath, 1024);
expect(result.content).toBeNull();
expect(result.skippedReason).toBe('binary-content');
});
test('should read valid UTF-8 multi-byte file without invoking isBinaryFile', async () => {
// Regression: prior to the UTF-8-first reorder, certain valid-UTF-8
// byte patterns triggered an O(n) protobuf-detector loop inside
// `isbinaryfile` that could spend seconds and ultimately throw
// `Invalid array length` (concrete trigger:
// `website/client/src/ko/guide/tips/best-practices.md`). The throw was
// caught by `readRawFile`'s outer try/catch and the file was silently
// dropped as `encoding-error`. After the reorder, valid UTF-8 with no
// NULL bytes must round-trip as text content without ever invoking
// `isBinaryFile`.
const filePath = path.join(testDir, 'korean.md');
// Korean Hangul syllables encode as 3-byte UTF-8 sequences (0xE0-0xEF
// lead bytes followed by two 0x80-0xBF continuation bytes); none of
// those bytes are NULL.
const content = `${'안녕하세요 '.repeat(200)}\n`; // ~3.6 KB of multi-byte UTF-8
await fs.writeFile(filePath, content, 'utf-8');
const result = await readRawFile(filePath, 1024 * 1024);
expect(result.content).toBe(content);
expect(result.skippedReason).toBeUndefined();
});
test('should not classify UTF-8 BOM file as binary even when followed by NULL', async () => {
// Regression: `isbinaryfile@5.0.2`'s `isBinaryCheck` short-circuits to
// "not binary" the moment it sees a UTF-8 BOM (`EF BB BF`), so a buffer
// like `EF BB BF 00 41` was packed as text before this PR. The cheap
// NULL-byte probe must mirror that exemption so this case keeps reaching
// the UTF-8 fast path instead of being newly skipped on the embedded NULL.
const filePath = path.join(testDir, 'utf8-bom-with-null.txt');
const utf8Bom = Buffer.from([0xef, 0xbb, 0xbf]);
const body = Buffer.from([0x00, 0x41]); // U+0000 then 'A'
await fs.writeFile(filePath, Buffer.concat([utf8Bom, body]));
const result = await readRawFile(filePath, 1024);
expect(result.skippedReason).toBeUndefined();
expect(result.content).toBe('\0A');
});
test('should decode UTF-16 LE BOM file despite embedded NULL bytes', async () => {
// Regression: the cheap NULL-byte binary probe ahead of the UTF-8 try
// would misclassify UTF-16/UTF-32 text files (whose ASCII characters
// encode with NULL high bytes) as binary. The probe must be skipped
// when the buffer starts with a UTF-16/UTF-32 BOM so jschardet+iconv
// can decode the file on the slow path, matching pre-change behavior.
const filePath = path.join(testDir, 'utf16le.txt');
// UTF-16 LE BOM (FF FE) followed by "Hello\n" encoded as 2 bytes/char.
const utf16LeBom = Buffer.from([0xff, 0xfe]);
const utf16LeBody = Buffer.from('Hello\n', 'utf16le');
await fs.writeFile(filePath, Buffer.concat([utf16LeBom, utf16LeBody]));
const result = await readRawFile(filePath, 1024);
expect(result.skippedReason).toBeUndefined();
expect(result.content).toBe('Hello\n');
});
});