--- /dev/null
+.. change::
+ :tags: bug, orm, sql
+ :tickets: 6824
+
+ Fixed issue where a bound parameter object that was "cloned" would cause a
+ name conflict in the compiler, if more than one clone of this parameter
+ were used at the same time in a single statement. This could occur in
+ particular with things like ORM single table inheritance queries that
+ indicated the same "discriminator" value multiple times in one query.
+
existing = self.binds[name]
if existing is not bindparam:
if (
- existing.unique or bindparam.unique
- ) and not existing.proxy_set.intersection(bindparam.proxy_set):
+ (existing.unique or bindparam.unique)
+ and not existing.proxy_set.intersection(
+ bindparam.proxy_set
+ )
+ and not existing._cloned_set.intersection(
+ bindparam._cloned_set
+ )
+ ):
raise exc.CompileError(
"Bind parameter '%s' conflicts with "
"unique bind parameter of the same name" % name
assert row.name == "Kurt"
assert row.employee_id == e1.employee_id
+ def test_discrim_bound_param_cloned_ok(self):
+ """Test #6824"""
+ Manager = self.classes.Manager
+
+ subq1 = select(Manager.employee_id).label("foo")
+ subq2 = select(Manager.employee_id).label("bar")
+ self.assert_compile(
+ select(subq1, subq2),
+ "SELECT (SELECT employees.employee_id FROM employees "
+ "WHERE employees.type IN ([POSTCOMPILE_type_1])) AS foo, "
+ "(SELECT employees.employee_id FROM employees "
+ "WHERE employees.type IN ([POSTCOMPILE_type_1])) AS bar",
+ )
+
def test_multi_qualification(self):
Manager, Engineer = (self.classes.Manager, self.classes.Engineer)
s,
)
+ def test_unique_binds_no_clone_collision(self):
+ """test #6824"""
+ bp = bindparam("foo", unique=True)
+
+ bpc1 = bp._clone(maintain_key=True)
+ bpc2 = bp._clone(maintain_key=True)
+
+ stmt1 = select(bp, bpc1, bpc2)
+
+ # OK, still strange that the double-dedupe logic is still *duping*
+ # the label name, but that's a different issue
+ self.assert_compile(
+ stmt1,
+ "SELECT :foo_1 AS anon_1, :foo_1 AS anon__1, :foo_1 AS anon__1",
+ )
+
def _test_binds_no_hash_collision(self):
"""test that construct_params doesn't corrupt dict
due to hash collisions"""