n8n/packages/editor-ui/src/composables/usePinnedData.test.ts

137 lines
4.5 KiB
TypeScript

import { describe, it, expect, vi, beforeEach } from 'vitest';
import { setActivePinia, createPinia } from 'pinia';
import { ref } from 'vue';
import { usePinnedData } from '@/composables/usePinnedData';
import type { INodeUi } from '@/Interface';
import { MAX_PINNED_DATA_SIZE } from '@/constants';
import { useWorkflowsStore } from '@/stores/workflows.store';
import { useTelemetry } from '@/composables/useTelemetry';
vi.mock('@/composables/useToast', () => ({ useToast: vi.fn(() => ({ showError: vi.fn() })) }));
vi.mock('@/composables/useI18n', () => ({
useI18n: vi.fn(() => ({ baseText: vi.fn((key) => key) })),
}));
vi.mock('@/composables/useExternalHooks', () => ({
useExternalHooks: vi.fn(() => ({
run: vi.fn(),
})),
}));
describe('usePinnedData', () => {
beforeEach(() => {
setActivePinia(createPinia());
});
describe('isValidJSON()', () => {
it('should return true for valid JSON', () => {
const { isValidJSON } = usePinnedData(ref(null));
expect(isValidJSON('{"key":"value"}')).toBe(true);
});
it('should return false for invalid JSON', () => {
const { isValidJSON } = usePinnedData(ref(null));
const result = isValidJSON('invalid json');
expect(result).toBe(false);
});
});
describe('isValidSize()', () => {
it('should return true when data size is at upper limit', () => {
const { isValidSize } = usePinnedData(ref({ name: 'testNode' } as INodeUi));
const largeData = new Array(MAX_PINNED_DATA_SIZE + 1).join('a');
expect(isValidSize(largeData)).toBe(true);
});
it('should return false when data size is too large', () => {
const { isValidSize } = usePinnedData(ref({ name: 'testNode' } as INodeUi));
const largeData = new Array(MAX_PINNED_DATA_SIZE + 2).join('a');
expect(isValidSize(largeData)).toBe(false);
});
});
describe('setData()', () => {
it('should throw if data is not valid JSON', () => {
const { setData } = usePinnedData(ref({ name: 'testNode' } as INodeUi));
expect(() => setData('invalid json', 'pin-icon-click')).toThrow();
});
it('should throw if data size is too large', () => {
const { setData } = usePinnedData(ref({ name: 'testNode' } as INodeUi));
const largeData = new Array(MAX_PINNED_DATA_SIZE + 2).join('a');
expect(() => setData(largeData, 'pin-icon-click')).toThrow();
});
it('should set data correctly for valid inputs', () => {
const workflowsStore = useWorkflowsStore();
const node = ref({ name: 'testNode' } as INodeUi);
const { setData } = usePinnedData(node);
const testData = [{ json: { key: 'value' } }];
expect(() => setData(testData, 'pin-icon-click')).not.toThrow();
expect(workflowsStore.workflow.pinData?.[node.value.name]).toEqual(testData);
});
});
describe('unsetData()', () => {
it('should unset data correctly', () => {
const workflowsStore = useWorkflowsStore();
const node = ref({ name: 'testNode' } as INodeUi);
const { setData, unsetData } = usePinnedData(node);
const testData = [{ json: { key: 'value' } }];
setData(testData, 'pin-icon-click');
unsetData('context-menu');
expect(workflowsStore.workflow.pinData?.[node.value.name]).toBeUndefined();
});
});
describe('onSetDataSuccess()', () => {
it('should trigger telemetry on successful data setting', async () => {
const telemetry = useTelemetry();
const spy = vi.spyOn(telemetry, 'track');
const pinnedData = usePinnedData(ref({ name: 'testNode', type: 'someType' } as INodeUi), {
displayMode: ref('json'),
runIndex: ref(0),
});
pinnedData.onSetDataSuccess({ source: 'pin-icon-click' });
expect(spy).toHaveBeenCalled();
});
});
describe('onSetDataError()', () => {
it('should trigger telemetry tracking on error in data setting', () => {
const telemetry = useTelemetry();
const spy = vi.spyOn(telemetry, 'track');
const pinnedData = usePinnedData(ref({ name: 'testNode', type: 'someType' } as INodeUi), {
displayMode: ref('json'),
runIndex: ref(0),
});
pinnedData.onSetDataError({ errorType: 'data-too-large', source: 'pin-icon-click' });
expect(spy).toHaveBeenCalled();
});
});
describe('onUnsetData()', () => {
it('should trigger telemetry on successful data unsetting', async () => {
const telemetry = useTelemetry();
const spy = vi.spyOn(telemetry, 'track');
const pinnedData = usePinnedData(ref({ name: 'testNode', type: 'someType' } as INodeUi), {
displayMode: ref('json'),
runIndex: ref(0),
});
pinnedData.onUnsetData({ source: 'context-menu' });
expect(spy).toHaveBeenCalled();
});
});
});