DictionaryImporterMediaLoader (#1860)

* Rename param for consistency

* Move media loading functionality into DictionaryImporterMediaLoader

* Create test class for media loading

* Remove unnecessary Blob/Image/URL functionality
This commit is contained in:
toasted-nutbread 2021-07-31 12:30:31 -04:00 committed by GitHub
parent 2d57d69b9e
commit 00c5ae7983
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
9 changed files with 91 additions and 119 deletions

View File

@ -30,63 +30,6 @@ const chrome = {
} }
}; };
class Image {
constructor() {
this._src = '';
this._loadCallbacks = [];
}
get src() {
return this._src;
}
set src(value) {
this._src = value;
this._delayTriggerLoad();
}
get naturalWidth() {
return 100;
}
get naturalHeight() {
return 100;
}
addEventListener(eventName, callback) {
if (eventName === 'load') {
this._loadCallbacks.push(callback);
}
}
removeEventListener(eventName, callback) {
if (eventName === 'load') {
const index = this._loadCallbacks.indexOf(callback);
if (index >= 0) {
this._loadCallbacks.splice(index, 1);
}
}
}
removeAttribute() {
// NOP
}
async _delayTriggerLoad() {
await Promise.resolve();
for (const callback of this._loadCallbacks) {
callback();
}
}
}
class Blob {
constructor(array, options) {
this._array = array;
this._options = options;
}
}
async function fetch(url2) { async function fetch(url2) {
const extDir = path.join(__dirname, '..', 'ext'); const extDir = path.join(__dirname, '..', 'ext');
let filePath; let filePath;
@ -114,8 +57,6 @@ class DatabaseVM extends VM {
constructor(globals={}) { constructor(globals={}) {
super(Object.assign({ super(Object.assign({
chrome, chrome,
Image,
Blob,
fetch, fetch,
indexedDB: global.indexedDB, indexedDB: global.indexedDB,
IDBKeyRange: global.IDBKeyRange, IDBKeyRange: global.IDBKeyRange,
@ -127,6 +68,14 @@ class DatabaseVM extends VM {
} }
} }
class DatabaseVMDictionaryImporterMediaLoader {
async getImageResolution() {
// Placeholder values
return {width: 100, height: 100};
}
}
module.exports = { module.exports = {
DatabaseVM DatabaseVM,
DatabaseVMDictionaryImporterMediaLoader
}; };

View File

@ -18,7 +18,7 @@
const fs = require('fs'); const fs = require('fs');
const path = require('path'); const path = require('path');
const assert = require('assert'); const assert = require('assert');
const {DatabaseVM} = require('./database-vm'); const {DatabaseVM, DatabaseVMDictionaryImporterMediaLoader} = require('./database-vm');
const {createDictionaryArchive} = require('./util'); const {createDictionaryArchive} = require('./util');
function clone(value) { function clone(value) {
@ -75,7 +75,8 @@ class TranslatorVM extends DatabaseVM {
const testDictionaryContent = await testDictionary.generateAsync({type: 'arraybuffer'}); const testDictionaryContent = await testDictionary.generateAsync({type: 'arraybuffer'});
// Setup database // Setup database
const dictionaryImporter = new DictionaryImporter(); const dictionaryImporterMediaLoader = new DatabaseVMDictionaryImporterMediaLoader();
const dictionaryImporter = new DictionaryImporter(dictionaryImporterMediaLoader);
const dictionaryDatabase = new DictionaryDatabase(); const dictionaryDatabase = new DictionaryDatabase();
await dictionaryDatabase.prepare(); await dictionaryDatabase.prepare();

View File

@ -116,7 +116,7 @@ function deepStrictEqual(actual, expected) {
} }
function createURLClass(urlMap) { function createURLClass() {
const BaseURL = URL; const BaseURL = URL;
const result = function URL(url) { const result = function URL(url) {
const u = new BaseURL(url); const u = new BaseURL(url);
@ -133,23 +133,13 @@ function createURLClass(urlMap) {
this.searchParams = u.searchParams; this.searchParams = u.searchParams;
this.username = u.username; this.username = u.username;
}; };
result.createObjectURL = (object) => {
const id = crypto.randomBytes(16).toString('hex');
const url = `blob:${id}`;
urlMap.set(url, object);
return url;
};
result.revokeObjectURL = (url) => {
urlMap.delete(url);
};
return result; return result;
} }
class VM { class VM {
constructor(context={}) { constructor(context={}) {
this._urlMap = new Map(); context.URL = createURLClass();
context.URL = createURLClass(this._urlMap);
context.crypto = { context.crypto = {
getRandomValues: (array) => { getRandomValues: (array) => {
const buffer = crypto.randomBytes(array.byteLength); const buffer = crypto.randomBytes(array.byteLength);
@ -205,10 +195,6 @@ class VM {
return single ? results[0] : results; return single ? results[0] : results;
} }
getUrlObject(url) {
return this._urlMap.get(url);
}
} }

View File

@ -0,0 +1,57 @@
/*
* Copyright (C) 2021 Yomichan Authors
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <https://www.gnu.org/licenses/>.
*/
/* global
* MediaUtil
*/
/**
* Class used for loading and validating media during the dictionary import process.
*/
class DictionaryImporterMediaLoader {
/**
* Attempts to load an image using a base64 encoded content and a media type
* and returns its resolution.
* @param mediaType The media type for the image content.
* @param content The binary content for the image, encoded in base64.
* @returns A Promise which resolves with {width, height} on success,
* otherwise an error is thrown.
*/
getImageResolution(mediaType, content) {
return new Promise((resolve, reject) => {
const image = new Image();
const eventListeners = new EventListenerCollection();
const cleanup = () => {
image.removeAttribute('src');
URL.revokeObjectURL(url);
eventListeners.removeAllEventListeners();
};
eventListeners.addEventListener(image, 'load', () => {
const {naturalWidth: width, naturalHeight: height} = image;
cleanup();
resolve({width, height});
}, false);
eventListeners.addEventListener(image, 'error', () => {
cleanup();
reject(new Error('Image failed to load'));
}, false);
const blob = MediaUtil.createBlobFromBase64Content(content, mediaType);
const url = URL.createObjectURL(blob);
image.src = url;
});
}
}

View File

@ -22,11 +22,12 @@
*/ */
class DictionaryImporter { class DictionaryImporter {
constructor() { constructor(mediaLoader) {
this._mediaLoader = mediaLoader;
this._schemas = new Map(); this._schemas = new Map();
} }
async importDictionary(dictionaryDatabase, archiveSource, details, onProgress) { async importDictionary(dictionaryDatabase, archiveContent, details, onProgress) {
if (!dictionaryDatabase) { if (!dictionaryDatabase) {
throw new Error('Invalid database'); throw new Error('Invalid database');
} }
@ -37,7 +38,7 @@ class DictionaryImporter {
const hasOnProgress = (typeof onProgress === 'function'); const hasOnProgress = (typeof onProgress === 'function');
// Read archive // Read archive
const archive = await JSZip.loadAsync(archiveSource); const archive = await JSZip.loadAsync(archiveContent);
// Read and validate index // Read and validate index
const indexFileName = 'index.json'; const indexFileName = 'index.json';
@ -469,7 +470,7 @@ class DictionaryImporter {
let width; let width;
let height; let height;
try { try {
({width, height} = await this._getImageResolution(mediaType, content)); ({width, height} = await this._mediaLoader.getImageResolution(mediaType, content));
} catch (e) { } catch (e) {
throw createError('Could not load image'); throw createError('Could not load image');
} }
@ -502,36 +503,4 @@ class DictionaryImporter {
} }
return await response.json(); return await response.json();
} }
/**
* Attempts to load an image using a base64 encoded content and a media type
* and returns its resolution.
* @param mediaType The media type for the image content.
* @param content The binary content for the image, encoded in base64.
* @returns A Promise which resolves with {width, height} on success,
* otherwise an error is thrown.
*/
_getImageResolution(mediaType, content) {
return new Promise((resolve, reject) => {
const image = new Image();
const eventListeners = new EventListenerCollection();
const cleanup = () => {
image.removeAttribute('src');
URL.revokeObjectURL(url);
eventListeners.removeAllEventListeners();
};
eventListeners.addEventListener(image, 'load', () => {
const {naturalWidth: width, naturalHeight: height} = image;
cleanup();
resolve({width, height});
}, false);
eventListeners.addEventListener(image, 'error', () => {
cleanup();
reject(new Error('Image failed to load'));
}, false);
const blob = MediaUtil.createBlobFromBase64Content(content, mediaType);
const url = URL.createObjectURL(blob);
image.src = url;
});
}
} }

View File

@ -19,6 +19,7 @@
* DictionaryController * DictionaryController
* DictionaryDatabase * DictionaryDatabase
* DictionaryImporter * DictionaryImporter
* DictionaryImporterMediaLoader
*/ */
class DictionaryImportController { class DictionaryImportController {
@ -184,7 +185,8 @@ class DictionaryImportController {
async _importDictionary(file, importDetails, onProgress) { async _importDictionary(file, importDetails, onProgress) {
const dictionaryDatabase = await this._getPreparedDictionaryDatabase(); const dictionaryDatabase = await this._getPreparedDictionaryDatabase();
try { try {
const dictionaryImporter = new DictionaryImporter(); const dictionaryImporterMediaLoader = new DictionaryImporterMediaLoader();
const dictionaryImporter = new DictionaryImporter(dictionaryImporterMediaLoader);
const archiveContent = await this._readFile(file); const archiveContent = await this._readFile(file);
const {result, errors} = await dictionaryImporter.importDictionary(dictionaryDatabase, archiveContent, importDetails, onProgress); const {result, errors} = await dictionaryImporter.importDictionary(dictionaryDatabase, archiveContent, importDetails, onProgress);
yomichan.api.triggerDatabaseUpdated('dictionary', 'import'); yomichan.api.triggerDatabaseUpdated('dictionary', 'import');

View File

@ -3451,6 +3451,7 @@
<script src="/js/input/hotkey-util.js"></script> <script src="/js/input/hotkey-util.js"></script>
<script src="/js/language/dictionary-database.js"></script> <script src="/js/language/dictionary-database.js"></script>
<script src="/js/language/dictionary-importer.js"></script> <script src="/js/language/dictionary-importer.js"></script>
<script src="/js/language/dictionary-importer-media-loader.js"></script>
<script src="/js/language/sandbox/dictionary-data-util.js"></script> <script src="/js/language/sandbox/dictionary-data-util.js"></script>
<script src="/js/language/sandbox/japanese-util.js"></script> <script src="/js/language/sandbox/japanese-util.js"></script>
<script src="/js/media/audio-system.js"></script> <script src="/js/media/audio-system.js"></script>

View File

@ -409,6 +409,7 @@
<script src="/js/input/hotkey-util.js"></script> <script src="/js/input/hotkey-util.js"></script>
<script src="/js/language/dictionary-database.js"></script> <script src="/js/language/dictionary-database.js"></script>
<script src="/js/language/dictionary-importer.js"></script> <script src="/js/language/dictionary-importer.js"></script>
<script src="/js/language/dictionary-importer-media-loader.js"></script>
<script src="/js/media/media-util.js"></script> <script src="/js/media/media-util.js"></script>
<script src="/js/pages/settings/dictionary-controller.js"></script> <script src="/js/pages/settings/dictionary-controller.js"></script>
<script src="/js/pages/settings/dictionary-import-controller.js"></script> <script src="/js/pages/settings/dictionary-import-controller.js"></script>

View File

@ -18,7 +18,7 @@
const path = require('path'); const path = require('path');
const assert = require('assert'); const assert = require('assert');
const {createDictionaryArchive, testMain} = require('../dev/util'); const {createDictionaryArchive, testMain} = require('../dev/util');
const {DatabaseVM} = require('../dev/database-vm'); const {DatabaseVM, DatabaseVMDictionaryImporterMediaLoader} = require('../dev/database-vm');
const vm = new DatabaseVM(); const vm = new DatabaseVM();
@ -41,6 +41,12 @@ function createTestDictionaryArchive(dictionary, dictionaryName) {
} }
function createDictionaryImporter() {
const dictionaryImporterMediaLoader = new DatabaseVMDictionaryImporterMediaLoader();
return new DictionaryImporter(dictionaryImporterMediaLoader);
}
function countDictionaryDatabaseEntriesWithTerm(dictionaryDatabaseEntries, term) { function countDictionaryDatabaseEntriesWithTerm(dictionaryDatabaseEntries, term) {
return dictionaryDatabaseEntries.reduce((i, v) => (i + (v.term === term ? 1 : 0)), 0); return dictionaryDatabaseEntries.reduce((i, v) => (i + (v.term === term ? 1 : 0)), 0);
} }
@ -125,7 +131,7 @@ async function testDatabase1() {
]; ];
// Setup database // Setup database
const dictionaryImporter = new DictionaryImporter(); const dictionaryImporter = createDictionaryImporter();
const dictionaryDatabase = new DictionaryDatabase(); const dictionaryDatabase = new DictionaryDatabase();
await dictionaryDatabase.prepare(); await dictionaryDatabase.prepare();
@ -775,7 +781,7 @@ async function testDatabase2() {
]); ]);
// Setup database // Setup database
const dictionaryImporter = new DictionaryImporter(); const dictionaryImporter = createDictionaryImporter();
const dictionaryDatabase = new DictionaryDatabase(); const dictionaryDatabase = new DictionaryDatabase();
// Error: not prepared // Error: not prepared
@ -817,7 +823,7 @@ async function testDatabase3() {
]; ];
// Setup database // Setup database
const dictionaryImporter = new DictionaryImporter(); const dictionaryImporter = createDictionaryImporter();
const dictionaryDatabase = new DictionaryDatabase(); const dictionaryDatabase = new DictionaryDatabase();
await dictionaryDatabase.prepare(); await dictionaryDatabase.prepare();