Sqlalchemy: подзапрос в FROM должен иметь псевдоним

Как я могу структурировать этот запрос sqlalchemy, чтобы он поступал правильно?

Я дал все, что я могу назвать псевдонимом, но я все еще получаю:

ProgrammingError: (psycopg2.ProgrammingError) subquery in FROM must have an alias LINE 4: FROM (SELECT foo.id AS foo_id, foo.version AS ... 

Кроме того, как указывал IMSoP, похоже, он пытается превратить его в кросс-соединение, но я просто хочу, чтобы он присоединился к таблице с группой по подзапросу в той же таблице.

Вот sqlalchemy:

(Примечание: я переписал его как отдельный файл, который является максимально полным и может быть запущен из оболочки python)

 from sqlalchemy import create_engine, func, select from sqlalchemy import Column, BigInteger, DateTime, Integer, String, SmallInteger from sqlalchemy.ext.declarative import declarative_base from sqlalchemy.orm import sessionmaker engine = create_engine('postgresql://postgres:#######@localhost:5435/foo1234') session = sessionmaker() session.configure(bind=engine) session = session() Base = declarative_base() class Foo(Base): __tablename__ = 'foo' __table_args__ = {'schema': 'public'} id = Column('id', BigInteger, primary_key=True) time = Column('time', DateTime(timezone=True)) version = Column('version', String) revision = Column('revision', SmallInteger) foo_max_time_q = select([ func.max(Foo.time).label('foo_max_time'), Foo.id.label('foo_id') ]).group_by(Foo.id ).alias('foo_max_time_q') foo_q = select([ Foo.id.label('foo_id'), Foo.version.label('foo_version'), Foo.revision.label('foo_revision'), foo_max_time_q.c.foo_max_time.label('foo_max_time') ]).join(foo_max_time_q, foo_max_time_q.c.foo_id == Foo.id ).alias('foo_q') thing = session.query(foo_q).all() print thing 

сгенерированный sql:

 SELECT foo_id AS foo_id, foo_version AS foo_version, foo_revision AS foo_revision, foo_max_time AS foo_max_time, foo_max_time_q.foo_max_time AS foo_max_time_q_foo_max_time, foo_max_time_q.foo_id AS foo_max_time_q_foo_id FROM (SELECT id AS foo_id, version AS foo_version, revision AS foo_revision, foo_max_time_q.foo_max_time AS foo_max_time FROM (SELECT max(time) AS foo_max_time, id AS foo_id GROUP BY id ) AS foo_max_time_q) JOIN (SELECT max(time) AS foo_max_time, id AS foo_id GROUP BY id ) AS foo_max_time_q ON foo_max_time_q.foo_id = id 

и вот таблица игрушек :

 CREATE TABLE foo ( id bigint , time timestamp with time zone, version character varying(32), revision smallint ); 

SQL, который я ожидал получить (желаемый SQL), будет примерно таким:

 SELECT foo.id AS foo_id, foo.version AS foo_version, foo.revision AS foo_revision, foo_max_time_q.foo_max_time AS foo_max_time FROM foo JOIN (SELECT max(time) AS foo_max_time, id AS foo_id GROUP BY id ) AS foo_max_time_q ON foo_max_time_q.foo_id = foo.id 

Заключительное примечание . Я надеюсь получить ответ, используя select () вместо session.query (), если это возможно. спасибо

    Вы почти там. Сделайте «подбираемый» подзапрос и присоедините его к основному запросу через join() :

     foo_max_time_q = select([func.max(Foo.time).label('foo_max_time'), Foo.id.label('foo_id') ]).group_by(Foo.id ).alias("foo_max_time_q") foo_q = session.query( Foo.id.label('foo_id'), Foo.version.label('foo_version'), Foo.revision.label('foo_revision'), foo_max_time_q.c.foo_max_time.label('foo_max_time') ).join(foo_max_time_q, foo_max_time_q.c.foo_id == Foo.id) print(foo_q.__str__()) 

    Отпечатки (приписаны вручную):

     SELECT foo.id AS foo_id, foo.version AS foo_version, foo.revision AS foo_revision, foo_max_time_q.foo_max_time AS foo_max_time FROM foo JOIN (SELECT max(foo.time) AS foo_max_time, foo.id AS foo_id FROM foo GROUP BY foo.id) AS foo_max_time_q ON foo_max_time_q.foo_id = foo.id 

    Полный рабочий код доступен в этом документе.