Итак, после небольшого прочтения и настройки я получил следующее. Прежде всего мне пришлось добавить код, найденный здесь, в качестве ответа Как использовать bindparam () в пользовательском скомпилированном выражении? . Код является измененной версией кода в предложении VALUES в SQLAlchemy .
from sqlalchemy import *
from sqlalchemy.ext.compiler import compiles
from sqlalchemy.sql.expression import FromClause, ClauseElement
from sqlalchemy.dialects.postgresql import array
from sqlalchemy.types import NULLTYPE
from pp.core.db import panels, engine
class values(FromClause):
named_with_column = True
def __init__(self, columns, *args, **kw):
self._column_args = columns
self.list = args
self.alias_name = self.name = kw.pop('alias_name', None)
def _populate_column_collection(self):
# self._columns.update((col.name, col) for col in self._column_args)
for c in self._column_args:
c._make_proxy(self, c.name)
@compiles(values)
def compile_values(clause, compiler, asfrom=False, **kw):
def decide(value, column):
add_type_hint = False
if isinstance(value, array) and not value.clauses: # for empty array literals
add_type_hint = True
if isinstance(value, ClauseElement):
intermediate = compiler.process(value)
if add_type_hint:
intermediate += '::' + str(column.type)
return intermediate
elif value is None:
return compiler.render_literal_value(
value,
NULLTYPE
) + '::' + str(column.type)
else:
return compiler.process(
bindparam(
None,
value=compiler.render_literal_value(
value,
column.type
).strip("'")
)
) + '::' + str(column.type)
columns = clause.columns
v = "VALUES %s" % ", ".join(
"(%s)" % ", ".join(
decide(elem, column)
for elem, column in zip(tup, columns))
for tup in clause.list
)
if asfrom:
if clause.alias_name:
v = "(%s) AS %s (%s)" % (v, clause.alias_name, (", ".join(c.name for c in clause.columns)))
else:
v = "(%s)" % v
return v
Оттуда следующий код работал для меня. Я не должен был использовать CTE в конце. Важнейшей частью было объявление имен столбцов для временной таблицы, которые не будут совпадать с именами столбцов моей таблицы панели, поскольку это приведет к «неоднозначной ошибке ссылки на столбец».
t1 = values(
[
column('given_id', Integer),
column('given_instance', Integer)
],
(1, 1),
(108, 23),
(203, 5),
alias_name='given'
)
stmt = select(['given_id']). \
select_from(t1.outerjoin(panels, and_(
panels.c.id == t1.c.given_id,
panels.c.instance_id == t1.c.given_instance
))). \
where(
panels.c.id.is_(None)
)
rs = engine.execute(stmt)