From 20177116e7e96a51a51540541edb95a32762cd3c Mon Sep 17 00:00:00 2001 From: Aiosa <469130@mail.muni.cz> Date: Tue, 22 Oct 2024 17:25:02 +0200 Subject: [PATCH] Integration tests: bugfixing of manipulation of tiles that share data: when tiles are loaded, when tiles are processed, also await async data preparation befre finishing the invalidation event. --- src/htmldrawer.js | 7 +- src/openseadragon.js | 16 +- src/tile.js | 14 +- src/tilecache.js | 28 ++-- src/tiledimage.js | 21 ++- src/viewer.js | 11 +- src/world.js | 66 ++++---- test/helpers/test.js | 24 +++ test/modules/data-manipulation.js | 259 ++++++++++++++++++++++++++++++ test/modules/drawer.js | 3 +- test/modules/tilecache.js | 1 - test/test.html | 1 + 12 files changed, 383 insertions(+), 68 deletions(-) create mode 100644 test/modules/data-manipulation.js diff --git a/src/htmldrawer.js b/src/htmldrawer.js index c4d07b67..f13856de 100644 --- a/src/htmldrawer.js +++ b/src/htmldrawer.js @@ -146,8 +146,7 @@ class HTMLDrawer extends OpenSeadragon.DrawerBase{ * @returns {Element} the div to draw into */ _createDrawingElement(){ - let canvas = $.makeNeutralElement("div"); - return canvas; + return $.makeNeutralElement("div"); } /** @@ -259,6 +258,10 @@ class HTMLDrawer extends OpenSeadragon.DrawerBase{ // content during animation of the container size. const dataObject = this.getDataToDraw(tile); + if (!dataObject) { + return; + } + if ( dataObject.element.parentNode !== container ) { container.appendChild( dataObject.element ); } diff --git a/src/openseadragon.js b/src/openseadragon.js index 1fa8bb1c..67a78028 100644 --- a/src/openseadragon.js +++ b/src/openseadragon.js @@ -2367,6 +2367,14 @@ function OpenSeadragon( options ){ }, /** + * Makes an AJAX request. + * @param {String} url - the url to request + * @param {Function} onSuccess + * @param {Function} onError + * @throws {Error} + * @returns {XMLHttpRequest} + * @deprecated deprecated way of calling this function + *//** * Makes an AJAX request. * @param {Object} options * @param {String} options.url - the url to request @@ -2379,14 +2387,6 @@ function OpenSeadragon( options ){ * @param {Boolean} [options.withCredentials=false] - whether to set the XHR's withCredentials * @throws {Error} * @returns {XMLHttpRequest} - *//** - * Makes an AJAX request. - * @param {String} url - the url to request - * @param {Function} onSuccess - * @param {Function} onError - * @throws {Error} - * @returns {XMLHttpRequest} - * @deprecated deprecated way of calling this function */ makeAjaxRequest: function( url, onSuccess, onError ) { var withCredentials; diff --git a/src/tile.js b/src/tile.js index bb7e35da..6e03affb 100644 --- a/src/tile.js +++ b/src/tile.js @@ -550,7 +550,10 @@ $.Tile.prototype = { /** * Optimizazion: prepare target cache for subsequent use in rendering, and perform updateRenderTarget() + * The main idea of this function is that it must be ASYNCHRONOUS since there might be additional processing + * happening due to underlying drawer requirements. * @private + * @return {OpenSeadragon.Promise>} */ updateRenderTargetWithDataTransform: function (drawerId, supportedFormats, usePrivateCache) { // Now, if working cache exists, we set main cache to the working cache --> prepare @@ -570,17 +573,20 @@ $.Tile.prototype = { return cache.prepareForRendering(drawerId, supportedFormats, usePrivateCache, this.processing); } - return null; + return $.Promise.resolve(); }, /** * Resolves render target: changes might've been made to the rendering pipeline: * - working cache is set: make sure main cache will be replaced * - working cache is unset: make sure main cache either gets updated to original data or stays (based on this.__restore) + * + * The main idea of this function is that it is SYNCHRONOUS, e.g. can perform in-place cache swap to update + * before any rendering occurs. * @private * @return */ - updateRenderTarget: function () { + updateRenderTarget: function (_allowTileNotLoaded = false) { // Check if we asked for restore, and make sure we set it to false since we update the whole cache state const requestedRestore = this.__restore; this.__restore = false; @@ -593,7 +599,8 @@ $.Tile.prototype = { this.tiledImage._tileCache.consumeCache({ tile: this, victimKey: this._wcKey, - consumerKey: newCacheKey + consumerKey: newCacheKey, + tileAllowNotLoaded: _allowTileNotLoaded }); this.cacheKey = newCacheKey; return; @@ -831,7 +838,6 @@ $.Tile.prototype = { } // Working key is never updated, it will be invalidated (but do not dereference cache, just fix the pointers) this._caches[newKey] = cache; - cache.AAA = true; delete this._caches[oldKey]; }, diff --git a/src/tilecache.js b/src/tilecache.js index 87aca50c..5889f9c5 100644 --- a/src/tilecache.js +++ b/src/tilecache.js @@ -238,6 +238,7 @@ } } } + return this; }; // if not internal copy and we have no data, or we are ready to render, exit @@ -273,8 +274,7 @@ const selectedFormat = conversionPath[conversionPath.length - 1].target.value; return $.convertor.convert(this._tRef, this.data, this.type, selectedFormat).then(data => { internalCache.setDataAs(data, selectedFormat); // synchronous, SimpleCacheRecord - fin(); - return this; + return fin(); }); } @@ -372,11 +372,10 @@ // make sure this gets destroyed even if loaded=false if (this.loaded) { this._destroySelfUnsafe(this._data, this._type); - } else { + } else if (this._promise) { const oldType = this._type; this._promise.then(x => this._destroySelfUnsafe(x, oldType)); } - this.loaded = false; } } @@ -389,6 +388,7 @@ if (!this._destroyed) { return; } + this.loaded = false; this._tiles = null; this._data = null; this._type = null; @@ -922,12 +922,14 @@ * inheriting its tiles and key. * @param {String} options.consumerKey - The cache that consumes the victim. In fact, it gets destroyed and * replaced by victim, which inherits all its metadata. - * @param {} + * @param {Boolean} options.tileAllowNotLoaded - if true, tile that is not loaded is also processed, + * this is internal parameter used in tile-loaded completion routine, as we need to prepare tile but + * it is not yet loaded and cannot be marked as so (otherwise the system would think it is ready) */ consumeCache(options) { const victim = this._cachesLoaded[options.victimKey], tile = options.tile; - if (!victim || (!tile.loaded && !tile.loading)) { + if (!victim || (!options.tileAllowNotLoaded && !tile.loaded && !tile.loading)) { $.console.warn("Attempt to consume non-existent cache: this is probably a bug!"); return; } @@ -935,11 +937,13 @@ let tiles = [...tile.getCache()._tiles]; if (consumer) { - // We need to avoid costly conversions: replace consumer. - // unloadCacheForTile() will modify the array, iterate over a copy - const iterateTiles = [...consumer._tiles]; + // We need to avoid async execution here: replace consumer instead of overwriting the data. + const iterateTiles = [...consumer._tiles]; // unloadCacheForTile() will modify the array, use a copy for (let tile of iterateTiles) { - this.unloadCacheForTile(tile, options.consumerKey, true); + if (tile.loaded) { + this.unloadCacheForTile(tile, options.consumerKey, true); + } + } } // Just swap victim to become new consumer @@ -951,8 +955,10 @@ if (resultCache) { // Only one cache got working item, other caches were idle: update cache: add the new cache // we can add since we removed above with unloadCacheForTile() + // Loading tiles are also accepted, since they might be in the process of finishing. However, + // note that they are not part of the unloading process above! for (let tile of tiles) { - if (tile !== options.tile && tile.loaded) { + if (tile !== options.tile && (tile.loaded || tile.loading)) { tile.addCache(options.consumerKey, resultCache.data, resultCache.type, true, false); } } diff --git a/src/tiledimage.js b/src/tiledimage.js index e5fd7424..43b5ef80 100644 --- a/src/tiledimage.js +++ b/src/tiledimage.js @@ -291,7 +291,7 @@ $.extend($.TiledImage.prototype, $.EventSource.prototype, /** @lends OpenSeadrag requestInvalidate: function (viewportOnly, tStamp, restoreTiles = true) { tStamp = tStamp || $.now(); const tiles = viewportOnly ? this._lastDrawn.map(x => x.tile) : this._tileCache.getLoadedTilesFor(this); - this.viewer.world.requestTileInvalidateEvent(tiles, tStamp, restoreTiles); + return this.viewer.world.requestTileInvalidateEvent(tiles, tStamp, restoreTiles); }, /** @@ -2106,7 +2106,12 @@ $.extend($.TiledImage.prototype, $.EventSource.prototype, /** @lends OpenSeadrag _setTileLoaded: function(tile, data, cutoff, tileRequest, dataType) { tile.tiledImage = this; //unloaded with tile.unload(), so we need to set it back // does nothing if tile.cacheKey already present - tile.addCache(tile.cacheKey, data, dataType, false, false); + + let tileCacheCreated = false; + tile.addCache(tile.cacheKey, () => { + tileCacheCreated = true; + return data; + }, dataType, false, false); let resolver = null, increment = 0, @@ -2125,7 +2130,7 @@ $.extend($.TiledImage.prototype, $.EventSource.prototype, /** @lends OpenSeadrag tile.hasTransparency = tile.hasTransparency || _this.source.hasTransparency( undefined, tile.getUrl(), tile.ajaxHeaders, tile.postData ); - tile.updateRenderTarget(); + tile.updateRenderTarget(true); //make sure cache data is ready for drawing, if not, request the desired format const cache = tile.getCache(tile.cacheKey), requiredTypes = _this._drawer.getSupportedDataFormats(); @@ -2162,6 +2167,16 @@ $.extend($.TiledImage.prototype, $.EventSource.prototype, /** @lends OpenSeadrag const fallbackCompletion = getCompletionCallback(); + if (!tileCacheCreated) { + // Tile-loaded not called on each tile, but only on tiles with new data! Verify we share the main cache + + // We could attempt to initialize the tile here (e.g. find another tile that has same key and if + // we find it in different main cache, we try to share it with current tile, but this process + // is also happening within tile cache logics (see last part of consumeCache(..)). + fallbackCompletion(); + return; + } + /** * Triggered when a tile has just been loaded in memory. That means that the * image has been downloaded and can be modified before being drawn to the canvas. diff --git a/src/viewer.js b/src/viewer.js index 2971f781..43cc05d0 100644 --- a/src/viewer.js +++ b/src/viewer.js @@ -769,18 +769,21 @@ $.extend( $.Viewer.prototype, $.EventSource.prototype, $.ControlDock.prototype, * @function * @param {Boolean} [restoreTiles=true] if true, tile processing starts from the tile original data * @fires OpenSeadragon.Viewer.event:tile-invalidated + * @return {OpenSeadragon.Promise>} */ requestInvalidate: function (restoreTiles = true) { if ( !THIS[ this.hash ] ) { //this viewer has already been destroyed: returning immediately - return; + return $.Promise.resolve(); } const tStamp = $.now(); - this.world.requestInvalidate(tStamp, restoreTiles); - if (this.navigator) { - this.navigator.world.requestInvalidate(tStamp, restoreTiles); + const worldPromise = this.world.requestInvalidate(tStamp, restoreTiles); + if (!this.navigator) { + return worldPromise; } + const navigatorPromise = this.navigator.world.requestInvalidate(tStamp, restoreTiles); + return $.Promise.all([worldPromise, navigatorPromise]); }, diff --git a/src/world.js b/src/world.js index 1b346456..33b85f78 100644 --- a/src/world.js +++ b/src/world.js @@ -242,16 +242,16 @@ $.extend( $.World.prototype, $.EventSource.prototype, /** @lends OpenSeadragon.W * @param {Boolean} [restoreTiles=true] if true, tile processing starts from the tile original data * @function * @fires OpenSeadragon.Viewer.event:tile-invalidated + * @return {OpenSeadragon.Promise>} */ requestInvalidate: function (tStamp, restoreTiles = true) { $.__updated = tStamp = tStamp || $.now(); - for ( let i = 0; i < this._items.length; i++ ) { - this._items[i].requestInvalidate(true, tStamp, restoreTiles); - } + + const promises = this._items.map(item => item.requestInvalidate(true, tStamp, restoreTiles)); const tiles = this.viewer.tileCache.getLoadedTilesFor(true); - // Delay processing of all tiles of all items to a later stage by increasing tstamp - this.requestTileInvalidateEvent(tiles, tStamp, restoreTiles); + promises.push(this.requestTileInvalidateEvent(tiles, tStamp, restoreTiles)); + return $.Promise.all(promises); }, /** @@ -263,42 +263,20 @@ $.extend( $.World.prototype, $.EventSource.prototype, /** @lends OpenSeadragon.W * changes are added to the cycle, else they await next iteration * @param {Boolean} [restoreTiles=true] if true, tile processing starts from the tile original data * @fires OpenSeadragon.Viewer.event:tile-invalidated + * @return {OpenSeadragon.Promise>} */ requestTileInvalidateEvent: function(tileList, tStamp, restoreTiles = true) { if (tileList.length < 1) { - return; + return $.Promise.resolve(); } if (this._queuedInvalidateTiles.length) { this._queuedInvalidateTiles.push(tileList); - return; + return $.Promise.resolve(); } // this.viewer.viewer is defined in navigator, ensure we call event on the parent viewer const eventTarget = this.viewer.viewer || this.viewer; - const finish = () => { - for (let tile of tileList) { - // pass update stamp on the new cache object to avoid needless updates - const newCache = tile.getCache(); - if (newCache) { - - newCache._updateStamp = tStamp; - for (let t of newCache._tiles) { - // Mark all as processing - t.processing = false; - } - } - } - - if (this._queuedInvalidateTiles.length) { - // Make space for other logics execution before we continue in processing - let list = this._queuedInvalidateTiles.splice(0, 1)[0]; - this.requestTileInvalidateEvent(list, tStamp, restoreTiles); - } else { - this.draw(); - } - }; - const supportedFormats = eventTarget.drawer.getSupportedDataFormats(); const keepInternalCacheCopy = eventTarget.drawer.options.usePrivateCache; const drawerId = eventTarget.drawer.getId(); @@ -320,7 +298,7 @@ $.extend( $.World.prototype, $.EventSource.prototype, /** @lends OpenSeadragon.W return true; }); - $.Promise.all(tileList.map(tile => { + const jobList = tileList.map(tile => { if (restoreTiles) { tile.restore(); } @@ -328,9 +306,31 @@ $.extend( $.World.prototype, $.EventSource.prototype, /** @lends OpenSeadragon.W tile: tile, tiledImage: tile.tiledImage, }).then(() => { - tile.updateRenderTargetWithDataTransform(drawerId, supportedFormats, keepInternalCacheCopy); + // asynchronous finisher + return tile.updateRenderTargetWithDataTransform(drawerId, supportedFormats, keepInternalCacheCopy); + }).catch(e => { + $.console.error("Update routine error:", e); }); - })).catch(finish).then(finish); + }); + + return $.Promise.all(jobList).then(() => { + for (let tile of tileList) { + // pass update stamp on the new cache object to avoid needless updates + const newCache = tile.getCache(); + if (newCache) { + newCache._updateStamp = tStamp; + tile.processing = false; + } + } + + if (this._queuedInvalidateTiles.length) { + // Make space for other logics execution before we continue in processing + let list = this._queuedInvalidateTiles.splice(0, 1)[0]; + this.requestTileInvalidateEvent(list, tStamp, restoreTiles); + } else { + this.draw(); + } + }); }, /** diff --git a/test/helpers/test.js b/test/helpers/test.js index d0251401..dcc6c128 100644 --- a/test/helpers/test.js +++ b/test/helpers/test.js @@ -240,5 +240,29 @@ }; OpenSeadragon.console = testConsole; + + OpenSeadragon.getBuiltInDrawersForTest = function() { + const drawers = []; + for (let property in OpenSeadragon) { + const drawer = OpenSeadragon[ property ], + proto = drawer.prototype; + if( proto && + proto instanceof OpenSeadragon.DrawerBase && + $.isFunction( proto.getType )){ + drawers.push(proto.getType.call( drawer )); + } + } + return drawers; + }; + + OpenSeadragon.Viewer.prototype.waitForFinishedJobsForTest = function () { + let finish; + let int = setInterval(() => { + if (this.imageLoader.jobsInProgress < 1) { + finish(); + } + }, 50); + return new OpenSeadragon.Promise((resolve) => finish = resolve); + }; } )(); diff --git a/test/modules/data-manipulation.js b/test/modules/data-manipulation.js new file mode 100644 index 00000000..3b10e504 --- /dev/null +++ b/test/modules/data-manipulation.js @@ -0,0 +1,259 @@ +/* global QUnit, testLog */ + +(function() { + + let viewer; + QUnit.module(`Data Manipulation Across Drawers`, { + beforeEach: function () { + $('
').appendTo("#qunit-fixture"); + testLog.reset(); + }, + afterEach: function () { + if (viewer && viewer.close) { + viewer.close(); + } + + viewer = null; + } + }); + + OpenSeadragon.getBuiltInDrawersForTest().forEach(testDrawer); + // If yuu want to debug a specific drawer, use instead: + // ['webgl'].forEach(testDrawer); + + function getPluginCode(overlayColor = "rgba(0,0,255,0.5)") { + return async function(e) { + const tile = e.tile; + const ctx = await tile.getData('context2d'), canvas = ctx.canvas; + ctx.fillStyle = overlayColor; + ctx.fillRect(0, 0, canvas.width, canvas.height); + await tile.setData(ctx, 'context2d'); + }; + } + + function getResetTileDataCode() { + return async function(e) { + const tile = e.tile; + tile.restore(); + }; + } + + function getTileDescription(t) { + return `${t.level}/${t.x}-${t.y}`; + } + + + function testDrawer(type) { + + function whiteViewport() { + viewer = OpenSeadragon({ + id: 'example', + prefixUrl: '/build/openseadragon/images/', + maxImageCacheCount: 200, + springStiffness: 100, + drawer: type + }); + + viewer.open({ + width: 24, + height: 24, + tileSize: 24, + minLevel: 1, + + // This is a crucial test feature: all tiles share the same URL, so there are plenty collisions + getTileUrl: (x, y, l) => "", + getTilePostData: () => "", + downloadTileStart: (context) => { + const canvas = document.createElement("canvas"); + const ctx = canvas.getContext("2d"); + canvas.width = context.tile.size.x; + canvas.height = context.tile.size.y; + ctx.fillStyle = "#ffffff"; + ctx.fillRect(0, 0, context.tile.size.x, context.tile.size.y); + + context.finish(ctx, null, "context2d"); + } + }); + } + const sleep = (ms) => new Promise((resolve) => setTimeout(resolve, ms)) + + // we test middle of the canvas, so that we can test both tiles or the output canvas of canvas drawer :) + async function readTileData() { + if (type === "canvas") { + //test with the underlying canvas instead + const canvas = viewer.drawer.canvas; + return viewer.drawer.canvas.getContext("2d").getImageData(canvas.width/2, canvas.height/2, 1, 1); + } + + await sleep(200); + + //else incompatible drawer for data getting + const tile = viewer.world.getItemAt(0).getTilesToDraw()[0]; + const cache = tile.tile.getCache(); + if (!cache || !cache.loaded) return null; + + const ctx = await cache.getDataAs("context2d"); + if (!ctx) return null; + return ctx.getImageData(ctx.canvas.width/2, ctx.canvas.height/2, 1, 1) + } + + QUnit.test(type + ' drawer: basic scenario.', function(assert) { + whiteViewport(); + const done = assert.async(); + const fnA = getPluginCode("rgba(0,0,255,1)"); + const fnB = getPluginCode("rgba(255,0,0,1)"); + + const crashTest = () => assert.ok(false, "Tile Invalidated event should not be called"); + + viewer.addHandler('tile-loaded', fnA); + viewer.addHandler('tile-invalidated', crashTest); + viewer.addHandler('tile-loaded', fnB); + viewer.addHandler('tile-invalidated', crashTest); + + viewer.addHandler('open', async () => { + await viewer.waitForFinishedJobsForTest(); + let data = await readTileData(); + assert.equal(data.data[0], 255); + assert.equal(data.data[1], 0); + assert.equal(data.data[2], 0); + assert.equal(data.data[3], 255); + + // Thorough testing of the cache state + for (let tile of viewer.tileCache._tilesLoaded) { + + const caches = Object.entries(tile._caches); + assert.equal(caches.length, 2, `Tile ${getTileDescription(tile)} has only two caches - main & original`); + for (let [key, value] of caches) { + assert.ok(value.loaded, `Attached cache '${key}' is ready.`); + assert.notOk(value._destroyed, `Attached cache '${key}' is not destroyed.`); + assert.ok(value._tiles.includes(tile), `Attached cache '${key}' reference is bidirectional.`); + } + assert.notOk(tile.getCache(tile._wcKey), "Tile cache working key is unset"); + } + + done(); + }); + }); + + QUnit.test(type + ' drawer: basic scenario with priorities + events addition.', function(assert) { + whiteViewport(); + const done = assert.async(); + // FNA gets applied last since it has low priority + const fnA = getPluginCode("rgba(0,0,255,1)"); + const fnB = getPluginCode("rgba(255,0,0,1)"); + + viewer.addHandler('tile-loaded', fnA); + viewer.addHandler('tile-invalidated', fnA); + viewer.addHandler('tile-loaded', fnB, null, 1); + viewer.addHandler('tile-invalidated', fnB, null, 1); + // const promise = viewer.requestInvalidate(); + + viewer.addHandler('open', async () => { + await viewer.waitForFinishedJobsForTest(); + + let data = await readTileData(); + assert.equal(data.data[0], 0); + assert.equal(data.data[1], 0); + assert.equal(data.data[2], 255); + assert.equal(data.data[3], 255); + + // Test swap + viewer.addHandler('tile-loaded', fnB); + viewer.addHandler('tile-invalidated', fnB); + await viewer.requestInvalidate(); + + data = await readTileData(); + // suddenly B is applied since it was added with same priority but later + assert.equal(data.data[0], 255); + assert.equal(data.data[1], 0); + assert.equal(data.data[2], 0); + assert.equal(data.data[3], 255); + + // Now B gets applied last! Red + viewer.addHandler('tile-loaded', fnB, null, -1); + viewer.addHandler('tile-invalidated', fnB, null, -1); + await viewer.requestInvalidate(); + // no change + data = await readTileData(); + assert.equal(data.data[0], 255); + assert.equal(data.data[1], 0); + assert.equal(data.data[2], 0); + assert.equal(data.data[3], 255); + + // Thorough testing of the cache state + for (let tile of viewer.tileCache._tilesLoaded) { + + const caches = Object.entries(tile._caches); + assert.equal(caches.length, 2, `Tile ${getTileDescription(tile)} has only two caches - main & original`); + for (let [key, value] of caches) { + assert.ok(value.loaded, `Attached cache '${key}' is ready.`); + assert.notOk(value._destroyed, `Attached cache '${key}' is not destroyed.`); + assert.ok(value._tiles.includes(tile), `Attached cache '${key}' reference is bidirectional.`); + } + assert.notOk(tile.getCache(tile._wcKey), "Tile cache working key is unset"); + } + + done(); + }); + }); + + QUnit.test(type + ' drawer: one calls tile restore.', function(assert) { + whiteViewport(); + + const done = assert.async(); + const fnA = getPluginCode("rgba(0,255,0,1)"); + const fnB = getResetTileDataCode(); + + viewer.addHandler('tile-loaded', fnA); + viewer.addHandler('tile-invalidated', fnA); + viewer.addHandler('tile-loaded', fnB, null, 1); + viewer.addHandler('tile-invalidated', fnB, null, 1); + // const promise = viewer.requestInvalidate(); + + viewer.addHandler('open', async () => { + await viewer.waitForFinishedJobsForTest(); + + let data = await readTileData(); + assert.equal(data.data[0], 0); + assert.equal(data.data[1], 255); + assert.equal(data.data[2], 0); + assert.equal(data.data[3], 255); + + // Test swap - suddenly B applied since it was added later + viewer.addHandler('tile-loaded', fnB); + viewer.addHandler('tile-invalidated', fnB); + await viewer.requestInvalidate(); + data = await readTileData(); + assert.equal(data.data[0], 255); + assert.equal(data.data[1], 255); + assert.equal(data.data[2], 255); + assert.equal(data.data[3], 255); + + viewer.addHandler('tile-loaded', fnB, null, -1); + viewer.addHandler('tile-invalidated', fnB, null, -1); + await viewer.requestInvalidate(); + data = await readTileData(); + //Erased! + assert.equal(data.data[0], 255); + assert.equal(data.data[1], 255); + assert.equal(data.data[2], 255); + assert.equal(data.data[3], 255); + + // Thorough testing of the cache state + for (let tile of viewer.tileCache._tilesLoaded) { + + const caches = Object.entries(tile._caches); + assert.equal(caches.length, 1, `Tile ${getTileDescription(tile)} has only single, original cache`); + for (let [key, value] of caches) { + assert.ok(value.loaded, `Attached cache '${key}' is ready.`); + assert.notOk(value._destroyed, `Attached cache '${key}' is not destroyed.`); + assert.ok(value._tiles.includes(tile), `Attached cache '${key}' reference is bidirectional.`); + } + assert.notOk(tile.getCache(tile._wcKey), "Tile cache working key is unset"); + } + + done(); + }); + }); + } +}()); diff --git a/test/modules/drawer.js b/test/modules/drawer.js index e7b8e8cf..45b3398d 100644 --- a/test/modules/drawer.js +++ b/test/modules/drawer.js @@ -2,8 +2,7 @@ (function() { var viewer; - const drawerTypes = ['webgl','canvas','html']; - drawerTypes.forEach(runDrawerTests); + OpenSeadragon.getBuiltInDrawersForTest().forEach(runDrawerTests); function runDrawerTests(drawerType){ diff --git a/test/modules/tilecache.js b/test/modules/tilecache.js index e7a86855..8140d70e 100644 --- a/test/modules/tilecache.js +++ b/test/modules/tilecache.js @@ -113,7 +113,6 @@ }); // ---------- - // TODO: this used to be async QUnit.test('basics', function(assert) { const done = assert.async(); const fakeViewer = MockSeadragon.getViewer( diff --git a/test/test.html b/test/test.html index 6c549f85..78ff6f7f 100644 --- a/test/test.html +++ b/test/test.html @@ -66,6 +66,7 @@ +