QGIS API Documentation 3.99.0-Master (d270888f95f)
Loading...
Searching...
No Matches
qgsserviceregistry.cpp
Go to the documentation of this file.
1/***************************************************************************
2 qgsserviceregistry.cpp
3
4 Class defining the service manager for QGIS server services.
5 -------------------
6 begin : 2016-12-05
7 copyright : (C) 2016 by David Marteau
8 email : david dot marteau at 3liz dot com
9 ***************************************************************************/
10
11/***************************************************************************
12 * *
13 * This program is free software; you can redistribute it and/or modify *
14 * it under the terms of the GNU General Public License as published by *
15 * the Free Software Foundation; either version 2 of the License, or *
16 * (at your option) any later version. *
17 * *
18 ***************************************************************************/
19
20#include "qgsserviceregistry.h"
21
22#include <algorithm>
23#include <functional>
24
25#include "qgsmessagelog.h"
26#include "qgsserverapi.h"
27#include "qgsservice.h"
28
29#include <QString>
30
31using namespace Qt::StringLiterals;
32
33namespace
34{
35
36 // Build a key entry from name and version
37 QString makeServiceKey( const QString &name, const QString &version )
38 {
39 return QString( "%1_%2" ).arg( name, version );
40 }
41
42 // Compare two version strings:
43 // The strings are split into dot separated segment
44 // Each segment are compared up to the shortest number of segment of the
45 // lists. Remaining segments are dropped.
46 // If both segments can be interpreted as numbers the are compared as numbers, otherwise
47 // They are compared lexicographically.
48 // Return true if v1 is greater than v2
49 bool isVersionGreater( const QString &v1, const QString &v2 )
50 {
51 QStringList l1 = v1.split( '.' );
52 QStringList l2 = v2.split( '.' );
53 QStringList::iterator it1 = l1.begin();
54 QStringList::iterator it2 = l2.begin();
55 bool isint;
56 while ( it1 != l1.end() && it2 != l2.end() )
57 {
58 if ( *it1 != *it2 )
59 {
60 // Compare as numbers
61 const int i1 = it1->toInt( &isint );
62 if ( isint )
63 {
64 const int i2 = it2->toInt( &isint );
65 if ( isint && i1 != i2 )
66 {
67 return i1 > i2;
68 }
69 }
70 // Compare lexicographically
71 if ( !isint )
72 {
73 return *it1 > *it2;
74 }
75 }
76 ++it1;
77 ++it2;
78 }
79 // We reach the end of one of the list
80 return false;
81 }
82
83 // Check that two versions are c
84
85
86} // namespace
87
92
93QgsService *QgsServiceRegistry::getService( const QString &name, const QString &version )
94{
95 QgsService *service = nullptr;
96 QString key;
97
98 // Check that we have a service of that name
99 const VersionTable::const_iterator v = mServiceVersions.constFind( name );
100 if ( v != mServiceVersions.constEnd() )
101 {
102 key = version.isEmpty() ? v->second : makeServiceKey( name, version );
103 const ServiceTable::const_iterator it = mServices.constFind( key );
104 if ( it != mServices.constEnd() )
105 {
106 service = it->get();
107 }
108 else
109 {
110 // Return the default version
111 QgsMessageLog::logMessage( QString( "Service %1 %2 not found, returning default" ).arg( name, version ), u"Server"_s, Qgis::MessageLevel::Warning );
112 service = mServices[v->second].get();
113 }
114 }
115 else
116 {
117 QgsMessageLog::logMessage( QString( "Service %1 is not registered" ).arg( name ), u"Server"_s, Qgis::MessageLevel::Critical );
118 }
119 return service;
120}
121
123{
124 const QString name = service->name();
125 const QString version = service->version();
126
127 // Test if service is already registered
128 const QString key = makeServiceKey( name, version );
129 if ( mServices.constFind( key ) != mServices.constEnd() )
130 {
131 QgsMessageLog::logMessage( u"Error Service %1 %2 is already registered"_s.arg( name, version ), u"Server"_s, Qgis::MessageLevel::Critical );
132 return;
133 }
134
135 QgsMessageLog::logMessage( u"Adding service %1 %2"_s.arg( name, version ), u"Server"_s, Qgis::MessageLevel::Info );
136 mServices.insert( key, std::shared_ptr<QgsService>( service ) );
137
138 // Check the default version
139 // The first inserted service of a given name
140 // is the default one.
141 // this will ensure that native services are always
142 // the defaults.
143 const VersionTable::const_iterator v = mServiceVersions.constFind( name );
144 if ( v == mServiceVersions.constEnd() )
145 {
146 // Insert the service as the default one
147 mServiceVersions.insert( name, VersionTable::mapped_type( version, key ) );
148 }
149 /*
150 if ( v != mVersions.constEnd() )
151 {
152 if ( isVersionGreater( version, v->first ) )
153 {
154 // Replace the default version key
155 mVersions.insert( name, VersionTable::mapped_type( version, key ) );
156 }
157 }
158 else
159 {
160 // Insert the service as the default one
161 mVersions.insert( name, VersionTable::mapped_type( version, key ) );
162 }*/
163}
164
165int QgsServiceRegistry::unregisterApi( const QString &name, const QString &version )
166{
167 // Check that we have an API of that name
168 int removed = 0;
169 const VersionTable::const_iterator v = mApiVersions.constFind( name );
170 if ( v != mApiVersions.constEnd() )
171 {
172 if ( version.isEmpty() )
173 {
174 // No version specified, remove all versions
175 ApiTable::iterator it = mApis.begin();
176 while ( it != mApis.end() )
177 {
178 if ( ( *it )->name() == name )
179 {
180 QgsMessageLog::logMessage( QString( "Unregistering API %1 %2" ).arg( name, ( *it )->version() ), u"Server"_s, Qgis::MessageLevel::Info );
181 it = mApis.erase( it );
182 ++removed;
183 }
184 else
185 {
186 ++it;
187 }
188 }
189 // Remove from version table
190 mApiVersions.remove( name );
191 }
192 else
193 {
194 const QString key = makeServiceKey( name, version );
195 const ApiTable::iterator found = mApis.find( key );
196 if ( found != mApis.end() )
197 {
198 QgsMessageLog::logMessage( QString( "Unregistering API %1 %2" ).arg( name, version ), u"Server"_s, Qgis::MessageLevel::Info );
199 mApis.erase( found );
200 removed = 1;
201
202 // Find if we have other services of that name
203 // but with different version
204 //
205 QString maxVer;
206 const std::function<void( const ApiTable::mapped_type & )>
207 findGreaterVersion = [name, &maxVer]( const ApiTable::mapped_type &api ) {
208 if ( api->name() == name && ( maxVer.isEmpty() || isVersionGreater( api->version(), maxVer ) ) )
209 maxVer = api->version();
210 };
211
212 mApiVersions.remove( name );
213
214 std::for_each( mApis.constBegin(), mApis.constEnd(), findGreaterVersion );
215 if ( !maxVer.isEmpty() )
216 {
217 // Set the new default service
218 const QString key = makeServiceKey( name, maxVer );
219 mApiVersions.insert( name, VersionTable::mapped_type( version, key ) );
220 }
221 }
222 }
223 }
224 return removed;
225}
226
228{
229 for ( const auto &api : mApis )
230 {
231 QgsMessageLog::logMessage( u"Trying URL path: '%1' for '%2'"_s.arg( request.url().path(), api->rootPath() ), u"Server"_s, Qgis::MessageLevel::Info );
232 if ( api->accept( request.url() ) )
233 {
234 Q_ASSERT( !api->name().isEmpty() );
235 QgsMessageLog::logMessage( u"API %1 accepts the URL path '%2' "_s.arg( api->name(), request.url().path() ), u"Server"_s, Qgis::MessageLevel::Info );
236 return api.get();
237 }
238 }
239 return nullptr;
240}
241
242QgsServerApi *QgsServiceRegistry::getApi( const QString &name, const QString &version )
243{
244 QgsServerApi *api = nullptr;
245 QString key;
246
247 // Check that we have an API with that name
248 const VersionTable::const_iterator v = mApiVersions.constFind( name );
249 if ( v != mApiVersions.constEnd() )
250 {
251 key = version.isEmpty() ? v->second : makeServiceKey( name, version );
252 const ApiTable::const_iterator it = mApis.constFind( key );
253 if ( it != mApis.constEnd() )
254 {
255 api = it->get();
256 }
257 else
258 {
259 // Return the default version
260 QgsMessageLog::logMessage( QString( "API %1 %2 not found, returning default" ).arg( name, version ), u"Server"_s, Qgis::MessageLevel::Warning );
261 api = mApis[v->second].get();
262 }
263 }
264 else
265 {
266 QgsMessageLog::logMessage( QString( "API %1 is not registered" ).arg( name ), u"Server"_s, Qgis::MessageLevel::Critical );
267 }
268 return api;
269}
270
271int QgsServiceRegistry::unregisterService( const QString &name, const QString &version )
272{
273 // Check that we have a service of that name
274 int removed = 0;
275 const VersionTable::const_iterator v = mServiceVersions.constFind( name );
276 if ( v != mServiceVersions.constEnd() )
277 {
278 if ( version.isEmpty() )
279 {
280 // No version specified, remove all versions
281 ServiceTable::iterator it = mServices.begin();
282 while ( it != mServices.end() )
283 {
284 if ( ( *it )->name() == name )
285 {
286 QgsMessageLog::logMessage( QString( "Unregistering service %1 %2" ).arg( name, ( *it )->version() ), u"Server"_s, Qgis::MessageLevel::Info );
287 it = mServices.erase( it );
288 ++removed;
289 }
290 else
291 {
292 ++it;
293 }
294 }
295 // Remove from version table
296 mServiceVersions.remove( name );
297 }
298 else
299 {
300 const QString key = makeServiceKey( name, version );
301 const ServiceTable::iterator found = mServices.find( key );
302 if ( found != mServices.end() )
303 {
304 QgsMessageLog::logMessage( QString( "Unregistering service %1 %2" ).arg( name, version ), u"Server"_s, Qgis::MessageLevel::Info );
305 mServices.erase( found );
306 removed = 1;
307
308 // Find if we have other services of that name
309 // but with different version
310 //
311 QString maxVer;
312 const std::function<void( const ServiceTable::mapped_type & )>
313 findGreaterVersion = [name, &maxVer]( const ServiceTable::mapped_type &service ) {
314 if ( service->name() == name && ( maxVer.isEmpty() || isVersionGreater( service->version(), maxVer ) ) )
315 maxVer = service->version();
316 };
317
318 mServiceVersions.remove( name );
319
320 std::for_each( mServices.constBegin(), mServices.constEnd(), findGreaterVersion );
321 if ( !maxVer.isEmpty() )
322 {
323 // Set the new default service
324 const QString key = makeServiceKey( name, maxVer );
325 mServiceVersions.insert( name, VersionTable::mapped_type( version, key ) );
326 }
327 }
328 }
329 }
330 return removed;
331}
332
333void QgsServiceRegistry::init( const QString &nativeModulePath, QgsServerInterface *serverIface )
334{
335 mNativeLoader.loadModules( nativeModulePath, *this, serverIface );
336}
337
339{
340 // Release all services
341 mServiceVersions.clear();
342 mServices.clear();
343 mApis.clear();
344 mNativeLoader.unloadModules();
345}
346
348{
349 const QString name = api->name();
350 const QString version = api->version();
351
352 // Test if service is already registered
353 const QString key = makeServiceKey( name, version );
354 if ( mApis.constFind( key ) != mApis.constEnd() )
355 {
356 QgsMessageLog::logMessage( u"Error API %1 %2 is already registered"_s.arg( name, version ), u"Server"_s, Qgis::MessageLevel::Critical );
357 return false;
358 }
359
360 QgsMessageLog::logMessage( u"Adding API %1 %2 - root path: %3"_s.arg( name, version, api->rootPath() ), QString(), Qgis::MessageLevel::Info );
361 mApis.insert( key, std::shared_ptr<QgsServerApi>( api ) );
362
363 // Check the default version
364 // The first inserted service of a given name
365 // is the default one.
366 // this will ensure that native services are always
367 // the defaults.
368 const VersionTable::const_iterator v = mApiVersions.constFind( name );
369 if ( v == mApiVersions.constEnd() )
370 {
371 // Insert the service as the default one
372 mApiVersions.insert( name, VersionTable::mapped_type( version, key ) );
373 }
374 return true;
375}
@ Warning
Warning message.
Definition qgis.h:161
@ Critical
Critical/error message.
Definition qgis.h:162
@ 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).
Server generic API endpoint abstract base class.
virtual const QString version() const
Returns the version of the service.
virtual const QString name() const =0
Returns the API name.
virtual const QString rootPath() const =0
Returns the root path for the API.
Defines interfaces exposed by QGIS Server and made available to plugins.
Defines requests passed to QgsService classes.
QUrl url() const
Returns the request URL as seen by QGIS server.
void cleanUp()
Clean up registered service and unregister modules.
bool registerApi(QgsServerApi *api)
Registers the QgsServerApi api.
int unregisterService(const QString &name, const QString &version=QString())
Unregister service from its name and version.
int unregisterApi(const QString &name, const QString &version=QString())
Unregisters API from its name and version.
QgsServerApi * apiForRequest(const QgsServerRequest &request) const
Searches the API register for an API matching the request and returns a (possibly NULL) pointer to it...
void init(const QString &nativeModulepath, QgsServerInterface *serverIface=nullptr)
Initialize registry, load modules and auto register services.
QgsService * getService(const QString &name, const QString &version=QString())
Retrieve a service from its name.
void registerService(QgsService *service)
Register a service by its name and version.
QgsServerApi * getApi(const QString &name, const QString &version=QString())
Retrieves an API from its name.
Defines interfaces for QGIS server services.
Definition qgsservice.h:38
virtual QString name() const =0
Returns the name of the service.
virtual QString version() const =0
Returns the version of the service.