34#include <QRegularExpression>
40 void appendOwsLayerStyles( QDomDocument &doc, QDomElement &layerElem,
QgsMapLayer *currentLayer );
42 void appendOwsLayersFromTreeGroup( QDomDocument &doc,
43 QDomElement &parentLayer,
46 const QgsWmsRequest &request,
49 const QString &strGroup );
51 void appendOwsGeneralAndResourceList( QDomDocument &doc, QDomElement &parentElement,
53 const QgsWmsRequest &request );
60#ifdef HAVE_SERVER_PYTHON_PLUGINS
65 const QDomDocument *contextDocument =
nullptr;
67#ifdef HAVE_SERVER_PYTHON_PLUGINS
69 if ( cacheManager && cacheManager->
getCachedDocument( &doc, project, request, accessControl ) )
71 contextDocument = &doc;
75 doc =
getContext( serverIface, project, request );
81 contextDocument = &doc;
84 doc =
getContext( serverIface, project, request );
85 contextDocument = &doc;
87 response.
setHeader( QStringLiteral(
"Content-Type" ), QStringLiteral(
"text/xml; charset=utf-8" ) );
88 response.
write( contextDocument->toByteArray() );
97 const QDomProcessingInstruction xmlDeclaration = doc.createProcessingInstruction( QStringLiteral(
"xml" ),
98 QStringLiteral(
"version=\"1.0\" encoding=\"utf-8\"" ) );
100 doc.appendChild( xmlDeclaration );
102 QDomElement owsContextElem = doc.createElement( QStringLiteral(
"OWSContext" ) );
103 owsContextElem.setAttribute( QStringLiteral(
"xmlns" ), QStringLiteral(
"http://www.opengis.net/ows-context" ) );
104 owsContextElem.setAttribute( QStringLiteral(
"xmlns:ows-context" ), QStringLiteral(
"http://www.opengis.net/ows-context" ) );
105 owsContextElem.setAttribute( QStringLiteral(
"xmlns:context" ), QStringLiteral(
"http://www.opengis.net/context" ) );
106 owsContextElem.setAttribute( QStringLiteral(
"xmlns:ows" ), QStringLiteral(
"http://www.opengis.net/ows" ) );
107 owsContextElem.setAttribute( QStringLiteral(
"xmlns:sld" ), QStringLiteral(
"http://www.opengis.net/sld" ) );
108 owsContextElem.setAttribute( QStringLiteral(
"xmlns:ogc" ), QStringLiteral(
"http://www.opengis.net/ogc" ) );
109 owsContextElem.setAttribute( QStringLiteral(
"xmlns:gml" ), QStringLiteral(
"http://www.opengis.net/gml" ) );
110 owsContextElem.setAttribute( QStringLiteral(
"xmlns:kml" ), QStringLiteral(
"http://www.opengis.net/kml/2.2" ) );
111 owsContextElem.setAttribute( QStringLiteral(
"xmlns:xlink" ), QStringLiteral(
"http://www.w3.org/1999/xlink" ) );
112 owsContextElem.setAttribute( QStringLiteral(
"xmlns:ns9" ), QStringLiteral(
"http://www.w3.org/2005/Atom" ) );
113 owsContextElem.setAttribute( QStringLiteral(
"xmlns:xal" ), QStringLiteral(
"urn:oasis:names:tc:ciq:xsdschema:xAL:2.0" ) );
114 owsContextElem.setAttribute( QStringLiteral(
"xmlns:ins" ), QStringLiteral(
"http://www.inspire.org" ) );
115 owsContextElem.setAttribute( QStringLiteral(
"version" ), QStringLiteral(
"0.3.1" ) );
116 doc.appendChild( owsContextElem );
118 appendOwsGeneralAndResourceList( doc, owsContextElem, serverIface, project, request );
124 void appendOwsGeneralAndResourceList( QDomDocument &doc, QDomElement &parentElement,
126 const QgsWmsRequest &request )
128 parentElement.setAttribute( QStringLiteral(
"id" ),
"ows-context-" + project->
baseName() );
131 QDomElement generalElem = doc.createElement( QStringLiteral(
"General" ) );
134 QDomElement windowElem = doc.createElement( QStringLiteral(
"Window" ) );
135 windowElem.setAttribute( QStringLiteral(
"height" ), QStringLiteral(
"600" ) );
136 windowElem.setAttribute( QStringLiteral(
"width" ), QStringLiteral(
"800" ) );
137 generalElem.appendChild( windowElem );
141 QDomElement titleElem = doc.createElement( QStringLiteral(
"ows:Title" ) );
142 const QDomText titleText = doc.createTextNode( title );
143 titleElem.appendChild( titleText );
144 generalElem.appendChild( titleElem );
148 if ( !abstract.isEmpty() )
150 QDomElement abstractElem = doc.createElement( QStringLiteral(
"ows:Abstract" ) );
151 const QDomText abstractText = doc.createCDATASection( abstract );
152 abstractElem.appendChild( abstractText );
153 generalElem.appendChild( abstractElem );
158 if ( !keywords.isEmpty() )
162 QDomElement keywordsElem = doc.createElement( QStringLiteral(
"ows:Keywords" ) );
164 for (
int i = 0; i < keywords.size(); ++i )
166 const QString keyword = keywords.at( i );
167 if ( !keyword.isEmpty() )
169 QDomElement keywordElem = doc.createElement( QStringLiteral(
"ows:Keyword" ) );
170 const QDomText keywordText = doc.createTextNode( keyword );
171 keywordElem.appendChild( keywordText );
174 keywordElem.setAttribute( QStringLiteral(
"vocabulary" ), QStringLiteral(
"SIA_Geo405" ) );
176 keywordsElem.appendChild( keywordElem );
179 generalElem.appendChild( keywordsElem );
183 parentElement.appendChild( generalElem );
186 QDomElement resourceListElem = doc.createElement( QStringLiteral(
"ResourceList" ) );
189 appendOwsLayersFromTreeGroup( doc, resourceListElem, serverIface, project, request, projectLayerTreeRoot, combinedBBox, QString() );
190 parentElement.appendChild( resourceListElem );
197 mapRect = combinedBBox;
199 QDomElement bboxElem = doc.createElement( QStringLiteral(
"ows:BoundingBox" ) );
200 bboxElem.setAttribute( QStringLiteral(
"crs" ), projectCrs.
authid() );
205 QDomElement lowerCornerElem = doc.createElement( QStringLiteral(
"ows:LowerCorner" ) );
206 const QDomText lowerCornerText = doc.createTextNode( QString::number( mapRect.
xMinimum() ) +
" " + QString::number( mapRect.
yMinimum() ) );
207 lowerCornerElem.appendChild( lowerCornerText );
208 bboxElem.appendChild( lowerCornerElem );
209 QDomElement upperCornerElem = doc.createElement( QStringLiteral(
"ows:UpperCorner" ) );
210 const QDomText upperCornerText = doc.createTextNode( QString::number( mapRect.
xMaximum() ) +
" " + QString::number( mapRect.
yMaximum() ) );
211 upperCornerElem.appendChild( upperCornerText );
212 bboxElem.appendChild( upperCornerElem );
213 generalElem.appendChild( bboxElem );
216 void appendOwsLayersFromTreeGroup( QDomDocument &doc,
217 QDomElement &parentLayer,
220 const QgsWmsRequest &request,
223 const QString &strGroup )
227 const QList< QgsLayerTreeNode * > layerTreeGroupChildren = layerTreeGroup->
children();
228 for (
int i = 0; i < layerTreeGroupChildren.size(); ++i )
236 const QString name = treeGroupChild->
name();
237 if ( restrictedLayers.contains( name ) )
243 if ( strGroup.isEmpty() )
249 group = strGroup +
"/" + name;
252 appendOwsLayersFromTreeGroup( doc, parentLayer, serverIface, project, request, treeGroupChild, combinedBBox, group );
258 if ( restrictedLayers.contains( l->
name() ) )
262#ifdef HAVE_SERVER_PYTHON_PLUGINS
269 QDomElement layerElem = doc.createElement( QStringLiteral(
"Layer" ) );
274 layerElem.setAttribute( QStringLiteral(
"queryable" ), QStringLiteral(
"false" ) );
278 layerElem.setAttribute( QStringLiteral(
"queryable" ), QStringLiteral(
"true" ) );
284 layerElem.setAttribute( QStringLiteral(
"hidden" ), QStringLiteral(
"false" ) );
288 layerElem.setAttribute( QStringLiteral(
"hidden" ), QStringLiteral(
"true" ) );
292 if ( !strGroup.isEmpty() )
294 layerElem.setAttribute( QStringLiteral(
"group" ), strGroup );
299 layerElem.setAttribute( QStringLiteral(
"opacity" ), 1 );
301 QString wmsName = l->
name();
311 layerElem.setAttribute( QStringLiteral(
"name" ), wmsName );
313 const thread_local QRegularExpression sRegEx( QStringLiteral(
"[\\W]" ), QRegularExpression::UseUnicodePropertiesOption );
314 layerElem.setAttribute( QStringLiteral(
"id" ), wmsName.replace( sRegEx, QStringLiteral(
"_" ) ) );
317 QDomElement titleElem = doc.createElement( QStringLiteral(
"ows:Title" ) );
318 QString title = l->
title();
319 if ( title.isEmpty() )
323 const QDomText titleText = doc.createTextNode( title );
324 titleElem.appendChild( titleText );
325 layerElem.appendChild( titleElem );
328 QDomElement formatElem = doc.createElement( QStringLiteral(
"ows:OutputFormat" ) );
329 const QDomText formatText = doc.createTextNode( QStringLiteral(
"image/png" ) );
330 formatElem.appendChild( formatText );
331 layerElem.appendChild( formatElem );
337 QString hrefString = href.toString();
338 hrefString.append( href.hasQuery() ?
"&" :
"?" );
341 QDomElement serverElem = doc.createElement( QStringLiteral(
"Server" ) );
342 serverElem.setAttribute( QStringLiteral(
"service" ), QStringLiteral(
"urn:ogc:serviceType:WMS" ) );
343 serverElem.setAttribute( QStringLiteral(
"version" ), QStringLiteral(
"1.3.0" ) );
344 serverElem.setAttribute( QStringLiteral(
"default" ), QStringLiteral(
"true" ) );
345 QDomElement orServerElem = doc.createElement( QStringLiteral(
"OnlineResource" ) );
346 orServerElem.setAttribute( QStringLiteral(
"xlink:href" ), hrefString );
347 serverElem.appendChild( orServerElem );
348 layerElem.appendChild( serverElem );
350 const QString abstract = l->
abstract();
351 if ( !abstract.isEmpty() )
353 QDomElement abstractElem = doc.createElement( QStringLiteral(
"ows:Abstract" ) );
354 const QDomText abstractText = doc.createTextNode( abstract );
355 abstractElem.appendChild( abstractText );
356 layerElem.appendChild( abstractElem );
362 const QString minScaleString = QString::number( l->
maximumScale() );
363 const QString maxScaleString = QString::number( l->
minimumScale() );
364 QDomElement minScaleElem = doc.createElement( QStringLiteral(
"sld:MinScaleDenominator" ) );
365 const QDomText minScaleText = doc.createTextNode( minScaleString );
366 minScaleElem.appendChild( minScaleText );
367 layerElem.appendChild( minScaleElem );
368 QDomElement maxScaleElem = doc.createElement( QStringLiteral(
"sld:MaxScaleDenominator" ) );
369 const QDomText maxScaleText = doc.createTextNode( maxScaleString );
370 maxScaleElem.appendChild( maxScaleText );
371 layerElem.appendChild( maxScaleElem );
375 appendOwsLayerStyles( doc, layerElem, l );
380 const QStringList keywordStringList = l->
keywordList().split(
',' );
383 QDomElement keywordsElem = doc.createElement( QStringLiteral(
"ows:Keywords" ) );
384 for (
int i = 0; i < keywordStringList.size(); ++i )
386 QDomElement keywordElem = doc.createElement( QStringLiteral(
"ows:Keyword" ) );
387 const QDomText keywordText = doc.createTextNode( keywordStringList.at( i ).trimmed() );
388 keywordElem.appendChild( keywordText );
391 keywordElem.setAttribute( QStringLiteral(
"vocabulary" ), QStringLiteral(
"SIA_Geo405" ) );
393 keywordsElem.appendChild( keywordElem );
395 layerElem.appendChild( keywordsElem );
399 const QString dataUrl = l->
dataUrl();
400 if ( !dataUrl.isEmpty() )
402 QDomElement dataUrlElem = doc.createElement( QStringLiteral(
"DataURL" ) );
404 dataUrlElem.setAttribute( QStringLiteral(
"format" ), dataUrlFormat );
405 QDomElement dataORElem = doc.createElement( QStringLiteral(
"OnlineResource" ) );
406 dataORElem.setAttribute( QStringLiteral(
"xmlns:xlink" ), QStringLiteral(
"http://www.w3.org/1999/xlink" ) );
407 dataORElem.setAttribute( QStringLiteral(
"xlink:type" ), QStringLiteral(
"simple" ) );
408 dataORElem.setAttribute( QStringLiteral(
"xlink:href" ), dataUrl );
409 dataUrlElem.appendChild( dataORElem );
410 layerElem.appendChild( dataUrlElem );
417 QDomElement metaUrlElem = doc.createElement( QStringLiteral(
"MetadataURL" ) );
418 metaUrlElem.setAttribute( QStringLiteral(
"format" ), url.format );
419 QDomElement metaUrlORElem = doc.createElement( QStringLiteral(
"OnlineResource" ) );
420 metaUrlORElem.setAttribute( QStringLiteral(
"xmlns:xlink" ), QStringLiteral(
"http://www.w3.org/1999/xlink" ) );
421 metaUrlORElem.setAttribute( QStringLiteral(
"xlink:type" ), QStringLiteral(
"simple" ) );
422 metaUrlORElem.setAttribute( QStringLiteral(
"xlink:href" ), url.url );
423 metaUrlElem.appendChild( metaUrlORElem );
424 layerElem.appendChild( metaUrlElem );
446 if ( parentLayer.hasChildNodes() )
448 parentLayer.insertBefore( layerElem, parentLayer.firstChild() );
452 parentLayer.appendChild( layerElem );
458 void appendOwsLayerStyles( QDomDocument &doc, QDomElement &layerElem,
QgsMapLayer *currentLayer )
462 QDomElement styleListElem = doc.createElement( QStringLiteral(
"StyleList" ) );
464 QDomElement styleElem = doc.createElement( QStringLiteral(
"Style" ) );
465 styleElem.setAttribute( QStringLiteral(
"current" ), QStringLiteral(
"true" ) );
466 QDomElement styleNameElem = doc.createElement( QStringLiteral(
"Name" ) );
467 const QDomText styleNameText = doc.createTextNode( styleName );
468 styleNameElem.appendChild( styleNameText );
469 QDomElement styleTitleElem = doc.createElement( QStringLiteral(
"Title" ) );
470 const QDomText styleTitleText = doc.createTextNode( styleName );
471 styleTitleElem.appendChild( styleTitleText );
472 styleElem.appendChild( styleNameElem );
473 styleElem.appendChild( styleTitleElem );
474 styleListElem.appendChild( styleElem );
475 layerElem.appendChild( styleListElem );
A helper class that centralizes restrictions given by all the access control filter plugins.
bool layerReadPermission(const QgsMapLayer *layer) const
Returns the layer read right.
This class represents a coordinate reference system (CRS).
bool hasAxisInverted() const
Returns whether axis is inverted (e.g., for WMS 1.3) for the CRS.
Custom exception class for Coordinate Reference System related exceptions.
Layer tree group node serves as a container for layers and further groups.
QString name() const override
Returns the group's name.
Layer tree node points to a map layer.
QgsMapLayer * layer() const
Returns the map layer associated with this node.
This class is a base class for nodes in a layer tree.
@ NodeGroup
Container of other groups and layers.
QList< QgsLayerTreeNode * > children()
Gets list of children of the node. Children are owned by the parent.
NodeType nodeType() const
Find out about type of the node. It is usually shorter to use convenience functions from QgsLayerTree...
bool itemVisibilityChecked() const
Returns whether a node is checked (independently of its ancestors or children)
Namespace with helper functions for layer tree operations.
QStringList styles() const
Returns list of all defined style names.
Base class for all map layer types.
virtual QgsRectangle extent() const
Returns the extent of the layer.
QgsMapLayer::LayerFlags flags() const
Returns the flags for this layer.
QgsCoordinateReferenceSystem crs
QgsMapLayerServerProperties * serverProperties()
Returns QGIS Server Properties for the map layer.
QString id() const
Returns the layer's unique ID, which is used to access this layer from QgsProject.
QString abstract() const
Returns the abstract of the layer used by QGIS Server in GetCapabilities request.
QString dataUrlFormat() const
Returns the DataUrl format of the layer used by QGIS Server in GetCapabilities request.
QString shortName() const
Returns the short name of the layer used by QGIS Server to identify the layer.
QString title() const
Returns the title of the layer used by QGIS Server in GetCapabilities request.
QString dataUrl() const
Returns the DataUrl of the layer used by QGIS Server in GetCapabilities request.
bool hasScaleBasedVisibility() const
Returns whether scale based visibility is enabled for the layer.
@ Identifiable
If the layer is identifiable using the identify map tool and as a WMS layer.
double minimumScale() const
Returns the minimum map scale (i.e.
QgsMapLayerStyleManager * styleManager() const
Gets access to the layer's style manager.
double maximumScale() const
Returns the maximum map scale (i.e.
QString keywordList() const
Returns the keyword list of the layer used by QGIS Server in GetCapabilities request.
Encapsulates a QGIS project, including sets of map layers and their styles, layouts,...
QString baseName() const
Returns the base name of the project file without the path and without extension - derived from fileN...
QgsLayerTree * layerTreeRoot() const
Returns pointer to the root (invisible) node of the project's layer tree.
QgsCoordinateReferenceSystem crs
A rectangle specified with double values.
double yMaximum() const SIP_HOLDGIL
Returns the y maximum value (top side of rectangle).
double xMaximum() const SIP_HOLDGIL
Returns the x maximum value (right side of rectangle).
double xMinimum() const SIP_HOLDGIL
Returns the x minimum value (left side of rectangle).
double yMinimum() const SIP_HOLDGIL
Returns the y minimum value (bottom side of rectangle).
void combineExtentWith(const QgsRectangle &rect)
Expands the rectangle so that it covers both the original rectangle and the given rectangle.
bool isEmpty() const
Returns true if the rectangle is empty.
void invert()
Swap x/y coordinates in the rectangle.
A helper class that centralizes caches accesses given by all the server cache filter plugins.
bool setCachedDocument(const QDomDocument *doc, const QgsProject *project, const QgsServerRequest &request, QgsAccessControl *accessControl) const
Updates or inserts the document in cache like capabilities.
bool getCachedDocument(QDomDocument *doc, const QgsProject *project, const QgsServerRequest &request, QgsAccessControl *accessControl) const
Returns cached document (or 0 if document not in cache) like capabilities.
QgsServerInterface Class defining interfaces exposed by QGIS Server and made available to plugins.
virtual QgsServerCacheManager * cacheManager() const =0
Gets the registered server cache filters.
virtual QgsAccessControl * accessControls() const =0
Gets the registered access control filters.
virtual QgsServerSettings * serverSettings()=0
Returns the server settings.
QgsServerResponse Class defining response interface passed to services QgsService::executeRequest() m...
virtual void write(const QString &data)
Write string This is a convenient method that will write directly to the underlying I/O device.
virtual void setHeader(const QString &key, const QString &value)=0
Set Header entry Add Header entry to the response Note that it is usually an error to set Header afte...
Class defining request interface passed to WMS service.
SERVER_EXPORT bool wmsInfoFormatSia2045(const QgsProject &project)
Returns if the info format is SIA20145.
SERVER_EXPORT QgsRectangle wmsExtent(const QgsProject &project)
Returns the WMS Extent restriction.
SERVER_EXPORT bool wmsUseLayerIds(const QgsProject &project)
Returns if layer ids are used as name in WMS.
SERVER_EXPORT QStringList owsServiceKeywords(const QgsProject &project)
Returns the owsService keywords defined in project.
SERVER_EXPORT QStringList wmsRestrictedLayers(const QgsProject &project)
Returns the restricted layer name list.
SERVER_EXPORT QString owsServiceTitle(const QgsProject &project)
Returns the owsService title defined in project.
SERVER_EXPORT QString owsServiceAbstract(const QgsProject &project)
Returns the owsService abstract defined in project.
Median cut implementation.
void writeGetContext(QgsServerInterface *serverIface, const QgsProject *project, const QgsWmsRequest &request, QgsServerResponse &response)
Output GetContext response.
QDomDocument getContext(QgsServerInterface *serverIface, const QgsProject *project, const QgsWmsRequest &request)
Returns XML document for the 'GetContext' request.
QUrl serviceUrl(const QgsServerRequest &request, const QgsProject *project, const QgsServerSettings &settings)
Returns WMS service URL.