Line data Source code
1 : /******************************************************************************
2 : *
3 : * Project: OpenGIS Simple Features Reference Implementation
4 : * Purpose: Implements OGRSQLiteDriver class.
5 : * Author: Frank Warmerdam, warmerdam@pobox.com
6 : *
7 : ******************************************************************************
8 : *
9 : * Contributor: Alessandro Furieri, a.furieri@lqt.it
10 : * Portions of this module properly supporting SpatiaLite DB creation
11 : * Developed for Faunalia ( http://www.faunalia.it) with funding from
12 : * Regione Toscana - Settore SISTEMA INFORMATIVO TERRITORIALE ED AMBIENTALE
13 : *
14 : ******************************************************************************
15 : * Copyright (c) 2004, Frank Warmerdam <warmerdam@pobox.com>
16 : * Copyright (c) 2007-2013, Even Rouault <even dot rouault at spatialys.com>
17 : *
18 : * SPDX-License-Identifier: MIT
19 : ****************************************************************************/
20 :
21 : #include "cpl_port.h"
22 : #include "ogr_sqlite.h"
23 :
24 : #include <cstring>
25 : #include <string>
26 :
27 : #include "cpl_conv.h"
28 : #include "cpl_error.h"
29 : #include "cpl_string.h"
30 : #include "cpl_vsi.h"
31 : #include "gdal.h"
32 : #include "gdal_priv.h"
33 : #include "ogr_core.h"
34 : #include "sqlite3.h"
35 :
36 : /************************************************************************/
37 : /* OGRSQLiteDriverIdentify() */
38 : /************************************************************************/
39 :
40 47688 : static int OGRSQLiteDriverIdentify(GDALOpenInfo *poOpenInfo)
41 :
42 : {
43 47688 : if (STARTS_WITH_CI(poOpenInfo->pszFilename, "SQLITE:"))
44 : {
45 156 : return TRUE;
46 : }
47 :
48 95064 : CPLString osExt(CPLGetExtensionSafe(poOpenInfo->pszFilename));
49 47532 : if (EQUAL(osExt, "gpkg") && GDALGetDriverByName("GPKG") != nullptr)
50 : {
51 1486 : return FALSE;
52 : }
53 46046 : if (EQUAL(osExt, "mbtiles") && GDALGetDriverByName("MBTILES") != nullptr)
54 : {
55 265 : if (CSLCount(poOpenInfo->papszAllowedDrivers) == 1 &&
56 74 : EQUAL(poOpenInfo->papszAllowedDrivers[0], "SQLite"))
57 : {
58 74 : return TRUE;
59 : }
60 117 : return FALSE;
61 : }
62 :
63 45857 : if (STARTS_WITH_CI(poOpenInfo->pszFilename, "VirtualShape:") &&
64 2 : EQUAL(osExt, "shp"))
65 : {
66 2 : return TRUE;
67 : }
68 :
69 : #ifdef HAVE_RASTERLITE2
70 : if (STARTS_WITH_CI(poOpenInfo->pszFilename, "RASTERLITE2:"))
71 : return poOpenInfo->nOpenFlags & GDAL_OF_RASTER;
72 : #endif
73 :
74 45853 : if (EQUAL(poOpenInfo->pszFilename, ":memory:"))
75 92 : return TRUE;
76 :
77 : #ifdef SQLITE_OPEN_URI
78 : // This code enables support for named memory databases in SQLite.
79 : // Named memory databases use file name format
80 : // file:name?mode=memory&cache=shared
81 : // SQLITE_USE_URI is checked only to enable backward compatibility, in case
82 : // we accidentally hijacked some other format.
83 45763 : if (STARTS_WITH(poOpenInfo->pszFilename, "file:") &&
84 2 : CPLTestBool(CPLGetConfigOption("SQLITE_USE_URI", "YES")))
85 : {
86 2 : char *queryparams = strchr(poOpenInfo->pszFilename, '?');
87 2 : if (queryparams)
88 : {
89 2 : if (strstr(queryparams, "mode=memory") != nullptr)
90 2 : return TRUE;
91 : }
92 : }
93 : #endif
94 :
95 : /* -------------------------------------------------------------------- */
96 : /* Verify that the target is a real file, and has an */
97 : /* appropriate magic string at the beginning. */
98 : /* -------------------------------------------------------------------- */
99 45759 : if (poOpenInfo->nHeaderBytes < 100)
100 42581 : return FALSE;
101 :
102 : #ifdef ENABLE_SQL_SQLITE_FORMAT
103 3178 : if (STARTS_WITH(reinterpret_cast<const char *>(poOpenInfo->pabyHeader),
104 : "-- SQL SQLITE"))
105 : {
106 2 : return TRUE;
107 : }
108 3176 : if (STARTS_WITH(reinterpret_cast<const char *>(poOpenInfo->pabyHeader),
109 : "-- SQL RASTERLITE"))
110 : {
111 2 : return -1;
112 : }
113 3174 : if (STARTS_WITH(reinterpret_cast<const char *>(poOpenInfo->pabyHeader),
114 : "-- SQL MBTILES"))
115 : {
116 0 : if (GDALGetDriverByName("MBTILES") != nullptr)
117 0 : return FALSE;
118 0 : if (poOpenInfo->eAccess == GA_Update)
119 0 : return FALSE;
120 0 : return -1;
121 : }
122 : #endif
123 :
124 3174 : if (!STARTS_WITH(reinterpret_cast<const char *>(poOpenInfo->pabyHeader),
125 : "SQLite format 3"))
126 2625 : return FALSE;
127 :
128 : // In case we are opening /vsizip/foo.zip with a .gpkg inside
129 1647 : if ((memcmp(poOpenInfo->pabyHeader + 68, "GP10", 4) == 0 ||
130 549 : memcmp(poOpenInfo->pabyHeader + 68, "GP11", 4) == 0 ||
131 1107 : memcmp(poOpenInfo->pabyHeader + 68, "GPKG", 4) == 0) &&
132 9 : GDALGetDriverByName("GPKG") != nullptr)
133 : {
134 9 : return FALSE;
135 : }
136 :
137 : // Could be a Rasterlite or MBTiles file as well
138 540 : return -1;
139 : }
140 :
141 : /************************************************************************/
142 : /* Open() */
143 : /************************************************************************/
144 :
145 424 : static GDALDataset *OGRSQLiteDriverOpen(GDALOpenInfo *poOpenInfo)
146 :
147 : {
148 424 : if (OGRSQLiteDriverIdentify(poOpenInfo) == FALSE)
149 0 : return nullptr;
150 :
151 : /* -------------------------------------------------------------------- */
152 : /* Check VirtualShape:xxx.shp syntax */
153 : /* -------------------------------------------------------------------- */
154 424 : const auto nLen = strlen(poOpenInfo->pszFilename);
155 424 : if (STARTS_WITH_CI(poOpenInfo->pszFilename, "VirtualShape:") && nLen > 4 &&
156 1 : EQUAL(poOpenInfo->pszFilename + nLen - 4, ".SHP"))
157 : {
158 2 : auto poDS = std::make_unique<OGRSQLiteDataSource>();
159 :
160 1 : char **papszOptions = CSLAddString(nullptr, "SPATIALITE=YES");
161 1 : int nRet = poDS->Create(":memory:", papszOptions);
162 1 : poDS->SetDescription(poOpenInfo->pszFilename);
163 1 : CSLDestroy(papszOptions);
164 1 : if (!nRet)
165 : {
166 0 : return nullptr;
167 : }
168 :
169 : char *pszSQLiteFilename =
170 1 : CPLStrdup(poOpenInfo->pszFilename + strlen("VirtualShape:"));
171 2 : if (!std::unique_ptr<GDALDataset>(GDALDataset::Open(
172 1 : pszSQLiteFilename, GDAL_OF_VECTOR, nullptr, nullptr, nullptr)))
173 : {
174 0 : CPLFree(pszSQLiteFilename);
175 0 : return nullptr;
176 : }
177 :
178 1 : char *pszLastDot = strrchr(pszSQLiteFilename, '.');
179 1 : if (pszLastDot)
180 1 : *pszLastDot = '\0';
181 :
182 2 : const std::string osTableName = CPLGetBasenameSafe(pszSQLiteFilename);
183 :
184 1 : char *pszSQL = CPLStrdup(CPLSPrintf(
185 : "CREATE VIRTUAL TABLE %s USING VirtualShape(%s, CP1252, -1)",
186 : osTableName.c_str(), pszSQLiteFilename));
187 1 : poDS->ExecuteSQL(pszSQL, nullptr, nullptr);
188 1 : CPLFree(pszSQL);
189 1 : CPLFree(pszSQLiteFilename);
190 1 : poDS->DisableUpdate();
191 1 : return poDS.release();
192 : }
193 :
194 : /* -------------------------------------------------------------------- */
195 : /* We think this is really an SQLite database, go ahead and try */
196 : /* and open it. */
197 : /* -------------------------------------------------------------------- */
198 423 : OGRSQLiteDataSource *poDS = new OGRSQLiteDataSource();
199 :
200 423 : if (!poDS->Open(poOpenInfo))
201 : {
202 4 : delete poDS;
203 4 : return nullptr;
204 : }
205 : else
206 419 : return poDS;
207 : }
208 :
209 : /************************************************************************/
210 : /* Create() */
211 : /************************************************************************/
212 :
213 235 : static GDALDataset *OGRSQLiteDriverCreate(const char *pszName, int nBands,
214 : CPL_UNUSED int nXSize,
215 : CPL_UNUSED int nYSize,
216 : CPL_UNUSED GDALDataType eDT,
217 : char **papszOptions)
218 : {
219 235 : if (nBands != 0)
220 : {
221 0 : CPLError(CE_Failure, CPLE_NotSupported,
222 : "Raster creation through Create() interface is not supported. "
223 : "Only CreateCopy() is supported");
224 0 : return nullptr;
225 : }
226 :
227 : /* -------------------------------------------------------------------- */
228 : /* First, ensure there isn't any such file yet. */
229 : /* -------------------------------------------------------------------- */
230 : VSIStatBufL sStatBuf;
231 :
232 235 : if (VSIStatL(pszName, &sStatBuf) == 0)
233 : {
234 0 : CPLError(CE_Failure, CPLE_AppDefined,
235 : "It seems a file system object called '%s' already exists.",
236 : pszName);
237 :
238 0 : return nullptr;
239 : }
240 :
241 : /* -------------------------------------------------------------------- */
242 : /* Try to create datasource. */
243 : /* -------------------------------------------------------------------- */
244 235 : OGRSQLiteDataSource *poDS = new OGRSQLiteDataSource();
245 :
246 235 : if (!poDS->Create(pszName, papszOptions))
247 : {
248 5 : delete poDS;
249 5 : return nullptr;
250 : }
251 :
252 230 : return poDS;
253 : }
254 :
255 : /************************************************************************/
256 : /* Delete() */
257 : /************************************************************************/
258 :
259 86 : static CPLErr OGRSQLiteDriverDelete(const char *pszName)
260 : {
261 86 : if (VSIUnlink(pszName) == 0)
262 64 : return CE_None;
263 : else
264 22 : return CE_Failure;
265 : }
266 :
267 : /************************************************************************/
268 : /* RegisterOGRSQLite() */
269 : /************************************************************************/
270 :
271 1682 : void RegisterOGRSQLite()
272 :
273 : {
274 1682 : if (!GDAL_CHECK_VERSION("SQLite driver"))
275 301 : return;
276 :
277 1682 : if (GDALGetDriverByName("SQLite") != nullptr)
278 301 : return;
279 :
280 1381 : GDALDriver *poDriver = new GDALDriver();
281 :
282 1381 : poDriver->SetDescription("SQLite");
283 1381 : poDriver->SetMetadataItem(GDAL_DCAP_VECTOR, "YES");
284 1381 : poDriver->SetMetadataItem(GDAL_DCAP_CREATE_LAYER, "YES");
285 1381 : poDriver->SetMetadataItem(GDAL_DCAP_DELETE_LAYER, "YES");
286 1381 : poDriver->SetMetadataItem(GDAL_DCAP_CREATE_FIELD, "YES");
287 1381 : poDriver->SetMetadataItem(GDAL_DCAP_DELETE_FIELD, "YES");
288 1381 : poDriver->SetMetadataItem(GDAL_DCAP_REORDER_FIELDS, "YES");
289 1381 : poDriver->SetMetadataItem(GDAL_DCAP_CURVE_GEOMETRIES, "YES");
290 1381 : poDriver->SetMetadataItem(GDAL_DCAP_MEASURED_GEOMETRIES, "YES");
291 1381 : poDriver->SetMetadataItem(GDAL_DCAP_Z_GEOMETRIES, "YES");
292 1381 : poDriver->SetMetadataItem(GDAL_DMD_SUPPORTED_SQL_DIALECTS, "SQLITE OGRSQL");
293 :
294 : #ifdef HAVE_RASTERLITE2
295 : poDriver->SetMetadataItem(GDAL_DCAP_RASTER, "YES");
296 : poDriver->SetMetadataItem(GDAL_DMD_LONGNAME,
297 : "SQLite / Spatialite / RasterLite2");
298 : #else
299 1381 : poDriver->SetMetadataItem(GDAL_DMD_LONGNAME, "SQLite / Spatialite");
300 : #endif
301 1381 : poDriver->SetMetadataItem(GDAL_DMD_HELPTOPIC, "drivers/vector/sqlite.html");
302 1381 : poDriver->SetMetadataItem(GDAL_DMD_EXTENSIONS, "sqlite db");
303 :
304 1381 : poDriver->SetMetadataItem(
305 : GDAL_DMD_OPENOPTIONLIST,
306 : "<OpenOptionList>"
307 : " <Option name='LIST_ALL_TABLES' type='boolean' description='Whether "
308 : "all tables, including non-spatial ones, should be listed' "
309 : "default='NO'/>"
310 : " <Option name='LIST_VIRTUAL_OGR' type='boolean' description='Whether "
311 : "VirtualOGR virtual tables should be listed. Should only be enabled on "
312 : "trusted datasources to avoid potential safety issues' default='NO'/>"
313 : " <Option name='PRELUDE_STATEMENTS' type='string' description='SQL "
314 : "statement(s) to send on the SQLite connection before any other ones'/>"
315 : #ifdef HAVE_RASTERLITE2
316 : " <Option name='1BIT_AS_8BIT' type='boolean' scope='raster' "
317 : "description='Whether to promote 1-bit monochrome raster as 8-bit, so "
318 : "as to have higher quality overviews' default='YES'/>"
319 : #endif
320 : " <Option name='OGR_SCHEMA' type='string' description='"
321 : "Partially or totally overrides the auto-detected schema to use for "
322 : "creating the layer. "
323 : "The overrides are defined as a JSON list of field definitions. "
324 : "This can be a filename or a JSON string or a URL.'/>"
325 1381 : "</OpenOptionList>");
326 :
327 : CPLString osCreationOptions(
328 : "<CreationOptionList>"
329 : #ifdef HAVE_SPATIALITE
330 : " <Option name='SPATIALITE' type='boolean' description='Whether to "
331 : "create a Spatialite database' default='NO'/>"
332 : #endif
333 : " <Option name='METADATA' type='boolean' description='Whether to "
334 : "create the geometry_columns and spatial_ref_sys tables' "
335 : "default='YES'/>"
336 : " <Option name='INIT_WITH_EPSG' type='boolean' description='Whether "
337 : "to insert the content of the EPSG CSV files into the spatial_ref_sys "
338 : "table ' default='NO'/>"
339 : #ifdef HAVE_RASTERLITE2
340 : " <Option name='APPEND_SUBDATASET' scope='raster' type='boolean' "
341 : "description='Whether to add the raster to the existing file' "
342 : "default='NO'/>"
343 : " <Option name='COVERAGE' scope='raster' type='string' "
344 : "description='Coverage name'/>"
345 : " <Option name='SECTION' scope='raster' type='string' "
346 : "description='Section name'/>"
347 : " <Option name='COMPRESS' scope='raster' type='string-select' "
348 : "description='Raster compression' default='NONE'>"
349 : " <Value>NONE</Value>"
350 : #endif
351 2762 : );
352 : #ifdef HAVE_RASTERLITE2
353 : if (rl2_is_supported_codec(RL2_COMPRESSION_DEFLATE))
354 : osCreationOptions += " <Value>DEFLATE</Value>";
355 : if (rl2_is_supported_codec(RL2_COMPRESSION_LZMA))
356 : osCreationOptions += " <Value>LZMA</Value>";
357 : if (rl2_is_supported_codec(RL2_COMPRESSION_PNG))
358 : osCreationOptions += " <Value>PNG</Value>";
359 : if (rl2_is_supported_codec(RL2_COMPRESSION_CCITTFAX4))
360 : osCreationOptions += " <Value>CCITTFAX4</Value>";
361 : if (rl2_is_supported_codec(RL2_COMPRESSION_JPEG))
362 : osCreationOptions += " <Value>JPEG</Value>";
363 : if (rl2_is_supported_codec(RL2_COMPRESSION_LOSSY_WEBP))
364 : osCreationOptions += " <Value>WEBP</Value>";
365 : if (rl2_is_supported_codec(RL2_COMPRESSION_LOSSY_JP2))
366 : osCreationOptions += " <Value>JPEG2000</Value>";
367 : #endif
368 : osCreationOptions +=
369 : #ifdef HAVE_RASTERLITE2
370 : " </Option>"
371 : " <Option name='QUALITY' scope='raster' type='int' description='Image "
372 : "quality for JPEG, WEBP and JPEG2000 compressions'/>"
373 : " <Option name='PIXEL_TYPE' scope='raster' type='string-select' "
374 : "description='Raster pixel type. Determines photometric "
375 : "interpretation'>"
376 : " <Value>MONOCHROME</Value>"
377 : " <Value>PALETTE</Value>"
378 : " <Value>GRAYSCALE</Value>"
379 : " <Value>RGB</Value>"
380 : " <Value>MULTIBAND</Value>"
381 : " <Value>DATAGRID</Value>"
382 : " </Option>"
383 : " <Option name='BLOCKXSIZE' scope='raster' type='int' "
384 : "description='Block width' default='512'/>"
385 : " <Option name='BLOCKYSIZE' scope='raster' type='int' "
386 : "description='Block height' default='512'/>"
387 : " <Option name='NBITS' scope='raster' type='int' description='Force "
388 : "bit width. 1, 2 or 4 are supported'/>"
389 : " <Option name='PYRAMIDIZE' scope='raster' type='boolean' "
390 : "description='Whether to automatically build relevant "
391 : "pyramids/overviews' default='NO'/>"
392 : #endif
393 1381 : "</CreationOptionList>";
394 :
395 1381 : poDriver->SetMetadataItem(GDAL_DMD_CREATIONOPTIONLIST, osCreationOptions);
396 :
397 1381 : poDriver->SetMetadataItem(
398 : GDAL_DS_LAYER_CREATIONOPTIONLIST,
399 : "<LayerCreationOptionList>"
400 : " <Option name='FORMAT' type='string-select' description='Format of "
401 : "geometry columns'>"
402 : " <Value>WKB</Value>"
403 : " <Value>WKT</Value>"
404 : #ifdef HAVE_SPATIALITE
405 : " <Value>SPATIALITE</Value>"
406 : #endif
407 : " </Option>"
408 : " <Option name='GEOMETRY_NAME' type='string' description='Name of "
409 : "geometry column. Defaults to WKT_GEOMETRY for FORMAT=WKT or GEOMETRY "
410 : "otherwise'/>"
411 : " <Option name='LAUNDER' type='boolean' description='Whether layer "
412 : "and field names will be laundered' default='YES'/>"
413 : #ifdef HAVE_SPATIALITE
414 : " <Option name='SPATIAL_INDEX' type='boolean' description='Whether to "
415 : "create a spatial index for Spatialite databases' default='YES'/>"
416 : " <Option name='COMPRESS_GEOM' type='boolean' description='Whether to "
417 : "use compressed format of Spatialite geometries' default='NO'/>"
418 : #endif
419 : " <Option name='SRID' type='int' description='Forced SRID of the "
420 : "layer'/>"
421 : " <Option name='COMPRESS_COLUMNS' type='string' "
422 : "description='=column_name1[,column_name2, ...]. list of (String) "
423 : "columns that must be compressed with ZLib DEFLATE algorithm'/>"
424 : " <Option name='OVERWRITE' type='boolean' description='Whether to "
425 : "overwrite an existing table with the layer name to be created' "
426 : "default='NO'/>"
427 : " <Option name='FID' type='string' description='Name of the FID "
428 : "column to create' default='OGC_FID'/>"
429 : #if SQLITE_VERSION_NUMBER >= 3037000
430 : " <Option name='STRICT' type='boolean' description='Whether to create "
431 : "the table in STRICT mode (only compatible of readers with sqlite >= "
432 : "3.37)' default='NO'/>"
433 : #endif
434 1381 : "</LayerCreationOptionList>");
435 :
436 1381 : poDriver->SetMetadataItem(GDAL_DMD_CREATIONFIELDDATATYPES,
437 : "Integer Integer64 Real String Date DateTime "
438 : "Time Binary IntegerList Integer64List "
439 1381 : "RealList StringList");
440 1381 : poDriver->SetMetadataItem(GDAL_DMD_CREATIONFIELDDATASUBTYPES,
441 1381 : "Boolean Int16 Float32");
442 1381 : poDriver->SetMetadataItem(GDAL_DMD_CREATION_FIELD_DEFN_FLAGS,
443 1381 : "WidthPrecision Nullable Default Unique");
444 1381 : poDriver->SetMetadataItem(
445 : GDAL_DMD_ALTER_FIELD_DEFN_FLAGS,
446 1381 : "Name Type WidthPrecision Nullable Default Unique");
447 :
448 : #ifdef HAVE_RASTERLITE2
449 : poDriver->SetMetadataItem(GDAL_DMD_CREATIONDATATYPES,
450 : "Byte Int8 UInt16 Int16 UInt32 Int32 Float32 "
451 : "Float64");
452 : #endif
453 1381 : poDriver->SetMetadataItem(GDAL_DCAP_NOTNULL_FIELDS, "YES");
454 1381 : poDriver->SetMetadataItem(GDAL_DCAP_DEFAULT_FIELDS, "YES");
455 1381 : poDriver->SetMetadataItem(GDAL_DCAP_UNIQUE_FIELDS, "YES");
456 1381 : poDriver->SetMetadataItem(GDAL_DCAP_NOTNULL_GEOMFIELDS, "YES");
457 1381 : poDriver->SetMetadataItem(GDAL_DCAP_VIRTUALIO, "YES");
458 1381 : poDriver->SetMetadataItem(GDAL_DCAP_MULTIPLE_VECTOR_LAYERS, "YES");
459 1381 : poDriver->SetMetadataItem(GDAL_DCAP_RELATIONSHIPS, "YES");
460 1381 : poDriver->SetMetadataItem(GDAL_DCAP_CREATE_RELATIONSHIP, "YES");
461 1381 : poDriver->SetMetadataItem(GDAL_DMD_RELATIONSHIP_FLAGS,
462 1381 : "OneToMany Association Composite");
463 1381 : poDriver->SetMetadataItem(GDAL_DMD_RELATIONSHIP_RELATED_TABLE_TYPES,
464 1381 : "features");
465 :
466 : #ifdef ENABLE_SQL_SQLITE_FORMAT
467 1381 : poDriver->SetMetadataItem("ENABLE_SQL_SQLITE_FORMAT", "YES");
468 : #endif
469 : #ifdef SQLITE_HAS_COLUMN_METADATA
470 1381 : poDriver->SetMetadataItem("SQLITE_HAS_COLUMN_METADATA", "YES");
471 : #endif
472 :
473 1381 : poDriver->pfnOpen = OGRSQLiteDriverOpen;
474 1381 : poDriver->pfnIdentify = OGRSQLiteDriverIdentify;
475 1381 : poDriver->pfnCreate = OGRSQLiteDriverCreate;
476 : #ifdef HAVE_RASTERLITE2
477 : poDriver->pfnCreateCopy = OGRSQLiteDriverCreateCopy;
478 : #endif
479 1381 : poDriver->pfnDelete = OGRSQLiteDriverDelete;
480 1381 : poDriver->pfnUnloadDriver = OGRSQLiteDriverUnload;
481 :
482 1381 : GetGDALDriverManager()->RegisterDriver(poDriver);
483 : }
|