UNPKG

terriajs

Version:

Geospatial data visualization platform.

735 lines (625 loc) 28.9 kB
'use strict'; /*global require*/ var URI = require('urijs'); var clone = require('terriajs-cesium/Source/Core/clone'); var defined = require('terriajs-cesium/Source/Core/defined'); var defineProperties = require('terriajs-cesium/Source/Core/defineProperties'); var formatError = require('terriajs-cesium/Source/Core/formatError'); var freezeObject = require('terriajs-cesium/Source/Core/freezeObject'); var knockout = require('terriajs-cesium/Source/ThirdParty/knockout'); var loadJson = require('../Core/loadJson'); var loadText = require('../Core/loadText'); var when = require('terriajs-cesium/Source/ThirdParty/when'); var CkanCatalogItem = require('./CkanCatalogItem'); var createRegexDeserializer = require('./createRegexDeserializer'); var createRegexSerializer = require('./createRegexSerializer'); var TerriaError = require('../Core/TerriaError'); var CatalogGroup = require('./CatalogGroup'); var inherit = require('../Core/inherit'); var proxyCatalogItemUrl = require('./proxyCatalogItemUrl'); var xml2json = require('../ThirdParty/xml2json'); /** * A {@link CatalogGroup} representing a collection of layers from a [CKAN](http://ckan.org) server. * * @alias CkanCatalogGroup * @constructor * @extends CatalogGroup * * @param {Terria} terria The Terria instance. */ var CkanCatalogGroup = function(terria) { CatalogGroup.call(this, terria, 'ckan'); /** * Gets or sets the URL of the CKAN server. This property is observable. * @type {String} */ this.url = ''; /** * Gets or sets a description of the custodian of the data sources in this group. * This property is an HTML string that must be sanitized before display to the user. * This property is observable. * @type {String} */ this.dataCustodian = undefined; /** * Gets or sets the filter query to pass to CKAN when querying the available data sources and their groups. Each item in the * array causes an independent request to the CKAN, and the results are concatenated. The * search string is equivalent to what would be in the parameters segment of the url calling the CKAN search api. * See the [Solr documentation](http://wiki.apache.org/solr/CommonQueryParameters#fq) for information about filter queries. * Each item can be either a URL-encoded string ("fq=res_format%3awms") or an object ({ fq: 'res_format:wms' }). The latter * format is easier to work with. * To get all the datasets with wms resources: [{ fq: 'res_format%3awms' }] * To get all wms/WMS datasets in the Surface Water group: [{q: 'groups=Surface Water', fq: 'res_format:WMS' }] * To get both wms and esri-mapService datasets: [{q: 'res_format:WMS'}, {q: 'res_format:"Esri REST"' }] * To get all datasets with no filter, you can use [''] * This property is required. * This property is observable. * @type {String[]|Object[]} * @editoritemstitle Filter */ this.filterQuery = undefined; /** * Gets or sets a hash of names of blacklisted groups and data sources. A group or data source that appears in this hash * will not be shown to the user. In this hash, the keys should be the names of the groups and data sources to blacklist, * and the values should be "true". This property is observable. * @type {Object} */ this.blacklist = undefined; /** * Gets or sets a value indicating whether the CKAN datasets should be filtered by querying GetCapabilities from each * referenced WMS server and excluding datasets not found therein. This property is observable. * @type {Boolean} */ this.filterByWmsGetCapabilities = false; /** * Gets or sets the minimum MaxScaleDenominator that is allowed for a WMS dataset to be included in this CKAN group. * If this property is undefined or if {@link CkanCatalogGroup#filterByWmsGetCapabilities} is false, no * filtering based on MaxScaleDenominator is performed. This property is observable. * @type {Number} */ this.minimumMaxScaleDenominator = undefined; /** * Gets or sets any extra wms parameters that should be added to the wms query urls in this CKAN group. * If this property is undefined then no extra parameters are added. * This property is observable. * @type {Object} */ this.wmsParameters = undefined; /** * Gets or sets a value indicating how datasets should be grouped. Valid values are: * * `none` - Datasets are put in a flat list; they are not grouped at all. * * `group` - Datasets are grouped according to their CKAN group. Datasets that are not in any groups are put at the top level. * * `organization` - Datasets are grouped by their CKAN organization. Datasets that are not associated with an organization are put at the top level. * @type {String} */ this.groupBy = 'group'; /** * Gets or sets a title for the group holding all items that don't have a group in CKAN. If the value is a blank string or undefined, * these items will be left at the top level, not grouped. * @type {String} */ this.ungroupedTitle = 'No group'; /** * Gets or sets a value indicating whether each catalog item's name should be populated from * individual resources instead of from the CKAN dataset. * @type {Boolean} */ this.useResourceName = false; /** * True to allow entire WMS servers (that is, WMS resources without a clearly-defined layer) to be * added to the catalog; otherwise, false. * @type {Boolean} * @default false */ this.allowEntireWmsServers = false; /** * True to allow entire WFS servers (that is, WFS resources without a clearly-defined layer) to be * added to the catalog; otherwise, false. * @type {Boolean} * @default false */ this.allowEntireWfsServers = false; /** * True to allow WMS resources to be added to the catalog; otherwise, false. * @type {Boolean} * @default true */ this.includeWms = true; /** * Gets or sets a regular expression that, when it matches a resource's format, indicates that the resource is a WMS resource. * @type {RegExp} */ this.wmsResourceFormat = /^wms$/i; /** * True to allow WFS resources to be added to the catalog; otherwise, false. * @type {Boolean} * @default true */ this.includeWfs = true; /** * Gets or sets a regular expression that, when it matches a resource's format, indicates that the resource is a WMS resource. * @type {RegExp} */ this.wfsResourceFormat = /^wfs$/i; /** * True to allow KML resources to be added to the catalog; otherwise, false. * @type {Boolean} * @default false */ this.includeKml = false; /** * Gets or sets a regular expression that, when it matches a resource's format, indicates that the resource is a KML resource. * @type {RegExp} */ this.kmlResourceFormat = /^km[lz]$/i; /** * True to allow CSV resources to be added to the catalog; otherwise, false. * @type {Boolean} */ this.includeCsv = false; /** * Gets or sets a regular expression that, when it matches a resource's format, indicates that the resource is a CSV resource. * @type {RegExp} */ this.csvResourceFormat = /^csv-geo-/i; /** * True to allow ESRI MapServer resources to be added to the catalog; otherwise, false. * @type {Boolean} * @default false */ this.includeEsriMapServer = false; /** * True to allow ESRI FeatureServer resources to be added to the catalog; otherwise, false. * @type {Boolean} * @default false */ this.includeEsriFeatureServer = false; /** * Gets or sets a regular expression that, when it matches a resource's format, indicates that the resource is an Esri MapServer resource. * @type {RegExp} */ this.esriMapServerResourceFormat = /^esri rest$/i; /** * Gets or sets a regular expression that, when it matches a resource's format, indicates that the resource is an Esri * MapServer or FeatureServer resource. A valid FeatureServer resource must also have `FeatureServer` in its URL. * @type {RegExp} */ this.esriFeatureServerResourceFormat = /^esri rest$/i; /** * True to allow GeoJSON resources to be added to the catalog; otherwise, false. * @type {Boolean} * @default false */ this.includeGeoJson = false; /** * Gets or sets a regular expression that, when it matches a resource's format, indicates that the resource is a GeoJSON resource. * @type {RegExp} */ this.geoJsonResourceFormat = /^geojson$/i; /** * True to allow CZML resources to be added to the catalog; otherwise, false. * @type {Boolean} * @default false */ this.includeCzml = false; /** * Gets or sets a regular expression that, when it matches a resource's format, indicates that the resource is a CZML resource. * @type {RegExp} */ this.czmlResourceFormat = /^czml$/i; /** * Gets or sets a hash of properties that will be set on each child item. * For example, { "treat404AsError": false } * @type {Object} */ this.itemProperties = undefined; knockout.track(this, [ 'url', 'dataCustodian', 'filterQuery', 'blacklist', 'wmsParameters', 'groupBy', 'ungroupedTitle', 'useResourceName', 'allowEntireWmsServers', 'allowEntireWfsServers', 'includeWms', 'includeWfs', 'includeKml', 'includeCsv', 'includeEsriMapServer', 'includeGeoJson', 'includeCzml', 'itemProperties' ]); }; inherit(CatalogGroup, CkanCatalogGroup); defineProperties(CkanCatalogGroup.prototype, { /** * Gets the type of data member represented by this instance. * @memberOf CkanCatalogGroup.prototype * @type {String} */ type : { get : function() { return 'ckan'; } }, /** * Gets a human-readable name for this type of data source, such as 'Web Map Service (WMS)'. * @memberOf CkanCatalogGroup.prototype * @type {String} */ typeName : { get : function() { return 'CKAN Server'; } }, /** * Gets the set of functions used to update individual properties in {@link CatalogMember#updateFromJson}. * When a property name in the returned object literal matches the name of a property on this instance, the value * will be called as a function and passed a reference to this instance, a reference to the source JSON object * literal, and the name of the property. * @memberOf CkanCatalogGroup.prototype * @type {Object} */ updaters : { get : function() { return CkanCatalogGroup.defaultUpdaters; } }, /** * Gets the set of functions used to serialize individual properties in {@link CatalogMember#serializeToJson}. * When a property name on the model matches the name of a property in the serializers object literal, * the value will be called as a function and passed a reference to the model, a reference to the destination * JSON object literal, and the name of the property. * @memberOf CkanCatalogGroup.prototype * @type {Object} */ serializers : { get : function() { return CkanCatalogGroup.defaultSerializers; } } }); /** * Gets or sets the set of default updater functions to use in {@link CatalogMember#updateFromJson}. Types derived from this type * should expose this instance - cloned and modified if necesary - through their {@link CatalogMember#updaters} property. * @type {Object} */ CkanCatalogGroup.defaultUpdaters = clone(CatalogGroup.defaultUpdaters); CkanCatalogGroup.defaultUpdaters.wmsResourceFormat = createRegexDeserializer('wmsResourceFormat'); CkanCatalogGroup.defaultUpdaters.wfsResourceFormat = createRegexDeserializer('wfsResourceFormat'); CkanCatalogGroup.defaultUpdaters.kmlResourceFormat = createRegexDeserializer('kmlResourceFormat'); CkanCatalogGroup.defaultUpdaters.csvResourceFormat = createRegexDeserializer('csvResourceFormat'); CkanCatalogGroup.defaultUpdaters.esriMapServerResourceFormat = createRegexDeserializer('esriMapServerResourceFormat'); CkanCatalogGroup.defaultUpdaters.esriFeatureServerResourceFormat = createRegexDeserializer('esriFeatureServerResourceFormat'); CkanCatalogGroup.defaultUpdaters.geoJsonResourceFormat = createRegexDeserializer('geoJsonResourceFormat'); CkanCatalogGroup.defaultUpdaters.czmlResourceFormat = createRegexDeserializer('czmlResourceFormat'); freezeObject(CkanCatalogGroup.defaultUpdaters); /** * Gets or sets the set of default serializer functions to use in {@link CatalogMember#serializeToJson}. Types derived from this type * should expose this instance - cloned and modified if necesary - through their {@link CatalogMember#serializers} property. * @type {Object} */ CkanCatalogGroup.defaultSerializers = clone(CatalogGroup.defaultSerializers); CkanCatalogGroup.defaultSerializers.items = CatalogGroup.enabledShareableItemsSerializer; CkanCatalogGroup.defaultSerializers.wmsResourceFormat = createRegexSerializer('wmsResourceFormat'); CkanCatalogGroup.defaultSerializers.wfsResourceFormat = createRegexSerializer('wfsResourceFormat'); CkanCatalogGroup.defaultSerializers.kmlResourceFormat = createRegexSerializer('kmlResourceFormat'); CkanCatalogGroup.defaultSerializers.csvResourceFormat = createRegexSerializer('csvResourceFormat'); CkanCatalogGroup.defaultSerializers.esriMapServerResourceFormat = createRegexSerializer('esriMapServerResourceFormat'); CkanCatalogGroup.defaultSerializers.esriFeatureServerResourceFormat = createRegexSerializer('esriFeatureServerResourceFormat'); CkanCatalogGroup.defaultSerializers.geoJsonResourceFormat = createRegexSerializer('geoJsonResourceFormat'); CkanCatalogGroup.defaultSerializers.czmlResourceFormat = createRegexSerializer('czmlResourceFormat'); freezeObject(CkanCatalogGroup.defaultSerializers); CkanCatalogGroup.prototype._getValuesThatInfluenceLoad = function() { return [ this.url, this.filterQuery, this.blacklist, this.filterByWmsGetCapabilities, this.minimumMaxScaleDenominator, this.allowEntireWmsServers, this.allowEntireWfsServers, this.includeWms, this.includeWfs, this.includeKml, this.includeCsv, this.includeEsriMapServer, this.includeGeoJson, this.includeCzml, this.groupBy, this.ungroupedTitle ]; }; CkanCatalogGroup.prototype._load = function() { if (!defined(this.url) || this.url.length === 0) { return undefined; } var that = this; var promises = []; if (!(defined(this.filterQuery) && Array.isArray(this.filterQuery) && (typeof this.filterQuery[0] === 'string' || typeof this.filterQuery[0] === 'object'))) { throw new TerriaError({ title: 'Error loading CKAN catalogue', message: 'The definition for this CKAN catalogue does not have a valid filter query.' }); } const results = []; this.filterQuery.forEach(function(query) { var uri = new URI(that.url) .segment('api/3/action/package_search') .addQuery({ rows: 100000, sort: 'metadata_created asc' }); if (typeof query === 'object') { // query is an object of non-encoded parameters, like { fq: "res_format:wms OR WMS" } Object.keys(query).forEach(key => uri.addQuery(key, query[key])); } let start = 0; function requestNext() { var nextUri = uri.clone().addQuery({ start: start }); var uristring = nextUri.toString(); if (typeof query === 'string') { // query is expected to be URL-encoded and begin with a query like 'fq=' uristring += '&' + query; } return loadJson(proxyCatalogItemUrl(that, uristring, '1d')).then(function(pageResults) { if (pageResults && pageResults.result && pageResults.result.results) { const thisPage = pageResults.result.results; for (let i = 0; i < thisPage.length; ++i) { results.push(thisPage[i]); } start += thisPage.length; if (start < pageResults.result.count) { return requestNext(); } } }); } promises.push(requestNext().then(function() { return results; })); }); return when.all(promises).then( function(queryResults) { if (!defined(queryResults)) { return; } var allResults = []; for (var p = 0; p < queryResults.length; p++) { var queryResult = queryResults[p]; for (var i = 0; i < queryResult.length; ++i) { allResults.push(queryResult[i]); } } if (that.filterByWmsGetCapabilities) { return when(filterResultsByGetCapabilities(that, allResults), function() { populateGroupFromResults(that, allResults); }); } else { populateGroupFromResults(that, allResults); } }).otherwise(function(e) { throw new TerriaError({ sender: that, title: that.name, message: '\ Couldn\'t retrieve packages from this CKAN server.<br/><br/>\ If you entered the URL manually, please double-check it.<br/><br/>\ If it\'s your server, make sure <a href="http://enable-cors.org/" target="_blank">CORS</a> is enabled.<br/><br/>\ Otherwise, if reloading doesn\'t fix it, please report the problem by sending an email to <a href="mailto:'+that.terria.supportEmail+'">'+that.terria.supportEmail+'</a> with the technical details below. Thank you!<br/><br/>\ <pre>' + formatError(e) + '</pre>' }); }); }; function filterResultsByGetCapabilities(ckanGroup, items) { var wmsServers = {}; for (var itemIndex = 0; itemIndex < items.length; ++itemIndex) { var item = items[itemIndex]; var resources = item.resources; for (var resourceIndex = 0; resourceIndex < resources.length; ++resourceIndex) { var resource = resources[resourceIndex]; if (!resource.format.match(ckanGroup.wmsResourceFormat)) { continue; } var wmsUrl = resource.wms_url; if (!defined(wmsUrl)) { wmsUrl = resource.url; if (!defined(wmsUrl)) { continue; } } // Extract the layer name from the WMS URL. var uri = new URI(wmsUrl); var params = uri.search(true); var layerName = params.LAYERS; // Remove the query portion of the WMS URL. uri.search(''); var url = uri.toString(); if (!defined(wmsServers[url])) { wmsServers[url] = {}; } wmsServers[url][layerName] = resource; } } var promises = []; for (var wmsServer in wmsServers) { if (wmsServers.hasOwnProperty(wmsServer)) { var getCapabilitiesUrl = ckanGroup.terria.corsProxy.getURLProxyIfNecessary(wmsServer + '?service=WMS&request=GetCapabilities', '1d'); promises.push(filterBasedOnGetCapabilities(ckanGroup, getCapabilitiesUrl, wmsServers[wmsServer])); } } return when.all(promises); } function filterBasedOnGetCapabilities(ckanGroup, getCapabilitiesUrl, resources) { // Initially assume all resources will be filtered. for (var name in resources) { if (resources.hasOwnProperty(name)) { resources[name].__filtered = true; } } return loadText(getCapabilitiesUrl).then(function(getCapabilitiesXml) { var getCapabilitiesJson = xml2json(getCapabilitiesXml); filterBasedOnGetCapabilitiesResponse(ckanGroup, getCapabilitiesJson.Capability.Layer, resources); }).otherwise(function() { // Do nothing - all resources will be filtered. }); } function filterBasedOnGetCapabilitiesResponse(ckanGroup, wmsLayersSource, resources) { if (defined(wmsLayersSource) && !(wmsLayersSource instanceof Array)) { wmsLayersSource = [wmsLayersSource]; } for (var i = 0; i < wmsLayersSource.length; ++i) { var layerSource = wmsLayersSource[i]; if (layerSource.Name) { var resource = resources[layerSource.Name]; if (resource) { if (!defined(ckanGroup.minimumMaxScaleDenominator) || !defined(layerSource.MaxScaleDenominator) || layerSource.MaxScaleDenominator >= ckanGroup.minimumMaxScaleDenominator) { resource.__filtered = false; } else { console.log('Provider Feedback: Filtering out ' + layerSource.Title + ' (' + layerSource.Name + ') because its MaxScaleDenominator is ' + layerSource.MaxScaleDenominator); } } } if (layerSource.Layer) { filterBasedOnGetCapabilitiesResponse(ckanGroup, layerSource.Layer, resources); } } } function createItemFromResource(resource, ckanGroup, itemData, extras, parent) { return CkanCatalogItem.createCatalogItemFromResource({ terria: ckanGroup.terria, itemData: itemData, resource: resource, extras: extras, parent: parent, ckanBaseUrl: ckanGroup.url, wmsResourceFormat: ckanGroup.includeWms ? ckanGroup.wmsResourceFormat : undefined, wfsResourceFormat: ckanGroup.includeWfs ? ckanGroup.wfsResourceFormat : undefined, kmlResourceFormat: ckanGroup.includeKml ? ckanGroup.kmlResourceFormat : undefined, csvResourceFormat: ckanGroup.includeCsv ? ckanGroup.csvResourceFormat : undefined, esriMapServerResourceFormat: ckanGroup.includeEsriMapServer ? ckanGroup.esriMapServerResourceFormat : undefined, esriFeatureServerResourceFormat: ckanGroup.includeEsriFeatureServer ? ckanGroup.esriFeatureServerResourceFormat : undefined, geoJsonResourceFormat: ckanGroup.includeGeoJson ? ckanGroup.geoJsonResourceFormat : undefined, czmlResourceFormat: ckanGroup.includeCzml ? ckanGroup.czmlResourceFormat : undefined, allowWmsGroups: ckanGroup.allowEntireWmsServers, allowWfsGroups: ckanGroup.allowEntireWfsServers, dataCustodian: ckanGroup.dataCustodian, itemProperties: ckanGroup.itemProperties, useResourceName: ckanGroup.useResourceName }); } function populateGroupFromResults(ckanGroup, items) { var ungrouped; for (var itemIndex = 0; itemIndex < items.length; ++itemIndex) { var item = items[itemIndex]; if (ckanGroup.blacklist && ckanGroup.blacklist[item.title]) { console.log('Provider Feedback: Filtering out ' + item.title + ' (' + item.name + ') because it is blacklisted.'); continue; } var extras = {}; if (defined(item.extras)) { for (var idx = 0; idx < item.extras.length; idx++) { extras[item.extras[idx].key] = item.extras[idx].value; } } var resourceItems = []; var resources = item.resources; for (var resourceIndex = 0; resourceIndex < resources.length; ++resourceIndex) { var resource = resources[resourceIndex]; var groups; if (ckanGroup.groupBy === 'group') { groups = item.groups; } else if (ckanGroup.groupBy === 'organization' && item.organization) { // item.organization is sometimes null groups = [item.organization]; } else { groups = undefined; } var addedItem; if (defined(groups) && groups.length > 0) { for (var groupIndex = 0; groupIndex < groups.length; ++groupIndex) { var group = groups[groupIndex]; var groupName = group.display_name || group.title; var groupId = ckanGroup.uniqueId + '/' + group.id; if (ckanGroup.blacklist && ckanGroup.blacklist[groupName]) { continue; } var groupToAdd = ckanGroup.terria.catalog.shareKeyIndex[groupId]; var updating = defined(groupToAdd); if (!defined(groupToAdd)) { groupToAdd = new CatalogGroup(ckanGroup.terria); groupToAdd.name = groupName; groupToAdd.id = groupId; } addedItem = addItem(resource, ckanGroup, item, extras, groupToAdd); if (!updating && groupToAdd.items.length) { ckanGroup.add(groupToAdd); } } } else { if (!ckanGroup.ungroupedTitle) { addedItem = addItem(resource, ckanGroup, item, extras, ckanGroup); } else { if (!defined(ungrouped)) { ungrouped = new CatalogGroup(ckanGroup.terria); ungrouped.name = ckanGroup.ungroupedTitle; ungrouped.id = ckanGroup.uniqueId + '/_ungrouped'; ckanGroup.add(ungrouped); } addedItem = addItem(resource, ckanGroup, item, extras, ungrouped); } } if (defined(addedItem)) { resourceItems.push(addedItem); } } // If there's more than one resource item, and we're not using the resource name to name // our items, then they'll all have the same name. Add the type to the name to help // distinguish them. if (resourceItems.length > 1 && !ckanGroup.useResourceName) { resourceItems.forEach(function(item) { var typeName = CkanCatalogItem.shortHumanReadableTypeNames[item.type] || 'Other'; item.name += ' (' + typeName + ')'; }); } } function compareNames(a, b) { var aName = a.name.toLowerCase(); var bName = b.name.toLowerCase(); if (aName < bName) { return -1; } else if (aName > bName) { return 1; } else { return 0; } } ckanGroup.items.sort(compareNames); for (var i = 0; i < ckanGroup.items.length; ++i) { if (defined(ckanGroup.items[i].items)) { ckanGroup.items[i].items.sort(compareNames); } } } /** * Creates a catalog item from the supplied resource and adds it to the supplied parent if necessary.. * @private * @param resource The Ckan resource * @param rootCkanGroup The root group of all items in this Ckan hierarchy * @param itemData The data of the item to build the catalog item from * @param extras * @param parent The parent group to add the item to once it's constructed - set this to rootCkanGroup for flat hierarchies. * @returns {CatalogItem} The catalog item added, or undefined if no catalog item was added. */ function addItem(resource, rootCkanGroup, itemData, extras, parent) { var item = rootCkanGroup.terria.catalog.shareKeyIndex[parent.uniqueId + '/' + resource.id]; var alreadyExists = defined(item); if (!alreadyExists) { item = createItemFromResource(resource, rootCkanGroup, itemData, extras, parent); if (item) { parent.add(item); } } return item; } module.exports = CkanCatalogGroup;