Restructure omni services and add Chatwoot research snapshot
This commit is contained in:
@@ -0,0 +1,79 @@
|
||||
import {
|
||||
getAssignee,
|
||||
isConversationUnassigned,
|
||||
isConversationAssignedToMe,
|
||||
isMessageFromCurrentUser,
|
||||
} from '../AudioMessageHelper';
|
||||
|
||||
describe('getAssignee', () => {
|
||||
it('should return assignee_id when present', () => {
|
||||
const message = { conversation: { assignee_id: 1 } };
|
||||
expect(getAssignee(message)).toBe(1);
|
||||
});
|
||||
|
||||
it('should return undefined when no assignee_id', () => {
|
||||
const message = { conversation: null };
|
||||
expect(getAssignee(message)).toBeUndefined();
|
||||
});
|
||||
|
||||
it('should handle null message', () => {
|
||||
expect(getAssignee(null)).toBeUndefined();
|
||||
});
|
||||
});
|
||||
|
||||
describe('isConversationUnassigned', () => {
|
||||
it('should return true when no assignee', () => {
|
||||
const message = { conversation: { assignee_id: null } };
|
||||
expect(isConversationUnassigned(message)).toBe(true);
|
||||
});
|
||||
|
||||
it('should return false when has assignee', () => {
|
||||
const message = { conversation: { assignee_id: 1 } };
|
||||
expect(isConversationUnassigned(message)).toBe(false);
|
||||
});
|
||||
|
||||
it('should handle null message', () => {
|
||||
expect(isConversationUnassigned(null)).toBe(true);
|
||||
});
|
||||
});
|
||||
|
||||
describe('isConversationAssignedToMe', () => {
|
||||
const currentUserId = 1;
|
||||
|
||||
it('should return true when assigned to current user', () => {
|
||||
const message = { conversation: { assignee_id: 1 } };
|
||||
expect(isConversationAssignedToMe(message, currentUserId)).toBe(true);
|
||||
});
|
||||
|
||||
it('should return false when assigned to different user', () => {
|
||||
const message = { conversation: { assignee_id: 2 } };
|
||||
expect(isConversationAssignedToMe(message, currentUserId)).toBe(false);
|
||||
});
|
||||
|
||||
it('should return false when unassigned', () => {
|
||||
const message = { conversation: { assignee_id: null } };
|
||||
expect(isConversationAssignedToMe(message, currentUserId)).toBe(false);
|
||||
});
|
||||
|
||||
it('should handle null message', () => {
|
||||
expect(isConversationAssignedToMe(null, currentUserId)).toBe(false);
|
||||
});
|
||||
});
|
||||
|
||||
describe('isMessageFromCurrentUser', () => {
|
||||
const currentUserId = 1;
|
||||
|
||||
it('should return true when message is from current user', () => {
|
||||
const message = { sender: { id: 1 } };
|
||||
expect(isMessageFromCurrentUser(message, currentUserId)).toBe(true);
|
||||
});
|
||||
|
||||
it('should return false when message is from different user', () => {
|
||||
const message = { sender: { id: 2 } };
|
||||
expect(isMessageFromCurrentUser(message, currentUserId)).toBe(false);
|
||||
});
|
||||
|
||||
it('should handle null message', () => {
|
||||
expect(isMessageFromCurrentUser(null, currentUserId)).toBe(false);
|
||||
});
|
||||
});
|
||||
@@ -0,0 +1,191 @@
|
||||
import AudioNotificationStore from '../AudioNotificationStore';
|
||||
import {
|
||||
ROLES,
|
||||
CONVERSATION_PERMISSIONS,
|
||||
} from 'dashboard/constants/permissions';
|
||||
import { getUserPermissions } from 'dashboard/helper/permissionsHelper';
|
||||
import wootConstants from 'dashboard/constants/globals';
|
||||
|
||||
vi.mock('dashboard/helper/permissionsHelper', () => ({
|
||||
getUserPermissions: vi.fn(),
|
||||
}));
|
||||
|
||||
describe('AudioNotificationStore', () => {
|
||||
let store;
|
||||
let audioNotificationStore;
|
||||
|
||||
beforeEach(() => {
|
||||
store = {
|
||||
getters: {
|
||||
getMineChats: vi.fn(),
|
||||
getSelectedChat: null,
|
||||
getCurrentAccountId: 1,
|
||||
getConversationById: vi.fn(),
|
||||
},
|
||||
};
|
||||
audioNotificationStore = new AudioNotificationStore(store);
|
||||
});
|
||||
|
||||
describe('hasUnreadConversation', () => {
|
||||
it('should return true when there are unread conversations', () => {
|
||||
store.getters.getMineChats.mockReturnValue([
|
||||
{ id: 1, unread_count: 2 },
|
||||
{ id: 2, unread_count: 0 },
|
||||
]);
|
||||
|
||||
expect(audioNotificationStore.hasUnreadConversation()).toBe(true);
|
||||
});
|
||||
|
||||
it('should return false when there are no unread conversations', () => {
|
||||
store.getters.getMineChats.mockReturnValue([
|
||||
{ id: 1, unread_count: 0 },
|
||||
{ id: 2, unread_count: 0 },
|
||||
]);
|
||||
|
||||
expect(audioNotificationStore.hasUnreadConversation()).toBe(false);
|
||||
});
|
||||
|
||||
it('should return false when there are no conversations', () => {
|
||||
store.getters.getMineChats.mockReturnValue([]);
|
||||
|
||||
expect(audioNotificationStore.hasUnreadConversation()).toBe(false);
|
||||
});
|
||||
|
||||
it('should call getMineChats with correct parameters', () => {
|
||||
store.getters.getMineChats.mockReturnValue([]);
|
||||
audioNotificationStore.hasUnreadConversation();
|
||||
|
||||
expect(store.getters.getMineChats).toHaveBeenCalledWith({
|
||||
assigneeType: 'me',
|
||||
status: 'open',
|
||||
});
|
||||
});
|
||||
});
|
||||
|
||||
describe('isMessageFromPendingConversation', () => {
|
||||
it('should return true when conversation status is pending', () => {
|
||||
store.getters.getConversationById.mockReturnValue({
|
||||
id: 123,
|
||||
status: wootConstants.STATUS_TYPE.PENDING,
|
||||
});
|
||||
const message = { conversation_id: 123 };
|
||||
|
||||
expect(
|
||||
audioNotificationStore.isMessageFromPendingConversation(message)
|
||||
).toBe(true);
|
||||
expect(store.getters.getConversationById).toHaveBeenCalledWith(123);
|
||||
});
|
||||
|
||||
it('should return false when conversation status is not pending', () => {
|
||||
store.getters.getConversationById.mockReturnValue({
|
||||
id: 123,
|
||||
status: wootConstants.STATUS_TYPE.OPEN,
|
||||
});
|
||||
const message = { conversation_id: 123 };
|
||||
|
||||
expect(
|
||||
audioNotificationStore.isMessageFromPendingConversation(message)
|
||||
).toBe(false);
|
||||
expect(store.getters.getConversationById).toHaveBeenCalledWith(123);
|
||||
});
|
||||
|
||||
it('should return false when conversation is not found', () => {
|
||||
store.getters.getConversationById.mockReturnValue(null);
|
||||
const message = { conversation_id: 123 };
|
||||
|
||||
expect(
|
||||
audioNotificationStore.isMessageFromPendingConversation(message)
|
||||
).toBe(false);
|
||||
expect(store.getters.getConversationById).toHaveBeenCalledWith(123);
|
||||
});
|
||||
|
||||
it('should return false when message has no conversation_id', () => {
|
||||
const message = {};
|
||||
|
||||
expect(
|
||||
audioNotificationStore.isMessageFromPendingConversation(message)
|
||||
).toBe(false);
|
||||
expect(store.getters.getConversationById).not.toHaveBeenCalled();
|
||||
});
|
||||
|
||||
it('should return false when message is null or undefined', () => {
|
||||
expect(
|
||||
audioNotificationStore.isMessageFromPendingConversation(null)
|
||||
).toBe(false);
|
||||
expect(
|
||||
audioNotificationStore.isMessageFromPendingConversation(undefined)
|
||||
).toBe(false);
|
||||
expect(store.getters.getConversationById).not.toHaveBeenCalled();
|
||||
});
|
||||
});
|
||||
|
||||
describe('isMessageFromCurrentConversation', () => {
|
||||
it('should return true when message is from selected chat', () => {
|
||||
store.getters.getSelectedChat = { id: 6179 };
|
||||
const message = { conversation_id: 6179 };
|
||||
|
||||
expect(
|
||||
audioNotificationStore.isMessageFromCurrentConversation(message)
|
||||
).toBe(true);
|
||||
});
|
||||
|
||||
it('should return false when message is from different chat', () => {
|
||||
store.getters.getSelectedChat = { id: 6179 };
|
||||
const message = { conversation_id: 1337 };
|
||||
|
||||
expect(
|
||||
audioNotificationStore.isMessageFromCurrentConversation(message)
|
||||
).toBe(false);
|
||||
});
|
||||
|
||||
it('should return false when no chat is selected', () => {
|
||||
store.getters.getSelectedChat = null;
|
||||
const message = { conversation_id: 6179 };
|
||||
|
||||
expect(
|
||||
audioNotificationStore.isMessageFromCurrentConversation(message)
|
||||
).toBe(false);
|
||||
});
|
||||
});
|
||||
|
||||
describe('hasConversationPermission', () => {
|
||||
const mockUser = { id: 'user123' };
|
||||
|
||||
beforeEach(() => {
|
||||
getUserPermissions.mockReset();
|
||||
});
|
||||
|
||||
it('should return true when user has a required role', () => {
|
||||
getUserPermissions.mockReturnValue([ROLES[0]]);
|
||||
|
||||
expect(audioNotificationStore.hasConversationPermission(mockUser)).toBe(
|
||||
true
|
||||
);
|
||||
expect(getUserPermissions).toHaveBeenCalledWith(mockUser, 1);
|
||||
});
|
||||
|
||||
it('should return true when user has a conversation permission', () => {
|
||||
getUserPermissions.mockReturnValue([CONVERSATION_PERMISSIONS[0]]);
|
||||
|
||||
expect(audioNotificationStore.hasConversationPermission(mockUser)).toBe(
|
||||
true
|
||||
);
|
||||
});
|
||||
|
||||
it('should return false when user has no required permissions', () => {
|
||||
getUserPermissions.mockReturnValue(['some-other-permission']);
|
||||
|
||||
expect(audioNotificationStore.hasConversationPermission(mockUser)).toBe(
|
||||
false
|
||||
);
|
||||
});
|
||||
|
||||
it('should return false when user has no permissions', () => {
|
||||
getUserPermissions.mockReturnValue([]);
|
||||
|
||||
expect(audioNotificationStore.hasConversationPermission(mockUser)).toBe(
|
||||
false
|
||||
);
|
||||
});
|
||||
});
|
||||
});
|
||||
@@ -0,0 +1,114 @@
|
||||
import { describe, it, expect, beforeEach, afterEach, vi } from 'vitest';
|
||||
import { WindowVisibilityHelper } from '../WindowVisibilityHelper';
|
||||
|
||||
describe('WindowVisibilityHelper', () => {
|
||||
let blurCallback;
|
||||
let focusCallback;
|
||||
let windowEventListeners;
|
||||
let documentHiddenValue = false;
|
||||
|
||||
beforeEach(() => {
|
||||
vi.resetModules();
|
||||
vi.resetAllMocks();
|
||||
|
||||
// Reset event listeners before each test
|
||||
windowEventListeners = {};
|
||||
|
||||
// Mock window.addEventListener
|
||||
window.addEventListener = vi.fn((event, callback) => {
|
||||
windowEventListeners[event] = callback;
|
||||
if (event === 'blur') blurCallback = callback;
|
||||
if (event === 'focus') focusCallback = callback;
|
||||
});
|
||||
|
||||
// Mock document.hidden with a getter that returns our controlled value
|
||||
Object.defineProperty(document, 'hidden', {
|
||||
configurable: true,
|
||||
get: () => documentHiddenValue,
|
||||
});
|
||||
});
|
||||
|
||||
afterEach(() => {
|
||||
vi.clearAllMocks();
|
||||
documentHiddenValue = false;
|
||||
});
|
||||
|
||||
describe('initialization', () => {
|
||||
it('should add blur and focus event listeners', () => {
|
||||
const helper = new WindowVisibilityHelper();
|
||||
expect(helper.isVisible).toBe(true);
|
||||
|
||||
expect(window.addEventListener).toHaveBeenCalledTimes(2);
|
||||
expect(window.addEventListener).toHaveBeenCalledWith(
|
||||
'blur',
|
||||
expect.any(Function)
|
||||
);
|
||||
expect(window.addEventListener).toHaveBeenCalledWith(
|
||||
'focus',
|
||||
expect.any(Function)
|
||||
);
|
||||
});
|
||||
});
|
||||
|
||||
describe('window events', () => {
|
||||
it('should set isVisible to false on blur', () => {
|
||||
const helper = new WindowVisibilityHelper();
|
||||
blurCallback();
|
||||
expect(helper.isVisible).toBe(false);
|
||||
});
|
||||
|
||||
it('should set isVisible to true on focus', () => {
|
||||
const helper = new WindowVisibilityHelper();
|
||||
blurCallback(); // First blur the window
|
||||
focusCallback(); // Then focus it
|
||||
expect(helper.isVisible).toBe(true);
|
||||
});
|
||||
|
||||
it('should handle multiple blur/focus events', () => {
|
||||
const helper = new WindowVisibilityHelper();
|
||||
|
||||
blurCallback();
|
||||
expect(helper.isVisible).toBe(false);
|
||||
|
||||
focusCallback();
|
||||
expect(helper.isVisible).toBe(true);
|
||||
|
||||
blurCallback();
|
||||
expect(helper.isVisible).toBe(false);
|
||||
});
|
||||
});
|
||||
|
||||
describe('isWindowVisible', () => {
|
||||
it('should return true when document is visible and window is focused', () => {
|
||||
const helper = new WindowVisibilityHelper();
|
||||
documentHiddenValue = false;
|
||||
helper.isVisible = true;
|
||||
|
||||
expect(helper.isWindowVisible()).toBe(true);
|
||||
});
|
||||
|
||||
it('should return false when document is hidden', () => {
|
||||
const helper = new WindowVisibilityHelper();
|
||||
documentHiddenValue = true;
|
||||
helper.isVisible = true;
|
||||
|
||||
expect(helper.isWindowVisible()).toBe(false);
|
||||
});
|
||||
|
||||
it('should return false when window is not focused', () => {
|
||||
const helper = new WindowVisibilityHelper();
|
||||
documentHiddenValue = false;
|
||||
helper.isVisible = false;
|
||||
|
||||
expect(helper.isWindowVisible()).toBe(false);
|
||||
});
|
||||
|
||||
it('should return false when both document is hidden and window is not focused', () => {
|
||||
const helper = new WindowVisibilityHelper();
|
||||
documentHiddenValue = true;
|
||||
helper.isVisible = false;
|
||||
|
||||
expect(helper.isWindowVisible()).toBe(false);
|
||||
});
|
||||
});
|
||||
});
|
||||
Reference in New Issue
Block a user