From 8737a482c4c1dc5c9896a52f4a4431f9ec51e74e Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Pawe=C5=82=20Smyrek?= Date: Fri, 18 Dec 2020 13:57:33 +0100 Subject: [PATCH 1/6] Use canvas.toBlob() instead of fetch() to create a blob from image src --- .../src/imageupload/imageuploadediting.js | 4 +- .../ckeditor5-image/src/imageupload/utils.js | 63 ++++++--- .../tests/imageupload/imageuploadediting.js | 125 ++++++++++++------ .../tests/manual/tickets/7957/1.html | 14 ++ .../tests/manual/tickets/7957/1.js | 42 ++++++ .../tests/manual/tickets/7957/1.md | 7 + 6 files changed, 189 insertions(+), 66 deletions(-) create mode 100644 packages/ckeditor5-paste-from-office/tests/manual/tickets/7957/1.html create mode 100644 packages/ckeditor5-paste-from-office/tests/manual/tickets/7957/1.js create mode 100644 packages/ckeditor5-paste-from-office/tests/manual/tickets/7957/1.md diff --git a/packages/ckeditor5-image/src/imageupload/imageuploadediting.js b/packages/ckeditor5-image/src/imageupload/imageuploadediting.js index 7dcaa143861..8bb894730c3 100644 --- a/packages/ckeditor5-image/src/imageupload/imageuploadediting.js +++ b/packages/ckeditor5-image/src/imageupload/imageuploadediting.js @@ -15,7 +15,7 @@ import UpcastWriter from '@ckeditor/ckeditor5-engine/src/view/upcastwriter'; import env from '@ckeditor/ckeditor5-utils/src/env'; import ImageUploadCommand from '../../src/imageupload/imageuploadcommand'; -import { fetchLocalImage, isLocalImage } from '../../src/imageupload/utils'; +import { createImageFile, isLocalImage } from '../../src/imageupload/utils'; import { createImageTypeRegExp } from './utils'; import { getViewImgFromWidget } from '../image/utils'; @@ -123,7 +123,7 @@ export default class ImageUploadEditing extends Plugin { this.listenTo( editor.plugins.get( Clipboard ), 'inputTransformation', ( evt, data ) => { const fetchableImages = Array.from( editor.editing.view.createRangeIn( data.content ) ) .filter( value => isLocalImage( value.item ) && !value.item.getAttribute( 'uploadProcessed' ) ) - .map( value => { return { promise: fetchLocalImage( value.item ), imageElement: value.item }; } ); + .map( value => { return { promise: createImageFile( value.item ), imageElement: value.item }; } ); if ( !fetchableImages.length ) { return; diff --git a/packages/ckeditor5-image/src/imageupload/utils.js b/packages/ckeditor5-image/src/imageupload/utils.js index 22cc43861a8..6b56cde6dea 100644 --- a/packages/ckeditor5-image/src/imageupload/utils.js +++ b/packages/ckeditor5-image/src/imageupload/utils.js @@ -7,7 +7,9 @@ * @module image/imageupload/utils */ -/* global fetch, File */ +import global from '@ckeditor/ckeditor5-utils/src/dom/global'; + +/* global File */ /** * Creates a regular expression used to test for image files. @@ -27,28 +29,22 @@ export function createImageTypeRegExp( types ) { } /** - * Creates a promise that fetches the image local source (Base64 or blob) and resolves with a `File` object. + * Creates a promise that converts the image local source (Base64 or blob) to a blob and resolves with a `File` object. * - * @param {module:engine/view/element~Element} image Image whose source to fetch. - * @returns {Promise.} A promise which resolves when an image source is fetched and converted to a `File` instance. + * @param {module:engine/view/element~Element} image Image whose source to convert. + * @returns {Promise.} A promise which resolves when an image source is converted to a `File` instance. * It resolves with a `File` object. If there were any errors during file processing, the promise will be rejected. */ -export function fetchLocalImage( image ) { - return new Promise( ( resolve, reject ) => { - const imageSrc = image.getAttribute( 'src' ); - - // Fetch works asynchronously and so does not block browser UI when processing data. - fetch( imageSrc ) - .then( resource => resource.blob() ) - .then( blob => { - const mimeType = getImageMimeType( blob, imageSrc ); - const ext = mimeType.replace( 'image/', '' ); - const filename = `image.${ ext }`; - const file = new File( [ blob ], filename, { type: mimeType } ); - - resolve( file ); - } ) - .catch( reject ); +export function createImageFile( image ) { + const imageSrc = image.getAttribute( 'src' ); + + // Conversion to blob works asynchronously, so it does not block browser UI when processing data. + return getBlobFromImage( imageSrc ).then( blob => { + const mimeType = getImageMimeType( blob, imageSrc ); + const ext = mimeType.replace( 'image/', '' ); + const filename = `image.${ ext }`; + + return new File( [ blob ], filename, { type: mimeType } ); } ); } @@ -67,6 +63,33 @@ export function isLocalImage( node ) { node.getAttribute( 'src' ).match( /^blob:/g ); } +// Creates a promise that resolves with a `Blob` object converted from the image source (Base64 or blob). +// +// @param {String} imageSrc Image `src` attribute value. +// @returns {Promise.} +function getBlobFromImage( imageSrc ) { + return new Promise( ( resolve, reject ) => { + const image = global.document.createElement( 'img' ); + + image.addEventListener( 'load', () => { + const canvas = global.document.createElement( 'canvas' ); + + canvas.width = image.width; + canvas.height = image.height; + + const ctx = canvas.getContext( '2d' ); + + ctx.drawImage( image, 0, 0 ); + + canvas.toBlob( blob => blob ? resolve( blob ) : reject() ); + } ); + + image.addEventListener( 'error', reject ); + + image.src = imageSrc; + } ); +} + // Extracts an image type based on its blob representation or its source. // // @param {String} src Image `src` attribute value. diff --git a/packages/ckeditor5-image/tests/imageupload/imageuploadediting.js b/packages/ckeditor5-image/tests/imageupload/imageuploadediting.js index 18c18f8ee87..cf437284854 100644 --- a/packages/ckeditor5-image/tests/imageupload/imageuploadediting.js +++ b/packages/ckeditor5-image/tests/imageupload/imageuploadediting.js @@ -3,7 +3,7 @@ * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license */ -/* globals window, setTimeout, atob, URL, Blob, console */ +/* globals document, window, setTimeout, atob, URL, Blob, HTMLCanvasElement, console */ import VirtualTestEditor from '@ckeditor/ckeditor5-core/tests/_utils/virtualtesteditor'; @@ -28,6 +28,7 @@ import Notification from '@ckeditor/ckeditor5-ui/src/notification/notification'; describe( 'ImageUploadEditing', () => { // eslint-disable-next-line max-len const base64Sample = ''; + const base64InvalidSample = '-DATA'; let adapterMocks = []; let editor, model, view, doc, fileRepository, viewDocument, nativeReaderMock, loader; @@ -671,7 +672,7 @@ describe( 'ImageUploadEditing', () => { expectModel( done, getModelData( model ), expected ); } ); - it( 'should not upload and remove image if fetch failed', done => { + it( 'should not upload and remove image if conversion to a blob failed', done => { const notification = editor.plugins.get( Notification ); // Prevent popping up alert window. @@ -681,17 +682,12 @@ describe( 'ImageUploadEditing', () => { setModelData( model, '[]foo' ); - const clipboardHtml = ``; + const clipboardHtml = ``; const dataTransfer = mockDataTransfer( clipboardHtml ); const targetRange = model.createRange( model.createPositionAt( doc.getRoot(), 1 ), model.createPositionAt( doc.getRoot(), 1 ) ); const targetViewRange = editor.editing.mapper.toViewRange( targetRange ); - // Stub `fetch` so it can be rejected. - sinon.stub( window, 'fetch' ).callsFake( () => { - return new Promise( ( res, rej ) => rej( 'could not fetch' ) ); - } ); - let content = null; editor.plugins.get( 'Clipboard' ).on( 'inputTransformation', ( evt, data ) => { content = data.content; @@ -709,7 +705,7 @@ describe( 'ImageUploadEditing', () => { ); } ); - it( 'should upload only images which were successfully fetched and remove failed ones', done => { + it( 'should upload only images which were successfully converted to a blob and remove failed ones', done => { const notification = editor.plugins.get( Notification ); // Prevent popping up alert window. @@ -729,25 +725,15 @@ describe( 'ImageUploadEditing', () => { setModelData( model, '[]foo' ); + // The first 2 images are valid ones, and the 3rd one fails. const clipboardHtml = `

bar

` + - ``; + `` + + ``; const dataTransfer = mockDataTransfer( clipboardHtml ); const targetRange = model.createRange( model.createPositionAt( doc.getRoot(), 1 ), model.createPositionAt( doc.getRoot(), 1 ) ); const targetViewRange = editor.editing.mapper.toViewRange( targetRange ); - // Stub `fetch` in a way that 2 first calls are successful and 3rd fails. - let counter = 0; - const fetch = window.fetch; - sinon.stub( window, 'fetch' ).callsFake( src => { - counter++; - if ( counter < 3 ) { - return fetch( src ); - } else { - return Promise.reject(); - } - } ); - let content = null; editor.plugins.get( 'Clipboard' ).on( 'inputTransformation', ( evt, data ) => { content = data.content; @@ -848,15 +834,6 @@ describe( 'ImageUploadEditing', () => { const targetRange = model.createRange( model.createPositionAt( doc.getRoot(), 1 ), model.createPositionAt( doc.getRoot(), 1 ) ); const targetViewRange = editor.editing.mapper.toViewRange( targetRange ); - // Stub `fetch` to return custom blob without type. - sinon.stub( window, 'fetch' ).callsFake( () => { - return new Promise( res => res( { - blob() { - return new Promise( res => res( new Blob( [ 'foo', 'bar' ] ) ) ); - } - } ) ); - } ); - viewDocument.fire( 'clipboardInput', { dataTransfer, targetRanges: [ targetViewRange ] } ); tryExpect( done, () => { @@ -873,15 +850,6 @@ describe( 'ImageUploadEditing', () => { const targetRange = model.createRange( model.createPositionAt( doc.getRoot(), 1 ), model.createPositionAt( doc.getRoot(), 1 ) ); const targetViewRange = editor.editing.mapper.toViewRange( targetRange ); - // Stub `fetch` to return custom blob without type. - sinon.stub( window, 'fetch' ).callsFake( () => { - return new Promise( res => res( { - blob() { - return new Promise( res => res( new Blob( [ 'foo', 'bar' ] ) ) ); - } - } ) ); - } ); - viewDocument.fire( 'clipboardInput', { dataTransfer, targetRanges: [ targetViewRange ] } ); tryExpect( done, () => { @@ -907,8 +875,8 @@ describe( 'ImageUploadEditing', () => { const targetRange = model.createRange( model.createPositionAt( doc.getRoot(), 1 ), model.createPositionAt( doc.getRoot(), 1 ) ); const targetViewRange = editor.editing.mapper.toViewRange( targetRange ); - // Stub `fetch` in a way that it always fails. - sinon.stub( window, 'fetch' ).callsFake( () => Promise.reject() ); + // Stub `HTMLCanvasElement#toBlob` to return invalid blob, so image conversion always fails. + sinon.stub( HTMLCanvasElement.prototype, 'toBlob' ).callsFake( fn => fn( null ) ); viewDocument.fire( 'clipboardInput', { dataTransfer, targetRanges: [ targetViewRange ] } ); @@ -923,13 +891,82 @@ describe( 'ImageUploadEditing', () => { } ); } ); + describe( 'with Content Security Policy rules', () => { + let metaElement; + let previousMetaContent; + + // Set the CSP rules before the first test in this block has been executed. + before( () => { + metaElement = document.querySelector( '[http-equiv=Content-Security-Policy]' ); + + if ( metaElement ) { + previousMetaContent = metaElement.getAttribute( 'content' ); + } else { + metaElement = document.createElement( 'meta' ); + metaElement.setAttribute( 'http-equiv', 'Content-Security-Policy' ); + + document.head.appendChild( metaElement ); + } + + metaElement.setAttribute( 'content', '' + + 'default-src \'none\'; ' + + 'connect-src \'self\'; ' + + 'script-src \'self\'; ' + + 'img-src * data: blob:;' + + 'style-src \'self\' \'unsafe-inline\'; ' + + 'frame-src *' + ); + } ); + + // Remove or restore the previous CSP rules after the last test in this block has been executed. + after( () => { + if ( previousMetaContent ) { + metaElement.setAttribute( 'content', previousMetaContent ); + } else { + document.head.removeChild( metaElement ); + } + } ); + + // See https://github.com/ckeditor/ckeditor5/issues/7957. + it( 'should upload image if strict CSP rules are defined', done => { + const spy = sinon.spy(); + const notification = editor.plugins.get( Notification ); + + notification.on( 'show:warning', evt => { + spy(); + evt.stop(); + }, { priority: 'high' } ); + + setModelData( model, '[]foo' ); + + const clipboardHtml = `

bar

`; + const dataTransfer = mockDataTransfer( clipboardHtml ); + + const targetRange = model.createRange( model.createPositionAt( doc.getRoot(), 1 ), model.createPositionAt( doc.getRoot(), 1 ) ); + const targetViewRange = editor.editing.mapper.toViewRange( targetRange ); + + viewDocument.fire( 'clipboardInput', { dataTransfer, targetRanges: [ targetViewRange ] } ); + + adapterMocks[ 0 ].loader.file.then( () => { + setTimeout( () => { + sinon.assert.notCalled( spy ); + done(); + } ); + } ).catch( () => { + setTimeout( () => { + expect.fail( 'Promise should be resolved.' ); + } ); + } ); + } ); + } ); + // Helper for validating clipboard and model data as a result of a paste operation. This function checks both clipboard // data and model data synchronously (`expectedClipboardData`, `expectedModel`) and then the model data after `loader.file` - // promise is resolved (so model state after successful/failed file fetch attempt). + // promise is resolved (so model state after successful/failed file conversion attempt). // // @param {String} expectedClipboardData Expected clipboard data on `inputTransformation` event. // @param {String} expectedModel Expected model data on `inputTransformation` event. - // @param {String} expectedModelOnFile Expected model data after all `file.loader` promises are fetched. + // @param {String} expectedModelOnFile Expected model data after all `file.loader` promises are fulfilled. // @param {DocumentFragment} content Content processed in inputTransformation // @param {Function} doneFn Callback function to be called when all assertions are done or error occures. // @param {Boolean} [onSuccess=true] If `expectedModelOnFile` data should be validated diff --git a/packages/ckeditor5-paste-from-office/tests/manual/tickets/7957/1.html b/packages/ckeditor5-paste-from-office/tests/manual/tickets/7957/1.html new file mode 100644 index 00000000000..f605f98b253 --- /dev/null +++ b/packages/ckeditor5-paste-from-office/tests/manual/tickets/7957/1.html @@ -0,0 +1,14 @@ + + + + +
+

Paste here:

+
diff --git a/packages/ckeditor5-paste-from-office/tests/manual/tickets/7957/1.js b/packages/ckeditor5-paste-from-office/tests/manual/tickets/7957/1.js new file mode 100644 index 00000000000..f74203075cc --- /dev/null +++ b/packages/ckeditor5-paste-from-office/tests/manual/tickets/7957/1.js @@ -0,0 +1,42 @@ +/** + * @license Copyright (c) 2003-2020, CKSource - Frederico Knabben. All rights reserved. + * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license + */ + +/* globals console, window, document */ + +import ClassicEditor from '@ckeditor/ckeditor5-editor-classic/src/classiceditor'; +import ArticlePluginSet from '@ckeditor/ckeditor5-core/tests/_utils/articlepluginset'; + +import Strikethrough from '@ckeditor/ckeditor5-basic-styles/src/strikethrough'; +import Underline from '@ckeditor/ckeditor5-basic-styles/src/underline'; +import Table from '@ckeditor/ckeditor5-table/src/table'; +import TableToolbar from '@ckeditor/ckeditor5-table/src/tabletoolbar'; +import EasyImage from '@ckeditor/ckeditor5-easy-image/src/easyimage'; +import FontColor from '@ckeditor/ckeditor5-font/src/fontcolor'; +import FontBackgroundColor from '@ckeditor/ckeditor5-font/src/fontbackgroundcolor'; +import PageBreak from '@ckeditor/ckeditor5-page-break/src/pagebreak'; +import TableProperties from '@ckeditor/ckeditor5-table/src/tableproperties'; +import TableCellProperties from '@ckeditor/ckeditor5-table/src/tablecellproperties'; + +import PasteFromOffice from '../../../../src/pastefromoffice'; + +import { CS_CONFIG } from '@ckeditor/ckeditor5-cloud-services/tests/_utils/cloud-services-config'; + +ClassicEditor + .create( document.querySelector( '#editor' ), { + plugins: [ ArticlePluginSet, Strikethrough, Underline, Table, TableToolbar, PageBreak, + TableProperties, TableCellProperties, EasyImage, PasteFromOffice, FontColor, FontBackgroundColor ], + toolbar: [ 'heading', '|', 'bold', 'italic', 'strikethrough', 'underline', 'link', + 'bulletedList', 'numberedList', 'blockQuote', 'insertTable', 'pageBreak', 'undo', 'redo' ], + table: { + contentToolbar: [ 'tableColumn', 'tableRow', 'mergeTableCells', 'tableProperties', 'tableCellProperties' ] + }, + cloudServices: CS_CONFIG + } ) + .then( editor => { + window.editor = editor; + } ) + .catch( err => { + console.error( err.stack ); + } ); diff --git a/packages/ckeditor5-paste-from-office/tests/manual/tickets/7957/1.md b/packages/ckeditor5-paste-from-office/tests/manual/tickets/7957/1.md new file mode 100644 index 00000000000..18c5a1f3ec2 --- /dev/null +++ b/packages/ckeditor5-paste-from-office/tests/manual/tickets/7957/1.md @@ -0,0 +1,7 @@ +## Paste from Office + +Test for Paste from Word, when strict CSP rules are configured. + +Check: + +1. Copy & paste some content from Word including at least one image. From a0af560396859f293883ff7e328a142f3834f2ea Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Pawe=C5=82=20Smyrek?= Date: Fri, 18 Dec 2020 15:02:26 +0100 Subject: [PATCH 2/6] Fixed code coverage --- .../ckeditor5-image/src/imageupload/utils.js | 25 +++++++++---------- 1 file changed, 12 insertions(+), 13 deletions(-) diff --git a/packages/ckeditor5-image/src/imageupload/utils.js b/packages/ckeditor5-image/src/imageupload/utils.js index 6b56cde6dea..92997eff111 100644 --- a/packages/ckeditor5-image/src/imageupload/utils.js +++ b/packages/ckeditor5-image/src/imageupload/utils.js @@ -40,7 +40,7 @@ export function createImageFile( image ) { // Conversion to blob works asynchronously, so it does not block browser UI when processing data. return getBlobFromImage( imageSrc ).then( blob => { - const mimeType = getImageMimeType( blob, imageSrc ); + const mimeType = getImageMimeType( imageSrc ); const ext = mimeType.replace( 'image/', '' ); const filename = `image.${ ext }`; @@ -81,7 +81,11 @@ function getBlobFromImage( imageSrc ) { ctx.drawImage( image, 0, 0 ); - canvas.toBlob( blob => blob ? resolve( blob ) : reject() ); + canvas.toBlob( + blob => blob ? resolve( blob ) : reject(), + getImageMimeType( imageSrc ), + 1 + ); } ); image.addEventListener( 'error', reject ); @@ -90,18 +94,13 @@ function getBlobFromImage( imageSrc ) { } ); } -// Extracts an image type based on its blob representation or its source. +// Extracts an image type based on its source. // // @param {String} src Image `src` attribute value. -// @param {Blob} blob Image blob representation. // @returns {String} -function getImageMimeType( blob, src ) { - if ( blob.type ) { - return blob.type; - } else if ( src.match( /data:(image\/\w+);base64/ ) ) { - return src.match( /data:(image\/\w+);base64/ )[ 1 ].toLowerCase(); - } else { - // Fallback to 'jpeg' as common extension. - return 'image/jpeg'; - } +function getImageMimeType( src ) { + const match = src.match( /data:(image\/\w+);base64/ ); + + // Fallback to 'jpeg' as common extension. + return match ? match[ 1 ].toLowerCase() : 'image/jpeg'; } From 8c6c7a15967ef284ba8e8e495277596c4b9536af Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Pawe=C5=82=20Smyrek?= Date: Tue, 22 Dec 2020 08:05:37 +0100 Subject: [PATCH 3/6] Call getImageMimeType() once --- packages/ckeditor5-image/src/imageupload/utils.js | 9 +++++---- 1 file changed, 5 insertions(+), 4 deletions(-) diff --git a/packages/ckeditor5-image/src/imageupload/utils.js b/packages/ckeditor5-image/src/imageupload/utils.js index 92997eff111..968b59de1d5 100644 --- a/packages/ckeditor5-image/src/imageupload/utils.js +++ b/packages/ckeditor5-image/src/imageupload/utils.js @@ -37,10 +37,10 @@ export function createImageTypeRegExp( types ) { */ export function createImageFile( image ) { const imageSrc = image.getAttribute( 'src' ); + const mimeType = getImageMimeType( imageSrc ); // Conversion to blob works asynchronously, so it does not block browser UI when processing data. - return getBlobFromImage( imageSrc ).then( blob => { - const mimeType = getImageMimeType( imageSrc ); + return getBlobFromImage( imageSrc, mimeType ).then( blob => { const ext = mimeType.replace( 'image/', '' ); const filename = `image.${ ext }`; @@ -66,8 +66,9 @@ export function isLocalImage( node ) { // Creates a promise that resolves with a `Blob` object converted from the image source (Base64 or blob). // // @param {String} imageSrc Image `src` attribute value. +// @param {String} mimeType Image type based on its source. // @returns {Promise.} -function getBlobFromImage( imageSrc ) { +function getBlobFromImage( imageSrc, mimeType ) { return new Promise( ( resolve, reject ) => { const image = global.document.createElement( 'img' ); @@ -83,7 +84,7 @@ function getBlobFromImage( imageSrc ) { canvas.toBlob( blob => blob ? resolve( blob ) : reject(), - getImageMimeType( imageSrc ), + mimeType, 1 ); } ); From 9079fcd1ab727d1640012b4d6f3df6d2d64c50a9 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Pawe=C5=82=20Smyrek?= Date: Mon, 28 Dec 2020 14:13:57 +0100 Subject: [PATCH 4/6] Use canvas conversion as a fallback only if fetch() has failed --- .../src/imageupload/imageuploadediting.js | 4 +- .../ckeditor5-image/src/imageupload/utils.js | 92 ++++++++++++------- .../tests/imageupload/imageuploadediting.js | 90 +++++++++++++++--- 3 files changed, 138 insertions(+), 48 deletions(-) diff --git a/packages/ckeditor5-image/src/imageupload/imageuploadediting.js b/packages/ckeditor5-image/src/imageupload/imageuploadediting.js index 8bb894730c3..7dcaa143861 100644 --- a/packages/ckeditor5-image/src/imageupload/imageuploadediting.js +++ b/packages/ckeditor5-image/src/imageupload/imageuploadediting.js @@ -15,7 +15,7 @@ import UpcastWriter from '@ckeditor/ckeditor5-engine/src/view/upcastwriter'; import env from '@ckeditor/ckeditor5-utils/src/env'; import ImageUploadCommand from '../../src/imageupload/imageuploadcommand'; -import { createImageFile, isLocalImage } from '../../src/imageupload/utils'; +import { fetchLocalImage, isLocalImage } from '../../src/imageupload/utils'; import { createImageTypeRegExp } from './utils'; import { getViewImgFromWidget } from '../image/utils'; @@ -123,7 +123,7 @@ export default class ImageUploadEditing extends Plugin { this.listenTo( editor.plugins.get( Clipboard ), 'inputTransformation', ( evt, data ) => { const fetchableImages = Array.from( editor.editing.view.createRangeIn( data.content ) ) .filter( value => isLocalImage( value.item ) && !value.item.getAttribute( 'uploadProcessed' ) ) - .map( value => { return { promise: createImageFile( value.item ), imageElement: value.item }; } ); + .map( value => { return { promise: fetchLocalImage( value.item ), imageElement: value.item }; } ); if ( !fetchableImages.length ) { return; diff --git a/packages/ckeditor5-image/src/imageupload/utils.js b/packages/ckeditor5-image/src/imageupload/utils.js index 968b59de1d5..1a31be50791 100644 --- a/packages/ckeditor5-image/src/imageupload/utils.js +++ b/packages/ckeditor5-image/src/imageupload/utils.js @@ -9,7 +9,7 @@ import global from '@ckeditor/ckeditor5-utils/src/dom/global'; -/* global File */ +/* global fetch File */ /** * Creates a regular expression used to test for image files. @@ -29,22 +29,35 @@ export function createImageTypeRegExp( types ) { } /** - * Creates a promise that converts the image local source (Base64 or blob) to a blob and resolves with a `File` object. + * Creates a promise that fetches the image local source (Base64 or blob) and resolves with a `File` object. * - * @param {module:engine/view/element~Element} image Image whose source to convert. - * @returns {Promise.} A promise which resolves when an image source is converted to a `File` instance. + * @param {module:engine/view/element~Element} image Image whose source to fetch. + * @returns {Promise.} A promise which resolves when an image source is fetched and converted to a `File` instance. * It resolves with a `File` object. If there were any errors during file processing, the promise will be rejected. */ -export function createImageFile( image ) { - const imageSrc = image.getAttribute( 'src' ); - const mimeType = getImageMimeType( imageSrc ); - - // Conversion to blob works asynchronously, so it does not block browser UI when processing data. - return getBlobFromImage( imageSrc, mimeType ).then( blob => { - const ext = mimeType.replace( 'image/', '' ); - const filename = `image.${ ext }`; - - return new File( [ blob ], filename, { type: mimeType } ); +export function fetchLocalImage( image ) { + return new Promise( ( resolve, reject ) => { + const imageSrc = image.getAttribute( 'src' ); + + // Fetch works asynchronously and so does not block browser UI when processing data. + fetch( imageSrc ) + .then( resource => resource.blob() ) + .then( blob => { + const mimeType = getImageMimeType( blob, imageSrc ); + const ext = mimeType.replace( 'image/', '' ); + const filename = `image.${ ext }`; + const file = new File( [ blob ], filename, { type: mimeType } ); + + resolve( file ); + } ) + .catch( err => { + // Fetch fails only, if it can't make a request due to a network failure or if anything prevented the request + // from completing, i.e. the Content Security Policy rules. It is not possible to detect the exact cause of failure, + // so we are just trying the fallback solution, if general TypeError is thrown. + return err && err.name === 'TypeError' ? + convertLocalImageOnCanvas( imageSrc ).then( resolve ).catch( reject ) : + reject( err ); + } ); } ); } @@ -63,12 +76,42 @@ export function isLocalImage( node ) { node.getAttribute( 'src' ).match( /^blob:/g ); } +// Extracts an image type based on its blob representation or its source. +// +// @param {String} src Image `src` attribute value. +// @param {Blob} blob Image blob representation. +// @returns {String} +function getImageMimeType( blob, src ) { + if ( blob.type ) { + return blob.type; + } else if ( src.match( /data:(image\/\w+);base64/ ) ) { + return src.match( /data:(image\/\w+);base64/ )[ 1 ].toLowerCase(); + } else { + // Fallback to 'jpeg' as common extension. + return 'image/jpeg'; + } +} + +// Creates a promise that converts the image local source (Base64 or blob) to a blob and resolves with a `File` object. +// +// @param {String} imageSrc Image `src` attribute value. +// @returns {Promise.} A promise which resolves when an image source is converted to a `File` instance. +// It resolves with a `File` object. If there were any errors during file processing, the promise will be rejected. +function convertLocalImageOnCanvas( imageSrc ) { + return getBlobFromCanvas( imageSrc ).then( blob => { + const mimeType = getImageMimeType( blob, imageSrc ); + const ext = mimeType.replace( 'image/', '' ); + const filename = `image.${ ext }`; + + return new File( [ blob ], filename, { type: mimeType } ); + } ); +} + // Creates a promise that resolves with a `Blob` object converted from the image source (Base64 or blob). // // @param {String} imageSrc Image `src` attribute value. -// @param {String} mimeType Image type based on its source. // @returns {Promise.} -function getBlobFromImage( imageSrc, mimeType ) { +function getBlobFromCanvas( imageSrc ) { return new Promise( ( resolve, reject ) => { const image = global.document.createElement( 'img' ); @@ -82,11 +125,7 @@ function getBlobFromImage( imageSrc, mimeType ) { ctx.drawImage( image, 0, 0 ); - canvas.toBlob( - blob => blob ? resolve( blob ) : reject(), - mimeType, - 1 - ); + canvas.toBlob( blob => blob ? resolve( blob ) : reject() ); } ); image.addEventListener( 'error', reject ); @@ -94,14 +133,3 @@ function getBlobFromImage( imageSrc, mimeType ) { image.src = imageSrc; } ); } - -// Extracts an image type based on its source. -// -// @param {String} src Image `src` attribute value. -// @returns {String} -function getImageMimeType( src ) { - const match = src.match( /data:(image\/\w+);base64/ ); - - // Fallback to 'jpeg' as common extension. - return match ? match[ 1 ].toLowerCase() : 'image/jpeg'; -} diff --git a/packages/ckeditor5-image/tests/imageupload/imageuploadediting.js b/packages/ckeditor5-image/tests/imageupload/imageuploadediting.js index cf437284854..5fe144efb66 100644 --- a/packages/ckeditor5-image/tests/imageupload/imageuploadediting.js +++ b/packages/ckeditor5-image/tests/imageupload/imageuploadediting.js @@ -28,7 +28,6 @@ import Notification from '@ckeditor/ckeditor5-ui/src/notification/notification'; describe( 'ImageUploadEditing', () => { // eslint-disable-next-line max-len const base64Sample = ''; - const base64InvalidSample = '-DATA'; let adapterMocks = []; let editor, model, view, doc, fileRepository, viewDocument, nativeReaderMock, loader; @@ -672,7 +671,7 @@ describe( 'ImageUploadEditing', () => { expectModel( done, getModelData( model ), expected ); } ); - it( 'should not upload and remove image if conversion to a blob failed', done => { + it( 'should not upload and remove image if fetch failed', done => { const notification = editor.plugins.get( Notification ); // Prevent popping up alert window. @@ -682,12 +681,17 @@ describe( 'ImageUploadEditing', () => { setModelData( model, '[]foo' ); - const clipboardHtml = ``; + const clipboardHtml = ``; const dataTransfer = mockDataTransfer( clipboardHtml ); const targetRange = model.createRange( model.createPositionAt( doc.getRoot(), 1 ), model.createPositionAt( doc.getRoot(), 1 ) ); const targetViewRange = editor.editing.mapper.toViewRange( targetRange ); + // Stub `fetch` so it can be rejected. + sinon.stub( window, 'fetch' ).callsFake( () => { + return new Promise( ( res, rej ) => rej( 'could not fetch' ) ); + } ); + let content = null; editor.plugins.get( 'Clipboard' ).on( 'inputTransformation', ( evt, data ) => { content = data.content; @@ -705,7 +709,7 @@ describe( 'ImageUploadEditing', () => { ); } ); - it( 'should upload only images which were successfully converted to a blob and remove failed ones', done => { + it( 'should upload only images which were successfully fetched and remove failed ones', done => { const notification = editor.plugins.get( Notification ); // Prevent popping up alert window. @@ -725,15 +729,25 @@ describe( 'ImageUploadEditing', () => { setModelData( model, '[]foo' ); - // The first 2 images are valid ones, and the 3rd one fails. const clipboardHtml = `

bar

` + - `` + - ``; + ``; const dataTransfer = mockDataTransfer( clipboardHtml ); const targetRange = model.createRange( model.createPositionAt( doc.getRoot(), 1 ), model.createPositionAt( doc.getRoot(), 1 ) ); const targetViewRange = editor.editing.mapper.toViewRange( targetRange ); + // Stub `fetch` in a way that 2 first calls are successful and 3rd fails. + let counter = 0; + const fetch = window.fetch; + sinon.stub( window, 'fetch' ).callsFake( src => { + counter++; + if ( counter < 3 ) { + return fetch( src ); + } else { + return Promise.reject(); + } + } ); + let content = null; editor.plugins.get( 'Clipboard' ).on( 'inputTransformation', ( evt, data ) => { content = data.content; @@ -834,6 +848,15 @@ describe( 'ImageUploadEditing', () => { const targetRange = model.createRange( model.createPositionAt( doc.getRoot(), 1 ), model.createPositionAt( doc.getRoot(), 1 ) ); const targetViewRange = editor.editing.mapper.toViewRange( targetRange ); + // Stub `fetch` to return custom blob without type. + sinon.stub( window, 'fetch' ).callsFake( () => { + return new Promise( res => res( { + blob() { + return new Promise( res => res( new Blob( [ 'foo', 'bar' ] ) ) ); + } + } ) ); + } ); + viewDocument.fire( 'clipboardInput', { dataTransfer, targetRanges: [ targetViewRange ] } ); tryExpect( done, () => { @@ -850,6 +873,15 @@ describe( 'ImageUploadEditing', () => { const targetRange = model.createRange( model.createPositionAt( doc.getRoot(), 1 ), model.createPositionAt( doc.getRoot(), 1 ) ); const targetViewRange = editor.editing.mapper.toViewRange( targetRange ); + // Stub `fetch` to return custom blob without type. + sinon.stub( window, 'fetch' ).callsFake( () => { + return new Promise( res => res( { + blob() { + return new Promise( res => res( new Blob( [ 'foo', 'bar' ] ) ) ); + } + } ) ); + } ); + viewDocument.fire( 'clipboardInput', { dataTransfer, targetRanges: [ targetViewRange ] } ); tryExpect( done, () => { @@ -875,8 +907,8 @@ describe( 'ImageUploadEditing', () => { const targetRange = model.createRange( model.createPositionAt( doc.getRoot(), 1 ), model.createPositionAt( doc.getRoot(), 1 ) ); const targetViewRange = editor.editing.mapper.toViewRange( targetRange ); - // Stub `HTMLCanvasElement#toBlob` to return invalid blob, so image conversion always fails. - sinon.stub( HTMLCanvasElement.prototype, 'toBlob' ).callsFake( fn => fn( null ) ); + // Stub `fetch` in a way that it always fails. + sinon.stub( window, 'fetch' ).callsFake( () => Promise.reject() ); viewDocument.fire( 'clipboardInput', { dataTransfer, targetRanges: [ targetViewRange ] } ); @@ -891,11 +923,12 @@ describe( 'ImageUploadEditing', () => { } ); } ); - describe( 'with Content Security Policy rules', () => { + describe( 'fallback image conversion on canvas', () => { let metaElement; let previousMetaContent; - // Set the CSP rules before the first test in this block has been executed. + // Set strict Content Security Policy (CSP) rules before the first test in this block has been executed. + // The CSP rules cause that fetch() fails and it triggers the fallback procedure. before( () => { metaElement = document.querySelector( '[http-equiv=Content-Security-Policy]' ); @@ -928,7 +961,7 @@ describe( 'ImageUploadEditing', () => { } ); // See https://github.com/ckeditor/ckeditor5/issues/7957. - it( 'should upload image if strict CSP rules are defined', done => { + it( 'should upload image using canvas conversion', done => { const spy = sinon.spy(); const notification = editor.plugins.get( Notification ); @@ -958,15 +991,44 @@ describe( 'ImageUploadEditing', () => { } ); } ); } ); + + it( 'should not upload and remove image if canvas convertion failed', done => { + setModelData( model, '[]foo' ); + + const clipboardHtml = ``; + const dataTransfer = mockDataTransfer( clipboardHtml ); + + const targetRange = model.createRange( model.createPositionAt( doc.getRoot(), 1 ), model.createPositionAt( doc.getRoot(), 1 ) ); + const targetViewRange = editor.editing.mapper.toViewRange( targetRange ); + + // Stub `HTMLCanvasElement#toBlob` to return invalid blob, so image conversion always fails. + sinon.stub( HTMLCanvasElement.prototype, 'toBlob' ).callsFake( fn => fn( null ) ); + + let content = null; + editor.plugins.get( 'Clipboard' ).on( 'inputTransformation', ( evt, data ) => { + content = data.content; + } ); + + viewDocument.fire( 'clipboardInput', { dataTransfer, targetRanges: [ targetViewRange ] } ); + + expectData( + '', + '[]foo', + '[]foo', + content, + done, + false + ); + } ); } ); // Helper for validating clipboard and model data as a result of a paste operation. This function checks both clipboard // data and model data synchronously (`expectedClipboardData`, `expectedModel`) and then the model data after `loader.file` - // promise is resolved (so model state after successful/failed file conversion attempt). + // promise is resolved (so model state after successful/failed file fetch attempt). // // @param {String} expectedClipboardData Expected clipboard data on `inputTransformation` event. // @param {String} expectedModel Expected model data on `inputTransformation` event. - // @param {String} expectedModelOnFile Expected model data after all `file.loader` promises are fulfilled. + // @param {String} expectedModelOnFile Expected model data after all `file.loader` promises are fetched. // @param {DocumentFragment} content Content processed in inputTransformation // @param {Function} doneFn Callback function to be called when all assertions are done or error occures. // @param {Boolean} [onSuccess=true] If `expectedModelOnFile` data should be validated From ac238166a3c10499cd4692b25305781e16a670f6 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Pawe=C5=82=20Smyrek?= Date: Mon, 28 Dec 2020 14:20:58 +0100 Subject: [PATCH 5/6] Cosmetic clean-up --- packages/ckeditor5-image/src/imageupload/utils.js | 7 ++++--- 1 file changed, 4 insertions(+), 3 deletions(-) diff --git a/packages/ckeditor5-image/src/imageupload/utils.js b/packages/ckeditor5-image/src/imageupload/utils.js index 1a31be50791..be8d1bd5e49 100644 --- a/packages/ckeditor5-image/src/imageupload/utils.js +++ b/packages/ckeditor5-image/src/imageupload/utils.js @@ -7,9 +7,9 @@ * @module image/imageupload/utils */ -import global from '@ckeditor/ckeditor5-utils/src/dom/global'; +/* global fetch, File */ -/* global fetch File */ +import global from '@ckeditor/ckeditor5-utils/src/dom/global'; /** * Creates a regular expression used to test for image files. @@ -92,7 +92,8 @@ function getImageMimeType( blob, src ) { } } -// Creates a promise that converts the image local source (Base64 or blob) to a blob and resolves with a `File` object. +// Creates a promise that converts the image local source (Base64 or blob) to a blob using canvas and resolves +// with a `File` object. // // @param {String} imageSrc Image `src` attribute value. // @returns {Promise.} A promise which resolves when an image source is converted to a `File` instance. From 402e793b24dc4246c7205aa8b58626ec21021deb Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Pawe=C5=82=20Smyrek?= Date: Tue, 12 Jan 2021 10:41:09 +0100 Subject: [PATCH 6/6] Prevent showing an alert() with event object --- .../ckeditor5-image/src/imageupload/utils.js | 2 +- .../tests/imageupload/imageuploadediting.js | 31 ++++++++++++++++++- 2 files changed, 31 insertions(+), 2 deletions(-) diff --git a/packages/ckeditor5-image/src/imageupload/utils.js b/packages/ckeditor5-image/src/imageupload/utils.js index be8d1bd5e49..9230bb0992b 100644 --- a/packages/ckeditor5-image/src/imageupload/utils.js +++ b/packages/ckeditor5-image/src/imageupload/utils.js @@ -129,7 +129,7 @@ function getBlobFromCanvas( imageSrc ) { canvas.toBlob( blob => blob ? resolve( blob ) : reject() ); } ); - image.addEventListener( 'error', reject ); + image.addEventListener( 'error', () => reject() ); image.src = imageSrc; } ); diff --git a/packages/ckeditor5-image/tests/imageupload/imageuploadediting.js b/packages/ckeditor5-image/tests/imageupload/imageuploadediting.js index 5fe144efb66..74da5ab5ba7 100644 --- a/packages/ckeditor5-image/tests/imageupload/imageuploadediting.js +++ b/packages/ckeditor5-image/tests/imageupload/imageuploadediting.js @@ -992,7 +992,7 @@ describe( 'ImageUploadEditing', () => { } ); } ); - it( 'should not upload and remove image if canvas convertion failed', done => { + it( 'should not upload and remove image if canvas conversion failed', done => { setModelData( model, '[]foo' ); const clipboardHtml = ``; @@ -1020,6 +1020,35 @@ describe( 'ImageUploadEditing', () => { false ); } ); + + it( 'should not show notification when image could not be loaded', done => { + const spy = sinon.spy(); + const notification = editor.plugins.get( Notification ); + + notification.on( 'show:warning', evt => { + spy(); + evt.stop(); + }, { priority: 'high' } ); + + setModelData( model, '[]foo' ); + + const clipboardHtml = ''; + const dataTransfer = mockDataTransfer( clipboardHtml ); + + const targetRange = model.createRange( model.createPositionAt( doc.getRoot(), 1 ), model.createPositionAt( doc.getRoot(), 1 ) ); + const targetViewRange = editor.editing.mapper.toViewRange( targetRange ); + + viewDocument.fire( 'clipboardInput', { dataTransfer, targetRanges: [ targetViewRange ] } ); + + adapterMocks[ 0 ].loader.file.then( () => { + expect.fail( 'Promise should be rejected.' ); + } ).catch( () => { + setTimeout( () => { + sinon.assert.notCalled( spy ); + done(); + } ); + } ); + } ); } ); // Helper for validating clipboard and model data as a result of a paste operation. This function checks both clipboard