我一直在使用amazon swf的流程框架,我希望能够运行优先级工作流程执行和正常的工作流程执行。如果存在优先级任务,则活动应在正常优先级任务之前获取优先级任务。完成此任务的最佳方法是什么?
我认为以下内容可能有效,但我想知道是否有更好/推荐的方法。
这种方法的问题在于无法保证在正常任务之前安排高优先级任务。
答案 0 :(得分:2)
这是一个很好的问题,它让我摸不着头脑。
当然,有很多方法可以给这只猫留下皮肤,并且存在许多有效的解决方案。我在这里集中讨论了我能想到的最简单的可能性,即在单个工作流程中按优先级顺序执行任务。
方案如下:我定义了一个活动工作者,为两个任务列表default_tasks
和urgent_tasks
提供服务,其中包含一个简单的逻辑:
urgent_tasks
列表中有待处理的任务,则从那里选择一个,default_tasks
问题是如何检查是否有任何高优先级的任务待定? CountPendingActivityTasks API拯救了!
我知道您使用Flow进行开发。我的示例是使用boto.swf.layer2
编写的,因为Python对于原型设计来说要容易得多 - 但这个想法保持不变,并且可以扩展到具有高优先级和低优先级工作流执行的更复杂场景。
因此,要使用boto.swf完成上述操作,请按以下步骤操作:
将凭据导出到环境
$ export AWS_ACCESS_KEY_ID=your access key
$ export AWS_SECRET_ACCESS_KEY= your secret key
获取代码段
为方便起见,您可以从github中分叉:
$ git clone git@github.com:oozie/stackoverflow.git
$ cd stackoverflow/amazon-swf/priority_tasks/
引导域和工作流程:
# domain_setup.py
import boto.swf.layer2 as swf
DOMAIN = 'stackoverflow'
VERSION = '1.0'
swf.Domain(name=DOMAIN).register()
swf.ActivityType(domain=DOMAIN, name='SomeActivity', version=VERSION, task_list='default_tasks').register()
swf.WorkflowType(domain=DOMAIN, name='MyWorkflow', version=VERSION, task_list='default_tasks').register()
决策者实施:
# decider.py
import boto.swf.layer2 as swf
DOMAIN = 'stackoverflow'
ACTIVITY = 'SomeActivity'
VERSION = '1.0'
class MyWorkflowDecider(swf.Decider):
domain = DOMAIN
task_list = 'default_tasks'
version = VERSION
def run(self):
history = self.poll()
print history
if 'events' in history:
# Get a list of non-decision events to see what event came in last.
workflow_events = [e for e in history['events']
if not e['eventType'].startswith('Decision')]
decisions = swf.Layer1Decisions()
last_event = workflow_events[-1]
last_event_type = last_event['eventType']
if last_event_type == 'WorkflowExecutionStarted':
# At the start, get the worker to fetch the first assignment.
decisions.schedule_activity_task(ACTIVITY+'1', ACTIVITY, VERSION, task_list='default_tasks')
decisions.schedule_activity_task(ACTIVITY+'2', ACTIVITY, VERSION, task_list='urgent_tasks')
decisions.schedule_activity_task(ACTIVITY+'3', ACTIVITY, VERSION, task_list='default_tasks')
decisions.schedule_activity_task(ACTIVITY+'4', ACTIVITY, VERSION, task_list='urgent_tasks')
decisions.schedule_activity_task(ACTIVITY+'5', ACTIVITY, VERSION, task_list='default_tasks')
elif last_event_type == 'ActivityTaskCompleted':
# Complete workflow execution after 5 completed activities.
closed_activity_count = sum(1 for wf_event in workflow_events if wf_event.get('eventType') == 'ActivityTaskCompleted')
if closed_activity_count == 5:
decisions.complete_workflow_execution()
self.complete(decisions=decisions)
return True
优先考虑员工实施:
# worker.py
import boto.swf.layer2 as swf
DOMAIN = 'stackoverflow'
VERSION = '1.0'
class PrioritizingWorker(swf.ActivityWorker):
domain = DOMAIN
version = VERSION
def run(self):
urgent_task_count = swf.Domain(name=DOMAIN).count_pending_activity_tasks('urgent_tasks').get('count', 0)
if urgent_task_count > 0:
self.task_list = 'urgent_tasks'
else:
self.task_list = 'default_tasks'
activity_task = self.poll()
if 'activityId' in activity_task:
print urgent_task_count, 'urgent tasks in the queue. Executing ' + activity_task.get('activityId')
self.complete()
return True
从交互式Python shell的三个实例运行工作流
运行决策程序:
$ python -i decider.py
>>> while MyWorkflowDecider().run(): pass
...
开始执行:
$ python -i decider.py
>>> swf.WorkflowType(domain='stackoverflow', name='MyWorkflow', version='1.0', task_list='default_tasks').start()
最后,启动工作人员并在执行任务时观察任务:
$ python -i worker.py
>>> while PrioritizingWorker().run(): pass
...
2 urgent tasks in the queue. Executing SomeActivity2
1 urgent tasks in the queue. Executing SomeActivity4
0 urgent tasks in the queue. Executing SomeActivity5
0 urgent tasks in the queue. Executing SomeActivity1
0 urgent tasks in the queue. Executing SomeActivity3
答案 1 :(得分:1)
事实证明,使用您必须首先检查的单独任务列表并不能很好地工作。
有几个问题。
首先,count API无法可靠地更新。因此,即使队列中有紧急任务,您也可能获得0个任务。
其次,如果没有可用任务,则轮询任务的调用将挂起。因此,当您对非紧急任务进行轮询时,这将会“坚持”#34;两分钟,或直到你有一个非紧急任务要做。
因此,这可能会导致工作流程中出现各种问题。
为此,SWF必须实现一个轮询API,该API可以从任务列表列表中返回第一个任务。那会更容易。