QGIS API Documentation 3.99.0-Master (e9821da5c6b)
Loading...
Searching...
No Matches
qgsserverogcapi.cpp
Go to the documentation of this file.
1/***************************************************************************
2 qgsserverogcapi.cpp - QgsServerOgcApi
3
4 ---------------------
5 begin : 10.7.2019
6 copyright : (C) 2019 by Alessandro Pasotti
7 email : elpaso at itopen dot it
8 ***************************************************************************
9 * *
10 * This program is free software; you can redistribute it and/or modify *
11 * it under the terms of the GNU General Public License as published by *
12 * the Free Software Foundation; either version 2 of the License, or *
13 * (at your option) any later version. *
14 * *
15 ***************************************************************************/
16
17#include "qgsserverogcapi.h"
18
19#include "qgsapplication.h"
20#include "qgsmessagelog.h"
22
23#include <QDebug>
24#include <QDir>
25#include <QString>
26#include <QtGlobal>
27
28#include "moc_qgsserverogcapi.cpp"
29
30using namespace Qt::StringLiterals;
31
32QMap<QgsServerOgcApi::ContentType, QStringList> QgsServerOgcApi::sContentTypeMime = []() -> QMap<QgsServerOgcApi::ContentType, QStringList> {
33 QMap<QgsServerOgcApi::ContentType, QStringList> map;
34 map[QgsServerOgcApi::ContentType::JSON] = QStringList { u"application/json"_s };
35 map[QgsServerOgcApi::ContentType::GEOJSON] = QStringList {
36 u"application/geo+json"_s,
37 u"application/vnd.geo+json"_s,
38 u"application/geojson"_s
39 };
40 map[QgsServerOgcApi::ContentType::HTML] = QStringList { u"text/html"_s };
41 map[QgsServerOgcApi::ContentType::OPENAPI3] = QStringList { u"application/vnd.oai.openapi+json;version=3.0"_s };
42 map[QgsServerOgcApi::ContentType::XML] = QStringList { u"application/xml"_s };
43 return map;
44}();
45
46QHash<QgsServerOgcApi::ContentType, QList<QgsServerOgcApi::ContentType>> QgsServerOgcApi::sContentTypeAliases = []() -> QHash<ContentType, QList<ContentType>> {
47 QHash<QgsServerOgcApi::ContentType, QList<QgsServerOgcApi::ContentType>> map;
49 return map;
50}();
51
52
53QgsServerOgcApi::QgsServerOgcApi( QgsServerInterface *serverIface, const QString &rootPath, const QString &name, const QString &description, const QString &version )
54 : QgsServerApi( serverIface ), mRootPath( rootPath ), mName( name ), mDescription( description ), mVersion( version )
55{
56}
57
59{
60 //qDebug() << "API destroyed: " << name();
61}
62
64{
65 std::shared_ptr<QgsServerOgcApiHandler> hp( handler );
66 mHandlers.emplace_back( std::move( hp ) );
67}
68
69QUrl QgsServerOgcApi::sanitizeUrl( const QUrl &url )
70{
71 // Since QT 5.12 NormalizePathSegments does not collapse double slashes
72 QUrl u { url.adjusted( QUrl::StripTrailingSlash | QUrl::NormalizePathSegments ) };
73 if ( u.path().contains( "//"_L1 ) )
74 {
75 u.setPath( u.path().replace( "//"_L1, QChar( '/' ) ) );
76 }
77 // Make sure the path starts with '/'
78 if ( !u.path().startsWith( '/' ) )
79 {
80 u.setPath( u.path().prepend( '/' ) );
81 }
82 return u;
83}
84
86{
87 // Get url
88 const auto path { sanitizeUrl( context.handlerPath() ).path() };
89 // Find matching handler
90 auto hasMatch { false };
91 for ( const auto &handler : mHandlers )
92 {
93 QgsMessageLog::logMessage( u"Checking API path %1 for %2 "_s.arg( path, handler->path().pattern() ), u"Server"_s, Qgis::MessageLevel::Info );
94 if ( handler->path().match( path ).hasMatch() )
95 {
96 hasMatch = true;
97 // Execute handler
98 QgsMessageLog::logMessage( u"API %1: found handler %2"_s.arg( name(), QString::fromStdString( handler->operationId() ) ), u"Server"_s, Qgis::MessageLevel::Info );
99 // May throw QgsServerApiBadRequestException or JSON exceptions on serializing
100 try
101 {
102 handler->handleRequest( context );
103 }
104 catch ( json::exception &ex )
105 {
106 throw QgsServerApiInternalServerError( u"The API handler returned an error: %1"_s.arg( ex.what() ) );
107 }
108 break;
109 }
110 }
111 // Throw
112 if ( !hasMatch )
113 {
114 throw QgsServerApiBadRequestException( u"Requested URI does not match any registered API handler"_s );
115 }
116}
117
118const QMap<QgsServerOgcApi::ContentType, QStringList> QgsServerOgcApi::contentTypeMimes()
119{
120 return sContentTypeMime;
121}
122
123const QHash<QgsServerOgcApi::ContentType, QList<QgsServerOgcApi::ContentType>> QgsServerOgcApi::contentTypeAliases()
124{
125 return sContentTypeAliases;
126}
127
128std::string QgsServerOgcApi::relToString( const Rel &rel )
129{
130 static const QMetaEnum metaEnum = QMetaEnum::fromType<QgsServerOgcApi::Rel>();
131 std::string val { metaEnum.valueToKey( rel ) };
132 std::replace( val.begin(), val.end(), '_', '-' );
133 return val;
134}
135
137{
138 static const QMetaEnum metaEnum = QMetaEnum::fromType<ContentType>();
139 QString result { metaEnum.valueToKey( ct ) };
140 return result.replace( '_', '-' );
141}
142
144{
145 static const QMetaEnum metaEnum = QMetaEnum::fromType<ContentType>();
146 return metaEnum.valueToKey( ct );
147}
148
150{
151 return contentTypeToString( ct ).toLower();
152}
153
155{
156 const QString exts = QString::fromStdString( extension );
157 const auto constMimeTypes( QgsServerOgcApi::contentTypeMimes() );
158 for ( auto it = constMimeTypes.constBegin();
159 it != constMimeTypes.constEnd();
160 ++it )
161 {
162 const auto constValues = it.value();
163 for ( const auto &value : constValues )
164 {
165 if ( value.contains( exts, Qt::CaseSensitivity::CaseInsensitive ) )
166 {
167 return it.key();
168 }
169 }
170 }
171 // Default to JSON, but log a warning!
172 QgsMessageLog::logMessage( u"Content type for extension %1 not found! Returning default (JSON)"_s.arg( exts ), u"Server"_s, Qgis::MessageLevel::Warning );
174}
175
177{
178 if ( !sContentTypeMime.contains( contentType ) )
179 {
180 return "";
181 }
182 return sContentTypeMime.value( contentType ).first().toStdString();
183}
184
185const std::vector<std::shared_ptr<QgsServerOgcApiHandler>> QgsServerOgcApi::handlers() const
186{
187 return mHandlers;
188}
@ Warning
Warning message.
Definition qgis.h:161
@ Info
Information message.
Definition qgis.h:160
static void logMessage(const QString &message, const QString &tag=QString(), Qgis::MessageLevel level=Qgis::MessageLevel::Warning, bool notifyUser=true, const char *file=__builtin_FILE(), const char *function=__builtin_FUNCTION(), int line=__builtin_LINE())
Adds a message to the log instance (and creates it if necessary).
Bad request error API exception.
Encapsulates the resources for a particular client request.
QString handlerPath() const
Returns the handler component of the URL path, i.e.
Internal server error API exception.
QgsServerApi(QgsServerInterface *serverIface)
Creates a QgsServerApi object.
QgsServerInterface * serverIface() const
Returns the server interface.
Defines interfaces exposed by QGIS Server and made available to plugins.
An abstract class which represents an OGC API handler to be registered in QgsServerOgcApi class.
static QUrl sanitizeUrl(const QUrl &url)
Returns a sanitized url with extra slashes removed and the path URL component that always starts with...
void executeRequest(const QgsServerApiContext &context) const override
Executes a request by passing the given context to the API handlers.
const QString rootPath() const override
Returns the root path for the API.
void registerHandler(Args... args)
Registers an OGC API handler passing Args to the constructor.
QgsServerOgcApi(QgsServerInterface *serverIface, const QString &rootPath, const QString &name, const QString &description=QString(), const QString &version=QString())
QgsServerOgcApi constructor.
static QString contentTypeToExtension(const QgsServerOgcApi::ContentType &ct)
Returns the file extension for a ct (Content-Type).
const QString description() const override
Returns the API description.
const std::vector< std::shared_ptr< QgsServerOgcApiHandler > > handlers() const
Returns registered handlers.
static const QMap< QgsServerOgcApi::ContentType, QStringList > contentTypeMimes()
Returns a map of contentType => list of mime types.
ContentType
Media types used for content negotiation, insert more specific first.
@ OPENAPI3
"application/openapi+json;version=3.0"
static QString contentTypeToString(const QgsServerOgcApi::ContentType &ct)
Returns the string representation of a ct (Content-Type) attribute.
Rel
Rel link types.
static std::string contentTypeToStdString(const QgsServerOgcApi::ContentType &ct)
Returns the string representation of a ct (Content-Type) attribute.
static std::string mimeType(const QgsServerOgcApi::ContentType &contentType)
Returns the mime-type for the contentType or an empty string if not found.
const QString version() const override
Returns the version of the service.
~QgsServerOgcApi() override
const QString name() const override
Returns the API name.
static std::string relToString(const QgsServerOgcApi::Rel &rel)
Returns the string representation of rel attribute.
static const QHash< QgsServerOgcApi::ContentType, QList< QgsServerOgcApi::ContentType > > contentTypeAliases()
Returns contentType specializations (e.g.
static QgsServerOgcApi::ContentType contentTypeFromExtension(const std::string &extension)
Returns the Content-Type value corresponding to extension.