mirror of
https://github.com/danny-avila/LibreChat.git
synced 2026-03-15 12:16:33 +01:00
* fix: add file size limits to conversation import multer instance * fix: address review findings for conversation import file size limits * fix: use local jest.mock for data-schemas instead of global moduleNameMapper The global @librechat/data-schemas mock in jest.config.js only provided logger, breaking all tests that depend on createModels from the same package. Replace with a virtual jest.mock scoped to the import spec file. * fix: move import to top of file, pre-compute upload middleware, assert logger.warn in tests * refactor: move resolveImportMaxFileSize to packages/api New backend logic belongs in packages/api as TypeScript. Delete the api/server/utils/import/limits.js wrapper and import directly from @librechat/api in convos.js and importConversations.js. Resolver unit tests move to packages/api; the api/ spec retains only multer behavior tests. * chore: rename importLimits to import * fix: stale type reference and mock isolation in import tests Update typeof import path from '../importLimits' to '../import' after the rename. Clear mockLogger.warn in beforeEach to prevent cross-test accumulation. * fix: add resolveImportMaxFileSize to @librechat/api mock in convos.spec.js * fix: resolve jest.mock hoisting issue in import tests jest.mock factories are hoisted above const declarations, so the mockLogger reference was undefined at factory evaluation time. Use a direct import of the mocked logger module instead. * fix: remove virtual flag from data-schemas mock for CI compatibility virtual: true prevents the mock from intercepting the real module in CI where @librechat/data-schemas is built, causing import.ts to use the real logger while the test asserts against the mock.
98 lines
2.9 KiB
JavaScript
98 lines
2.9 KiB
JavaScript
const express = require('express');
|
|
const request = require('supertest');
|
|
const multer = require('multer');
|
|
|
|
const importFileFilter = (req, file, cb) => {
|
|
if (file.mimetype === 'application/json') {
|
|
cb(null, true);
|
|
} else {
|
|
cb(new Error('Only JSON files are allowed'), false);
|
|
}
|
|
};
|
|
|
|
/** Proxy app that mirrors the production multer + error-handling pattern */
|
|
function createImportApp(fileSize) {
|
|
const app = express();
|
|
const upload = multer({
|
|
storage: multer.memoryStorage(),
|
|
fileFilter: importFileFilter,
|
|
limits: { fileSize },
|
|
});
|
|
const uploadSingle = upload.single('file');
|
|
|
|
function handleUpload(req, res, next) {
|
|
uploadSingle(req, res, (err) => {
|
|
if (err && err.code === 'LIMIT_FILE_SIZE') {
|
|
return res.status(413).json({ message: 'File exceeds the maximum allowed size' });
|
|
}
|
|
if (err) {
|
|
return next(err);
|
|
}
|
|
next();
|
|
});
|
|
}
|
|
|
|
app.post('/import', handleUpload, (req, res) => {
|
|
res.status(201).json({ message: 'success', size: req.file.size });
|
|
});
|
|
|
|
app.use((err, _req, res, _next) => {
|
|
res.status(400).json({ error: err.message });
|
|
});
|
|
|
|
return app;
|
|
}
|
|
|
|
describe('Conversation Import - Multer File Size Limits', () => {
|
|
describe('multer rejects files exceeding the configured limit', () => {
|
|
it('returns 413 for files larger than the limit', async () => {
|
|
const limit = 1024;
|
|
const app = createImportApp(limit);
|
|
const oversized = Buffer.alloc(limit + 512, 'x');
|
|
|
|
const res = await request(app)
|
|
.post('/import')
|
|
.attach('file', oversized, { filename: 'import.json', contentType: 'application/json' });
|
|
|
|
expect(res.status).toBe(413);
|
|
expect(res.body.message).toBe('File exceeds the maximum allowed size');
|
|
});
|
|
|
|
it('accepts files within the limit', async () => {
|
|
const limit = 4096;
|
|
const app = createImportApp(limit);
|
|
const valid = Buffer.from(JSON.stringify({ title: 'test' }));
|
|
|
|
const res = await request(app)
|
|
.post('/import')
|
|
.attach('file', valid, { filename: 'import.json', contentType: 'application/json' });
|
|
|
|
expect(res.status).toBe(201);
|
|
expect(res.body.message).toBe('success');
|
|
});
|
|
|
|
it('rejects at the exact boundary (limit + 1 byte)', async () => {
|
|
const limit = 512;
|
|
const app = createImportApp(limit);
|
|
const boundary = Buffer.alloc(limit + 1, 'a');
|
|
|
|
const res = await request(app)
|
|
.post('/import')
|
|
.attach('file', boundary, { filename: 'import.json', contentType: 'application/json' });
|
|
|
|
expect(res.status).toBe(413);
|
|
});
|
|
|
|
it('accepts a file just under the limit', async () => {
|
|
const limit = 512;
|
|
const app = createImportApp(limit);
|
|
const underLimit = Buffer.alloc(limit - 1, 'b');
|
|
|
|
const res = await request(app)
|
|
.post('/import')
|
|
.attach('file', underLimit, { filename: 'import.json', contentType: 'application/json' });
|
|
|
|
expect(res.status).toBe(201);
|
|
});
|
|
});
|
|
});
|