Skip to content

Commit f6ea049

Browse files
authored
Merge pull request #288 from Mytherin/uppercaseops
Fix #284: quote uppercase identifiers when shipping queries to Postgres
2 parents 9b0dbd1 + e548757 commit f6ea049

7 files changed

+53
-9
lines changed

src/include/postgres_utils.hpp

+1
Original file line numberDiff line numberDiff line change
@@ -68,6 +68,7 @@ class PostgresUtils {
6868
static bool SupportedPostgresOid(const LogicalType &input);
6969
static LogicalType RemoveAlias(const LogicalType &type);
7070
static PostgresType CreateEmptyPostgresType(const LogicalType &type);
71+
static string QuotePostgresIdentifier(const string &text);
7172

7273
static PostgresVersion ExtractPostgresVersion(const string &version);
7374
};

src/postgres_utils.cpp

+4
Original file line numberDiff line numberDiff line change
@@ -475,4 +475,8 @@ PostgresVersion PostgresUtils::ExtractPostgresVersion(const string &version_str)
475475
return result;
476476
}
477477

478+
string PostgresUtils::QuotePostgresIdentifier(const string &text) {
479+
return KeywordHelper::WriteOptionallyQuoted(text, '"', false);
480+
}
481+
478482
} // namespace duckdb

src/storage/postgres_delete.cpp

+1-1
Original file line numberDiff line numberDiff line change
@@ -19,7 +19,7 @@ string GetDeleteSQL(const PostgresTableEntry &table, const string &ctid_list) {
1919
string result;
2020
result = "DELETE FROM ";
2121
result += KeywordHelper::WriteQuoted(table.schema.name, '"') + ".";
22-
result += KeywordHelper::WriteOptionallyQuoted(table.name);
22+
result += PostgresUtils::QuotePostgresIdentifier(table.name);
2323
result += " WHERE ctid IN (" + ctid_list + ")";
2424
return result;
2525
}

src/storage/postgres_index_set.cpp

+3-3
Original file line numberDiff line numberDiff line change
@@ -61,10 +61,10 @@ string PGGetCreateIndexSQL(CreateIndexInfo &info, TableCatalogEntry &tbl) {
6161
sql += " UNIQUE";
6262
}
6363
sql += " INDEX ";
64-
sql += KeywordHelper::WriteOptionallyQuoted(info.index_name);
64+
sql += PostgresUtils::QuotePostgresIdentifier(info.index_name);
6565
sql += " ON ";
66-
sql += KeywordHelper::WriteOptionallyQuoted(tbl.schema.name) + ".";
67-
sql += KeywordHelper::WriteOptionallyQuoted(tbl.name);
66+
sql += PostgresUtils::QuotePostgresIdentifier(tbl.schema.name) + ".";
67+
sql += PostgresUtils::QuotePostgresIdentifier(tbl.name);
6868
sql += "(";
6969
for (idx_t i = 0; i < info.parsed_expressions.size(); i++) {
7070
if (i > 0) {

src/storage/postgres_schema_entry.cpp

+3-3
Original file line numberDiff line numberDiff line change
@@ -73,8 +73,8 @@ optional_ptr<CatalogEntry> PostgresSchemaEntry::CreateIndex(CatalogTransaction t
7373
string PGGetCreateViewSQL(PostgresSchemaEntry &schema, CreateViewInfo &info) {
7474
string sql;
7575
sql = "CREATE VIEW ";
76-
sql += KeywordHelper::WriteOptionallyQuoted(schema.name) + ".";
77-
sql += KeywordHelper::WriteOptionallyQuoted(info.view_name);
76+
sql += PostgresUtils::QuotePostgresIdentifier(schema.name) + ".";
77+
sql += PostgresUtils::QuotePostgresIdentifier(info.view_name);
7878
sql += " ";
7979
if (!info.aliases.empty()) {
8080
sql += "(";
@@ -83,7 +83,7 @@ string PGGetCreateViewSQL(PostgresSchemaEntry &schema, CreateViewInfo &info) {
8383
sql += ", ";
8484
}
8585
auto &alias = info.aliases[i];
86-
sql += KeywordHelper::WriteOptionallyQuoted(alias);
86+
sql += PostgresUtils::QuotePostgresIdentifier(alias);
8787
}
8888
sql += ") ";
8989
}

src/storage/postgres_update.cpp

+2-2
Original file line numberDiff line numberDiff line change
@@ -30,7 +30,7 @@ class PostgresUpdateGlobalState : public GlobalSinkState {
3030

3131
string CreateUpdateTable(const string &name, PostgresTableEntry &table, const vector<PhysicalIndex> &index) {
3232
string result;
33-
result = "CREATE LOCAL TEMPORARY TABLE " + KeywordHelper::WriteOptionallyQuoted(name);
33+
result = "CREATE LOCAL TEMPORARY TABLE " + PostgresUtils::QuotePostgresIdentifier(name);
3434
result += "(";
3535
for (idx_t i = 0; i < index.size(); i++) {
3636
if (i > 0) {
@@ -63,7 +63,7 @@ string GetUpdateSQL(const string &name, PostgresTableEntry &table, const vector<
6363
result += ".";
6464
result += KeywordHelper::WriteQuoted(column_name, '"');
6565
}
66-
result += " FROM " + KeywordHelper::WriteOptionallyQuoted(name);
66+
result += " FROM " + PostgresUtils::QuotePostgresIdentifier(name);
6767
result += " WHERE ";
6868
result += KeywordHelper::WriteQuoted(table.name, '"');
6969
result += ".ctid=__page_id_string::TID";
+39
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,39 @@
1+
# name: test/sql/storage/attach_upper_case.test
2+
# description: Test modifying an upper-case table
3+
# group: [storage]
4+
5+
require postgres_scanner
6+
7+
require-env POSTGRES_TEST_DATABASE_AVAILABLE
8+
9+
statement ok
10+
PRAGMA enable_verification
11+
12+
statement ok
13+
ATTACH 'dbname=postgresscanner' AS s1 (TYPE POSTGRES)
14+
15+
statement ok
16+
USE s1
17+
18+
statement ok
19+
DROP SCHEMA IF EXISTS SCHEM01 CASCADE
20+
21+
statement ok
22+
CREATE SCHEMA SCHEM01
23+
24+
statement ok
25+
create or replace table SCHEM01.TAB01(COL01 VARCHAR);
26+
27+
statement ok
28+
insert into SCHEM01.TAB01 values ('abc')
29+
30+
statement ok
31+
update SCHEM01.TAB01 set COL01='zxcv'
32+
33+
query I
34+
SELECT COL01 FROM SCHEM01.TAB01
35+
----
36+
zxcv
37+
38+
statement ok
39+
truncate SCHEM01.tab01

0 commit comments

Comments
 (0)