mirror of
https://github.com/n8n-io/n8n
synced 2026-04-19 13:05:54 +02:00
Signed-off-by: Oleg Ivaniv <me@olegivaniv.com> Co-authored-by: Albert Alises <albert.alises@gmail.com> Co-authored-by: Jaakko Husso <jaakko@n8n.io> Co-authored-by: Dimitri Lavrenük <20122620+dlavrenuek@users.noreply.github.com> Co-authored-by: cubic-dev-ai[bot] <191113872+cubic-dev-ai[bot]@users.noreply.github.com> Co-authored-by: Tuukka Kantola <Tuukkaa@users.noreply.github.com> Co-authored-by: Claude Opus 4.6 <noreply@anthropic.com> Co-authored-by: Mutasem Aldmour <4711238+mutdmour@users.noreply.github.com> Co-authored-by: Raúl Gómez Morales <raul00gm@gmail.com> Co-authored-by: Elias Meire <elias@meire.dev> Co-authored-by: Dimitri Lavrenük <dimitri.lavrenuek@n8n.io> Co-authored-by: Tomi Turtiainen <10324676+tomi@users.noreply.github.com> Co-authored-by: Mutasem Aldmour <mutasem@n8n.io>
110 lines
3.2 KiB
TypeScript
110 lines
3.2 KiB
TypeScript
import * as fs from 'node:fs/promises';
|
|
|
|
import { textOf } from '../test-utils';
|
|
import { writeFileTool } from './write-file';
|
|
|
|
jest.mock('node:fs/promises');
|
|
|
|
const CONTEXT = { dir: '/base' };
|
|
|
|
function mockMkdir(): void {
|
|
(fs.mkdir as jest.Mock).mockResolvedValue(undefined);
|
|
}
|
|
|
|
function mockWriteFile(): void {
|
|
(fs.writeFile as jest.Mock).mockResolvedValue(undefined);
|
|
}
|
|
|
|
describe('writeFileTool', () => {
|
|
beforeEach(() => {
|
|
jest.resetAllMocks();
|
|
(fs.realpath as jest.Mock).mockImplementation(async (p: string) => {
|
|
if (p === '/base') return await Promise.resolve('/base');
|
|
throw Object.assign(new Error('ENOENT'), { code: 'ENOENT' });
|
|
});
|
|
});
|
|
|
|
describe('metadata', () => {
|
|
it('has the correct name', () => {
|
|
expect(writeFileTool.name).toBe('write_file');
|
|
});
|
|
|
|
it('has a non-empty description', () => {
|
|
expect(writeFileTool.description).not.toBe('');
|
|
});
|
|
});
|
|
|
|
describe('inputSchema validation', () => {
|
|
it('accepts valid input', () => {
|
|
expect(() =>
|
|
writeFileTool.inputSchema.parse({ filePath: 'src/index.ts', content: 'hello' }),
|
|
).not.toThrow();
|
|
});
|
|
|
|
it('throws when filePath is missing', () => {
|
|
expect(() => writeFileTool.inputSchema.parse({ content: 'hello' })).toThrow();
|
|
});
|
|
|
|
it('throws when content is missing', () => {
|
|
expect(() => writeFileTool.inputSchema.parse({ filePath: 'src/index.ts' })).toThrow();
|
|
});
|
|
|
|
it('throws when filePath is not a string', () => {
|
|
expect(() => writeFileTool.inputSchema.parse({ filePath: 99, content: 'hello' })).toThrow();
|
|
});
|
|
});
|
|
|
|
describe('execute', () => {
|
|
it('creates parent directories and writes the file', async () => {
|
|
mockMkdir();
|
|
mockWriteFile();
|
|
|
|
const result = await writeFileTool.execute(
|
|
{ filePath: 'subdir/hello.txt', content: 'hello world' },
|
|
CONTEXT,
|
|
);
|
|
|
|
// eslint-disable-next-line n8n-local-rules/no-uncaught-json-parse
|
|
const data = JSON.parse(textOf(result)) as { path: string };
|
|
expect(data.path).toBe('subdir/hello.txt');
|
|
expect(fs.mkdir).toHaveBeenCalledWith('/base/subdir', { recursive: true });
|
|
expect(fs.writeFile).toHaveBeenCalledWith('/base/subdir/hello.txt', 'hello world', 'utf-8');
|
|
});
|
|
|
|
it('returns a single text content block', async () => {
|
|
mockMkdir();
|
|
mockWriteFile();
|
|
|
|
const result = await writeFileTool.execute({ filePath: 'hello.txt', content: 'hi' }, CONTEXT);
|
|
|
|
expect(result.content).toHaveLength(1);
|
|
expect(result.content[0].type).toBe('text');
|
|
});
|
|
|
|
it('overwrites a file that already exists', async () => {
|
|
mockMkdir();
|
|
mockWriteFile();
|
|
|
|
await expect(
|
|
writeFileTool.execute({ filePath: 'existing.txt', content: 'new data' }, CONTEXT),
|
|
).resolves.not.toThrow();
|
|
|
|
expect(fs.writeFile).toHaveBeenCalledWith('/base/existing.txt', 'new data', 'utf-8');
|
|
});
|
|
|
|
it('rejects content larger than 512 KB', async () => {
|
|
const largeContent = 'x'.repeat(600 * 1024);
|
|
|
|
await expect(
|
|
writeFileTool.execute({ filePath: 'large.txt', content: largeContent }, CONTEXT),
|
|
).rejects.toThrow('too large');
|
|
});
|
|
|
|
it('rejects path traversal', async () => {
|
|
await expect(
|
|
writeFileTool.execute({ filePath: '../../../etc/passwd', content: 'bad' }, CONTEXT),
|
|
).rejects.toThrow('escapes');
|
|
});
|
|
});
|
|
});
|