Line data Source code
1 : /******************************************************************************
2 : *
3 : * Project: OpenGIS Simple Features Reference Implementation
4 : * Purpose: Run SQL requests with SQLite SQL engine
5 : * Author: Even Rouault, even dot rouault at spatialys.com
6 : *
7 : ******************************************************************************
8 : * Copyright (c) 2012-2013, Even Rouault <even dot rouault at spatialys.com>
9 : *
10 : * SPDX-License-Identifier: MIT
11 : ****************************************************************************/
12 :
13 : #include "ogr_sqlite.h"
14 : #include "ogrsqliteexecutesql.h"
15 : #include "ogrsqlitevirtualogr.h"
16 : #include "ogrsqliteutility.h"
17 :
18 : #include <cctype>
19 : #include <cstdio>
20 : #include <cstring>
21 :
22 : #include "cpl_conv.h"
23 : #include "cpl_error.h"
24 : #include "cpl_multiproc.h"
25 : #include "cpl_port.h"
26 : #include "cpl_string.h"
27 : #include "cpl_vsi.h"
28 : #include "gdal_priv.h"
29 : #include "ogr_api.h"
30 : #include "ogr_core.h"
31 : #include "ogr_feature.h"
32 : #include "ogr_geometry.h"
33 : #include "ogr_spatialref.h"
34 : #include "sqlite3.h"
35 :
36 : /************************************************************************/
37 : /* OGR2SQLITEExtractUnquotedString() */
38 : /************************************************************************/
39 :
40 389 : static CPLString OGR2SQLITEExtractUnquotedString(const char **ppszSQLCommand)
41 : {
42 389 : CPLString osRet;
43 389 : const char *pszSQLCommand = *ppszSQLCommand;
44 389 : if (*pszSQLCommand == '"' || *pszSQLCommand == '\'')
45 : {
46 19 : const char chQuoteChar = *pszSQLCommand;
47 19 : pszSQLCommand++;
48 :
49 193 : while (*pszSQLCommand != '\0')
50 : {
51 193 : if (*pszSQLCommand == chQuoteChar &&
52 19 : pszSQLCommand[1] == chQuoteChar)
53 : {
54 0 : pszSQLCommand++;
55 0 : osRet += chQuoteChar;
56 : }
57 193 : else if (*pszSQLCommand == chQuoteChar)
58 : {
59 19 : pszSQLCommand++;
60 19 : break;
61 : }
62 : else
63 174 : osRet += *pszSQLCommand;
64 :
65 174 : pszSQLCommand++;
66 19 : }
67 : }
68 : else
69 : {
70 370 : bool bNotATableName = false;
71 370 : char chQuoteChar = 0;
72 370 : int nParenthesisLevel = 0;
73 2974 : while (*pszSQLCommand != '\0')
74 : {
75 2931 : if (*pszSQLCommand == chQuoteChar &&
76 0 : pszSQLCommand[1] == chQuoteChar)
77 : {
78 0 : osRet += *pszSQLCommand;
79 0 : pszSQLCommand++;
80 : }
81 2931 : else if (*pszSQLCommand == chQuoteChar)
82 : {
83 0 : chQuoteChar = 0;
84 : }
85 2931 : else if (chQuoteChar == 0)
86 : {
87 2931 : if (*pszSQLCommand == '(')
88 : {
89 2 : bNotATableName = true;
90 2 : nParenthesisLevel++;
91 : }
92 2929 : else if (*pszSQLCommand == ')')
93 : {
94 6 : nParenthesisLevel--;
95 6 : if (nParenthesisLevel < 0)
96 4 : break;
97 : }
98 2923 : else if (*pszSQLCommand == '"' || *pszSQLCommand == '\'')
99 : {
100 0 : chQuoteChar = *pszSQLCommand;
101 : }
102 2923 : else if (nParenthesisLevel == 0 &&
103 2907 : (isspace(static_cast<unsigned char>(*pszSQLCommand)) ||
104 2588 : *pszSQLCommand == '.' || *pszSQLCommand == ','))
105 : {
106 : break;
107 : }
108 : }
109 :
110 2604 : osRet += *pszSQLCommand;
111 2604 : pszSQLCommand++;
112 : }
113 370 : if (bNotATableName)
114 2 : osRet.clear();
115 : }
116 :
117 389 : *ppszSQLCommand = pszSQLCommand;
118 :
119 389 : return osRet;
120 : }
121 :
122 : /************************************************************************/
123 : /* OGR2SQLITEExtractLayerDesc() */
124 : /************************************************************************/
125 :
126 374 : static LayerDesc OGR2SQLITEExtractLayerDesc(const char **ppszSQLCommand)
127 : {
128 748 : std::string osStr;
129 374 : const char *pszSQLCommand = *ppszSQLCommand;
130 374 : LayerDesc oLayerDesc;
131 :
132 374 : while (isspace(static_cast<unsigned char>(*pszSQLCommand)))
133 0 : pszSQLCommand++;
134 :
135 374 : const char *pszOriginalStrStart = pszSQLCommand;
136 374 : oLayerDesc.osOriginalStr = pszSQLCommand;
137 :
138 374 : osStr = OGR2SQLITEExtractUnquotedString(&pszSQLCommand);
139 :
140 374 : if (*pszSQLCommand == '.')
141 : {
142 7 : oLayerDesc.osDSName = osStr;
143 7 : pszSQLCommand++;
144 : oLayerDesc.osLayerName =
145 7 : OGR2SQLITEExtractUnquotedString(&pszSQLCommand);
146 : }
147 : else
148 : {
149 367 : oLayerDesc.osLayerName = std::move(osStr);
150 : }
151 :
152 374 : oLayerDesc.osOriginalStr.resize(pszSQLCommand - pszOriginalStrStart);
153 :
154 374 : *ppszSQLCommand = pszSQLCommand;
155 :
156 748 : return oLayerDesc;
157 : }
158 :
159 : /************************************************************************/
160 : /* OGR2SQLITEAddLayer() */
161 : /************************************************************************/
162 :
163 374 : static void OGR2SQLITEAddLayer(const char *&pszStart, int &nNum,
164 : const char *&pszSQLCommand,
165 : std::set<LayerDesc> &oSet,
166 : CPLString &osModifiedSQL)
167 : {
168 748 : CPLString osTruncated(pszStart);
169 374 : osTruncated.resize(pszSQLCommand - pszStart);
170 374 : osModifiedSQL += osTruncated;
171 374 : pszStart = pszSQLCommand;
172 374 : LayerDesc oLayerDesc = OGR2SQLITEExtractLayerDesc(&pszSQLCommand);
173 374 : bool bInsert = true;
174 374 : if (oLayerDesc.osDSName.empty())
175 : {
176 367 : osTruncated = pszStart;
177 367 : osTruncated.resize(pszSQLCommand - pszStart);
178 367 : osModifiedSQL += osTruncated;
179 : }
180 : else
181 : {
182 7 : std::set<LayerDesc>::iterator oIter = oSet.find(oLayerDesc);
183 7 : if (oIter == oSet.end())
184 : {
185 : oLayerDesc.osSubstitutedName =
186 7 : CPLString().Printf("_OGR_%d", nNum++);
187 7 : osModifiedSQL += "\"";
188 7 : osModifiedSQL += oLayerDesc.osSubstitutedName;
189 7 : osModifiedSQL += "\"";
190 : }
191 : else
192 : {
193 0 : osModifiedSQL += (*oIter).osSubstitutedName;
194 0 : bInsert = false;
195 : }
196 : }
197 374 : if (bInsert)
198 : {
199 374 : oSet.insert(oLayerDesc);
200 : }
201 374 : pszStart = pszSQLCommand;
202 374 : }
203 :
204 : /************************************************************************/
205 : /* StartsAsSQLITEKeyWord() */
206 : /************************************************************************/
207 :
208 294 : static int StartsAsSQLITEKeyWord(const char *pszStr)
209 : {
210 85 : for (const char *pszKeyword : {"WHERE", "GROUP", "ORDER", "JOIN", "UNION",
211 379 : "INTERSECT", "EXCEPT", "LIMIT"})
212 : {
213 372 : if (STARTS_WITH_CI(pszStr, pszKeyword))
214 287 : return TRUE;
215 : }
216 7 : return FALSE;
217 : }
218 :
219 : /************************************************************************/
220 : /* OGR2SQLITEGetPotentialLayerNames() */
221 : /************************************************************************/
222 :
223 640 : static void OGR2SQLITEGetPotentialLayerNamesInternal(
224 : const char **ppszSQLCommand, std::set<LayerDesc> &oSetLayers,
225 : std::set<CPLString> &oSetSpatialIndex, CPLString &osModifiedSQL, int &nNum)
226 : {
227 640 : const char *pszSQLCommand = *ppszSQLCommand;
228 640 : const char *pszStart = pszSQLCommand;
229 640 : char ch = '\0';
230 640 : int nParenthesisLevel = 0;
231 640 : bool bLookforFTableName = false;
232 :
233 24764 : while ((ch = *pszSQLCommand) != '\0')
234 : {
235 24126 : if (ch == '(')
236 545 : nParenthesisLevel++;
237 23581 : else if (ch == ')')
238 : {
239 557 : nParenthesisLevel--;
240 557 : if (nParenthesisLevel < 0)
241 : {
242 2 : pszSQLCommand++;
243 2 : break;
244 : }
245 : }
246 :
247 : /* Skip literals and strings */
248 24124 : if (ch == '\'' || ch == '"')
249 : {
250 495 : char chEscapeChar = ch;
251 495 : pszSQLCommand++;
252 16948 : while ((ch = *pszSQLCommand) != '\0')
253 : {
254 16948 : if (ch == chEscapeChar && pszSQLCommand[1] == chEscapeChar)
255 0 : pszSQLCommand++;
256 16948 : else if (ch == chEscapeChar)
257 : {
258 495 : pszSQLCommand++;
259 495 : break;
260 : }
261 16453 : pszSQLCommand++;
262 495 : }
263 : }
264 :
265 23629 : else if (STARTS_WITH_CI(pszSQLCommand, "ogr_layer_"))
266 : {
267 182 : while (*pszSQLCommand != '\0' && *pszSQLCommand != '(')
268 172 : pszSQLCommand++;
269 :
270 10 : if (*pszSQLCommand != '(')
271 0 : break;
272 :
273 10 : pszSQLCommand++;
274 10 : nParenthesisLevel++;
275 :
276 10 : while (isspace(static_cast<unsigned char>(*pszSQLCommand)))
277 0 : pszSQLCommand++;
278 :
279 10 : OGR2SQLITEAddLayer(pszStart, nNum, pszSQLCommand, oSetLayers,
280 : osModifiedSQL);
281 : }
282 :
283 23619 : else if (bLookforFTableName &&
284 7 : STARTS_WITH_CI(pszSQLCommand, "f_table_name") &&
285 1 : (pszSQLCommand[strlen("f_table_name")] == '=' ||
286 1 : isspace(static_cast<unsigned char>(
287 1 : pszSQLCommand[strlen("f_table_name")]))))
288 : {
289 1 : pszSQLCommand += strlen("f_table_name");
290 :
291 2 : while (isspace(static_cast<unsigned char>(*pszSQLCommand)))
292 1 : pszSQLCommand++;
293 :
294 1 : if (*pszSQLCommand == '=')
295 : {
296 1 : pszSQLCommand++;
297 :
298 2 : while (isspace(static_cast<unsigned char>(*pszSQLCommand)))
299 1 : pszSQLCommand++;
300 :
301 : oSetSpatialIndex.insert(
302 1 : OGR2SQLITEExtractUnquotedString(&pszSQLCommand));
303 : }
304 :
305 1 : bLookforFTableName = false;
306 : }
307 :
308 23618 : else if (STARTS_WITH_CI(pszSQLCommand, "FROM") &&
309 467 : isspace(
310 467 : static_cast<unsigned char>(pszSQLCommand[strlen("FROM")])))
311 : {
312 338 : pszSQLCommand += strlen("FROM") + 1;
313 :
314 338 : while (isspace(static_cast<unsigned char>(*pszSQLCommand)))
315 0 : pszSQLCommand++;
316 :
317 338 : if (STARTS_WITH_CI(pszSQLCommand, "SpatialIndex") &&
318 1 : isspace(static_cast<unsigned char>(
319 1 : pszSQLCommand[strlen("SpatialIndex")])))
320 : {
321 1 : pszSQLCommand += strlen("SpatialIndex") + 1;
322 :
323 1 : bLookforFTableName = true;
324 :
325 1 : continue;
326 : }
327 :
328 337 : if (*pszSQLCommand == '(')
329 : {
330 2 : pszSQLCommand++;
331 :
332 2 : CPLString osTruncated(pszStart);
333 2 : osTruncated.resize(pszSQLCommand - pszStart);
334 2 : osModifiedSQL += osTruncated;
335 :
336 2 : OGR2SQLITEGetPotentialLayerNamesInternal(
337 : &pszSQLCommand, oSetLayers, oSetSpatialIndex, osModifiedSQL,
338 : nNum);
339 :
340 2 : pszStart = pszSQLCommand;
341 : }
342 : else
343 335 : OGR2SQLITEAddLayer(pszStart, nNum, pszSQLCommand, oSetLayers,
344 : osModifiedSQL);
345 :
346 348 : while (*pszSQLCommand != '\0')
347 : {
348 302 : if (isspace(static_cast<unsigned char>(*pszSQLCommand)))
349 : {
350 294 : pszSQLCommand++;
351 294 : while (isspace(static_cast<unsigned char>(*pszSQLCommand)))
352 0 : pszSQLCommand++;
353 :
354 294 : if (STARTS_WITH_CI(pszSQLCommand, "AS"))
355 : {
356 0 : pszSQLCommand += 2;
357 0 : while (
358 0 : isspace(static_cast<unsigned char>(*pszSQLCommand)))
359 0 : pszSQLCommand++;
360 : }
361 :
362 : /* Skip alias */
363 294 : if (*pszSQLCommand != '\0' && *pszSQLCommand != ',')
364 : {
365 294 : if (StartsAsSQLITEKeyWord(pszSQLCommand))
366 287 : break;
367 7 : OGR2SQLITEExtractUnquotedString(&pszSQLCommand);
368 : }
369 : }
370 8 : else if (*pszSQLCommand == ',')
371 : {
372 4 : pszSQLCommand++;
373 8 : while (isspace(static_cast<unsigned char>(*pszSQLCommand)))
374 4 : pszSQLCommand++;
375 :
376 4 : if (*pszSQLCommand == '(')
377 : {
378 0 : pszSQLCommand++;
379 :
380 0 : CPLString osTruncated(pszStart);
381 0 : osTruncated.resize(pszSQLCommand - pszStart);
382 0 : osModifiedSQL += osTruncated;
383 :
384 0 : OGR2SQLITEGetPotentialLayerNamesInternal(
385 : &pszSQLCommand, oSetLayers, oSetSpatialIndex,
386 : osModifiedSQL, nNum);
387 :
388 0 : pszStart = pszSQLCommand;
389 : }
390 : else
391 4 : OGR2SQLITEAddLayer(pszStart, nNum, pszSQLCommand,
392 : oSetLayers, osModifiedSQL);
393 : }
394 : else
395 4 : break;
396 337 : }
397 : }
398 23280 : else if (STARTS_WITH_CI(pszSQLCommand, "JOIN") &&
399 9 : isspace(
400 9 : static_cast<unsigned char>(pszSQLCommand[strlen("JOIN")])))
401 : {
402 5 : pszSQLCommand += strlen("JOIN") + 1;
403 5 : OGR2SQLITEAddLayer(pszStart, nNum, pszSQLCommand, oSetLayers,
404 : osModifiedSQL);
405 : }
406 23275 : else if (STARTS_WITH_CI(pszSQLCommand, "INTO") &&
407 12 : isspace(
408 12 : static_cast<unsigned char>(pszSQLCommand[strlen("INTO")])))
409 : {
410 12 : pszSQLCommand += strlen("INTO") + 1;
411 12 : OGR2SQLITEAddLayer(pszStart, nNum, pszSQLCommand, oSetLayers,
412 : osModifiedSQL);
413 : }
414 23263 : else if (STARTS_WITH_CI(pszSQLCommand, "UPDATE") &&
415 8 : isspace(static_cast<unsigned char>(
416 8 : pszSQLCommand[strlen("UPDATE")])))
417 : {
418 8 : pszSQLCommand += strlen("UPDATE") + 1;
419 8 : OGR2SQLITEAddLayer(pszStart, nNum, pszSQLCommand, oSetLayers,
420 : osModifiedSQL);
421 : }
422 : else
423 23255 : pszSQLCommand++;
424 : }
425 :
426 640 : CPLString osTruncated(pszStart);
427 640 : osTruncated.resize(pszSQLCommand - pszStart);
428 640 : osModifiedSQL += osTruncated;
429 :
430 640 : *ppszSQLCommand = pszSQLCommand;
431 640 : }
432 :
433 638 : static void OGR2SQLITEGetPotentialLayerNames(
434 : const char *pszSQLCommand, std::set<LayerDesc> &oSetLayers,
435 : std::set<CPLString> &oSetSpatialIndex, CPLString &osModifiedSQL)
436 : {
437 638 : int nNum = 1;
438 638 : OGR2SQLITEGetPotentialLayerNamesInternal(
439 : &pszSQLCommand, oSetLayers, oSetSpatialIndex, osModifiedSQL, nNum);
440 638 : }
441 :
442 : /************************************************************************/
443 : /* OGRSQLiteGetReferencedLayers() */
444 : /************************************************************************/
445 :
446 1 : std::set<LayerDesc> OGRSQLiteGetReferencedLayers(const char *pszStatement)
447 : {
448 : /* -------------------------------------------------------------------- */
449 : /* Analyze the statement to determine which tables will be used. */
450 : /* -------------------------------------------------------------------- */
451 1 : std::set<LayerDesc> oSetLayers;
452 2 : std::set<CPLString> oSetSpatialIndex;
453 2 : CPLString osModifiedSQL;
454 1 : OGR2SQLITEGetPotentialLayerNames(pszStatement, oSetLayers, oSetSpatialIndex,
455 : osModifiedSQL);
456 :
457 2 : return oSetLayers;
458 : }
459 :
460 : #ifndef HAVE_SQLITE3EXT_H
461 : OGRLayer *OGRSQLiteExecuteSQL(GDALDataset *, const char *, OGRGeometry *,
462 : const char *)
463 : {
464 : CPLError(CE_Failure, CPLE_NotSupported,
465 : "SQL SQLite dialect not supported due to GDAL being built "
466 : "without sqlite3ext.h header");
467 : return nullptr;
468 : }
469 :
470 : #else
471 :
472 : /************************************************************************/
473 : /* OGRSQLiteExecuteSQLLayer */
474 : /************************************************************************/
475 :
476 : class OGRSQLiteExecuteSQLLayer final : public OGRSQLiteSelectLayer
477 : {
478 : char *m_pszTmpDBName = nullptr;
479 : bool m_bStringsAsUTF8 = false;
480 :
481 : CPL_DISALLOW_COPY_ASSIGN(OGRSQLiteExecuteSQLLayer)
482 :
483 : public:
484 : OGRSQLiteExecuteSQLLayer(char *pszTmpDBName, OGRSQLiteDataSource *poDS,
485 : const CPLString &osSQL, sqlite3_stmt *hStmt,
486 : bool bUseStatementForGetNextFeature,
487 : bool bEmptyLayer, bool bCanReopenBaseDS,
488 : bool bStringsAsUTF8);
489 : virtual ~OGRSQLiteExecuteSQLLayer();
490 :
491 : int TestCapability(const char *pszCap) override;
492 : };
493 :
494 : /************************************************************************/
495 : /* OGRSQLiteExecuteSQLLayer() */
496 : /************************************************************************/
497 :
498 596 : OGRSQLiteExecuteSQLLayer::OGRSQLiteExecuteSQLLayer(
499 : char *pszTmpDBNameIn, OGRSQLiteDataSource *poDSIn, const CPLString &osSQL,
500 : sqlite3_stmt *hStmtIn, bool bUseStatementForGetNextFeature,
501 596 : bool bEmptyLayer, bool bCanReopenBaseDS, bool bStringsAsUTF8)
502 : : OGRSQLiteSelectLayer(poDSIn, osSQL, hStmtIn,
503 : bUseStatementForGetNextFeature, bEmptyLayer, true,
504 : bCanReopenBaseDS),
505 596 : m_pszTmpDBName(pszTmpDBNameIn), m_bStringsAsUTF8(bStringsAsUTF8)
506 : {
507 596 : }
508 :
509 : /************************************************************************/
510 : /* ~OGRSQLiteExecuteSQLLayer() */
511 : /************************************************************************/
512 :
513 1190 : OGRSQLiteExecuteSQLLayer::~OGRSQLiteExecuteSQLLayer()
514 : {
515 : // This is a bit peculiar: we must "finalize" the OGRLayer, since
516 : // it has objects that depend on the datasource, that we are just
517 : // going to destroy afterwards. The issue here is that we destroy
518 : // our own datasource,
519 595 : Finalize();
520 :
521 595 : delete m_poDS;
522 595 : VSIUnlink(m_pszTmpDBName);
523 595 : CPLFree(m_pszTmpDBName);
524 1190 : }
525 :
526 : /************************************************************************/
527 : /* TestCapability() */
528 : /************************************************************************/
529 :
530 1 : int OGRSQLiteExecuteSQLLayer::TestCapability(const char *pszCap)
531 : {
532 1 : if (EQUAL(pszCap, OLCStringsAsUTF8))
533 1 : return m_bStringsAsUTF8;
534 0 : return OGRSQLiteSelectLayer::TestCapability(pszCap);
535 : }
536 :
537 : /************************************************************************/
538 : /* OGR2SQLITE_IgnoreAllFieldsExceptGeometry() */
539 : /************************************************************************/
540 :
541 : #ifdef HAVE_SPATIALITE
542 2 : static void OGR2SQLITE_IgnoreAllFieldsExceptGeometry(OGRLayer *poLayer)
543 : {
544 2 : char **papszIgnored = nullptr;
545 2 : papszIgnored = CSLAddString(papszIgnored, "OGR_STYLE");
546 2 : OGRFeatureDefn *poFeatureDefn = poLayer->GetLayerDefn();
547 4 : for (int i = 0; i < poFeatureDefn->GetFieldCount(); i++)
548 : {
549 2 : papszIgnored = CSLAddString(
550 2 : papszIgnored, poFeatureDefn->GetFieldDefn(i)->GetNameRef());
551 : }
552 2 : poLayer->SetIgnoredFields(const_cast<const char **>(papszIgnored));
553 2 : CSLDestroy(papszIgnored);
554 2 : }
555 : #endif
556 :
557 : /************************************************************************/
558 : /* OGR2SQLITEDealWithSpatialColumn() */
559 : /************************************************************************/
560 : #if HAVE_SPATIALITE
561 : #define WHEN_SPATIALITE(arg) arg
562 : #else
563 : #define WHEN_SPATIALITE(arg)
564 : #endif
565 :
566 251 : static int OGR2SQLITEDealWithSpatialColumn(
567 : OGRLayer *poLayer, int iGeomCol, const LayerDesc &oLayerDesc,
568 : const CPLString &osTableName, OGRSQLiteDataSource *poSQLiteDS, sqlite3 *hDB,
569 : bool bSpatialiteDB, const std::set<LayerDesc> &WHEN_SPATIALITE(oSetLayers),
570 : const std::set<CPLString> &WHEN_SPATIALITE(oSetSpatialIndex))
571 : {
572 : OGRGeomFieldDefn *poGeomField =
573 251 : poLayer->GetLayerDefn()->GetGeomFieldDefn(iGeomCol);
574 502 : CPLString osGeomColRaw;
575 251 : if (iGeomCol == 0)
576 246 : osGeomColRaw = OGR2SQLITE_GetNameForGeometryColumn(poLayer);
577 : else
578 5 : osGeomColRaw = poGeomField->GetNameRef();
579 251 : const char *pszGeomColRaw = osGeomColRaw.c_str();
580 :
581 502 : CPLString osGeomColEscaped(SQLEscapeLiteral(pszGeomColRaw));
582 251 : const char *pszGeomColEscaped = osGeomColEscaped.c_str();
583 :
584 502 : CPLString osLayerNameEscaped(SQLEscapeLiteral(osTableName));
585 251 : const char *pszLayerNameEscaped = osLayerNameEscaped.c_str();
586 :
587 : CPLString osIdxNameRaw(
588 502 : CPLSPrintf("idx_%s_%s", oLayerDesc.osLayerName.c_str(), pszGeomColRaw));
589 502 : CPLString osIdxNameEscaped(SQLEscapeName(osIdxNameRaw));
590 :
591 : /* Make sure that the SRS is injected in spatial_ref_sys */
592 251 : const OGRSpatialReference *poSRS = poGeomField->GetSpatialRef();
593 251 : if (iGeomCol == 0 && poSRS == nullptr)
594 221 : poSRS = poLayer->GetSpatialRef();
595 251 : int nSRSId = poSQLiteDS->GetUndefinedSRID();
596 251 : if (poSRS != nullptr)
597 30 : nSRSId = poSQLiteDS->FetchSRSId(poSRS);
598 :
599 502 : CPLString osSQL;
600 : #ifdef HAVE_SPATIALITE
601 251 : bool bCreateSpatialIndex = false;
602 : #endif
603 251 : if (!bSpatialiteDB)
604 : {
605 : osSQL.Printf("INSERT INTO geometry_columns (f_table_name, "
606 : "f_geometry_column, geometry_format, geometry_type, "
607 : "coord_dimension, srid) "
608 : "VALUES ('%s','%s','SpatiaLite',%d,%d,%d)",
609 : pszLayerNameEscaped, pszGeomColEscaped,
610 58 : static_cast<int>(wkbFlatten(poLayer->GetGeomType())),
611 116 : wkbHasZ(poLayer->GetGeomType()) ? 3 : 2, nSRSId);
612 : }
613 : #ifdef HAVE_SPATIALITE
614 : else
615 : {
616 : /* We detect the need for creating a spatial index by 2 means : */
617 :
618 : /* 1) if there's an explicit reference to a
619 : * 'idx_layername_geometrycolumn' */
620 : /* table in the SQL --> old/traditional way of requesting spatial
621 : * indices */
622 : /* with spatialite. */
623 :
624 193 : std::set<LayerDesc>::const_iterator oIter2 = oSetLayers.begin();
625 392 : for (; oIter2 != oSetLayers.end(); ++oIter2)
626 : {
627 200 : const LayerDesc &oLayerDescIter = *oIter2;
628 200 : if (EQUAL(oLayerDescIter.osLayerName, osIdxNameRaw))
629 : {
630 1 : bCreateSpatialIndex = true;
631 1 : break;
632 : }
633 : }
634 :
635 : /* 2) or if there's a SELECT FROM SpatialIndex WHERE f_table_name =
636 : * 'layername' */
637 193 : if (!bCreateSpatialIndex)
638 : {
639 : std::set<CPLString>::const_iterator oIter3 =
640 192 : oSetSpatialIndex.begin();
641 193 : for (; oIter3 != oSetSpatialIndex.end(); ++oIter3)
642 : {
643 2 : const CPLString &osNameIter = *oIter3;
644 2 : if (EQUAL(osNameIter, oLayerDesc.osLayerName))
645 : {
646 1 : bCreateSpatialIndex = true;
647 1 : break;
648 : }
649 : }
650 : }
651 :
652 193 : if (poSQLiteDS->HasSpatialite4Layout())
653 : {
654 193 : const auto eGeomType = poLayer->GetGeomType();
655 193 : int nGeomType = eGeomType;
656 193 : int nCoordDimension = 2;
657 193 : if (wkbHasZ(eGeomType))
658 : {
659 1 : nGeomType += 1000;
660 1 : nCoordDimension = 3;
661 : }
662 :
663 : osSQL.Printf("INSERT INTO geometry_columns (f_table_name, "
664 : "f_geometry_column, geometry_type, coord_dimension, "
665 : "srid, spatial_index_enabled) "
666 : "VALUES (Lower('%s'),Lower('%s'),%d ,%d ,%d, %d)",
667 : pszLayerNameEscaped, pszGeomColEscaped, nGeomType,
668 : nCoordDimension, nSRSId,
669 193 : static_cast<int>(bCreateSpatialIndex));
670 : }
671 : else
672 : {
673 : const char *pszGeometryType =
674 0 : OGRToOGCGeomType(poLayer->GetGeomType());
675 0 : if (pszGeometryType[0] == '\0')
676 0 : pszGeometryType = "GEOMETRY";
677 :
678 : osSQL.Printf("INSERT INTO geometry_columns (f_table_name, "
679 : "f_geometry_column, type, coord_dimension, "
680 : "srid, spatial_index_enabled) "
681 : "VALUES ('%s','%s','%s','%s',%d, %d)",
682 : pszLayerNameEscaped, pszGeomColEscaped,
683 : pszGeometryType,
684 0 : wkbHasZ(poLayer->GetGeomType()) ? "XYZ" : "XY", nSRSId,
685 0 : static_cast<int>(bCreateSpatialIndex));
686 : }
687 : }
688 : #endif // HAVE_SPATIALITE
689 251 : char *pszErrMsg = nullptr;
690 251 : int rc = sqlite3_exec(hDB, osSQL.c_str(), nullptr, nullptr, &pszErrMsg);
691 251 : if (pszErrMsg != nullptr)
692 : {
693 2 : CPLDebug("SQLITE", "%s -> %s", osSQL.c_str(), pszErrMsg);
694 2 : sqlite3_free(pszErrMsg);
695 : }
696 :
697 : #ifdef HAVE_SPATIALITE
698 : /* -------------------------------------------------------------------- */
699 : /* Should we create a spatial index ?. */
700 : /* -------------------------------------------------------------------- */
701 251 : if (!bSpatialiteDB || !bCreateSpatialIndex)
702 249 : return rc == SQLITE_OK;
703 :
704 2 : CPLDebug("SQLITE", "Create spatial index %s", osIdxNameRaw.c_str());
705 :
706 : /* ENABLE_VIRTUAL_OGR_SPATIAL_INDEX is not defined */
707 : #ifdef ENABLE_VIRTUAL_OGR_SPATIAL_INDEX
708 : osSQL.Printf(
709 : "CREATE VIRTUAL TABLE \"%s\" USING "
710 : "VirtualOGRSpatialIndex(%d, '%s', pkid, xmin, xmax, ymin, ymax)",
711 : osIdxNameEscaped.c_str(), nExtraDS,
712 : SQLEscapeLiteral(oLayerDesc.osLayerName).c_str());
713 :
714 : rc = sqlite3_exec(hDB, osSQL.c_str(), NULL, NULL, NULL);
715 : if (rc != SQLITE_OK)
716 : {
717 : CPLDebug("SQLITE", "Error occurred during spatial index creation : %s",
718 : sqlite3_errmsg(hDB));
719 : }
720 : #else // ENABLE_VIRTUAL_OGR_SPATIAL_INDEX
721 2 : rc = sqlite3_exec(hDB, "BEGIN", nullptr, nullptr, nullptr);
722 :
723 : osSQL.Printf("CREATE VIRTUAL TABLE \"%s\" "
724 : "USING rtree(pkid, xmin, xmax, ymin, ymax)",
725 2 : osIdxNameEscaped.c_str());
726 :
727 2 : if (rc == SQLITE_OK)
728 2 : rc = sqlite3_exec(hDB, osSQL.c_str(), nullptr, nullptr, nullptr);
729 :
730 2 : sqlite3_stmt *hStmt = nullptr;
731 2 : if (rc == SQLITE_OK)
732 : {
733 : const char *pszInsertInto =
734 2 : CPLSPrintf("INSERT INTO \"%s\" (pkid, xmin, xmax, ymin, ymax) "
735 : "VALUES (?,?,?,?,?)",
736 : osIdxNameEscaped.c_str());
737 2 : rc = sqlite3_prepare_v2(hDB, pszInsertInto, -1, &hStmt, nullptr);
738 : }
739 :
740 2 : OGR2SQLITE_IgnoreAllFieldsExceptGeometry(poLayer);
741 2 : poLayer->ResetReading();
742 :
743 2 : OGRFeature *poFeature = nullptr;
744 2 : OGREnvelope sEnvelope;
745 12 : while (rc == SQLITE_OK &&
746 6 : (poFeature = poLayer->GetNextFeature()) != nullptr)
747 : {
748 4 : OGRGeometry *poGeom = poFeature->GetGeometryRef();
749 4 : if (poGeom != nullptr && !poGeom->IsEmpty())
750 : {
751 4 : poGeom->getEnvelope(&sEnvelope);
752 4 : sqlite3_bind_int64(hStmt, 1,
753 4 : static_cast<sqlite3_int64>(poFeature->GetFID()));
754 4 : sqlite3_bind_double(hStmt, 2, sEnvelope.MinX);
755 4 : sqlite3_bind_double(hStmt, 3, sEnvelope.MaxX);
756 4 : sqlite3_bind_double(hStmt, 4, sEnvelope.MinY);
757 4 : sqlite3_bind_double(hStmt, 5, sEnvelope.MaxY);
758 4 : rc = sqlite3_step(hStmt);
759 4 : if (rc == SQLITE_OK || rc == SQLITE_DONE)
760 4 : rc = sqlite3_reset(hStmt);
761 : }
762 4 : delete poFeature;
763 : }
764 :
765 2 : poLayer->SetIgnoredFields(nullptr);
766 :
767 2 : sqlite3_finalize(hStmt);
768 :
769 2 : if (rc == SQLITE_OK)
770 2 : rc = sqlite3_exec(hDB, "COMMIT", nullptr, nullptr, nullptr);
771 : else
772 : {
773 0 : CPLDebug("SQLITE", "Error occurred during spatial index creation : %s",
774 : sqlite3_errmsg(hDB));
775 0 : rc = sqlite3_exec(hDB, "ROLLBACK", nullptr, nullptr, nullptr);
776 : }
777 : #endif // ENABLE_VIRTUAL_OGR_SPATIAL_INDEX
778 :
779 : #endif // HAVE_SPATIALITE
780 :
781 2 : return rc == SQLITE_OK;
782 : }
783 :
784 : /************************************************************************/
785 : /* OGRSQLiteExecuteSQL() */
786 : /************************************************************************/
787 :
788 644 : OGRLayer *OGRSQLiteExecuteSQL(GDALDataset *poDS, const char *pszStatement,
789 : OGRGeometry *poSpatialFilter,
790 : CPL_UNUSED const char *pszDialect)
791 : {
792 644 : while (*pszStatement != '\0' &&
793 644 : isspace(static_cast<unsigned char>(*pszStatement)))
794 0 : pszStatement++;
795 :
796 644 : if (STARTS_WITH_CI(pszStatement, "ALTER TABLE ") ||
797 643 : STARTS_WITH_CI(pszStatement, "DROP TABLE ") ||
798 642 : STARTS_WITH_CI(pszStatement, "CREATE INDEX ") ||
799 641 : STARTS_WITH_CI(pszStatement, "DROP INDEX "))
800 : {
801 4 : CPLError(CE_Failure, CPLE_NotSupported,
802 : "SQL command not supported with SQLite dialect. "
803 : "Use OGRSQL dialect instead.");
804 4 : return nullptr;
805 : }
806 648 : else if (STARTS_WITH_CI(pszStatement, "CREATE VIRTUAL TABLE ") &&
807 8 : CPLTestBool(CPLGetConfigOption(
808 : "OGR_SQLITE_DIALECT_ALLOW_CREATE_VIRTUAL_TABLE", "NO")))
809 : {
810 : // for ogr_virtualogr.py::ogr_virtualogr_run_sql() only. This is
811 : // just a convenient way of testing VirtualOGR() with
812 : // CREATE VIRTUAL TABLE ... USING VirtualOGR(...)
813 : // but there is no possible use of that given we run that into
814 : // an ephemeral database.
815 : }
816 : else
817 : {
818 633 : bool bUnderstoodStatement = false;
819 134 : for (const char *pszKeyword : {"SELECT", "WITH", "EXPLAIN", "INSERT",
820 767 : "UPDATE", "DELETE", "REPLACE"})
821 : {
822 764 : if (STARTS_WITH_CI(pszStatement, pszKeyword) &&
823 630 : std::isspace(static_cast<unsigned char>(
824 630 : pszStatement[strlen(pszKeyword)])))
825 : {
826 630 : bUnderstoodStatement = true;
827 630 : break;
828 : }
829 : }
830 633 : if (!bUnderstoodStatement)
831 : {
832 3 : CPLError(CE_Failure, CPLE_NotSupported, "Unsupported SQL command.");
833 3 : return nullptr;
834 : }
835 : }
836 :
837 637 : char *pszTmpDBName = static_cast<char *>(CPLMalloc(256));
838 637 : snprintf(pszTmpDBName, 256, "%s", VSIMemGenerateHiddenFilename("temp.db"));
839 :
840 637 : OGRSQLiteDataSource *poSQLiteDS = nullptr;
841 637 : bool bSpatialiteDB = false;
842 :
843 : /* -------------------------------------------------------------------- */
844 : /* Create in-memory sqlite/spatialite DB */
845 : /* -------------------------------------------------------------------- */
846 :
847 : #ifdef HAVE_SPATIALITE
848 :
849 : /* -------------------------------------------------------------------- */
850 : /* Creating an empty SpatiaLite DB (with spatial_ref_sys populated */
851 : /* has a significant cost. So at the first attempt, let's make */
852 : /* one and cache it for later use. */
853 : /* -------------------------------------------------------------------- */
854 : #if 1
855 : static size_t nEmptyDBSize = 0;
856 : static GByte *pabyEmptyDB = nullptr;
857 : {
858 : static CPLMutex *hMutex = nullptr;
859 1274 : CPLMutexHolder oMutexHolder(&hMutex);
860 : static bool bTried = false;
861 637 : if (!bTried && CPLTestBool(CPLGetConfigOption(
862 : "OGR_SQLITE_DIALECT_USE_SPATIALITE", "YES")))
863 : {
864 4 : bTried = true;
865 4 : char *pszCachedFilename = static_cast<char *>(CPLMalloc(256));
866 4 : snprintf(pszCachedFilename, 256, "%s",
867 : VSIMemGenerateHiddenFilename("reference.db"));
868 4 : char **papszOptions = CSLAddString(nullptr, "SPATIALITE=YES");
869 4 : OGRSQLiteDataSource *poCachedDS = new OGRSQLiteDataSource();
870 : const int nRet =
871 4 : poCachedDS->Create(pszCachedFilename, papszOptions);
872 4 : CSLDestroy(papszOptions);
873 4 : papszOptions = nullptr;
874 4 : delete poCachedDS;
875 4 : if (nRet)
876 : {
877 : /* Note: the reference file keeps the ownership of the data, so
878 : * that */
879 : /* it gets released with VSICleanupFileManager() */
880 4 : vsi_l_offset nEmptyDBSizeLarge = 0;
881 4 : pabyEmptyDB = VSIGetMemFileBuffer(pszCachedFilename,
882 : &nEmptyDBSizeLarge, FALSE);
883 4 : nEmptyDBSize = static_cast<size_t>(nEmptyDBSizeLarge);
884 : }
885 4 : CPLFree(pszCachedFilename);
886 : }
887 : }
888 :
889 : /* The following configuration option is useful mostly for debugging/testing
890 : */
891 1274 : if (pabyEmptyDB != nullptr &&
892 637 : CPLTestBool(
893 : CPLGetConfigOption("OGR_SQLITE_DIALECT_USE_SPATIALITE", "YES")))
894 : {
895 : GByte *pabyEmptyDBClone =
896 527 : static_cast<GByte *>(VSI_MALLOC_VERBOSE(nEmptyDBSize));
897 527 : if (pabyEmptyDBClone == nullptr)
898 : {
899 0 : CPLFree(pszTmpDBName);
900 0 : return nullptr;
901 : }
902 527 : memcpy(pabyEmptyDBClone, pabyEmptyDB, nEmptyDBSize);
903 527 : VSIFCloseL(VSIFileFromMemBuffer(pszTmpDBName, pabyEmptyDBClone,
904 : nEmptyDBSize, TRUE));
905 :
906 527 : poSQLiteDS = new OGRSQLiteDataSource();
907 527 : GDALOpenInfo oOpenInfo(pszTmpDBName, GDAL_OF_VECTOR | GDAL_OF_UPDATE);
908 : CPLConfigOptionSetter oSetter("OGR_SQLITE_STATIC_VIRTUAL_OGR", "NO",
909 527 : false);
910 527 : const int nRet = poSQLiteDS->Open(&oOpenInfo);
911 527 : if (!nRet)
912 : {
913 : /* should not happen really ! */
914 0 : delete poSQLiteDS;
915 0 : VSIUnlink(pszTmpDBName);
916 0 : CPLFree(pszTmpDBName);
917 0 : return nullptr;
918 : }
919 527 : bSpatialiteDB = true;
920 : }
921 : #else
922 : /* No caching version */
923 : poSQLiteDS = new OGRSQLiteDataSource();
924 : char **papszOptions = CSLAddString(NULL, "SPATIALITE=YES");
925 : {
926 : CPLConfigOptionSetter oSetter("OGR_SQLITE_STATIC_VIRTUAL_OGR", "NO",
927 : false);
928 : const int nRet = poSQLiteDS->Create(pszTmpDBName, papszOptions);
929 : CSLDestroy(papszOptions);
930 : papszOptions = nullptr;
931 : if (nRet)
932 : {
933 : bSpatialiteDB = true;
934 : }
935 : }
936 : #endif
937 :
938 : else
939 : {
940 110 : delete poSQLiteDS;
941 110 : poSQLiteDS = nullptr;
942 : #else // HAVE_SPATIALITE
943 : if (true)
944 : {
945 : #endif // HAVE_SPATIALITE
946 :
947 : // cppcheck-suppress redundantAssignment
948 110 : poSQLiteDS = new OGRSQLiteDataSource();
949 : CPLConfigOptionSetter oSetter("OGR_SQLITE_STATIC_VIRTUAL_OGR", "NO",
950 110 : false);
951 110 : const int nRet = poSQLiteDS->Create(pszTmpDBName, nullptr);
952 110 : if (!nRet)
953 : {
954 0 : delete poSQLiteDS;
955 0 : VSIUnlink(pszTmpDBName);
956 0 : CPLFree(pszTmpDBName);
957 0 : return nullptr;
958 : }
959 : }
960 :
961 : /* -------------------------------------------------------------------- */
962 : /* Attach the Virtual Table OGR2SQLITE module to it. */
963 : /* -------------------------------------------------------------------- */
964 637 : OGR2SQLITEModule *poModule = OGR2SQLITE_Setup(poDS, poSQLiteDS);
965 637 : if (!poModule)
966 : {
967 0 : delete poSQLiteDS;
968 0 : VSIUnlink(pszTmpDBName);
969 0 : CPLFree(pszTmpDBName);
970 0 : return nullptr;
971 : }
972 637 : sqlite3 *hDB = poSQLiteDS->GetDB();
973 :
974 : /* -------------------------------------------------------------------- */
975 : /* Analysze the statement to determine which tables will be used. */
976 : /* -------------------------------------------------------------------- */
977 1274 : std::set<LayerDesc> oSetLayers;
978 1274 : std::set<CPLString> oSetSpatialIndex;
979 1274 : CPLString osModifiedSQL;
980 637 : OGR2SQLITEGetPotentialLayerNames(pszStatement, oSetLayers, oSetSpatialIndex,
981 : osModifiedSQL);
982 637 : std::set<LayerDesc>::iterator oIter = oSetLayers.begin();
983 :
984 637 : if (strcmp(pszStatement, osModifiedSQL.c_str()) != 0)
985 6 : CPLDebug("OGR", "Modified SQL: %s", osModifiedSQL.c_str());
986 637 : pszStatement = osModifiedSQL.c_str(); /* do not use it anymore */
987 :
988 : const bool bFoundOGRStyle =
989 637 : (osModifiedSQL.ifind("OGR_STYLE") != std::string::npos);
990 :
991 : /* -------------------------------------------------------------------- */
992 : /* For each of those tables, create a Virtual Table. */
993 : /* -------------------------------------------------------------------- */
994 637 : OGRLayer *poSingleSrcLayer = nullptr;
995 637 : bool bStringsAsUTF8 = true;
996 1006 : for (; oIter != oSetLayers.end(); ++oIter)
997 : {
998 371 : const LayerDesc &oLayerDesc = *oIter;
999 : /*CPLDebug("OGR", "Layer desc : %s, %s, %s, %s",
1000 : oLayerDesc.osOriginalStr.c_str(),
1001 : oLayerDesc.osSubstitutedName.c_str(),
1002 : oLayerDesc.osDSName.c_str(),
1003 : oLayerDesc.osLayerName.c_str());*/
1004 :
1005 371 : CPLString osSQL;
1006 371 : OGRLayer *poLayer = nullptr;
1007 371 : CPLString osTableName;
1008 371 : int nExtraDS = -1;
1009 371 : if (oLayerDesc.osDSName.empty())
1010 : {
1011 364 : poLayer = poDS->GetLayerByName(oLayerDesc.osLayerName);
1012 : /* Might be a false positive (unlikely) */
1013 364 : if (poLayer == nullptr)
1014 8 : continue;
1015 :
1016 356 : osTableName = oLayerDesc.osLayerName;
1017 : }
1018 : else
1019 : {
1020 : auto poOtherDS = std::unique_ptr<GDALDataset>(
1021 : GDALDataset::Open(oLayerDesc.osDSName, GDAL_OF_VECTOR, nullptr,
1022 7 : nullptr, nullptr));
1023 7 : if (poOtherDS == nullptr)
1024 : {
1025 1 : CPLError(CE_Failure, CPLE_AppDefined,
1026 : "Cannot open datasource '%s'",
1027 : oLayerDesc.osDSName.c_str());
1028 1 : delete poSQLiteDS;
1029 1 : VSIUnlink(pszTmpDBName);
1030 1 : CPLFree(pszTmpDBName);
1031 1 : return nullptr;
1032 : }
1033 :
1034 6 : poLayer = poOtherDS->GetLayerByName(oLayerDesc.osLayerName);
1035 6 : if (poLayer == nullptr)
1036 : {
1037 1 : CPLError(CE_Failure, CPLE_AppDefined,
1038 : "Cannot find layer '%s' in '%s'",
1039 : oLayerDesc.osLayerName.c_str(),
1040 : oLayerDesc.osDSName.c_str());
1041 1 : delete poSQLiteDS;
1042 1 : VSIUnlink(pszTmpDBName);
1043 1 : CPLFree(pszTmpDBName);
1044 1 : return nullptr;
1045 : }
1046 :
1047 5 : osTableName = oLayerDesc.osSubstitutedName;
1048 :
1049 5 : nExtraDS = OGR2SQLITE_AddExtraDS(poModule, poOtherDS.release());
1050 : }
1051 :
1052 361 : if (!poLayer->TestCapability(OLCStringsAsUTF8))
1053 348 : bStringsAsUTF8 = false;
1054 :
1055 361 : if (oSetLayers.size() == 1)
1056 347 : poSingleSrcLayer = poLayer;
1057 :
1058 : osSQL.Printf(
1059 : "CREATE VIRTUAL TABLE \"%s\" USING VirtualOGR(%d,'%s',%d,%d)",
1060 722 : SQLEscapeName(osTableName).c_str(), nExtraDS,
1061 722 : SQLEscapeLiteral(oLayerDesc.osLayerName).c_str(),
1062 1083 : bFoundOGRStyle ? 1 : 0, TRUE /*bExposeOGRNativeData*/);
1063 :
1064 361 : char *pszErrMsg = nullptr;
1065 361 : int rc = sqlite3_exec(hDB, osSQL.c_str(), nullptr, nullptr, &pszErrMsg);
1066 361 : if (rc != SQLITE_OK)
1067 : {
1068 0 : CPLError(CE_Failure, CPLE_AppDefined,
1069 : "Cannot create virtual table for layer '%s' : %s",
1070 : osTableName.c_str(), pszErrMsg);
1071 0 : sqlite3_free(pszErrMsg);
1072 0 : continue;
1073 : }
1074 :
1075 612 : for (int i = 0; i < poLayer->GetLayerDefn()->GetGeomFieldCount(); i++)
1076 : {
1077 251 : OGR2SQLITEDealWithSpatialColumn(poLayer, i, oLayerDesc, osTableName,
1078 : poSQLiteDS, hDB, bSpatialiteDB,
1079 : oSetLayers, oSetSpatialIndex);
1080 : }
1081 : }
1082 :
1083 : /* -------------------------------------------------------------------- */
1084 : /* Reload, so that virtual tables are recognized */
1085 : /* -------------------------------------------------------------------- */
1086 635 : poSQLiteDS->ReloadLayers();
1087 :
1088 : /* -------------------------------------------------------------------- */
1089 : /* Prepare the statement. */
1090 : /* -------------------------------------------------------------------- */
1091 : /* This will speed-up layer creation */
1092 : /* ORDER BY are costly to evaluate and are not necessary to establish */
1093 : /* the layer definition. */
1094 635 : bool bUseStatementForGetNextFeature = true;
1095 635 : bool bEmptyLayer = false;
1096 :
1097 635 : sqlite3_stmt *hSQLStmt = nullptr;
1098 635 : int rc = sqlite3_prepare_v2(hDB, pszStatement, -1, &hSQLStmt, nullptr);
1099 :
1100 635 : if (rc != SQLITE_OK)
1101 : {
1102 3 : CPLError(CE_Failure, CPLE_AppDefined,
1103 : "In ExecuteSQL(): sqlite3_prepare_v2(%s):\n %s", pszStatement,
1104 : sqlite3_errmsg(hDB));
1105 :
1106 3 : if (hSQLStmt != nullptr)
1107 : {
1108 0 : sqlite3_finalize(hSQLStmt);
1109 : }
1110 :
1111 3 : delete poSQLiteDS;
1112 3 : VSIUnlink(pszTmpDBName);
1113 3 : CPLFree(pszTmpDBName);
1114 :
1115 3 : return nullptr;
1116 : }
1117 :
1118 : /* -------------------------------------------------------------------- */
1119 : /* Do we get a resultset? */
1120 : /* -------------------------------------------------------------------- */
1121 632 : rc = sqlite3_step(hSQLStmt);
1122 632 : if (rc != SQLITE_ROW)
1123 : {
1124 161 : if (rc != SQLITE_DONE)
1125 : {
1126 1 : CPLError(CE_Failure, CPLE_AppDefined,
1127 : "In ExecuteSQL(): sqlite3_step(%s):\n %s", pszStatement,
1128 : sqlite3_errmsg(hDB));
1129 :
1130 1 : sqlite3_finalize(hSQLStmt);
1131 :
1132 1 : delete poSQLiteDS;
1133 1 : VSIUnlink(pszTmpDBName);
1134 1 : CPLFree(pszTmpDBName);
1135 :
1136 1 : return nullptr;
1137 : }
1138 :
1139 160 : if (!STARTS_WITH_CI(pszStatement, "SELECT "))
1140 : {
1141 :
1142 35 : sqlite3_finalize(hSQLStmt);
1143 :
1144 35 : delete poSQLiteDS;
1145 35 : VSIUnlink(pszTmpDBName);
1146 35 : CPLFree(pszTmpDBName);
1147 :
1148 35 : return nullptr;
1149 : }
1150 :
1151 125 : bUseStatementForGetNextFeature = false;
1152 125 : bEmptyLayer = true;
1153 : }
1154 :
1155 : /* -------------------------------------------------------------------- */
1156 : /* Create layer. */
1157 : /* -------------------------------------------------------------------- */
1158 :
1159 596 : auto poDrv = poDS->GetDriver();
1160 : const bool bCanReopenBaseDS =
1161 596 : !(poDrv && EQUAL(poDrv->GetDescription(), "Memory"));
1162 : OGRSQLiteSelectLayer *poLayer = new OGRSQLiteExecuteSQLLayer(
1163 : pszTmpDBName, poSQLiteDS, pszStatement, hSQLStmt,
1164 : bUseStatementForGetNextFeature, bEmptyLayer, bCanReopenBaseDS,
1165 596 : bStringsAsUTF8);
1166 :
1167 596 : if (poSpatialFilter != nullptr)
1168 : {
1169 6 : const auto nErrorCounter = CPLGetErrorCounter();
1170 6 : poLayer->SetSpatialFilter(0, poSpatialFilter);
1171 8 : if (CPLGetErrorCounter() > nErrorCounter &&
1172 2 : CPLGetLastErrorType() != CE_None)
1173 : {
1174 2 : delete poLayer;
1175 2 : return nullptr;
1176 : }
1177 : }
1178 :
1179 594 : if (poSingleSrcLayer != nullptr)
1180 316 : poLayer->SetMetadata(poSingleSrcLayer->GetMetadata("NATIVE_DATA"),
1181 316 : "NATIVE_DATA");
1182 :
1183 594 : return poLayer;
1184 : }
1185 :
1186 : #endif // HAVE_SQLITE3EXT_H
|