mirror of
https://github.com/danny-avila/LibreChat.git
synced 2025-12-17 17:00:15 +01:00
feat: allow config of DALL-E-3 System Prompt via env 🎨 (#1150)
This commit is contained in:
parent
4f3b66756a
commit
43d7a751d6
4 changed files with 214 additions and 7 deletions
199
api/app/clients/tools/structured/specs/DALLE3.spec.js
Normal file
199
api/app/clients/tools/structured/specs/DALLE3.spec.js
Normal file
|
|
@ -0,0 +1,199 @@
|
|||
const fs = require('fs');
|
||||
const path = require('path');
|
||||
const OpenAI = require('openai');
|
||||
const DALLE3 = require('../DALLE3');
|
||||
const saveImageFromUrl = require('../../saveImageFromUrl');
|
||||
|
||||
jest.mock('openai');
|
||||
|
||||
const generate = jest.fn();
|
||||
OpenAI.mockImplementation(() => ({
|
||||
images: {
|
||||
generate,
|
||||
},
|
||||
}));
|
||||
|
||||
jest.mock('fs', () => {
|
||||
return {
|
||||
existsSync: jest.fn(),
|
||||
mkdirSync: jest.fn(),
|
||||
};
|
||||
});
|
||||
|
||||
jest.mock('../../saveImageFromUrl', () => {
|
||||
return jest.fn();
|
||||
});
|
||||
|
||||
jest.mock('path', () => {
|
||||
return {
|
||||
resolve: jest.fn(),
|
||||
join: jest.fn(),
|
||||
relative: jest.fn(),
|
||||
};
|
||||
});
|
||||
|
||||
describe('DALLE3', () => {
|
||||
let originalEnv;
|
||||
let dalle; // Keep this declaration if you need to use dalle in other tests
|
||||
const mockApiKey = 'mock_api_key';
|
||||
|
||||
beforeAll(() => {
|
||||
// Save the original process.env
|
||||
originalEnv = { ...process.env };
|
||||
});
|
||||
|
||||
beforeEach(() => {
|
||||
// Reset the process.env before each test
|
||||
jest.resetModules();
|
||||
process.env = { ...originalEnv, DALLE_API_KEY: mockApiKey };
|
||||
// Instantiate DALLE3 for tests that do not depend on DALLE3_SYSTEM_PROMPT
|
||||
dalle = new DALLE3();
|
||||
});
|
||||
|
||||
afterEach(() => {
|
||||
jest.clearAllMocks();
|
||||
// Restore the original process.env after each test
|
||||
process.env = originalEnv;
|
||||
});
|
||||
|
||||
it('should throw an error if DALLE_API_KEY is missing', () => {
|
||||
delete process.env.DALLE_API_KEY;
|
||||
expect(() => new DALLE3()).toThrow('Missing DALLE_API_KEY environment variable.');
|
||||
});
|
||||
|
||||
it('should replace unwanted characters in input string', () => {
|
||||
const input = 'This is a test\nstring with "quotes" and new lines.';
|
||||
const expectedOutput = 'This is a test string with quotes and new lines.';
|
||||
expect(dalle.replaceUnwantedChars(input)).toBe(expectedOutput);
|
||||
});
|
||||
|
||||
it('should generate markdown image URL correctly', () => {
|
||||
const imageName = 'test.png';
|
||||
path.join.mockReturnValue('images/test.png');
|
||||
path.relative.mockReturnValue('images/test.png');
|
||||
const markdownImage = dalle.getMarkdownImageUrl(imageName);
|
||||
expect(markdownImage).toBe('');
|
||||
});
|
||||
|
||||
it('should call OpenAI API with correct parameters', async () => {
|
||||
const mockData = {
|
||||
prompt: 'A test prompt',
|
||||
quality: 'standard',
|
||||
size: '1024x1024',
|
||||
style: 'vivid',
|
||||
};
|
||||
|
||||
const mockResponse = {
|
||||
data: [
|
||||
{
|
||||
url: 'http://example.com/img-test.png',
|
||||
},
|
||||
],
|
||||
};
|
||||
|
||||
generate.mockResolvedValue(mockResponse);
|
||||
saveImageFromUrl.mockResolvedValue(true);
|
||||
fs.existsSync.mockReturnValue(true);
|
||||
path.resolve.mockReturnValue('/fakepath/images');
|
||||
path.join.mockReturnValue('/fakepath/images/img-test.png');
|
||||
path.relative.mockReturnValue('images/img-test.png');
|
||||
|
||||
const result = await dalle._call(mockData);
|
||||
|
||||
expect(generate).toHaveBeenCalledWith({
|
||||
model: 'dall-e-3',
|
||||
quality: mockData.quality,
|
||||
style: mockData.style,
|
||||
size: mockData.size,
|
||||
prompt: mockData.prompt,
|
||||
n: 1,
|
||||
});
|
||||
expect(result).toContain('![generated image]');
|
||||
});
|
||||
|
||||
it('should use the system prompt if provided', () => {
|
||||
process.env.DALLE3_SYSTEM_PROMPT = 'System prompt for testing';
|
||||
jest.resetModules(); // This will ensure the module is fresh and will read the new env var
|
||||
const DALLE3 = require('../DALLE3'); // Re-require after setting the env var
|
||||
const dalleWithSystemPrompt = new DALLE3();
|
||||
expect(dalleWithSystemPrompt.description_for_model).toBe('System prompt for testing');
|
||||
});
|
||||
|
||||
it('should not use the system prompt if not provided', async () => {
|
||||
delete process.env.DALLE3_SYSTEM_PROMPT;
|
||||
const dalleWithoutSystemPrompt = new DALLE3();
|
||||
expect(dalleWithoutSystemPrompt.description_for_model).not.toBe('System prompt for testing');
|
||||
});
|
||||
|
||||
it('should throw an error if prompt is missing', async () => {
|
||||
const mockData = {
|
||||
quality: 'standard',
|
||||
size: '1024x1024',
|
||||
style: 'vivid',
|
||||
};
|
||||
await expect(dalle._call(mockData)).rejects.toThrow('Missing required field: prompt');
|
||||
});
|
||||
|
||||
it('should throw an error if no image URL is returned from OpenAI API', async () => {
|
||||
const mockData = {
|
||||
prompt: 'A test prompt',
|
||||
};
|
||||
// Simulate a response with an object that has a `url` property set to `undefined`
|
||||
generate.mockResolvedValue({ data: [{ url: undefined }] });
|
||||
await expect(dalle._call(mockData)).rejects.toThrow('No image URL returned from OpenAI API.');
|
||||
});
|
||||
|
||||
it('should log to console if no image name is found in the URL', async () => {
|
||||
const mockData = {
|
||||
prompt: 'A test prompt',
|
||||
};
|
||||
const mockResponse = {
|
||||
data: [
|
||||
{
|
||||
url: 'http://example.com/invalid-url',
|
||||
},
|
||||
],
|
||||
};
|
||||
console.log = jest.fn(); // Mock console.log
|
||||
generate.mockResolvedValue(mockResponse);
|
||||
await dalle._call(mockData);
|
||||
expect(console.log).toHaveBeenCalledWith('No image name found in the string.');
|
||||
});
|
||||
|
||||
it('should create the directory if it does not exist', async () => {
|
||||
const mockData = {
|
||||
prompt: 'A test prompt',
|
||||
};
|
||||
const mockResponse = {
|
||||
data: [
|
||||
{
|
||||
url: 'http://example.com/img-test.png',
|
||||
},
|
||||
],
|
||||
};
|
||||
generate.mockResolvedValue(mockResponse);
|
||||
fs.existsSync.mockReturnValue(false); // Simulate directory does not exist
|
||||
await dalle._call(mockData);
|
||||
expect(fs.mkdirSync).toHaveBeenCalledWith(expect.any(String), { recursive: true });
|
||||
});
|
||||
|
||||
it('should log an error and return the image URL if there is an error saving the image', async () => {
|
||||
const mockData = {
|
||||
prompt: 'A test prompt',
|
||||
};
|
||||
const mockResponse = {
|
||||
data: [
|
||||
{
|
||||
url: 'http://example.com/img-test.png',
|
||||
},
|
||||
],
|
||||
};
|
||||
const error = new Error('Error while saving the image');
|
||||
generate.mockResolvedValue(mockResponse);
|
||||
saveImageFromUrl.mockRejectedValue(error);
|
||||
console.error = jest.fn(); // Mock console.error
|
||||
const result = await dalle._call(mockData);
|
||||
expect(console.error).toHaveBeenCalledWith('Error while saving the image:', error);
|
||||
expect(result).toBe(mockResponse.data[0].url);
|
||||
});
|
||||
});
|
||||
Loading…
Add table
Add a link
Reference in a new issue