Python SQLalchemy 0.7的任何示例,使用from_语句()更新

Python SQLalchemy 0.7的任何示例,使用from_语句()更新,python,mysql,sqlalchemy,Python,Mysql,Sqlalchemy,我正在编写一个快速的一次性迁移脚本,用于更新一个包含50万行的表中的单个字段 由于我没有计划为连接编写完整的模型,我正在获取最初的25000行数据,因此我一直在试图找出如何使用from_statement()调用和使用我自己的原始sql来执行UPDATE语句,但我找不到任何示例 除此之外,SQLalchemy还抛出了一个错误。以下是我的呼叫和错误示例: mydb = self.session() mydb.query().from_statement( """ UPDATE my

我正在编写一个快速的一次性迁移脚本,用于更新一个包含50万行的表中的单个字段

由于我没有计划为连接编写完整的模型,我正在获取最初的25000行数据,因此我一直在试图找出如何使用from_statement()调用和使用我自己的原始sql来执行UPDATE语句,但我找不到任何示例

除此之外,SQLalchemy还抛出了一个错误。以下是我的呼叫和错误示例:

mydb = self.session()
mydb.query().from_statement(
    """
    UPDATE my_table
    SET settings=mysettings
    WHERE user_id=myuserid AND setting_id=123
    """).params(mysettings=new_settings, myuserid=user_id).all()
我得到的错误是:

Traceback (most recent call last):
  File "./sample_script.py", line 111, in <module>
    main()
  File "./sample_script.py", line 108, in main
    migrate.set_migration_data()
  File "./sample_script.py", line 100, in set_migration_data
    """).params(mysettings=new_settings, myuserid=user_id).all()
  File "/usr/lib/pymodules/python2.6/sqlalchemy/orm/query.py", line 1267, in all
    return list(self)
  File "/usr/lib/pymodules/python2.6/sqlalchemy/orm/query.py", line 1361, in __iter__
    return self._execute_and_instances(context)
  File "/usr/lib/pymodules/python2.6/sqlalchemy/orm/query.py", line 1364, in _execute_and_instances
    result = self.session.execute(querycontext.statement, params=self._params, mapper=self._mapper_zero_or_none())
  File "/usr/lib/pymodules/python2.6/sqlalchemy/orm/query.py", line 251, in _mapper_zero_or_none
    if not getattr(self._entities[0], 'primary_entity', False):
IndexError: list index out of range
这没有效果。我没有得到任何错误,但语句似乎没有实际执行,因为行没有更改。如果我手动剪切并粘贴从echo=True选项中记录的查询,则行在数据库中会很好地更新

更新-已解决

Samy的建议是正确的,但是.execute()调用只在“引擎”上有效,而不是在“会话”上有效,所以这很好:

self.engine.execute(
    "UPDATE mytable SET settings=:mysettings WHERE user_id=:userid AND setting_id=123",
    {'userid': user_id, 'mysettings': new_settings}
    )

您需要使用正确的参数语法;格式完全取决于数据库适配器。例如,某些适配器支持
:name
参数,在这种情况下,查询中缺少以下冒号:

mydb.query().from_statement(
    """
    UPDATE my_table
    SET settings=:mysettings
    WHERE user_id=:myuserid AND setting_id=123
    """).params(mysettings=new_settings, myuserid=user_id).all()

支持多种格式,包括带有
%s
占位符的位置参数,以及上述格式中的命名参数和格式为
%(名称)s
的命名参数。您需要查看您的数据库适配器文档,以了解支持的内容。

根据文档,这相当奇怪,用于
SELECT
语句

执行给定的SELECT语句并返回结果

我可能看错了函数,或者可能使用了其他类型的语句,我不是很确定

您可以使用
execute
,因为它可以执行任何类型的语句,下面是一个简单的示例

from sqlalchemy import create_engine
from sqlalchemy.orm import sessionmaker

session = sessionmaker(bind = create_engine('sqlite://'), autocommit = True)()

_ = session.execute('CREATE TABLE my_table (user_id int, setting_id int, settings string)')
for id in xrange(200):
    _ = session.execute('INSERT INTO my_table (user_id, setting_id) VALUES (:user_id, :setting_id)',
        {'user_id':id, 'setting_id':id})

_ = session.execute(
"""
    UPDATE my_table
    SET settings = :mysettings
    WHERE user_id = :user_id AND setting_id = 123
""", {'user_id':123, 'mysettings':'test'})

r = session.execute('SELECT * FROM my_table WHERE user_id = :user_id', {'user_id':123}).fetchall()
print r

[(123, 123, u'test')]

请注意,这并不是使用
sqlalchemy
的最佳方式,它旨在创建一个干燥的环境,与特定的数据库后端分离,尽管您可能有使用原始sql而不是ORM的理由。

谢谢Martijn。添加冒号没有帮助。我已经用MySQL标记了我的问题,但应该在我的问题文本中指定它。我将尝试查找所需的文档,但可能只是切换到原始mysql驱动程序以更快地完成此操作。
mysqldb
使用
%s
参数(位置)。谢谢,也尝试了,错误消失了,但实际上没有写入数据库。您提交了事务吗`嗨,Samy,我也试着按照你的建议使用execute()(请参阅我原始问题中的更新),但它似乎没有实际运行。请确保刷新会话,
mydb.flush()
我没有尝试session.flush,但使用引擎本身可以很好地执行查询。再次感谢您的帮助。没问题,很高兴我能提供帮助,希望您继续使用/享受sqlalchemy,它很棒。
from sqlalchemy import create_engine
from sqlalchemy.orm import sessionmaker

session = sessionmaker(bind = create_engine('sqlite://'), autocommit = True)()

_ = session.execute('CREATE TABLE my_table (user_id int, setting_id int, settings string)')
for id in xrange(200):
    _ = session.execute('INSERT INTO my_table (user_id, setting_id) VALUES (:user_id, :setting_id)',
        {'user_id':id, 'setting_id':id})

_ = session.execute(
"""
    UPDATE my_table
    SET settings = :mysettings
    WHERE user_id = :user_id AND setting_id = 123
""", {'user_id':123, 'mysettings':'test'})

r = session.execute('SELECT * FROM my_table WHERE user_id = :user_id', {'user_id':123}).fetchall()
print r

[(123, 123, u'test')]