return new_func.alias(name=name, joins_implicitly=joins_implicitly)
- def column_valued(self, name=None):
+ def column_valued(self, name=None, joins_implicitly=False):
"""Return this :class:`_functions.FunctionElement` as a column expression that
selects from itself as a FROM clause.
gs = func.generate_series(1, 5, -1).alias().column
+ :param name: optional name to assign to the alias name that's generated.
+ If omitted, a unique anonymizing name is used.
+
+ :param joins_implicitly: when True, the "table" portion of the column
+ valued function may be a member of the FROM clause without any
+ explicit JOIN to other tables in the SQL query, and no "cartesian
+ product" warning will be generated. May be useful for SQL functions
+ such as ``func.json_array_elements()``.
+
+ .. versionadded:: 1.4.46
.. seealso::
""" # noqa: 501
- return self.alias(name=name).column
+ return self.alias(name=name, joins_implicitly=joins_implicitly).column
@property
def columns(self):
assert start is p3
assert froms == {p1}
- @testing.combinations(
- "render_derived", "alias", None, argnames="additional_transformation"
+ @testing.variation("additional_transformation", ["alias", "none"])
+ @testing.variation("joins_implicitly", [True, False])
+ @testing.variation(
+ "type_", ["table_valued", "table_valued_derived", "column_valued"]
)
- @testing.combinations(True, False, argnames="joins_implicitly")
- def test_table_valued(
- self,
- joins_implicitly,
- additional_transformation,
+ def test_fn_valued(
+ self, joins_implicitly, additional_transformation, type_
):
- """test #7845"""
+ """test #7845, #9009"""
+
my_table = table(
"tbl",
column("id", Integer),
sub_dict = my_table.c.data["d"]
- tv = func.json_each(sub_dict)
+ if type_.table_valued or type_.table_valued_derived:
+ tv = func.json_each(sub_dict)
+
+ tv = tv.table_valued("key", joins_implicitly=joins_implicitly)
+
+ if type_.table_valued_derived:
+ tv = tv.render_derived(name="tv", with_types=True)
+
+ if additional_transformation.alias:
+ tv = tv.alias()
+
+ has_key = tv.c.key == "f"
+ stmt = select(my_table.c.id).where(has_key)
+ elif type_.column_valued:
+ tv = func.json_array_elements(sub_dict)
- tv = tv.table_valued("key", joins_implicitly=joins_implicitly)
+ if additional_transformation.alias:
+ tv = tv.alias(joins_implicitly=joins_implicitly).column
+ else:
+ tv = tv.column_valued("key", joins_implicitly=joins_implicitly)
- if additional_transformation == "render_derived":
- tv = tv.render_derived(name="tv", with_types=True)
- elif additional_transformation == "alias":
- tv = tv.alias()
+ stmt = select(my_table.c.id, tv)
+ else:
+ type_.fail()
- has_key = tv.c.key == "f"
- stmt = select(my_table.c.id).where(has_key)
froms, start = find_unmatching_froms(stmt, my_table)
if joins_implicitly:
is_(start, None)
is_(froms, None)
- else:
+ elif type_.column_valued:
+ assert start == my_table
+ assert froms == {tv.scalar_alias}
+
+ elif type_.table_valued or type_.table_valued_derived:
assert start == my_table
assert froms == {tv}
+ else:
+ type_.fail()
def test_count_non_eq_comparison_operators(self):
query = select(self.a).where(self.a.c.col_a > self.b.c.col_b)