summaryrefslogtreecommitdiff
path: root/lib/sqlalchemy/sql/compiler.py
diff options
context:
space:
mode:
authormike bayer <mike_mp@zzzcomputing.com>2021-04-29 19:53:02 +0000
committerGerrit Code Review <gerrit@ci3.zzzcomputing.com>2021-04-29 19:53:02 +0000
commitdc5485b7ecdbe1cbed34fcb8d748fbe975aee140 (patch)
tree7f0456f166b53fecf881c6e214b69dc7db4944e3 /lib/sqlalchemy/sql/compiler.py
parent28493bf4bc35a4802b57b02a8b389cec7b6dcbb6 (diff)
parentaba308868544b21bafa0b3435701ddc908654b0a (diff)
downloadsqlalchemy-dc5485b7ecdbe1cbed34fcb8d748fbe975aee140.tar.gz
Merge "Use non-subquery form for empty IN"
Diffstat (limited to 'lib/sqlalchemy/sql/compiler.py')
-rw-r--r--lib/sqlalchemy/sql/compiler.py50
1 files changed, 46 insertions, 4 deletions
diff --git a/lib/sqlalchemy/sql/compiler.py b/lib/sqlalchemy/sql/compiler.py
index f3ae8c44f..57ffdf86b 100644
--- a/lib/sqlalchemy/sql/compiler.py
+++ b/lib/sqlalchemy/sql/compiler.py
@@ -1904,6 +1904,45 @@ class SQLCompiler(Compiled):
binary, override_operator=operators.match_op
)
+ def visit_in_op_binary(self, binary, operator, **kw):
+ return self._render_in_expr_w_bindparam(binary, operator, **kw)
+
+ def visit_not_in_op_binary(self, binary, operator, **kw):
+ return self._render_in_expr_w_bindparam(binary, operator, **kw)
+
+ def _render_in_expr_w_bindparam(self, binary, operator, **kw):
+ opstring = OPERATORS[operator]
+
+ if isinstance(binary.right, elements.BindParameter):
+ if not binary.right.expanding or not binary.right.expand_op:
+ # note that by cloning here, we rely upon the
+ # _cache_key_bind_match dictionary to resolve
+ # clones of bindparam() objects to the ones that are
+ # present in our cache key.
+ binary.right = binary.right._clone(maintain_key=True)
+ binary.right.expanding = True
+ binary.right.expand_op = operator
+
+ return self._generate_generic_binary(binary, opstring, **kw)
+
+ def visit_empty_set_op_expr(self, type_, expand_op):
+ if expand_op is operators.not_in_op:
+ if len(type_) > 1:
+ return "(%s)) OR (1 = 1" % (
+ ", ".join("NULL" for element in type_)
+ )
+ else:
+ return "NULL) OR (1 = 1"
+ elif expand_op is operators.in_op:
+ if len(type_) > 1:
+ return "(%s)) AND (1 != 1" % (
+ ", ".join("NULL" for element in type_)
+ )
+ else:
+ return "NULL) AND (1 != 1"
+ else:
+ return self.visit_empty_set_expr(type_)
+
def visit_empty_set_expr(self, element_types):
raise NotImplementedError(
"Dialect '%s' does not support empty set expression."
@@ -1960,12 +1999,12 @@ class SQLCompiler(Compiled):
to_update = []
if parameter.type._is_tuple_type:
- replacement_expression = self.visit_empty_set_expr(
- parameter.type.types
+ replacement_expression = self.visit_empty_set_op_expr(
+ parameter.type.types, parameter.expand_op
)
else:
- replacement_expression = self.visit_empty_set_expr(
- [parameter.type]
+ replacement_expression = self.visit_empty_set_op_expr(
+ [parameter.type], parameter.expand_op
)
elif isinstance(values[0], (tuple, list)):
@@ -3900,6 +3939,9 @@ class StrSQLCompiler(SQLCompiler):
for t in extra_froms
)
+ def visit_empty_set_op_expr(self, type_, expand_op):
+ return self.visit_empty_set_expr(type_)
+
def visit_empty_set_expr(self, type_):
return "SELECT 1 WHERE 1!=1"