我正在编写一个快速的一次性迁移脚本,该脚本更新具有半百万行的表中的单个字段。
由于我没有计划为要获取初始 ~25000 行数据的连接编写完整模型,因此我一直试图弄清楚如何使用 from_statement() 调用和使用来执行 UPDATE 语句我自己的原始 sql,但我找不到任何示例。
与此同时,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
更新
我正在使用 MySQL。
根据 Samy 的建议,我尝试了这个:
mydb.execute(
"UPDATE mytable SET settings=:mysettings WHERE user_id=:userid AND setting_id=123",
{'userid': user_id, 'mysettings': new_settings}
)
这没有效果。我没有收到任何错误,但该语句似乎并没有真正执行,因为该行没有改变。如果我手动剪切并粘贴从 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}
)