如何选择ORM属性的MIN()和文字日期时间?

时间:2011-08-15 14:13:28

标签: mysql sqlite datetime sqlalchemy portability

如何在SQL Alchemy v0.6.4中获取datetime列的MIN()和文字日期时间?我想将日期时间结果限制在特定范围内。

我一直在使用sqlalchemy.func.min(column, literal_datetime)。这似乎在SQLite中运行良好但与MySQL完全没有关系,这无疑意味着我正在犯这个错误。 MySQL的错误是:

sqlalchemy.exc.ProgrammingError: (ProgrammingError) (1064, "You have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to use near ' '2011-06-14 12:30:00') AS min_1 \nFROM example' at line 1") 'SELECT min(example.my_datetime, %s) AS min_1 \nFROM example' (datetime.datetime(2011, 6, 14, 12, 30),)
  • 如何以便携方式钳制日期时间结果?
  • (我真的必须在我的申请中这样做吗?)

以下是我一直用来探索问题的方法 - 这里提供的工作正常(使用基于内存的SQLite DB),不适用于MySQL:

#!/usr/bin/env python

import random
from datetime import datetime

import sqlalchemy as sa
import sqlalchemy.orm as orm
from sqlalchemy.ext.declarative import declarative_base

orm_base = declarative_base()

class MyClass( orm_base ):
    __tablename__ = "example"
    pri_key  = sa.Column(sa.Integer(), primary_key=True)
    my_datetime  = sa.Column(sa.DateTime(), index=True)

engine = sa.create_engine("sqlite:///:memory:", echo=True)

Session = orm.sessionmaker( bind=engine )

orm_base.metadata.bind = engine
orm_base.metadata.create_all()

# Create test-data

session = Session()

random.seed(1234567980)
for i in range(100):
    month = random.randint(1, 12)
    day = random.randint(1, 28)
    hour = random.randint(0, 23)
    minute = random.randint(0, 59)

    my_instance = MyClass()
    my_instance.my_datetime = datetime(2011, month, day, hour, minute)

    session.add( my_instance )

session.commit()
session.close()

# Problem starts here

session = Session()
literal_datetime = datetime(2011, 06, 14, 12, 30)
print session.query( sa.func.min(MyClass.my_datetime) ).one()  # OK
print session.query( sa.func.min(MyClass.my_datetime, literal_datetime) ).all()  # Error in MySQL

我不是SQL专家,但数据库可移植性对我来说很重要,所以欢迎任何有关如何避免将来陷阱的提示。

1 个答案:

答案 0 :(得分:0)

使用mySqlSQLite支持的sqlalchemy.sql.expression.case表达式。

min/max的{​​{1}}函数支持对多个值的非聚合操作。但SQLite不支持此功能。