mirror of
https://github.com/danny-avila/LibreChat.git
synced 2025-12-16 16:30:15 +01:00
📸 fix: Avatar Handling for Social Login (#8993)
- Updated `handleExistingUser` function to improve avatar handling logic, including checks for manual flags and null/undefined avatars. - Introduced a new test suite for `handleExistingUser` covering various scenarios, ensuring robust functionality for avatar updates in both local and non-local storage contexts.
This commit is contained in:
parent
7e4c8a5d0d
commit
8cefa566da
2 changed files with 169 additions and 2 deletions
|
|
@ -22,9 +22,12 @@ const handleExistingUser = async (oldUser, avatarUrl) => {
|
||||||
const isLocal = fileStrategy === FileSources.local;
|
const isLocal = fileStrategy === FileSources.local;
|
||||||
|
|
||||||
let updatedAvatar = false;
|
let updatedAvatar = false;
|
||||||
if (isLocal && (oldUser.avatar === null || !oldUser.avatar.includes('?manual=true'))) {
|
const hasManualFlag =
|
||||||
|
typeof oldUser?.avatar === 'string' && oldUser.avatar.includes('?manual=true');
|
||||||
|
|
||||||
|
if (isLocal && (!oldUser?.avatar || !hasManualFlag)) {
|
||||||
updatedAvatar = avatarUrl;
|
updatedAvatar = avatarUrl;
|
||||||
} else if (!isLocal && (oldUser.avatar === null || !oldUser.avatar.includes('?manual=true'))) {
|
} else if (!isLocal && (!oldUser?.avatar || !hasManualFlag)) {
|
||||||
const userId = oldUser._id;
|
const userId = oldUser._id;
|
||||||
const resizedBuffer = await resizeAvatar({
|
const resizedBuffer = await resizeAvatar({
|
||||||
userId,
|
userId,
|
||||||
|
|
|
||||||
164
api/strategies/process.test.js
Normal file
164
api/strategies/process.test.js
Normal file
|
|
@ -0,0 +1,164 @@
|
||||||
|
const { FileSources } = require('librechat-data-provider');
|
||||||
|
const { handleExistingUser } = require('./process');
|
||||||
|
|
||||||
|
jest.mock('~/server/services/Files/strategies', () => ({
|
||||||
|
getStrategyFunctions: jest.fn(),
|
||||||
|
}));
|
||||||
|
|
||||||
|
jest.mock('~/server/services/Files/images/avatar', () => ({
|
||||||
|
resizeAvatar: jest.fn(),
|
||||||
|
}));
|
||||||
|
|
||||||
|
jest.mock('~/models', () => ({
|
||||||
|
updateUser: jest.fn(),
|
||||||
|
createUser: jest.fn(),
|
||||||
|
getUserById: jest.fn(),
|
||||||
|
}));
|
||||||
|
|
||||||
|
jest.mock('~/server/services/Config', () => ({
|
||||||
|
getBalanceConfig: jest.fn(),
|
||||||
|
}));
|
||||||
|
|
||||||
|
const { getStrategyFunctions } = require('~/server/services/Files/strategies');
|
||||||
|
const { resizeAvatar } = require('~/server/services/Files/images/avatar');
|
||||||
|
const { updateUser } = require('~/models');
|
||||||
|
|
||||||
|
describe('handleExistingUser', () => {
|
||||||
|
beforeEach(() => {
|
||||||
|
jest.clearAllMocks();
|
||||||
|
process.env.CDN_PROVIDER = FileSources.local;
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should handle null avatar without throwing error', async () => {
|
||||||
|
const oldUser = {
|
||||||
|
_id: 'user123',
|
||||||
|
avatar: null,
|
||||||
|
};
|
||||||
|
const avatarUrl = 'https://example.com/avatar.png';
|
||||||
|
|
||||||
|
await handleExistingUser(oldUser, avatarUrl);
|
||||||
|
|
||||||
|
expect(updateUser).toHaveBeenCalledWith('user123', { avatar: avatarUrl });
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should handle undefined avatar without throwing error', async () => {
|
||||||
|
const oldUser = {
|
||||||
|
_id: 'user123',
|
||||||
|
// avatar is undefined
|
||||||
|
};
|
||||||
|
const avatarUrl = 'https://example.com/avatar.png';
|
||||||
|
|
||||||
|
await handleExistingUser(oldUser, avatarUrl);
|
||||||
|
|
||||||
|
expect(updateUser).toHaveBeenCalledWith('user123', { avatar: avatarUrl });
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should not update avatar if it has manual=true flag', async () => {
|
||||||
|
const oldUser = {
|
||||||
|
_id: 'user123',
|
||||||
|
avatar: 'https://example.com/avatar.png?manual=true',
|
||||||
|
};
|
||||||
|
const avatarUrl = 'https://example.com/new-avatar.png';
|
||||||
|
|
||||||
|
await handleExistingUser(oldUser, avatarUrl);
|
||||||
|
|
||||||
|
expect(updateUser).not.toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should update avatar for local storage when avatar has no manual flag', async () => {
|
||||||
|
const oldUser = {
|
||||||
|
_id: 'user123',
|
||||||
|
avatar: 'https://example.com/old-avatar.png',
|
||||||
|
};
|
||||||
|
const avatarUrl = 'https://example.com/new-avatar.png';
|
||||||
|
|
||||||
|
await handleExistingUser(oldUser, avatarUrl);
|
||||||
|
|
||||||
|
expect(updateUser).toHaveBeenCalledWith('user123', { avatar: avatarUrl });
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should process avatar for non-local storage', async () => {
|
||||||
|
process.env.CDN_PROVIDER = 's3';
|
||||||
|
|
||||||
|
const mockProcessAvatar = jest.fn().mockResolvedValue('processed-avatar-url');
|
||||||
|
getStrategyFunctions.mockReturnValue({ processAvatar: mockProcessAvatar });
|
||||||
|
resizeAvatar.mockResolvedValue(Buffer.from('resized-image'));
|
||||||
|
|
||||||
|
const oldUser = {
|
||||||
|
_id: 'user123',
|
||||||
|
avatar: null,
|
||||||
|
};
|
||||||
|
const avatarUrl = 'https://example.com/avatar.png';
|
||||||
|
|
||||||
|
await handleExistingUser(oldUser, avatarUrl);
|
||||||
|
|
||||||
|
expect(resizeAvatar).toHaveBeenCalledWith({
|
||||||
|
userId: 'user123',
|
||||||
|
input: avatarUrl,
|
||||||
|
});
|
||||||
|
expect(mockProcessAvatar).toHaveBeenCalledWith({
|
||||||
|
buffer: Buffer.from('resized-image'),
|
||||||
|
userId: 'user123',
|
||||||
|
manual: 'false',
|
||||||
|
});
|
||||||
|
expect(updateUser).toHaveBeenCalledWith('user123', { avatar: 'processed-avatar-url' });
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should not update if avatar already has manual flag in non-local storage', async () => {
|
||||||
|
process.env.CDN_PROVIDER = 's3';
|
||||||
|
|
||||||
|
const oldUser = {
|
||||||
|
_id: 'user123',
|
||||||
|
avatar: 'https://cdn.example.com/avatar.png?manual=true',
|
||||||
|
};
|
||||||
|
const avatarUrl = 'https://example.com/new-avatar.png';
|
||||||
|
|
||||||
|
await handleExistingUser(oldUser, avatarUrl);
|
||||||
|
|
||||||
|
expect(resizeAvatar).not.toHaveBeenCalled();
|
||||||
|
expect(updateUser).not.toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should handle avatar with query parameters but without manual flag', async () => {
|
||||||
|
const oldUser = {
|
||||||
|
_id: 'user123',
|
||||||
|
avatar: 'https://example.com/avatar.png?size=large&format=webp',
|
||||||
|
};
|
||||||
|
const avatarUrl = 'https://example.com/new-avatar.png';
|
||||||
|
|
||||||
|
await handleExistingUser(oldUser, avatarUrl);
|
||||||
|
|
||||||
|
expect(updateUser).toHaveBeenCalledWith('user123', { avatar: avatarUrl });
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should handle empty string avatar', async () => {
|
||||||
|
const oldUser = {
|
||||||
|
_id: 'user123',
|
||||||
|
avatar: '',
|
||||||
|
};
|
||||||
|
const avatarUrl = 'https://example.com/avatar.png';
|
||||||
|
|
||||||
|
await handleExistingUser(oldUser, avatarUrl);
|
||||||
|
|
||||||
|
expect(updateUser).toHaveBeenCalledWith('user123', { avatar: avatarUrl });
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should handle avatar with manual=false parameter', async () => {
|
||||||
|
const oldUser = {
|
||||||
|
_id: 'user123',
|
||||||
|
avatar: 'https://example.com/avatar.png?manual=false',
|
||||||
|
};
|
||||||
|
const avatarUrl = 'https://example.com/new-avatar.png';
|
||||||
|
|
||||||
|
await handleExistingUser(oldUser, avatarUrl);
|
||||||
|
|
||||||
|
expect(updateUser).toHaveBeenCalledWith('user123', { avatar: avatarUrl });
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should handle oldUser being null gracefully', async () => {
|
||||||
|
const avatarUrl = 'https://example.com/avatar.png';
|
||||||
|
|
||||||
|
// This should throw an error when trying to access oldUser._id
|
||||||
|
await expect(handleExistingUser(null, avatarUrl)).rejects.toThrow();
|
||||||
|
});
|
||||||
|
});
|
||||||
Loading…
Add table
Add a link
Reference in a new issue