2013-03-22 21 views
5

Jak mogę napisać następującą instrukcję sql z sqlalchemy w pythonie. Szukałem przez 30 minut, ale wciąż nie mogłem znaleźć żadnych rozwiązań.Korzystanie z DATEADD w sqlalchemy

DATEADD(NOW(), INTERVAL 1 DAY) 

lub

INSERT INTO dates (expire) 
VALUES(DATEADD(NOW(), INTERVAL 1 DAY)) 

Dzięki z góry

Odpowiedz

6

SQLAlchemy terminach automagicznie map do Pythona obiektów datetime, więc należy po prostu być w stanie to zrobić:

from sqlalchemy import Table, Column, MetaData, DateTime 
from datetime import datetime, timedelta 

metadata = MetaData() 
example = Table('users', metadata, 
    Column('expire', DateTime) 
) 

tomorrow = datetime.now() + timedelta(days=1) 

ins = example.insert().values(expire=tomorrow) 
+0

niestety System data sqlalchemy nie będzie automatycznie zmusić do platformy konkretnych funcitons jak DateAdd() teraz. Tylko na Postgresql z psycopg2 (i może także mysql) wykonuje prostą arytmetyczną datę jak powyżej, tłumacząc to, czego oczekuje DB. – zzzeek

3

Doobeh mnie pokonać do tego podczas pisania, tutaj jest przykład kolby-sqlalchemy, który zamierzałem opublikować (komplement zwykły przykład sqlalchemy):

from flask.ext.sqlalchemy import SQLAlchemy 
from datetime import datetime, timedelta 

db = SQLAlchemy() 

class Thing(db.Model): 
    id = db.Column(db.Integer, primary_key=True) 
    created = db.Column(db.DateTime) 

c = Thing(created = datetime.utcnow() + timedelta(days=1)) 
print repr(c.created) 
# datetime.datetime(2013, 3, 23, 15, 5, 48, 136583) 

można przekazać domyślnie jako wywoływalnym też:

from flask.ext.sqlalchemy import SQLAlchemy 
from datetime import datetime, timedelta 

db = SQLAlchemy() 

def tomorrow(): 
    return datetime.utcnow() + timedelta(days=1) 

class Thing(db.Model): 
    id = db.Column(db.Integer, primary_key=True) 
    publish_date = db.Column(db.DateTime, default=tomorrow) 
22

Dla dobra kompletności, tutaj jest jak chcesz wygenerować dokładnie to SQL z wykorzystaniem sqlalchemy.sql.func:

from sqlalchemy.sql import func 
from sqlalchemy.sql.expression import bindparam 
from sqlalchemy import Interval 

tomorrow = func.dateadd(func.now(), bindparam('tomorrow', timedelta(days=1), Interval())) 

co skutkuje:

>>> from sqlalchemy.sql import func 
>>> func.dateadd(func.now(), bindparam('tomorrow', timedelta(days=1), Interval(native=True))) 
<sqlalchemy.sql.expression.Function at 0x100f559d0; dateadd> 
>>> str(func.dateadd(func.now(), bindparam('tomorrow', timedelta(days=1), Interval(native=True)))) 
'dateadd(now(), :tomorrow)' 

Alternatywnie można użyć text() obiektu, aby określić przedział zamiast:

from sqlalchemy.sql import func 
from sqlalchemy.sql.expression import text 

tomorrow = func.dateadd(func.now(), text('interval 1 day')) 
+0

Warto zauważyć, że [dokumenty mówią, że 'Interval'] (http://docs.sqlalchemy.org/en/latest/core/type_basics.html#sqlalchemy.types.Interval) działa tylko z pewnymi bazami danych i tekstem' text 'wersja może być wymagana dla tych. – wbyoung

+0

@wbyoung: ponieważ OP określił dokładne wyrażenie SQL, które chciał użyć w pytaniu, można bezpiecznie założyć, że korzystał z bazy danych, która obsługuje składnię. –

+0

inne bazy danych obsługują 'DATEADD (TERAZ(), INTERWAŁ 1 DZIEŃ)', tj. MySQL, ale SQLAlchemy nie robi, ponieważ adapter ma problemy z typem. – wbyoung

0
from datetime import datetime, timedelta 
from dateutil import tz 

new_date = datetime.now(tz=tz.tzlocal()) + timedelta(days=1) 
new_item = Expire(expire=new_date) 
session.save(new_item) 
session.commit()