diff --git a/superset/sql_parse.py b/superset/sql_parse.py index 1130763a372c0..26b4bfd2a4e92 100644 --- a/superset/sql_parse.py +++ b/superset/sql_parse.py @@ -15,6 +15,7 @@ # specific language governing permissions and limitations # under the License. import logging +import re from dataclasses import dataclass from enum import Enum from typing import List, Optional, Set @@ -41,6 +42,16 @@ logger = logging.getLogger(__name__) +# TODO: Workaround for https://github.com/andialbrecht/sqlparse/issues/652. +sqlparse.keywords.SQL_REGEX.insert( + 0, + ( + re.compile(r"'(''|\\\\|\\|[^'])*'", sqlparse.keywords.FLAGS).match, + sqlparse.tokens.String.Single, + ), +) + + class CtasMethod(str, Enum): TABLE = "TABLE" VIEW = "VIEW" diff --git a/tests/unit_tests/sql_parse_tests.py b/tests/unit_tests/sql_parse_tests.py index f405b9fcda426..92917df3b4697 100644 --- a/tests/unit_tests/sql_parse_tests.py +++ b/tests/unit_tests/sql_parse_tests.py @@ -1199,3 +1199,9 @@ def test_validate_filter_clause_comment(): def test_validate_filter_clause_subquery_comment(): with pytest.raises(QueryClauseValidationException): validate_filter_clause("(1 = 1 -- comment\n)") + + +def test_sqlparse_issue_652(): + stmt = sqlparse.parse(r"foo = '\' AND bar = 'baz'")[0] + assert len(stmt.tokens) == 5 + assert str(stmt.tokens[0]) == "foo = '\\'"