我正在尝试构建一个 Pyramid 应用程序。我从 SQLAlchemy 脚手架开始。我遇到了一个问题,我想知道解决它的最佳方法是什么。在我的一个观点中,我需要从两个不相关的表中选择很多行。我需要确保在从第一个表中选择行和从第二个表中选择行之间没有任何行插入到第二个表中。
我有三个模型,Node
,Test
和Tasking
。两者Nodes
都有Tests
相当多的元数据。给定一个 的列表Nodes
和一个 的列表,可以创建Tests
一个 的全局列表。Taskings
例如,我们可以有三个Nodes
, a
,b
和c
两个Tests
“我们需要一个节点来执行任务P
”和“我们需要两个节点来执行任务Q
”。
根据这些信息,Tasks
应该创建三个。例如:
- “节点
a
应该做任务P
” - “节点
b
应该做任务Q
” - “节点
c
应该做任务Q
”
现在,我正在尝试为此提供一个 REST API。绝大多数时间客户将请求 的列表Tasks
,因此需要快速。但是,有时客户可能会添加 aNode
或Test
. 发生这种情况时,我需要重新生成整个列表Tasks
。
这是一个粗略的例子:
@view_config(route_name='list_taskings')
def list_taskings(request):
return DBSession.Query(Tasking).all()
@view_config(route_name='add_node')
def add_node(request):
DBSession.add(Node())
_update_taskings()
@view_config(route_name='add_test')
def add_test(request):
DBSession.add(Test())
_update_taskings()
def _update_taskings():
nodes = DBSession.query(Node).all()
tests = DBSession.query(Test).all()
# Process...
Tasking.query.delete()
for t in taskings:
DBSession.add(t)
我正在使用默认的 Pyramid SQLAlchemy 脚手架。因此,每个请求都会自动启动一个事务。因此,如果_update_tasking
从一个请求(例如add_node
)调用,那么新节点将被添加到本地DBSession
,并且查询 allNodes
和Tests
in_update_tasking
将返回该新元素。此外,删除所有现有的Taskings
并添加新计算的也是安全的。
我有两个问题:
如果在
Tests
我得到我的 list ofnodes
和我的 list oftests
in之间将新行添加到表中会发生_update_taskings
什么?在我的现实世界生产系统中,这些选择很接近,但并不相邻。存在竞争条件的可能性。如何确保将更新的两个请求
Taskings
不会相互覆盖?例如,想象一下如果我们现有的系统有一个Node
和一个Test
. 两个请求同时进来,一个添加 aNode
,一个添加 aTest
。即使问题 #1 不是问题,并且我知道每个请求的一对选择代表“数据库中的单个时间实例”,仍然存在一个请求覆盖另一个请求的问题。如果第一个请求首先以 now twoNodes
and one完成Test
,则第二个请求仍将选择旧数据(可能)并将生成一个Taskings
包含 oneNode
和 two的列表Tests
。
那么,处理这个问题的最佳方法是什么?我在开发中使用 SQLite,在生产中使用 PostgreSQL,但我想要一个与数据库无关的解决方案。我不担心其他应用程序访问这个数据库。我的 REST API 将是唯一的访问机制。我应该锁定任何改变数据库的请求(添加 aNode
或 a Test
)吗?我应该以某种方式锁定数据库吗?
谢谢你的帮助!