import ImageTile from '../../../../src/ol/ImageTile.js'; import {get as getProjection} from '../../../../src/ol/proj.js'; import TileWMS from '../../../../src/ol/source/TileWMS.js'; import {createXYZ} from '../../../../src/ol/tilegrid.js'; import TileGrid from '../../../../src/ol/tilegrid/TileGrid.js'; describe('ol.source.TileWMS', function() { let options, optionsReproj; beforeEach(function() { options = { params: { 'LAYERS': 'layer' }, url: 'http://example.com/wms' }; optionsReproj = { params: { 'LAYERS': 'layer' }, url: 'http://example.com/wms', projection: 'EPSG:4326' }; }); describe('constructor', function() { it('can be constructed without url or urls params', function() { const source = new TileWMS({ projection: 'EPSG:3857', tileGrid: createXYZ({maxZoom: 6}) }); expect(source).to.be.an(TileWMS); }); }); describe('#getTile', function() { it('returns a tile with the expected URL', function() { const source = new TileWMS(options); const tile = source.getTile(3, 2, 6, 1, getProjection('EPSG:3857')); expect(tile).to.be.an(ImageTile); const uri = new URL(tile.src_); expect(uri.protocol).to.be('http:'); expect(uri.hostname).to.be('example.com'); expect(uri.pathname).to.be('/wms'); const queryData = uri.searchParams; const bbox = queryData.get('BBOX').split(',').map(parseFloat); expect(bbox[0]).roughlyEqual(-10018754.171394622, 1e-9); expect(bbox[1]).roughlyEqual(-15028131.257091936, 1e-9); expect(bbox[2]).roughlyEqual(-5009377.085697311, 1e-9); expect(bbox[3]).roughlyEqual(-10018754.171394624, 1e-9); expect(queryData.get('CRS')).to.be('EPSG:3857'); expect(queryData.get('FORMAT')).to.be('image/png'); expect(queryData.get('HEIGHT')).to.be('256'); expect(queryData.get('LAYERS')).to.be('layer'); expect(queryData.get('REQUEST')).to.be('GetMap'); expect(queryData.get('SERVICE')).to.be('WMS'); expect(queryData.get('SRS')).to.be(null); expect(queryData.get('STYLES')).to.be(''); expect(queryData.get('TRANSPARENT')).to.be('true'); expect(queryData.get('VERSION')).to.be('1.3.0'); expect(queryData.get('WIDTH')).to.be('256'); expect(uri.hash.replace('#', '')).to.be.empty(); }); it('returns a larger tile when a gutter is specified', function() { options.gutter = 16; const source = new TileWMS(options); const tile = source.getTile(3, 2, 6, 1, getProjection('EPSG:3857')); expect(tile).to.be.an(ImageTile); const uri = new URL(tile.src_); const queryData = uri.searchParams; const bbox = queryData.get('BBOX').split(','); const expected = [-10331840.239250705, -15341217.324948018, -4696291.017841229, -9705668.103538541]; for (let i = 0, ii = bbox.length; i < ii; ++i) { expect(parseFloat(bbox[i])).to.roughlyEqual(expected[i], 1e-9); } expect(queryData.get('HEIGHT')).to.be('288'); expect(queryData.get('WIDTH')).to.be('288'); }); it('sets the SRS query value instead of CRS if version < 1.3', function() { options.params.VERSION = '1.2'; const source = new TileWMS(options); const tile = source.getTile(3, 2, 2, 1, getProjection('EPSG:4326')); const uri = new URL(tile.src_); const queryData = uri.searchParams; expect(queryData.get('CRS')).to.be(null); expect(queryData.get('SRS')).to.be('EPSG:4326'); }); it('allows various parameters to be overridden', function() { options.params.FORMAT = 'image/jpeg'; options.params.TRANSPARENT = false; const source = new TileWMS(options); const tile = source.getTile(3, 2, 2, 1, getProjection('EPSG:4326')); const uri = new URL(tile.src_); const queryData = uri.searchParams; expect(queryData.get('FORMAT')).to.be('image/jpeg'); expect(queryData.get('TRANSPARENT')).to.be('false'); }); it('does not add a STYLES= option if one is specified', function() { options.params.STYLES = 'foo'; const source = new TileWMS(options); const tile = source.getTile(3, 2, 2, 1, getProjection('EPSG:4326')); const uri = new URL(tile.src_); const queryData = uri.searchParams; expect(queryData.get('STYLES')).to.be('foo'); }); it('changes the BBOX order for EN axis orientations', function() { const source = new TileWMS(options); const tile = source.getTile(3, 2, 2, 1, getProjection('EPSG:4326')); const uri = new URL(tile.src_); const queryData = uri.searchParams; expect(queryData.get('BBOX')).to.be('-45,-90,0,-45'); }); it('uses EN BBOX order if version < 1.3', function() { options.params.VERSION = '1.1.0'; const source = new TileWMS(options); const tile = source.getTile(3, 2, 2, 1, getProjection('CRS:84')); const uri = new URL(tile.src_); const queryData = uri.searchParams; expect(queryData.get('BBOX')).to.be('-90,-45,-45,0'); }); it('sets FORMAT_OPTIONS when the server is GeoServer', function() { options.serverType = 'geoserver'; const source = new TileWMS(options); const tile = source.getTile(3, 2, 2, 2, getProjection('CRS:84')); const uri = new URL(tile.src_); const queryData = uri.searchParams; expect(queryData.get('FORMAT_OPTIONS')).to.be('dpi:180'); }); it('extends FORMAT_OPTIONS if it is already present', function() { options.serverType = 'geoserver'; const source = new TileWMS(options); options.params.FORMAT_OPTIONS = 'param1:value1'; const tile = source.getTile(3, 2, 2, 2, getProjection('CRS:84')); const uri = new URL(tile.src_); const queryData = uri.searchParams; expect(queryData.get('FORMAT_OPTIONS')).to.be('param1:value1;dpi:180'); }); it('rounds FORMAT_OPTIONS to an integer when the server is GeoServer', function() { options.serverType = 'geoserver'; const source = new TileWMS(options); const tile = source.getTile(3, 2, 2, 1.325, getProjection('CRS:84')); const uri = new URL(tile.src_); const queryData = uri.searchParams; expect(queryData.get('FORMAT_OPTIONS')).to.be('dpi:119'); }); }); describe('#tileUrlFunction', function() { it('returns a tile if it is contained within layers extent', function() { options.extent = [-80, -40, -50, -10]; const source = new TileWMS(options); const tileCoord = [3, 2, 2]; const url = source.tileUrlFunction(tileCoord, 1, getProjection('EPSG:4326')); const uri = new URL(url); const queryData = uri.searchParams; expect(queryData.get('BBOX')).to.be('-45,-90,0,-45'); }); it('returns a tile if it intersects layers extent', function() { options.extent = [-80, -40, -40, -10]; const source = new TileWMS(options); const tileCoord = [3, 3, 2]; const url = source.tileUrlFunction(tileCoord, 1, getProjection('EPSG:4326')); const uri = new URL(url); const queryData = uri.searchParams; expect(queryData.get('BBOX')).to.be('-45,-45,0,0'); }); it('works with non-square tiles', function() { options.tileGrid = new TileGrid({ tileSize: [640, 320], resolutions: [1.40625, 0.703125, 0.3515625, 0.17578125], origin: [-180, -90] }); const source = new TileWMS(options); const tileCoord = [3, 3, 2]; const url = source.tileUrlFunction(tileCoord, 1, getProjection('EPSG:4326')); const uri = new URL(url); const queryData = uri.searchParams; expect(queryData.get('WIDTH')).to.be('640'); expect(queryData.get('HEIGHT')).to.be('320'); }); }); describe('#getGetFeatureInfoUrl', function() { it('returns the expected GetFeatureInfo URL', function() { const source = new TileWMS(options); source.pixelRatio_ = 1; const url = source.getGetFeatureInfoUrl( [-7000000, -12000000], 19567.87924100512, getProjection('EPSG:3857'), {INFO_FORMAT: 'text/plain'}); const uri = new URL(url); expect(uri.protocol).to.be('http:'); expect(uri.hostname).to.be('example.com'); expect(uri.pathname).to.be('/wms'); const queryData = uri.searchParams; const bbox = queryData.get('BBOX').split(',').map(parseFloat); expect(bbox[0]).roughlyEqual(-10018754.171394622, 1e-9); expect(bbox[1]).roughlyEqual(-15028131.257091936, 1e-9); expect(bbox[2]).roughlyEqual(-5009377.085697311, 1e-9); expect(bbox[3]).roughlyEqual(-10018754.171394624, 1e-9); expect(queryData.get('CRS')).to.be('EPSG:3857'); expect(queryData.get('FORMAT')).to.be('image/png'); expect(queryData.get('HEIGHT')).to.be('256'); expect(queryData.get('I')).to.be('154'); expect(queryData.get('J')).to.be('101'); expect(queryData.get('LAYERS')).to.be('layer'); expect(queryData.get('QUERY_LAYERS')).to.be('layer'); expect(queryData.get('REQUEST')).to.be('GetFeatureInfo'); expect(queryData.get('SERVICE')).to.be('WMS'); expect(queryData.get('SRS')).to.be(null); expect(queryData.get('STYLES')).to.be(''); expect(queryData.get('TRANSPARENT')).to.be('true'); expect(queryData.get('VERSION')).to.be('1.3.0'); expect(queryData.get('WIDTH')).to.be('256'); expect(uri.hash.replace('#', '')).to.be.empty(); }); it('returns the expected GetFeatureInfo URL when source\'s projection is different from the parameter', function() { const source = new TileWMS(optionsReproj); source.pixelRatio_ = 1; const url = source.getGetFeatureInfoUrl( [-7000000, -12000000], 19567.87924100512, getProjection('EPSG:3857'), {INFO_FORMAT: 'text/plain'}); const uri = new URL(url); expect(uri.protocol).to.be('http:'); expect(uri.hostname).to.be('example.com'); expect(uri.pathname).to.be('/wms'); const queryData = uri.searchParams; expect(queryData.get('BBOX')).to.be('-79.17133464081945,-90,-66.51326044311186,-45'); expect(queryData.get('CRS')).to.be('EPSG:4326'); expect(queryData.get('FORMAT')).to.be('image/png'); expect(queryData.get('HEIGHT')).to.be('256'); expect(queryData.get('I')).to.be('517'); expect(queryData.get('J')).to.be('117'); expect(queryData.get('LAYERS')).to.be('layer'); expect(queryData.get('QUERY_LAYERS')).to.be('layer'); expect(queryData.get('REQUEST')).to.be('GetFeatureInfo'); expect(queryData.get('SERVICE')).to.be('WMS'); expect(queryData.get('SRS')).to.be(null); expect(queryData.get('STYLES')).to.be(''); expect(queryData.get('TRANSPARENT')).to.be('true'); expect(queryData.get('VERSION')).to.be('1.3.0'); expect(queryData.get('WIDTH')).to.be('256'); expect(uri.hash.replace('#', '')).to.be.empty(); }); it('sets the QUERY_LAYERS param as expected', function() { const source = new TileWMS(options); source.pixelRatio_ = 1; const url = source.getGetFeatureInfoUrl( [-7000000, -12000000], 19567.87924100512, getProjection('EPSG:3857'), {INFO_FORMAT: 'text/plain', QUERY_LAYERS: 'foo,bar'}); const uri = new URL(url); expect(uri.protocol).to.be('http:'); expect(uri.hostname).to.be('example.com'); expect(uri.pathname).to.be('/wms'); const queryData = uri.searchParams; const bbox = queryData.get('BBOX').split(',').map(parseFloat); expect(bbox[0]).roughlyEqual(-10018754.171394622, 1e-9); expect(bbox[1]).roughlyEqual(-15028131.257091936, 1e-9); expect(bbox[2]).roughlyEqual(-5009377.085697311, 1e-9); expect(bbox[3]).roughlyEqual(-10018754.171394624, 1e-9); expect(queryData.get('CRS')).to.be('EPSG:3857'); expect(queryData.get('FORMAT')).to.be('image/png'); expect(queryData.get('HEIGHT')).to.be('256'); expect(queryData.get('I')).to.be('154'); expect(queryData.get('J')).to.be('101'); expect(queryData.get('LAYERS')).to.be('layer'); expect(queryData.get('QUERY_LAYERS')).to.be('foo,bar'); expect(queryData.get('REQUEST')).to.be('GetFeatureInfo'); expect(queryData.get('SERVICE')).to.be('WMS'); expect(queryData.get('SRS')).to.be(null); expect(queryData.get('STYLES')).to.be(''); expect(queryData.get('TRANSPARENT')).to.be('true'); expect(queryData.get('VERSION')).to.be('1.3.0'); expect(queryData.get('WIDTH')).to.be('256'); expect(uri.hash.replace('#', '')).to.be.empty(); }); }); describe('#setUrl()', function() { it('sets the correct url', function() { const source = new TileWMS(options); const url = 'http://foo/'; source.setUrl(url); const tileUrl = source.tileUrlFunction([0, 0, 0], 1, getProjection('EPSG:4326')); expect(tileUrl.indexOf(url)).to.be(0); }); }); describe('#setUrls()', function() { it('updates the source key', function() { const source = new TileWMS({ urls: ['u1', 'u2'] }); const originalKey = source.getKey(); source.setUrls(['u3', 'u4']); expect(source.getKey() !== originalKey).to.be(true); }); }); });