QGIS API Documentation 4.1.0-Master (5bf3c20f3c9)
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 { u"application/geo+json"_s, u"application/vnd.geo+json"_s, u"application/geojson"_s };
36 map[QgsServerOgcApi::ContentType::HTML] = QStringList { u"text/html"_s };
37 map[QgsServerOgcApi::ContentType::OPENAPI3] = QStringList { u"application/vnd.oai.openapi+json;version=3.0"_s };
38 map[QgsServerOgcApi::ContentType::XML] = QStringList { u"application/xml"_s };
39 return map;
40}();
41
42QHash<QgsServerOgcApi::ContentType, QList<QgsServerOgcApi::ContentType>> QgsServerOgcApi::sContentTypeAliases = []() -> QHash<ContentType, QList<ContentType>> {
43 QHash<QgsServerOgcApi::ContentType, QList<QgsServerOgcApi::ContentType>> map;
45 return map;
46}();
47
48
49QgsServerOgcApi::QgsServerOgcApi( QgsServerInterface *serverIface, const QString &rootPath, const QString &name, const QString &description, const QString &version )
51 , mRootPath( rootPath )
52 , mName( name )
53 , mDescription( description )
54 , mVersion( version )
55{}
56
58{
59 //qDebug() << "API destroyed: " << name();
60}
61
63{
64 std::shared_ptr<QgsServerOgcApiHandler> hp( handler );
65 mHandlers.emplace_back( std::move( hp ) );
66}
67
68QUrl QgsServerOgcApi::sanitizeUrl( const QUrl &url )
69{
70 // Since QT 5.12 NormalizePathSegments does not collapse double slashes
71 QUrl u { url.adjusted( QUrl::StripTrailingSlash | QUrl::NormalizePathSegments ) };
72 if ( u.path().contains( "//"_L1 ) )
73 {
74 u.setPath( u.path().replace( "//"_L1, QChar( '/' ) ) );
75 }
76 // Make sure the path starts with '/'
77 if ( !u.path().startsWith( '/' ) )
78 {
79 u.setPath( u.path().prepend( '/' ) );
80 }
81 return u;
82}
83
85{
86 // Get url
87 const auto path { sanitizeUrl( context.handlerPath() ).path() };
88 // Find matching handler
89 auto hasMatch { false };
90 for ( const auto &handler : mHandlers )
91 {
92 QgsMessageLog::logMessage( u"Checking API path %1 for %2 "_s.arg( path, handler->path().pattern() ), u"Server"_s, Qgis::MessageLevel::Info );
93 if ( handler->path().match( path ).hasMatch() )
94 {
95 hasMatch = true;
96 // Execute handler
97 QgsMessageLog::logMessage( u"API %1: found handler %2"_s.arg( name(), QString::fromStdString( handler->operationId() ) ), u"Server"_s, Qgis::MessageLevel::Info );
98 // May throw QgsServerApiBadRequestException or JSON exceptions on serializing
99 try
100 {
101 handler->handleRequest( context );
102 }
103 catch ( json::exception &ex )
104 {
105 throw QgsServerApiInternalServerError( u"The API handler returned an error: %1"_s.arg( ex.what() ) );
106 }
107 break;
108 }
109 }
110 // Throw
111 if ( !hasMatch )
112 {
113 throw QgsServerApiBadRequestException( u"Requested URI does not match any registered API handler"_s );
114 }
115}
116
117const QMap<QgsServerOgcApi::ContentType, QStringList> QgsServerOgcApi::contentTypeMimes()
118{
119 return sContentTypeMime;
120}
121
122const QHash<QgsServerOgcApi::ContentType, QList<QgsServerOgcApi::ContentType>> QgsServerOgcApi::contentTypeAliases()
123{
124 return sContentTypeAliases;
125}
126
127std::string QgsServerOgcApi::relToString( const Rel &rel )
128{
129 static const QMetaEnum metaEnum = QMetaEnum::fromType<QgsServerOgcApi::Rel>();
130 std::string val { metaEnum.valueToKey( rel ) };
131 std::replace( val.begin(), val.end(), '_', '-' );
132 return val;
133}
134
136{
137 static const QMetaEnum metaEnum = QMetaEnum::fromType<ContentType>();
138 QString result { metaEnum.valueToKey( ct ) };
139 return result.replace( '_', '-' );
140}
141
143{
144 static const QMetaEnum metaEnum = QMetaEnum::fromType<ContentType>();
145 return metaEnum.valueToKey( ct );
146}
147
149{
150 return contentTypeToString( ct ).toLower();
151}
152
154{
155 const QString exts = QString::fromStdString( extension );
156 const auto constMimeTypes( QgsServerOgcApi::contentTypeMimes() );
157 for ( auto it = constMimeTypes.constBegin(); it != constMimeTypes.constEnd(); ++it )
158 {
159 const auto constValues = it.value();
160 for ( const auto &value : constValues )
161 {
162 if ( value.contains( exts, Qt::CaseSensitivity::CaseInsensitive ) )
163 {
164 return it.key();
165 }
166 }
167 }
168 // Default to JSON, but log a warning!
169 QgsMessageLog::logMessage( u"Content type for extension %1 not found! Returning default (JSON)"_s.arg( exts ), u"Server"_s, Qgis::MessageLevel::Warning );
171}
172
174{
175 if ( !sContentTypeMime.contains( contentType ) )
176 {
177 return "";
178 }
179 return sContentTypeMime.value( contentType ).first().toStdString();
180}
181
182const std::vector<std::shared_ptr<QgsServerOgcApiHandler>> QgsServerOgcApi::handlers() const
183{
184 return mHandlers;
185}
@ Warning
Warning message.
Definition qgis.h:162
@ Info
Information message.
Definition qgis.h:161
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(), Qgis::StringFormat format=Qgis::StringFormat::PlainText)
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.