import axios from 'axios'; import MockAdapter from 'axios-mock-adapter'; import { once } from 'lodash'; import waitForPromises from 'helpers/wait_for_promises'; import Attachment from '~/content_editor/extensions/attachment'; import Image from '~/content_editor/extensions/image'; import Link from '~/content_editor/extensions/link'; import Loading from '~/content_editor/extensions/loading'; import httpStatus from '~/lib/utils/http_status'; import { loadMarkdownApiResult } from '../markdown_processing_examples'; import { createTestEditor, createDocBuilder } from '../test_utils'; describe('content_editor/extensions/attachment', () => { let tiptapEditor; let eq; let doc; let p; let image; let loading; let link; let renderMarkdown; let mock; const uploadsPath = '/uploads/'; const imageFile = new File(['foo'], 'test-file.png', { type: 'image/png' }); const attachmentFile = new File(['foo'], 'test-file.zip', { type: 'application/zip' }); beforeEach(() => { renderMarkdown = jest.fn(); tiptapEditor = createTestEditor({ extensions: [Loading, Link, Image, Attachment.configure({ renderMarkdown, uploadsPath })], }); ({ builders: { doc, p, image, loading, link }, eq, } = createDocBuilder({ tiptapEditor, names: { loading: { markType: Loading.name }, image: { nodeType: Image.name }, link: { nodeType: Link.name }, }, })); mock = new MockAdapter(axios); }); afterEach(() => { mock.reset(); }); it.each` eventType | propName | eventData | output ${'paste'} | ${'handlePaste'} | ${{ clipboardData: { files: [attachmentFile] } }} | ${true} ${'paste'} | ${'handlePaste'} | ${{ clipboardData: { files: [] } }} | ${undefined} ${'drop'} | ${'handleDrop'} | ${{ dataTransfer: { files: [attachmentFile] } }} | ${true} `('handles $eventType properly', ({ eventType, propName, eventData, output }) => { const event = Object.assign(new Event(eventType), eventData); const handled = tiptapEditor.view.someProp(propName, (eventHandler) => { return eventHandler(tiptapEditor.view, event); }); expect(handled).toBe(output); }); describe('uploadAttachment command', () => { let initialDoc; beforeEach(() => { initialDoc = doc(p('')); tiptapEditor.commands.setContent(initialDoc.toJSON()); }); describe('when the file has image mime type', () => { const base64EncodedFile = ''; beforeEach(() => { renderMarkdown.mockResolvedValue( loadMarkdownApiResult('project_wiki_attachment_image').body, ); }); describe('when uploading succeeds', () => { const successResponse = { link: { markdown: '![test-file](test-file.png)', }, }; beforeEach(() => { mock.onPost().reply(httpStatus.OK, successResponse); }); it('inserts an image with src set to the encoded image file and uploading true', (done) => { const expectedDoc = doc(p(image({ uploading: true, src: base64EncodedFile }))); tiptapEditor.on( 'update', once(() => { expect(eq(tiptapEditor.state.doc, expectedDoc)).toBe(true); done(); }), ); tiptapEditor.commands.uploadAttachment({ file: imageFile }); }); it('updates the inserted image with canonicalSrc when upload is successful', async () => { const expectedDoc = doc( p( image({ canonicalSrc: 'test-file.png', src: base64EncodedFile, alt: 'test-file', uploading: false, }), ), ); tiptapEditor.commands.uploadAttachment({ file: imageFile }); await waitForPromises(); expect(eq(tiptapEditor.state.doc, expectedDoc)).toBe(true); }); }); describe('when uploading request fails', () => { beforeEach(() => { mock.onPost().reply(httpStatus.INTERNAL_SERVER_ERROR); }); it('resets the doc to orginal state', async () => { const expectedDoc = doc(p('')); tiptapEditor.commands.uploadAttachment({ file: imageFile }); await waitForPromises(); expect(eq(tiptapEditor.state.doc, expectedDoc)).toBe(true); }); it('emits an error event that includes an error message', (done) => { tiptapEditor.commands.uploadAttachment({ file: imageFile }); tiptapEditor.on('error', ({ error }) => { expect(error).toBe('An error occurred while uploading the image. Please try again.'); done(); }); }); }); }); describe('when the file has a zip (or any other attachment) mime type', () => { const markdownApiResult = loadMarkdownApiResult('project_wiki_attachment_link').body; beforeEach(() => { renderMarkdown.mockResolvedValue(markdownApiResult); }); describe('when uploading succeeds', () => { const successResponse = { link: { markdown: '[test-file](test-file.zip)', }, }; beforeEach(() => { mock.onPost().reply(httpStatus.OK, successResponse); }); it('inserts a loading mark', (done) => { const expectedDoc = doc(p(loading({ label: 'test-file' }))); tiptapEditor.on( 'update', once(() => { expect(eq(tiptapEditor.state.doc, expectedDoc)).toBe(true); done(); }), ); tiptapEditor.commands.uploadAttachment({ file: attachmentFile }); }); it('updates the loading mark with a link with canonicalSrc and href attrs', async () => { const [, group, project] = markdownApiResult.match(/\/(group[0-9]+)\/(project[0-9]+)\//); const expectedDoc = doc( p( link( { canonicalSrc: 'test-file.zip', href: `/${group}/${project}/-/wikis/test-file.zip`, }, 'test-file', ), ), ); tiptapEditor.commands.uploadAttachment({ file: attachmentFile }); await waitForPromises(); expect(eq(tiptapEditor.state.doc, expectedDoc)).toBe(true); }); }); describe('when uploading request fails', () => { beforeEach(() => { mock.onPost().reply(httpStatus.INTERNAL_SERVER_ERROR); }); it('resets the doc to orginal state', async () => { const expectedDoc = doc(p('')); tiptapEditor.commands.uploadAttachment({ file: attachmentFile }); await waitForPromises(); expect(eq(tiptapEditor.state.doc, expectedDoc)).toBe(true); }); it('emits an error event that includes an error message', (done) => { tiptapEditor.commands.uploadAttachment({ file: attachmentFile }); tiptapEditor.on('error', ({ error }) => { expect(error).toBe('An error occurred while uploading the file. Please try again.'); done(); }); }); }); }); }); });