UNPKG

maplibre-gl

Version:

BSD licensed community fork of mapbox-gl, a WebGL interactive maps library

377 lines (338 loc) 13.9 kB
import {describe, beforeEach, afterEach, test, expect, vi, it} from 'vitest'; import {fakeServer, type FakeServer} from 'nise'; import {RasterDEMTileSource} from './raster_dem_tile_source'; import {OverscaledTileID} from '../tile/tile_id'; import {RequestManager} from '../util/request_manager'; import {type Tile} from '../tile/tile'; import {getMockDispatcher} from '../util/test/util'; import {sleep, waitForEvent, waitForMetadataEvent} from '../util/test/util'; import type {MapSourceDataEvent} from '../ui/events'; function createSource(options, transformCallback?) { const source = new RasterDEMTileSource('id', options, getMockDispatcher(), options.eventedParent); source.onAdd({ transform: {angle: 0, pitch: 0, showCollisionBoxes: false}, _getMapId: () => 1, _requestManager: new RequestManager(transformCallback), getPixelRatio() { return 1; } } as any); source.on('error', (e) => { throw e.error; }); return source; } describe('RasterDEMTileSource', () => { let server: FakeServer; beforeEach(() => { global.fetch = null; server = fakeServer.create(); }); afterEach(() => { server.restore(); }); test('transforms request for TileJSON URL', () => { server.respondWith('/source.json', JSON.stringify({ minzoom: 0, maxzoom: 22, attribution: 'MapLibre', tiles: ['http://example.com/{z}/{x}/{y}.pngraw'], bounds: [-47, -7, -45, -5] })); const transformSpy = vi.fn().mockImplementation((url) => { return {url}; }); createSource({url: '/source.json'}, transformSpy); server.respond(); expect(transformSpy.mock.calls[0][0]).toBe('/source.json'); expect(transformSpy.mock.calls[0][1]).toBe('Source'); }); test('can asynchronously transform request for TileJSON URL', async () => { server.respondWith('/source.json', JSON.stringify({ minzoom: 0, maxzoom: 22, attribution: 'MapLibre', tiles: ['http://example.com/{z}/{x}/{y}.pngraw'], bounds: [-47, -7, -45, -5] })); const source = createSource({url: '/source.json'}, async (url) => ({ url, headers: {Authorization: 'Bearer token'} })); const promise = waitForMetadataEvent(source); await sleep(0); server.respond(); await promise; expect(server.requests[0].url).toBe('/source.json'); expect(server.requests[0].requestHeaders.Authorization).toBe('Bearer token'); }); test('transforms tile urls before requesting', async () => { server.respondWith('/source.json', JSON.stringify({ minzoom: 0, maxzoom: 22, attribution: 'MapLibre', tiles: ['http://example.com/{z}/{x}/{y}.png'], bounds: [-47, -7, -45, -5] })); const source = createSource({url: '/source.json'}); const transformSpy = vi.spyOn(source.map._requestManager, 'transformRequest'); const promise = waitForMetadataEvent(source); await sleep(0); server.respond(); await promise; const tile = { tileID: new OverscaledTileID(10, 0, 10, 5, 5), state: 'loading', loadVectorData () {}, setExpiryData() {} } as any as Tile; source.loadTile(tile); expect(transformSpy).toHaveBeenCalledTimes(1); expect(transformSpy.mock.calls[0][0]).toBe('http://example.com/10/5/5.png'); expect(transformSpy.mock.calls[0][1]).toBe('Tile'); }); test('can asynchronously transform tile request', async () => { server.respondWith('http://example.com/10/5/5.png', [200, {'Content-Type': 'image/png', 'Content-Length': 1, 'Cache-Control': 'max-age=100'}, '0'] ); const source = createSource({ tiles: ['http://example.com/{z}/{x}/{y}.png'] }, async (url) => ({ url, headers: {Authorization: 'Bearer token'} })); source.map.painter = {context: {}, getTileTexture: () => { return {update: () => {}}; }} as any; await waitForMetadataEvent(source); const tile = { tileID: new OverscaledTileID(10, 0, 10, 5, 5), state: 'loading', setExpiryData() {}, actor: 1 } as any as Tile; const promise = source.loadTile(tile); await sleep(0); server.respond(); await promise; expect(server.requests[0].url).toBe('http://example.com/10/5/5.png'); expect(server.requests[0].requestHeaders.Authorization).toBe('Bearer token'); }); test('populates neighboringTiles', async () => { server.respondWith('/source.json', JSON.stringify({ minzoom: 0, maxzoom: 22, attribution: 'MapLibre', tiles: ['http://example.com/{z}/{x}/{y}.png'] })); const source = createSource({url: '/source.json'}); const promise = waitForMetadataEvent(source); await sleep(0); server.respond(); await promise; const tile = { tileID: new OverscaledTileID(10, 0, 10, 5, 5), state: 'loading', loadVectorData () {}, setExpiryData() {} } as any as Tile; source.loadTile(tile); await sleep(0); expect(Object.keys(tile.neighboringTiles)).toEqual([ new OverscaledTileID(10, 0, 10, 4, 5).key, new OverscaledTileID(10, 0, 10, 6, 5).key, new OverscaledTileID(10, 0, 10, 4, 4).key, new OverscaledTileID(10, 0, 10, 5, 4).key, new OverscaledTileID(10, 0, 10, 6, 4).key, new OverscaledTileID(10, 0, 10, 4, 6).key, new OverscaledTileID(10, 0, 10, 5, 6).key, new OverscaledTileID(10, 0, 10, 6, 6).key ]); }); test('populates neighboringTiles with wrapped tiles', async () => { server.respondWith('/source.json', JSON.stringify({ minzoom: 0, maxzoom: 22, attribution: 'MapLibre', tiles: ['http://example.com/{z}/{x}/{y}.png'] })); const source = createSource({url: '/source.json'}); const promise = waitForMetadataEvent(source); await sleep(0); server.respond(); await promise; const tile = { tileID: new OverscaledTileID(5, 0, 5, 31, 5), state: 'loading', loadVectorData() {}, setExpiryData() {} } as any as Tile; source.loadTile(tile); await sleep(0); expect(Object.keys(tile.neighboringTiles)).toEqual([ new OverscaledTileID(5, 0, 5, 30, 6).key, new OverscaledTileID(5, 0, 5, 31, 6).key, new OverscaledTileID(5, 0, 5, 30, 5).key, new OverscaledTileID(5, 1, 5, 0, 5).key, new OverscaledTileID(5, 0, 5, 30, 4).key, new OverscaledTileID(5, 0, 5, 31, 4).key, new OverscaledTileID(5, 1, 5, 0, 4).key, new OverscaledTileID(5, 1, 5, 0, 6).key ]); }); it('serializes options', () => { const source = createSource({ tiles: ['http://localhost:2900/raster-dem/{z}/{x}/{y}.png'], minzoom: 2, maxzoom: 10 }); expect(source.serialize()).toStrictEqual({ type: 'raster-dem', tiles: ['http://localhost:2900/raster-dem/{z}/{x}/{y}.png'], minzoom: 2, maxzoom: 10 }); }); test('Tile expiry data is set when "Cache-Control" is set but not "Expires"', async () => { server.respondWith('/source.json', JSON.stringify({ minzoom: 0, maxzoom: 22, attribution: 'MapLibre', tiles: ['http://example.com/{z}/{x}/{y}.png'], bounds: [-47, -7, -45, -5] })); server.respondWith('http://example.com/10/5/5.png', [200, {'Content-Type': 'image/png', 'Content-Length': 1, 'Cache-Control': 'max-age=100'}, '0'] ); const source = createSource({url: '/source.json'}); source.map.painter = {context: {}, getTileTexture: () => { return {update: () => {}}; }} as any; source.map._refreshExpiredTiles = true; const promise = waitForEvent(source, 'data', (e: MapSourceDataEvent) => e.sourceDataType === 'metadata'); await sleep(0); server.respond(); await promise; const tile = { tileID: new OverscaledTileID(10, 0, 10, 5, 5), state: 'loading', setExpiryData() {}, actor: source.dispatcher.getActor() } as any as Tile; const expiryDataSpy = vi.spyOn(tile, 'setExpiryData'); const tilePromise = source.loadTile(tile); await sleep(0); server.respond(); await tilePromise; expect(expiryDataSpy).toHaveBeenCalledTimes(1); }); test('Tile expiry data is set when "Expires" is set but not "Cache-Control"', async () => { server.respondWith('/source.json', JSON.stringify({ minzoom: 0, maxzoom: 22, attribution: 'MapLibre', tiles: ['http://example.com/{z}/{x}/{y}.png'], bounds: [-47, -7, -45, -5] })); server.respondWith('http://example.com/10/5/5.png', [200, {'Content-Type': 'image/png', 'Content-Length': 1, 'Expires': 'Wed, 21 Oct 2015 07:28:00 GMT'}, '0'] ); const source = createSource({url: '/source.json'}); source.map.painter = {context: {}, getTileTexture: () => { return {update: () => {}}; }} as any; source.map._refreshExpiredTiles = true; const promise = waitForEvent(source, 'data', (e: MapSourceDataEvent) => e.sourceDataType === 'metadata'); await sleep(0); server.respond(); await promise; const tile = { tileID: new OverscaledTileID(10, 0, 10, 5, 5), state: 'loading', setExpiryData() {}, actor: source.dispatcher.getActor() } as any as Tile; const expiryDataSpy = vi.spyOn(tile, 'setExpiryData'); const tilePromise = source.loadTile(tile); await sleep(0); server.respond(); await tilePromise; expect(expiryDataSpy).toHaveBeenCalledTimes(1); }); test('Tile expiry data is set when "Expires" is set and "Cache-Control" is an empty string', async () => { server.respondWith('/source.json', JSON.stringify({ minzoom: 0, maxzoom: 22, attribution: 'MapLibre', tiles: ['http://example.com/{z}/{x}/{y}.png'], bounds: [-47, -7, -45, -5] })); server.respondWith('http://example.com/10/5/5.png', [200, {'Content-Type': 'image/png', 'Content-Length': 1, 'Cache-Control': '', 'Expires': 'Wed, 21 Oct 2015 07:28:00 GMT'}, '0'] ); const source = createSource({url: '/source.json'}); source.map.painter = {context: {}, getTileTexture: () => { return {update: () => {}}; }} as any; source.map._refreshExpiredTiles = true; const promise = waitForEvent(source, 'data', (e: MapSourceDataEvent) => e.sourceDataType === 'metadata'); await sleep(0); server.respond(); await promise; const tile = { tileID: new OverscaledTileID(10, 0, 10, 5, 5), state: 'loading', setExpiryData() {}, actor: source.dispatcher.getActor() } as any as Tile; const expiryDataSpy = vi.spyOn(tile, 'setExpiryData'); const tilePromise = source.loadTile(tile); await sleep(0); server.respond(); await tilePromise; expect(expiryDataSpy).toHaveBeenCalledTimes(1); }); test('does not throw when tile is aborted', async () => { server.respondWith('/source.json', JSON.stringify({ minzoom: 0, maxzoom: 22, attribution: 'MapLibre', tiles: ['http://example.com/{z}/{x}/{y}.png'] })); const source = createSource({url: '/source.json'}); const promise = waitForMetadataEvent(source); await sleep(0); server.respond(); await promise; const tile = { tileID: new OverscaledTileID(5, 0, 5, 31, 5), state: 'loading', loadVectorData() {}, setExpiryData() {} } as any as Tile; const loadPromise = source.loadTile(tile); await sleep(0); tile.abortController.abort(); tile.aborted = true; await expect(loadPromise).resolves.toBeUndefined(); expect(tile.state).toBe('unloaded'); }); test('reloads tile in reloading state', async () => { server.respondWith('/source.json', JSON.stringify({ minzoom: 0, maxzoom: 22, attribution: 'MapLibre', tiles: ['http://example.com/{z}/{x}/{y}.png'] })); server.respondWith('http://example.com/5/31/5.png', [200, {'Content-Type': 'image/png', 'Content-Length': 1}, '0'] ); const source = createSource({url: '/source.json'}); const promise = waitForMetadataEvent(source); await sleep(0); server.respond(); await promise; const tile = { tileID: new OverscaledTileID(5, 0, 5, 31, 5), state: 'reloading', actor: source.dispatcher.getActor(), loadVectorData() {}, setExpiryData() {} } as any as Tile; const tilePromise = source.loadTile(tile); await sleep(0); server.respond(); await tilePromise; expect(tile.state).toBe('loaded'); }); });