Fixing engine transaction model and error handling

* Transaction in on_action_complete() must not be splitted into 2 parts,
  it caused the bug with after task completion logic
* Fix executor behavior so that it doesn't send an error back to engine
  if a error came from engine itself. It should report back only errors
  occurred with an action itself.
* YAQL and other expected Mistral exceptions in transitions should not
  lead to transaction rollback and rollback of action result. For example
  if action result came and it's valid but while evaluating transition
  conditions we got a YAQL exception then action result should be stored
  normally w/o transaction rollback and corresponding task and workflow
  should fail with corresponding state_info.
* Fixed all tests
* Minor cosmetic changes

Closes-Bug: #1524477

Change-Id: I09086e40a5902bbb6c977bf195cb035e31f21246
This commit is contained in:
Renat Akhmerov 2016-03-28 13:44:54 +06:00
parent 9a1a157274
commit ad07ba0d68
11 changed files with 474 additions and 331 deletions

28
AUTHORS
View File

@ -1,43 +1,71 @@
Abhishek Chanda <abhishek@cloudscaling.com> Abhishek Chanda <abhishek@cloudscaling.com>
Alexander Kuznetsov <akuznetsov@mirantis.com> Alexander Kuznetsov <akuznetsov@mirantis.com>
Anastasia Kuznetsova <akuznetsova@mirantis.com> Anastasia Kuznetsova <akuznetsova@mirantis.com>
Andreas Jaeger <aj@suse.com>
Angus Salkeld <angus.salkeld@rackspace.com> Angus Salkeld <angus.salkeld@rackspace.com>
Ankita Wagh <ankita_wagh@symmactoolkit-c02lr80ufd57.symc.symantec.com> Ankita Wagh <ankita_wagh@symmactoolkit-c02lr80ufd57.symc.symantec.com>
Antoine Musso <hashar@free.fr> Antoine Musso <hashar@free.fr>
Bertrand Lallau <bertrand.lallau@gmail.com>
Bhaskar Duvvuri <dbasu84@gmail.com>
Boris Pavlovic <boris@pavlovic.me> Boris Pavlovic <boris@pavlovic.me>
Bryan Havenstein <bryan.havenstein@ericsson.com> Bryan Havenstein <bryan.havenstein@ericsson.com>
Chaozhe.Chen <chaozhe.chen@easystack.cn>
Christian Berendt <berendt@b1-systems.de> Christian Berendt <berendt@b1-systems.de>
Claudiu Belu <cbelu@cloudbasesolutions.com> Claudiu Belu <cbelu@cloudbasesolutions.com>
Dan Prince <dprince@redhat.com>
Daryl Mowrer <dmowrer@us.ibm.com>
David C Kennedy <david.c.kennedy@hp.com> David C Kennedy <david.c.kennedy@hp.com>
David Charles Kennedy <dkennedy@hp.com> David Charles Kennedy <dkennedy@hp.com>
Dawid Deja <dawid.deja@intel.com>
Dmitri Zimine <dz@stackstorm.com> Dmitri Zimine <dz@stackstorm.com>
Doug Hellmann <doug@doughellmann.com> Doug Hellmann <doug@doughellmann.com>
Ed Cranford <ed.cranford@rackspace.com> Ed Cranford <ed.cranford@rackspace.com>
Gal Margalit <gal.margalit@alcatel-lucent.com>
Guy Paz <guy.paz@alcatel-lucent.com> Guy Paz <guy.paz@alcatel-lucent.com>
Jeremy Stanley <fungi@yuggoth.org> Jeremy Stanley <fungi@yuggoth.org>
Jiri Tomasek <jtomasek@redhat.com>
Kevin Pouget <kpouget@altair.com>
Kirill Izotov <enykeev@stackstorm.com> Kirill Izotov <enykeev@stackstorm.com>
Lakshmi Kannan <lakshmi@stackstorm.com> Lakshmi Kannan <lakshmi@stackstorm.com>
Limor <limor.bortman@nokia.com>
Limor Stotland <limor.bortman@alcatel-lucent.com> Limor Stotland <limor.bortman@alcatel-lucent.com>
Lingxian Kong <konglingxian@huawei.com> Lingxian Kong <konglingxian@huawei.com>
Liu Sheng <liusheng@huawei.com> Liu Sheng <liusheng@huawei.com>
LiuNanke <nanke.liu@easystack.cn>
Manas Kelshikar <manas@stackstorm.com> Manas Kelshikar <manas@stackstorm.com>
Michael Krotscheck <krotscheck@gmail.com>
Michal Gershenzon <michal.gershenzon@alcatel-lucent.com>
Monty Taylor <mordred@inaugust.com> Monty Taylor <mordred@inaugust.com>
Moshe Elisha <moshe.elisha@alcatel-lucent.com> Moshe Elisha <moshe.elisha@alcatel-lucent.com>
Nikolay Mahotkin <nmakhotkin@mirantis.com> Nikolay Mahotkin <nmakhotkin@mirantis.com>
Noa Koffman <noa.koffman@alcatel-lucent.com> Noa Koffman <noa.koffman@alcatel-lucent.com>
Oleksii Chuprykov <ochuprykov@mirantis.com>
Pierre-Arthur MATHIEU <pierre-arthur.mathieu@hp.com> Pierre-Arthur MATHIEU <pierre-arthur.mathieu@hp.com>
Ray Chen <chenrano2002@gmail.com> Ray Chen <chenrano2002@gmail.com>
Renat Akhmerov <rakhmerov@mirantis.com> Renat Akhmerov <rakhmerov@mirantis.com>
Renat Akhmerov <renat.akhmerov@gmail.com>
Rico Lin <rico.lin.guanyu@gmail.com> Rico Lin <rico.lin.guanyu@gmail.com>
Rinat Sabitov <rinat.sabitov@gmail.com> Rinat Sabitov <rinat.sabitov@gmail.com>
Sergey Kolekonov <skolekonov@mirantis.com> Sergey Kolekonov <skolekonov@mirantis.com>
Sergey Murashov <smurashov@mirantis.com> Sergey Murashov <smurashov@mirantis.com>
Shuquan Huang <huang.shuquan@99cloud.net>
Thierry Carrez <thierry@openstack.org>
Thomas Herve <therve@redhat.com>
Timur Nurlygayanov <tnurlygayanov@mirantis.com> Timur Nurlygayanov <tnurlygayanov@mirantis.com>
Venkata Mahesh Kotha <venkatamaheshkotha@gmail.com> Venkata Mahesh Kotha <venkatamaheshkotha@gmail.com>
Winson Chan <wcchan@stackstorm.com> Winson Chan <wcchan@stackstorm.com>
Yaroslav Lobankov <ylobankov@mirantis.com> Yaroslav Lobankov <ylobankov@mirantis.com>
Zhao Lei <zhaolei@cn.fujitsu.com> Zhao Lei <zhaolei@cn.fujitsu.com>
Zhenguo Niu <niuzhenguo@huawei.com> Zhenguo Niu <niuzhenguo@huawei.com>
ZhiQiang Fan <aji.zqfan@gmail.com>
ZhiQiang Fan <zhiqiang.fan@huawei.com> ZhiQiang Fan <zhiqiang.fan@huawei.com>
Zhu Rong <zhu.rong@99cloud.net> Zhu Rong <zhu.rong@99cloud.net>
caoyue <yue.cao@easystack.cn>
cheneydc <dongc@neunn.com>
hardik <hardik.parekh@nectechnologies.in> hardik <hardik.parekh@nectechnologies.in>
hparekh <hardik.parekh@nectechnologies.in>
keliang <ke.liang@easystack.cn>
syed ahsan shamim zaidi <ahsanmohsin04@yahoo.com>
tengqm <tengqim@cn.ibm.com>
wangzhh <wangzhh@awcloud.com>
zhangguoqing <zhang.guoqing@99cloud.net>

View File

@ -29,7 +29,7 @@ from mistral.workflow import utils as wf_utils
def create_action_execution(action_def, action_input, task_ex=None, def create_action_execution(action_def, action_input, task_ex=None,
index=0, description=''): index=0, description=''):
# TODO(rakhmerov): We can avoid hitting DB at all when calling something # TODO(rakhmerov): We can avoid hitting DB at all when calling things like
# create_action_execution(), these operations can be just done using # create_action_execution(), these operations can be just done using
# SQLAlchemy session (1-level cache) and session flush (on TX commit) would # SQLAlchemy session (1-level cache) and session flush (on TX commit) would
# send necessary SQL queries to DB. Currently, session flush happens # send necessary SQL queries to DB. Currently, session flush happens

View File

@ -24,6 +24,7 @@ from mistral.engine import action_handler
from mistral.engine import base from mistral.engine import base
from mistral.engine import task_handler from mistral.engine import task_handler
from mistral.engine import workflow_handler as wf_handler from mistral.engine import workflow_handler as wf_handler
from mistral import exceptions as exc
from mistral.services import action_manager as a_m from mistral.services import action_manager as a_m
from mistral.services import executions as wf_ex_service from mistral.services import executions as wf_ex_service
from mistral.services import workflows as wf_service from mistral.services import workflows as wf_service
@ -55,6 +56,9 @@ class DefaultEngine(base.Engine, coordination.Service):
wf_ex_id = None wf_ex_id = None
try: try:
# Create a persistent workflow execution in a separate transaction
# so that we can return it even in case of unexpected errors that
# lead to transaction rollback.
with db_api.transaction(): with db_api.transaction():
# The new workflow execution will be in an IDLE # The new workflow execution will be in an IDLE
# state on initial record creation. # state on initial record creation.
@ -65,10 +69,6 @@ class DefaultEngine(base.Engine, coordination.Service):
params params
) )
# Separate workflow execution creation and dispatching command
# transactions in order to be able to return workflow execution
# with corresponding error message in state_info when error occurs
# at dispatching commands.
with db_api.transaction(): with db_api.transaction():
wf_ex = db_api.get_workflow_execution(wf_ex_id) wf_ex = db_api.get_workflow_execution(wf_ex_id)
wf_handler.set_execution_state(wf_ex, states.RUNNING) wf_handler.set_execution_state(wf_ex, states.RUNNING)
@ -161,14 +161,10 @@ class DefaultEngine(base.Engine, coordination.Service):
self._on_task_state_change(task_ex, wf_ex, wf_spec) self._on_task_state_change(task_ex, wf_ex, wf_spec)
def _on_task_state_change(self, task_ex, wf_ex, wf_spec, def _on_task_state_change(self, task_ex, wf_ex, wf_spec):
task_state=states.SUCCESS):
task_spec = wf_spec.get_tasks()[task_ex.name] task_spec = wf_spec.get_tasks()[task_ex.name]
# We must be sure that if task is completed, if task_handler.is_task_completed(task_ex, task_spec):
# it was also completed in previous transaction.
if (task_handler.is_task_completed(task_ex, task_spec)
and states.is_completed(task_state)):
task_handler.after_task_complete(task_ex, task_spec, wf_spec) task_handler.after_task_complete(task_ex, task_spec, wf_spec)
# Ignore DELAYED state. # Ignore DELAYED state.
@ -178,8 +174,21 @@ class DefaultEngine(base.Engine, coordination.Service):
wf_ctrl = wf_base.get_controller(wf_ex, wf_spec) wf_ctrl = wf_base.get_controller(wf_ex, wf_spec)
# Calculate commands to process next. # Calculate commands to process next.
try:
cmds = wf_ctrl.continue_workflow() cmds = wf_ctrl.continue_workflow()
except exc.YaqlEvaluationException as e:
LOG.error(
'YAQL error occurred while calculating next workflow '
'commands [wf_ex_id=%s, task_ex_id=%s]: %s',
wf_ex.id, task_ex.id, e
)
wf_handler.fail_workflow(wf_ex, str(e))
return
# Mark task as processed after all decisions have been made
# upon its completion.
task_ex.processed = True task_ex.processed = True
self._dispatch_workflow_commands(wf_ex, cmds, wf_spec) self._dispatch_workflow_commands(wf_ex, cmds, wf_spec)
@ -235,6 +244,7 @@ class DefaultEngine(base.Engine, coordination.Service):
wf_ex_id = action_ex.task_execution.workflow_execution_id wf_ex_id = action_ex.task_execution.workflow_execution_id
wf_ex = wf_handler.lock_workflow_execution(wf_ex_id) wf_ex = wf_handler.lock_workflow_execution(wf_ex_id)
wf_spec = spec_parser.get_workflow_spec(wf_ex.spec) wf_spec = spec_parser.get_workflow_spec(wf_ex.spec)
task_ex = task_handler.on_action_complete( task_ex = task_handler.on_action_complete(
@ -248,30 +258,13 @@ class DefaultEngine(base.Engine, coordination.Service):
if states.is_paused_or_completed(wf_ex.state): if states.is_paused_or_completed(wf_ex.state):
return action_ex.get_clone() return action_ex.get_clone()
prev_task_state = task_ex.state self._on_task_state_change(task_ex, wf_ex, wf_spec)
# Separate the task transition in a separate transaction. The task
# has already completed for better or worst. The task state should
# not be affected by errors during transition on conditions such as
# on-success and on-error.
with db_api.transaction():
wf_ex = wf_handler.lock_workflow_execution(wf_ex_id)
action_ex = db_api.get_action_execution(action_ex_id)
task_ex = action_ex.task_execution
self._on_task_state_change(
task_ex,
wf_ex,
wf_spec,
task_state=prev_task_state
)
return action_ex.get_clone() return action_ex.get_clone()
except Exception as e: except Exception as e:
# TODO(dzimine): try to find out which command caused failure.
# TODO(rakhmerov): Need to refactor logging in a more elegant way. # TODO(rakhmerov): Need to refactor logging in a more elegant way.
LOG.error( LOG.error(
"Failed to handle action execution result [id=%s]: %s\n%s", 'Failed to handle action execution result [id=%s]: %s\n%s',
action_ex_id, e, traceback.format_exc() action_ex_id, e, traceback.format_exc()
) )
@ -301,12 +294,13 @@ class DefaultEngine(base.Engine, coordination.Service):
wf_ctrl = wf_base.get_controller(wf_ex) wf_ctrl = wf_base.get_controller(wf_ex)
# TODO(rakhmerov): Add YAQL error handling.
# Calculate commands to process next. # Calculate commands to process next.
cmds = wf_ctrl.continue_workflow(task_ex=task_ex, reset=reset, env=env) cmds = wf_ctrl.continue_workflow(task_ex=task_ex, reset=reset, env=env)
# When resuming a workflow we need to ignore all 'pause' # When resuming a workflow we need to ignore all 'pause'
# commands because workflow controller takes tasks that # commands because workflow controller takes tasks that
# completed within the period when the workflow was pause. # completed within the period when the workflow was paused.
cmds = list( cmds = list(
filter( filter(
lambda c: not isinstance(c, commands.PauseWorkflow), lambda c: not isinstance(c, commands.PauseWorkflow),
@ -323,6 +317,7 @@ class DefaultEngine(base.Engine, coordination.Service):
t_ex.processed = True t_ex.processed = True
wf_spec = spec_parser.get_workflow_spec(wf_ex.spec) wf_spec = spec_parser.get_workflow_spec(wf_ex.spec)
self._dispatch_workflow_commands(wf_ex, cmds, wf_spec) self._dispatch_workflow_commands(wf_ex, cmds, wf_spec)
if not cmds: if not cmds:
@ -378,9 +373,9 @@ class DefaultEngine(base.Engine, coordination.Service):
raise e raise e
@u.log_exec(LOG) @u.log_exec(LOG)
def stop_workflow(self, execution_id, state, message=None): def stop_workflow(self, wf_ex_id, state, message=None):
with db_api.transaction(): with db_api.transaction():
wf_ex = wf_handler.lock_workflow_execution(execution_id) wf_ex = wf_handler.lock_workflow_execution(wf_ex_id)
return self._stop_workflow(wf_ex, state, message) return self._stop_workflow(wf_ex, state, message)
@ -390,13 +385,16 @@ class DefaultEngine(base.Engine, coordination.Service):
wf_ctrl = wf_base.get_controller(wf_ex) wf_ctrl = wf_base.get_controller(wf_ex)
final_context = {} final_context = {}
try: try:
final_context = wf_ctrl.evaluate_workflow_final_context() final_context = wf_ctrl.evaluate_workflow_final_context()
except Exception as e: except Exception as e:
LOG.warning( LOG.warning(
"Failed to get final context for %s: %s" % (wf_ex, e) 'Failed to get final context for %s: %s' % (wf_ex, e)
) )
wf_spec = spec_parser.get_workflow_spec(wf_ex.spec) wf_spec = spec_parser.get_workflow_spec(wf_ex.spec)
return wf_handler.succeed_workflow( return wf_handler.succeed_workflow(
wf_ex, wf_ex,
final_context, final_context,
@ -409,7 +407,7 @@ class DefaultEngine(base.Engine, coordination.Service):
return wf_ex return wf_ex
@u.log_exec(LOG) @u.log_exec(LOG)
def rollback_workflow(self, execution_id): def rollback_workflow(self, wf_ex_id):
# TODO(rakhmerov): Implement. # TODO(rakhmerov): Implement.
raise NotImplementedError raise NotImplementedError
@ -421,12 +419,26 @@ class DefaultEngine(base.Engine, coordination.Service):
if isinstance(cmd, commands.RunTask) and cmd.is_waiting(): if isinstance(cmd, commands.RunTask) and cmd.is_waiting():
task_handler.defer_task(cmd) task_handler.defer_task(cmd)
elif isinstance(cmd, commands.RunTask): elif isinstance(cmd, commands.RunTask):
task_handler.run_new_task(cmd, wf_spec) task_ex = task_handler.run_new_task(cmd, wf_spec)
if task_ex.state == states.ERROR:
wf_handler.fail_workflow(
wf_ex,
'Failed to start task [task_ex=%s]: %s' %
(task_ex, task_ex.state_info)
)
elif isinstance(cmd, commands.RunExistingTask): elif isinstance(cmd, commands.RunExistingTask):
task_handler.run_existing_task( task_ex = task_handler.run_existing_task(
cmd.task_ex.id, cmd.task_ex.id,
reset=cmd.reset reset=cmd.reset
) )
if task_ex.state == states.ERROR:
wf_handler.fail_workflow(
wf_ex,
'Failed to start task [task_ex=%s]: %s' %
(task_ex, task_ex.state_info)
)
elif isinstance(cmd, commands.SetWorkflowState): elif isinstance(cmd, commands.SetWorkflowState):
if states.is_completed(cmd.new_state): if states.is_completed(cmd.new_state):
self._stop_workflow(cmd.wf_ex, cmd.new_state, cmd.msg) self._stop_workflow(cmd.wf_ex, cmd.new_state, cmd.msg)
@ -441,33 +453,28 @@ class DefaultEngine(base.Engine, coordination.Service):
if wf_ex.state != states.RUNNING: if wf_ex.state != states.RUNNING:
break break
# TODO(rakhmerov): This method may not be needed at all because error
# handling is now implemented too roughly w/o distinguishing different
# errors. On most errors (like YAQLException) we shouldn't rollback
# transactions, we just need to fail corresponding execution objects
# where a problem happened (action, task or workflow).
@staticmethod @staticmethod
def _fail_workflow(wf_ex_id, err, action_ex_id=None): def _fail_workflow(wf_ex_id, exc):
"""Private helper to fail workflow on exceptions.""" """Private helper to fail workflow on exceptions."""
err_msg = str(err)
with db_api.transaction(): with db_api.transaction():
wf_ex = db_api.load_workflow_execution(wf_ex_id) wf_ex = db_api.load_workflow_execution(wf_ex_id)
if wf_ex is None: if wf_ex is None:
LOG.error( LOG.error(
"Cant fail workflow execution with id='%s': not found.", "Can't fail workflow execution with id='%s': not found.",
wf_ex_id wf_ex_id
) )
return return None
wf_handler.set_execution_state(wf_ex, states.ERROR, err_msg) wf_ex = wf_handler.lock_workflow_execution(wf_ex_id)
if action_ex_id: if not states.is_paused_or_completed(wf_ex.state):
# Note(dzimine): Don't call self.engine_client: wf_handler.set_execution_state(wf_ex, states.ERROR, str(exc))
# 1) to avoid computing and triggering next tasks
# 2) to avoid a loop in case of error in transport
action_ex = db_api.get_action_execution(action_ex_id)
task_handler.on_action_complete(
action_ex,
spec_parser.get_workflow_spec(wf_ex.spec),
wf_utils.Result(error=err_msg)
)
return wf_ex return wf_ex

View File

@ -19,7 +19,6 @@ from oslo_log import log as logging
from mistral.actions import action_factory as a_f from mistral.actions import action_factory as a_f
from mistral import coordination from mistral import coordination
from mistral.engine import base from mistral.engine import base
from mistral import exceptions as exc
from mistral.utils import inspect_utils as i_u from mistral.utils import inspect_utils as i_u
from mistral.workflow import utils as wf_utils from mistral.workflow import utils as wf_utils
@ -37,7 +36,7 @@ class DefaultExecutor(base.Executor, coordination.Service):
action_params): action_params):
"""Runs action. """Runs action.
:param action_ex_id: Corresponding task id. :param action_ex_id: Action execution id.
:param action_class_str: Path to action class in dot notation. :param action_class_str: Path to action class in dot notation.
:param attributes: Attributes of action class which will be set to. :param attributes: Attributes of action class which will be set to.
:param action_params: Action parameters. :param action_params: Action parameters.
@ -51,14 +50,29 @@ class DefaultExecutor(base.Executor, coordination.Service):
action_ex_id, action_ex_id,
error_result error_result
) )
else:
return None
return error_result return error_result
action_cls = a_f.construct_action_class(action_class_str, attributes) action_cls = a_f.construct_action_class(action_class_str, attributes)
# Instantiate action.
try: try:
action = action_cls(**action_params) action = action_cls(**action_params)
except Exception as e:
msg = ("Failed to initialize action %s. Action init params = %s."
" Actual init params = %s. More info: %s"
% (action_class_str, i_u.get_arg_list(action_cls.__init__),
action_params.keys(), e))
LOG.warning(msg)
return send_error_back(msg)
# Run action.
try:
result = action.run() result = action.run()
# Note: it's made for backwards compatibility with already # Note: it's made for backwards compatibility with already
@ -67,24 +81,25 @@ class DefaultExecutor(base.Executor, coordination.Service):
if not isinstance(result, wf_utils.Result): if not isinstance(result, wf_utils.Result):
result = wf_utils.Result(data=result) result = wf_utils.Result(data=result)
if action_ex_id and (action.is_sync() or result.is_error()): except Exception as e:
self._engine_client.on_action_complete(action_ex_id, result)
return result
except TypeError as e:
msg = ("Failed to initialize action %s. Action init params = %s."
" Actual init params = %s. More info: %s"
% (action_class_str, i_u.get_arg_list(action_cls.__init__),
action_params.keys(), e))
LOG.warning(msg)
except exc.ActionException as e:
msg = ("Failed to run action [action_ex_id=%s, action_cls='%s'," msg = ("Failed to run action [action_ex_id=%s, action_cls='%s',"
" attributes='%s', params='%s']\n %s" " attributes='%s', params='%s']\n %s"
% (action_ex_id, action_cls, attributes, action_params, e)) % (action_ex_id, action_cls, attributes, action_params, e))
LOG.exception(msg) LOG.exception(msg)
except Exception as e:
msg = str(e)
# Send error info to engine.
return send_error_back(msg) return send_error_back(msg)
# Send action result.
try:
if action_ex_id and (action.is_sync() or result.is_error()):
self._engine_client.on_action_complete(action_ex_id, result)
except Exception as e:
msg = ("Exception occurred when calling engine on_action_complete"
" [action_ex_id=%s, action_cls='%s',"
" attributes='%s', params='%s']\n %s"
% (action_ex_id, action_cls, attributes, action_params, e))
LOG.exception(msg)
return result

View File

@ -24,6 +24,7 @@ from mistral.engine import base
from mistral import exceptions as exc from mistral import exceptions as exc
from mistral.workflow import utils as wf_utils from mistral.workflow import utils as wf_utils
LOG = logging.getLogger(__name__) LOG = logging.getLogger(__name__)
@ -164,6 +165,8 @@ class EngineServer(object):
:param rpc_ctx: RPC request context. :param rpc_ctx: RPC request context.
:param action_ex_id: Action execution id. :param action_ex_id: Action execution id.
:param result_data: Action result data.
:param result_error: Action result error.
:return: Action execution. :return: Action execution.
""" """

View File

@ -46,6 +46,9 @@ def run_existing_task(task_ex_id, reset=True):
"""This function runs existing task execution. """This function runs existing task execution.
It is needed mostly by scheduler. It is needed mostly by scheduler.
:param task_ex_id: Task execution id.
:param reset: Reset action executions for the task.
""" """
task_ex = db_api.get_task_execution(task_ex_id) task_ex = db_api.get_task_execution(task_ex_id)
task_spec = spec_parser.get_task_spec(task_ex.spec) task_spec = spec_parser.get_task_spec(task_ex.spec)
@ -54,15 +57,16 @@ def run_existing_task(task_ex_id, reset=True):
# Throw exception if the existing task already succeeded. # Throw exception if the existing task already succeeded.
if task_ex.state == states.SUCCESS: if task_ex.state == states.SUCCESS:
raise exc.EngineException('Reruning existing task that already ' raise exc.EngineException(
'succeeded is not supported.') 'Rerunning existing task that already succeeded is not supported.'
)
# Exit if the existing task failed and reset is not instructed. # Exit if the existing task failed and reset is not instructed.
# For a with-items task without reset, re-running the existing # For a with-items task without reset, re-running the existing
# task will re-run the failed and unstarted items. # task will re-run the failed and unstarted items.
if (task_ex.state == states.ERROR and not reset and if (task_ex.state == states.ERROR and not reset and
not task_spec.get_with_items()): not task_spec.get_with_items()):
return return task_ex
# Reset nested executions only if task is not already RUNNING. # Reset nested executions only if task is not already RUNNING.
if task_ex.state != states.RUNNING: if task_ex.state != states.RUNNING:
@ -84,14 +88,27 @@ def run_existing_task(task_ex_id, reset=True):
_run_existing_task(task_ex, task_spec, wf_spec) _run_existing_task(task_ex, task_spec, wf_spec)
return task_ex
def _run_existing_task(task_ex, task_spec, wf_spec): def _run_existing_task(task_ex, task_spec, wf_spec):
try:
input_dicts = _get_input_dictionaries( input_dicts = _get_input_dictionaries(
wf_spec, wf_spec,
task_ex, task_ex,
task_spec, task_spec,
task_ex.in_context task_ex.in_context
) )
except exc.MistralException as e:
LOG.error(
'An error while calculating task action inputs'
' [task_execution_id=%s]: %s',
task_ex.id, e
)
set_task_state(task_ex, states.ERROR, str(e))
return
# In some cases we can have no input, e.g. in case of 'with-items'. # In some cases we can have no input, e.g. in case of 'with-items'.
if input_dicts: if input_dicts:
@ -113,8 +130,15 @@ def defer_task(wf_cmd):
wf_ex = wf_cmd.wf_ex wf_ex = wf_cmd.wf_ex
task_spec = wf_cmd.task_spec task_spec = wf_cmd.task_spec
if not wf_utils.find_task_executions_by_spec(wf_ex, task_spec): if wf_utils.find_task_executions_by_spec(wf_ex, task_spec):
_create_task_execution(wf_ex, task_spec, ctx, state=states.WAITING) return None
return _create_task_execution(
wf_ex,
task_spec,
ctx,
state=states.WAITING
)
def run_new_task(wf_cmd, wf_spec): def run_new_task(wf_cmd, wf_spec):
@ -149,23 +173,25 @@ def run_new_task(wf_cmd, wf_spec):
# Policies could possibly change task state. # Policies could possibly change task state.
if task_ex.state != states.RUNNING: if task_ex.state != states.RUNNING:
return return task_ex
_run_existing_task(task_ex, task_spec, wf_spec) _run_existing_task(task_ex, task_spec, wf_spec)
return task_ex
def on_action_complete(action_ex, wf_spec, result): def on_action_complete(action_ex, wf_spec, result):
"""Handles event of action result arrival. """Handles event of action result arrival.
Given action result this method performs analysis of the workflow Given action result this method changes corresponding task execution
execution and identifies commands (including tasks) that can be object. This method must never be called for the case of individual
scheduled for execution. action which is not associated with any tasks.
:param action_ex: Action execution objects the result belongs to. :param action_ex: Action execution objects the result belongs to.
:param wf_spec: Workflow specification. :param wf_spec: Workflow specification.
:param result: Task action/workflow output wrapped into :param result: Task action/workflow output wrapped into
mistral.workflow.utils.Result instance. mistral.workflow.utils.Result instance.
:return List of engine commands that need to be performed. :return Task execution object.
""" """
task_ex = action_ex.task_execution task_ex = action_ex.task_execution
@ -177,7 +203,18 @@ def on_action_complete(action_ex, wf_spec, result):
task_spec = wf_spec.get_tasks()[task_ex.name] task_spec = wf_spec.get_tasks()[task_ex.name]
try:
result = action_handler.transform_result(result, task_ex, task_spec) result = action_handler.transform_result(result, task_ex, task_spec)
except exc.YaqlEvaluationException as e:
err_msg = str(e)
LOG.error(
'YAQL error while transforming action result'
' [action_execution_id=%s, result=%s]: %s',
action_ex.id, result, err_msg
)
result = wf_utils.Result(error=err_msg)
# Ignore workflow executions because they're handled during # Ignore workflow executions because they're handled during
# workflow completion. # workflow completion.
@ -195,6 +232,7 @@ def on_action_complete(action_ex, wf_spec, result):
_complete_task(task_ex, task_spec, task_state, task_state_info) _complete_task(task_ex, task_spec, task_state, task_state_info)
else: else:
with_items.increase_capacity(task_ex) with_items.increase_capacity(task_ex)
if with_items.is_completed(task_ex): if with_items.is_completed(task_ex):
_complete_task( _complete_task(
task_ex, task_ex,
@ -405,7 +443,10 @@ def _schedule_run_action(task_ex, task_spec, action_input, index, wf_spec):
) )
action_ex = action_handler.create_action_execution( action_ex = action_handler.create_action_execution(
action_def, action_input, task_ex, index action_def,
action_input,
task_ex,
index
) )
target = expr.evaluate_recursively( target = expr.evaluate_recursively(
@ -506,11 +547,14 @@ def _complete_task(task_ex, task_spec, state, state_info=None):
set_task_state(task_ex, state, state_info) set_task_state(task_ex, state, state_info)
try: try:
data_flow.publish_variables( data_flow.publish_variables(task_ex, task_spec)
task_ex, except exc.MistralException as e:
task_spec LOG.error(
'An error while publishing task variables'
' [task_execution_id=%s]: %s',
task_ex.id, str(e)
) )
except Exception as e:
set_task_state(task_ex, states.ERROR, str(e)) set_task_state(task_ex, states.ERROR, str(e))
if not task_spec.get_keep_result(): if not task_spec.get_keep_result():
@ -518,7 +562,6 @@ def _complete_task(task_ex, task_spec, state, state_info=None):
def set_task_state(task_ex, state, state_info, processed=None): def set_task_state(task_ex, state, state_info, processed=None):
# TODO(rakhmerov): How do we log task result?
wf_trace.info( wf_trace.info(
task_ex.workflow_execution, task_ex.workflow_execution,
"Task execution '%s' [%s -> %s]" % "Task execution '%s' [%s -> %s]" %

View File

@ -146,17 +146,17 @@ class EngineTestCase(base.DbTestCase):
def is_task_in_state(self, task_ex_id, state): def is_task_in_state(self, task_ex_id, state):
return db_api.get_task_execution(task_ex_id).state == state return db_api.get_task_execution(task_ex_id).state == state
def is_execution_in_state(self, wf_ex_id, state): def is_execution_in_state(self, ex_id, state):
return db_api.get_workflow_execution(wf_ex_id).state == state return db_api.get_workflow_execution(ex_id).state == state
def is_execution_success(self, wf_ex_id): def is_execution_success(self, ex_id):
return self.is_execution_in_state(wf_ex_id, states.SUCCESS) return self.is_execution_in_state(ex_id, states.SUCCESS)
def is_execution_error(self, wf_ex_id): def is_execution_error(self, ex_id):
return self.is_execution_in_state(wf_ex_id, states.ERROR) return self.is_execution_in_state(ex_id, states.ERROR)
def is_execution_paused(self, wf_ex_id): def is_execution_paused(self, ex_id):
return self.is_execution_in_state(wf_ex_id, states.PAUSED) return self.is_execution_in_state(ex_id, states.PAUSED)
def is_task_success(self, task_ex_id): def is_task_success(self, task_ex_id):
return self.is_task_in_state(task_ex_id, states.SUCCESS) return self.is_task_in_state(task_ex_id, states.SUCCESS)

View File

@ -29,12 +29,14 @@ cfg.CONF.set_default('auth_enable', False, group='pecan')
class DirectWorkflowEngineTest(base.EngineTestCase): class DirectWorkflowEngineTest(base.EngineTestCase):
def _run_workflow(self, workflow_yaml, state=states.ERROR): def _run_workflow(self, wf_text, expected_state=states.ERROR):
wf_service.create_workflows(workflow_yaml) wf_service.create_workflows(wf_text)
wf_ex = self.engine.start_workflow('wf', {}) wf_ex = self.engine.start_workflow('wf', {})
self._await(lambda: self.is_execution_in_state(wf_ex.id, state)) self._await(
lambda: self.is_execution_in_state(wf_ex.id, expected_state)
)
return db_api.get_workflow_execution(wf_ex.id) return db_api.get_workflow_execution(wf_ex.id)
@ -274,17 +276,18 @@ class DirectWorkflowEngineTest(base.EngineTestCase):
self.assertEqual(states.ERROR, wf_ex.state) self.assertEqual(states.ERROR, wf_ex.state)
self.assertIn('Can not evaluate YAQL expression', wf_ex.state_info) self.assertIn('Can not evaluate YAQL expression', wf_ex.state_info)
# Assert that there is only one task execution and it's SUCCESS. task_execs = wf_ex.task_executions
self.assertEqual(1, len(wf_ex.task_executions))
self.assertEqual(2, len(task_execs))
# 'task1' should be in SUCCESS.
task_1_ex = self._assert_single_item( task_1_ex = self._assert_single_item(
wf_ex.task_executions, task_execs,
name='task1' name='task1',
state=states.SUCCESS
) )
self.assertEqual(states.SUCCESS, task_1_ex.state) # 'task1' should have exactly one action execution (in SUCCESS).
# Assert that there is only one action execution and it's SUCCESS.
task_1_action_exs = db_api.get_action_executions( task_1_action_exs = db_api.get_action_executions(
task_execution_id=task_1_ex.id task_execution_id=task_1_ex.id
) )
@ -292,6 +295,19 @@ class DirectWorkflowEngineTest(base.EngineTestCase):
self.assertEqual(1, len(task_1_action_exs)) self.assertEqual(1, len(task_1_action_exs))
self.assertEqual(states.SUCCESS, task_1_action_exs[0].state) self.assertEqual(states.SUCCESS, task_1_action_exs[0].state)
# 'task2' should exist but in ERROR.
task_2_ex = self._assert_single_item(
task_execs,
name='task2',
state=states.ERROR
)
# 'task2' must not have action executions.
self.assertEqual(
0,
len(db_api.get_action_executions(task_execution_id=task_2_ex.id))
)
def test_async_next_task_with_input_yaql_error(self): def test_async_next_task_with_input_yaql_error(self):
wf_text = """ wf_text = """
version: '2.0' version: '2.0'
@ -331,29 +347,28 @@ class DirectWorkflowEngineTest(base.EngineTestCase):
self.assertEqual(states.RUNNING, task_1_action_exs[0].state) self.assertEqual(states.RUNNING, task_1_action_exs[0].state)
# Update async action execution result. # Update async action execution result.
result = wf_utils.Result(data='foobar') self.engine.on_action_complete(
self.assertRaises(
exc.YaqlEvaluationException,
self.engine.on_action_complete,
task_1_action_exs[0].id, task_1_action_exs[0].id,
result wf_utils.Result(data='foobar')
) )
# Assert that task1 is SUCCESS and workflow is ERROR.
wf_ex = db_api.get_workflow_execution(wf_ex.id) wf_ex = db_api.get_workflow_execution(wf_ex.id)
self.assertEqual(states.ERROR, wf_ex.state) self.assertEqual(states.ERROR, wf_ex.state)
self.assertIn('Can not evaluate YAQL expression', wf_ex.state_info) self.assertIn('Can not evaluate YAQL expression', wf_ex.state_info)
self.assertEqual(1, len(wf_ex.task_executions))
task_execs = wf_ex.task_executions
self.assertEqual(2, len(task_execs))
# 'task1' must be in SUCCESS.
task_1_ex = self._assert_single_item( task_1_ex = self._assert_single_item(
wf_ex.task_executions, task_execs,
name='task1' name='task1',
state=states.SUCCESS
) )
self.assertEqual(states.SUCCESS, task_1_ex.state) # 'task1' must have exactly one action execution (in SUCCESS).
task_1_action_exs = db_api.get_action_executions( task_1_action_exs = db_api.get_action_executions(
task_execution_id=task_1_ex.id task_execution_id=task_1_ex.id
) )
@ -361,6 +376,19 @@ class DirectWorkflowEngineTest(base.EngineTestCase):
self.assertEqual(1, len(task_1_action_exs)) self.assertEqual(1, len(task_1_action_exs))
self.assertEqual(states.SUCCESS, task_1_action_exs[0].state) self.assertEqual(states.SUCCESS, task_1_action_exs[0].state)
# 'task2' must be in ERROR.
task_2_ex = self._assert_single_item(
task_execs,
name='task2',
state=states.ERROR
)
# 'task2' must not have action executions.
self.assertEqual(
0,
len(db_api.get_action_executions(task_execution_id=task_2_ex.id))
)
def test_messed_yaql_in_first_task(self): def test_messed_yaql_in_first_task(self):
wf_text = """ wf_text = """
version: '2.0' version: '2.0'
@ -511,13 +539,9 @@ class DirectWorkflowEngineTest(base.EngineTestCase):
self.assertEqual(states.RUNNING, task_1_action_exs[0].state) self.assertEqual(states.RUNNING, task_1_action_exs[0].state)
# Update async action execution result. # Update async action execution result.
result = wf_utils.Result(data='foobar') self.engine.on_action_complete(
self.assertRaises(
exc.YaqlEvaluationException,
self.engine.on_action_complete,
task_1_action_exs[0].id, task_1_action_exs[0].id,
result wf_utils.Result(data='foobar')
) )
# Assert that task1 is SUCCESS and workflow is ERROR. # Assert that task1 is SUCCESS and workflow is ERROR.

View File

@ -37,6 +37,7 @@ wf:
input: input:
- workflow_input: '__WORKFLOW_INPUT__' - workflow_input: '__WORKFLOW_INPUT__'
- action_output_length: 0 - action_output_length: 0
tasks: tasks:
task1: task1:
action: my_action action: my_action
@ -79,8 +80,10 @@ def expect_size_limit_exception(field_name):
def generate_workflow(tokens): def generate_workflow(tokens):
new_wf = WF new_wf = WF
long_string = ''.join('A' for _ in range(1024)) long_string = ''.join('A' for _ in range(1024))
for token in tokens: for token in tokens:
new_wf = new_wf.replace(token, long_string) new_wf = new_wf.replace(token, long_string)
return new_wf return new_wf
@ -136,11 +139,11 @@ class ExecutionFieldsSizeLimitTest(base.EngineTestCase):
# Start workflow. # Start workflow.
wf_ex = self.engine.start_workflow('wf', {}) wf_ex = self.engine.start_workflow('wf', {})
self.assertEqual(states.ERROR, wf_ex.state)
self.assertIn( self.assertIn(
"Size of 'input' is 1KB which exceeds the limit of 0KB", "Size of 'input' is 1KB which exceeds the limit of 0KB",
wf_ex.state_info wf_ex.state_info
) )
self.assertEqual(states.ERROR, wf_ex.state)
def test_action_output_limit(self): def test_action_output_limit(self):
wf_service.create_workflows(WF) wf_service.create_workflows(WF)
@ -175,7 +178,7 @@ class ExecutionFieldsSizeLimitTest(base.EngineTestCase):
wf_ex = db_api.get_workflow_execution(wf_ex.id) wf_ex = db_api.get_workflow_execution(wf_ex.id)
self.assertIn( self.assertIn(
"Failure caused by error in tasks: task1", 'Failure caused by error in tasks: task1',
wf_ex.state_info wf_ex.state_info
) )

View File

@ -33,7 +33,7 @@ from mistral.workflow import utils as wf_utils
# the change in value is not permanent. # the change in value is not permanent.
cfg.CONF.set_default('auth_enable', False, group='pecan') cfg.CONF.set_default('auth_enable', False, group='pecan')
WORKBOOK = """ WB = """
--- ---
version: "2.0" version: "2.0"
@ -55,7 +55,7 @@ workflows:
""" """
WORKBOOK_WITH_STATIC_VAR = """ WB_WITH_STATIC_VAR = """
--- ---
version: "2.0" version: "2.0"
@ -78,7 +78,7 @@ workflows:
""" """
WORKBOOK_MULTI_ARRAY = """ WB_MULTI_ARRAY = """
--- ---
version: "2.0" version: "2.0"
@ -104,7 +104,7 @@ workflows:
""" """
WORKBOOK_ACTION_CONTEXT = """ WB_ACTION_CONTEXT = """
--- ---
version: "2.0" version: "2.0"
name: wb1 name: wb1
@ -123,7 +123,7 @@ workflows:
""" """
WORKFLOW_INPUT = { WF_INPUT = {
'names_info': [ 'names_info': [
{'name': 'John'}, {'name': 'John'},
{'name': 'Ivan'}, {'name': 'Ivan'},
@ -140,7 +140,7 @@ WF_INPUT_URLS = {
] ]
} }
WORKFLOW_INPUT_ONE_ITEM = { WF_INPUT_ONE_ITEM = {
'names_info': [ 'names_info': [
{'name': 'Guy'} {'name': 'Guy'}
] ]
@ -153,6 +153,7 @@ class RandomSleepEchoAction(action_base.Action):
def run(self): def run(self):
utils.random_sleep(1) utils.random_sleep(1)
return self.output return self.output
def test(self): def test(self):
@ -176,10 +177,10 @@ class WithItemsEngineTest(base.EngineTestCase):
if ex.state == states.RUNNING]) if ex.state == states.RUNNING])
def test_with_items_simple(self): def test_with_items_simple(self):
wb_service.create_workbook_v2(WORKBOOK) wb_service.create_workbook_v2(WB)
# Start workflow. # Start workflow.
wf_ex = self.engine.start_workflow('wb1.with_items', WORKFLOW_INPUT) wf_ex = self.engine.start_workflow('wb1.with_items', WF_INPUT)
self._await( self._await(
lambda: self.is_execution_success(wf_ex.id), lambda: self.is_execution_success(wf_ex.id),
@ -188,15 +189,17 @@ class WithItemsEngineTest(base.EngineTestCase):
# Note: We need to reread execution to access related tasks. # Note: We need to reread execution to access related tasks.
wf_ex = db_api.get_workflow_execution(wf_ex.id) wf_ex = db_api.get_workflow_execution(wf_ex.id)
tasks = wf_ex.task_executions task_execs = wf_ex.task_executions
task1 = self._assert_single_item(tasks, name='task1')
with_items_context = task1.runtime_context['with_items_context']
self.assertEqual(3, with_items_context['count']) task1_ex = self._assert_single_item(task_execs, name='task1')
with_items_ctx = task1_ex.runtime_context['with_items_context']
self.assertEqual(3, with_items_ctx['count'])
# Since we know that we can receive results in random order, # Since we know that we can receive results in random order,
# check is not depend on order of items. # check is not depend on order of items.
result = data_flow.get_task_execution_result(task1) result = data_flow.get_task_execution_result(task1_ex)
self.assertIsInstance(result, list) self.assertIsInstance(result, list)
@ -204,15 +207,15 @@ class WithItemsEngineTest(base.EngineTestCase):
self.assertIn('Ivan', result) self.assertIn('Ivan', result)
self.assertIn('Mistral', result) self.assertIn('Mistral', result)
published = task1.published published = task1_ex.published
self.assertIn(published['result'], ['John', 'Ivan', 'Mistral']) self.assertIn(published['result'], ['John', 'Ivan', 'Mistral'])
self.assertEqual(1, len(tasks)) self.assertEqual(1, len(task_execs))
self.assertEqual(states.SUCCESS, task1.state) self.assertEqual(states.SUCCESS, task1_ex.state)
def test_with_items_fail(self): def test_with_items_fail(self):
workflow = """--- wf_text = """---
version: "2.0" version: "2.0"
with_items: with_items:
@ -227,23 +230,21 @@ class WithItemsEngineTest(base.EngineTestCase):
task2: task2:
action: std.echo output="With-items failed" action: std.echo output="With-items failed"
""" """
wf_service.create_workflows(workflow)
wf_service.create_workflows(wf_text)
# Start workflow. # Start workflow.
wf_ex = self.engine.start_workflow('with_items', {}) wf_ex = self.engine.start_workflow('with_items', {})
self._await( self._await(lambda: self.is_execution_success(wf_ex.id))
lambda: self.is_execution_success(wf_ex.id),
)
# Note: We need to reread execution to access related tasks. # Note: We need to reread execution to access related tasks.
wf_ex = db_api.get_workflow_execution(wf_ex.id) wf_ex = db_api.get_workflow_execution(wf_ex.id)
tasks = wf_ex.task_executions self.assertEqual(2, len(wf_ex.task_executions))
self.assertEqual(2, len(tasks))
def test_with_items_sub_workflow_fail(self): def test_with_items_sub_workflow_fail(self):
workbook = """--- wb_text = """---
version: "2.0" version: "2.0"
name: wb1 name: wb1
@ -263,36 +264,34 @@ class WithItemsEngineTest(base.EngineTestCase):
subworkflow: subworkflow:
type: direct type: direct
tasks: tasks:
fail: fail:
action: std.fail action: std.fail
""" """
wb_service.create_workbook_v2(workbook)
wb_service.create_workbook_v2(wb_text)
# Start workflow. # Start workflow.
wf_ex = self.engine.start_workflow('wb1.with_items', {}) wf_ex = self.engine.start_workflow('wb1.with_items', {})
self._await( self._await(lambda: self.is_execution_success(wf_ex.id))
lambda: self.is_execution_success(wf_ex.id),
)
# Note: We need to reread execution to access related tasks. # Note: We need to reread execution to access related tasks.
wf_ex = db_api.get_workflow_execution(wf_ex.id) wf_ex = db_api.get_workflow_execution(wf_ex.id)
tasks = wf_ex.task_executions self.assertEqual(2, len(wf_ex.task_executions))
self.assertEqual(2, len(tasks))
def test_with_items_static_var(self): def test_with_items_static_var(self):
wb_service.create_workbook_v2(WORKBOOK_WITH_STATIC_VAR) wb_service.create_workbook_v2(WB_WITH_STATIC_VAR)
wf_input = copy.deepcopy(WORKFLOW_INPUT) wf_input = copy.deepcopy(WF_INPUT)
wf_input.update({'greeting': 'Hello'}) wf_input.update({'greeting': 'Hello'})
# Start workflow. # Start workflow.
wf_ex = self.engine.start_workflow('wb1.with_items', wf_input) wf_ex = self.engine.start_workflow('wb1.with_items', wf_input)
self._await( self._await(lambda: self.is_execution_success(wf_ex.id))
lambda: self.is_execution_success(wf_ex.id),
)
# Note: We need to reread execution to access related tasks. # Note: We need to reread execution to access related tasks.
wf_ex = db_api.get_workflow_execution(wf_ex.id) wf_ex = db_api.get_workflow_execution(wf_ex.id)
@ -311,26 +310,25 @@ class WithItemsEngineTest(base.EngineTestCase):
self.assertEqual(states.SUCCESS, task1.state) self.assertEqual(states.SUCCESS, task1.state)
def test_with_items_multi_array(self): def test_with_items_multi_array(self):
wb_service.create_workbook_v2(WORKBOOK_MULTI_ARRAY) wb_service.create_workbook_v2(WB_MULTI_ARRAY)
wf_input = {'arrayI': ['a', 'b', 'c'], 'arrayJ': [1, 2, 3]} wf_input = {'arrayI': ['a', 'b', 'c'], 'arrayJ': [1, 2, 3]}
# Start workflow. # Start workflow.
wf_ex = self.engine.start_workflow('wb1.with_items', wf_input) wf_ex = self.engine.start_workflow('wb1.with_items', wf_input)
self._await( self._await(lambda: self.is_execution_success(wf_ex.id))
lambda: self.is_execution_success(wf_ex.id),
)
# Note: We need to reread execution to access related tasks. # Note: We need to reread execution to access related tasks.
wf_ex = db_api.get_workflow_execution(wf_ex.id) wf_ex = db_api.get_workflow_execution(wf_ex.id)
tasks = wf_ex.task_executions task_execs = wf_ex.task_executions
task1 = self._assert_single_item(tasks, name='task1')
task1_ex = self._assert_single_item(task_execs, name='task1')
# Since we know that we can receive results in random order, # Since we know that we can receive results in random order,
# check is not depend on order of items. # check is not depend on order of items.
result = data_flow.get_task_execution_result(task1) result = data_flow.get_task_execution_result(task1_ex)
self.assertIsInstance(result, list) self.assertIsInstance(result, list)
@ -338,30 +336,28 @@ class WithItemsEngineTest(base.EngineTestCase):
self.assertIn('b 2', result) self.assertIn('b 2', result)
self.assertIn('c 3', result) self.assertIn('c 3', result)
self.assertEqual(1, len(tasks)) self.assertEqual(1, len(task_execs))
self.assertEqual(states.SUCCESS, task1.state) self.assertEqual(states.SUCCESS, task1_ex.state)
def test_with_items_action_context(self): def test_with_items_action_context(self):
wb_service.create_workbook_v2(WORKBOOK_ACTION_CONTEXT) wb_service.create_workbook_v2(WB_ACTION_CONTEXT)
# Start workflow. # Start workflow.
wf_ex = self.engine.start_workflow( wf_ex = self.engine.start_workflow('wb1.wf1_with_items', WF_INPUT_URLS)
'wb1.wf1_with_items', WF_INPUT_URLS
)
wf_ex = db_api.get_workflow_execution(wf_ex.id) wf_ex = db_api.get_workflow_execution(wf_ex.id)
task_ex = wf_ex.task_executions[0] task_ex = wf_ex.task_executions[0]
act_exs = task_ex.executions act_exs = task_ex.executions
self.engine.on_action_complete(act_exs[0].id, wf_utils.Result("Ivan")) self.engine.on_action_complete(act_exs[0].id, wf_utils.Result("Ivan"))
self.engine.on_action_complete(act_exs[1].id, wf_utils.Result("John")) self.engine.on_action_complete(act_exs[1].id, wf_utils.Result("John"))
self.engine.on_action_complete( self.engine.on_action_complete(
act_exs[2].id, wf_utils.Result("Mistral") act_exs[2].id,
wf_utils.Result("Mistral")
) )
self._await( self._await(lambda: self.is_execution_success(wf_ex.id))
lambda: self.is_execution_success(wf_ex.id),
)
# Note: We need to reread execution to access related tasks. # Note: We need to reread execution to access related tasks.
wf_ex = db_api.get_workflow_execution(wf_ex.id) wf_ex = db_api.get_workflow_execution(wf_ex.id)
@ -378,7 +374,7 @@ class WithItemsEngineTest(base.EngineTestCase):
self.assertEqual(states.SUCCESS, task_ex.state) self.assertEqual(states.SUCCESS, task_ex.state)
def test_with_items_empty_list(self): def test_with_items_empty_list(self):
workbook = """--- wb_text = """---
version: "2.0" version: "2.0"
name: wb1 name: wb1
@ -400,29 +396,29 @@ class WithItemsEngineTest(base.EngineTestCase):
task2: task2:
action: std.echo output="Hi!" action: std.echo output="Hi!"
""" """
wb_service.create_workbook_v2(workbook)
wb_service.create_workbook_v2(wb_text)
# Start workflow. # Start workflow.
wf_input = {'names_info': []} wf_input = {'names_info': []}
wf_ex = self.engine.start_workflow('wb1.with_items', wf_input) wf_ex = self.engine.start_workflow('wb1.with_items', wf_input)
self._await( self._await(lambda: self.is_execution_success(wf_ex.id))
lambda: self.is_execution_success(wf_ex.id),
)
# Note: We need to reread execution to access related tasks. # Note: We need to reread execution to access related tasks.
wf_ex = db_api.get_workflow_execution(wf_ex.id) wf_ex = db_api.get_workflow_execution(wf_ex.id)
tasks = wf_ex.task_executions task_execs = wf_ex.task_executions
task1 = self._assert_single_item(tasks, name='task1')
task2 = self._assert_single_item(tasks, name='task2')
self.assertEqual(2, len(tasks)) task1_ex = self._assert_single_item(task_execs, name='task1')
self.assertEqual(states.SUCCESS, task1.state) task2_ex = self._assert_single_item(task_execs, name='task2')
self.assertEqual(states.SUCCESS, task2.state)
self.assertEqual(2, len(task_execs))
self.assertEqual(states.SUCCESS, task1_ex.state)
self.assertEqual(states.SUCCESS, task2_ex.state)
def test_with_items_plain_list(self): def test_with_items_plain_list(self):
workbook = """--- wb_text = """---
version: "2.0" version: "2.0"
name: wb1 name: wb1
@ -436,7 +432,8 @@ class WithItemsEngineTest(base.EngineTestCase):
with-items: i in [1, 2, 3] with-items: i in [1, 2, 3]
action: std.echo output=<% $.i %> action: std.echo output=<% $.i %>
""" """
wb_service.create_workbook_v2(workbook)
wb_service.create_workbook_v2(wb_text)
# Start workflow. # Start workflow.
wf_ex = self.engine.start_workflow('wb1.with_items', {}) wf_ex = self.engine.start_workflow('wb1.with_items', {})
@ -446,11 +443,13 @@ class WithItemsEngineTest(base.EngineTestCase):
# Note: We need to reread execution to access related tasks. # Note: We need to reread execution to access related tasks.
wf_ex = db_api.get_workflow_execution(wf_ex.id) wf_ex = db_api.get_workflow_execution(wf_ex.id)
tasks = wf_ex.task_executions task1_ex = self._assert_single_item(
task1 = self._assert_single_item(tasks, name='task1') wf_ex.task_executions,
self.assertEqual(states.SUCCESS, task1.state) name='task1',
state=states.SUCCESS
)
result = data_flow.get_task_execution_result(task1) result = data_flow.get_task_execution_result(task1_ex)
# Since we know that we can receive results in random order, # Since we know that we can receive results in random order,
# check is not depend on order of items. # check is not depend on order of items.
@ -459,7 +458,7 @@ class WithItemsEngineTest(base.EngineTestCase):
self.assertIn(3, result) self.assertIn(3, result)
def test_with_items_plain_list_wrong(self): def test_with_items_plain_list_wrong(self):
workbook = """--- wb_text = """---
version: "2.0" version: "2.0"
name: wb1 name: wb1
@ -477,13 +476,13 @@ class WithItemsEngineTest(base.EngineTestCase):
exception = self.assertRaises( exception = self.assertRaises(
exc.InvalidModelException, exc.InvalidModelException,
wb_service.create_workbook_v2, workbook wb_service.create_workbook_v2, wb_text
) )
self.assertIn("Invalid array in 'with-items'", exception.message) self.assertIn("Invalid array in 'with-items'", exception.message)
def test_with_items_results_order(self): def test_with_items_results_order(self):
workbook = """--- wb_text = """---
version: "2.0" version: "2.0"
name: wb1 name: wb1
@ -502,60 +501,57 @@ class WithItemsEngineTest(base.EngineTestCase):
# Register random sleep action in the DB. # Register random sleep action in the DB.
test_base.register_action_class('sleep_echo', RandomSleepEchoAction) test_base.register_action_class('sleep_echo', RandomSleepEchoAction)
wb_service.create_workbook_v2(workbook) wb_service.create_workbook_v2(wb_text)
# Start workflow. # Start workflow.
wf_ex = self.engine.start_workflow('wb1.with_items', {}) wf_ex = self.engine.start_workflow('wb1.with_items', {})
self._await( self._await(lambda: self.is_execution_success(wf_ex.id))
lambda: self.is_execution_success(wf_ex.id),
)
# Note: We need to reread execution to access related tasks. # Note: We need to reread execution to access related tasks.
wf_ex = db_api.get_workflow_execution(wf_ex.id) wf_ex = db_api.get_workflow_execution(wf_ex.id)
tasks = wf_ex.task_executions task1_ex = self._assert_single_item(
task1 = self._assert_single_item(tasks, name='task1') wf_ex.task_executions,
name='task1',
state=states.SUCCESS
)
self.assertEqual(states.SUCCESS, task1.state) published = task1_ex.published
published = task1.published
# Now we can check order of results explicitly. # Now we can check order of results explicitly.
self.assertEqual([1, 2, 3], published['one_two_three']) self.assertEqual([1, 2, 3], published['one_two_three'])
def test_with_items_results_one_item_as_list(self): def test_with_items_results_one_item_as_list(self):
wb_service.create_workbook_v2(WORKBOOK) wb_service.create_workbook_v2(WB)
# Start workflow. # Start workflow.
wf_ex = self.engine.start_workflow('wb1.with_items', wf_ex = self.engine.start_workflow('wb1.with_items', WF_INPUT_ONE_ITEM)
WORKFLOW_INPUT_ONE_ITEM)
self._await( self._await(lambda: self.is_execution_success(wf_ex.id))
lambda: self.is_execution_success(wf_ex.id),
)
# Note: We need to reread execution to access related tasks. # Note: We need to reread execution to access related tasks.
wf_ex = db_api.get_workflow_execution(wf_ex.id) wf_ex = db_api.get_workflow_execution(wf_ex.id)
tasks = wf_ex.task_executions task_execs = wf_ex.task_executions
task1 = self._assert_single_item(tasks, name='task1')
result = data_flow.get_task_execution_result(task1) self.assertEqual(1, len(task_execs))
task1_ex = self._assert_single_item(
task_execs,
name='task1',
state=states.SUCCESS
)
result = data_flow.get_task_execution_result(task1_ex)
self.assertIsInstance(result, list) self.assertIsInstance(result, list)
self.assertIn('Guy', result) self.assertIn('Guy', result)
published = task1.published self.assertIn(task1_ex.published['result'], ['Guy'])
self.assertIn(published['result'], ['Guy'])
self.assertEqual(1, len(tasks))
self.assertEqual(states.SUCCESS, task1.state)
def test_with_items_concurrency_1(self): def test_with_items_concurrency_1(self):
workflow_with_concurrency_1 = """--- wf_with_concurrency_1 = """---
version: "2.0" version: "2.0"
concurrency_test: concurrency_test:
@ -569,16 +565,18 @@ class WithItemsEngineTest(base.EngineTestCase):
action: std.async_noop action: std.async_noop
with-items: name in <% $.names %> with-items: name in <% $.names %>
concurrency: 1 concurrency: 1
""" """
wf_service.create_workflows(workflow_with_concurrency_1)
wf_service.create_workflows(wf_with_concurrency_1)
# Start workflow. # Start workflow.
wf_ex = self.engine.start_workflow('concurrency_test', {}) wf_ex = self.engine.start_workflow('concurrency_test', {})
wf_ex = db_api.get_execution(wf_ex.id)
task_ex = wf_ex.task_executions[0]
wf_ex = db_api.get_execution(wf_ex.id)
task_ex = wf_ex.task_executions[0]
task_ex = db_api.get_task_execution(task_ex.id) task_ex = db_api.get_task_execution(task_ex.id)
self.assert_capacity(0, task_ex) self.assert_capacity(0, task_ex)
self.assertEqual(1, self.get_running_action_exs_number(task_ex)) self.assertEqual(1, self.get_running_action_exs_number(task_ex))
@ -589,6 +587,7 @@ class WithItemsEngineTest(base.EngineTestCase):
) )
task_ex = db_api.get_task_execution(task_ex.id) task_ex = db_api.get_task_execution(task_ex.id)
self.assert_capacity(0, task_ex) self.assert_capacity(0, task_ex)
self.assertEqual(1, self.get_running_action_exs_number(task_ex)) self.assertEqual(1, self.get_running_action_exs_number(task_ex))
@ -599,6 +598,7 @@ class WithItemsEngineTest(base.EngineTestCase):
) )
task_ex = db_api.get_task_execution(task_ex.id) task_ex = db_api.get_task_execution(task_ex.id)
self.assert_capacity(0, task_ex) self.assert_capacity(0, task_ex)
self.assertEqual(1, self.get_running_action_exs_number(task_ex)) self.assertEqual(1, self.get_running_action_exs_number(task_ex))
@ -609,13 +609,13 @@ class WithItemsEngineTest(base.EngineTestCase):
) )
task_ex = db_api.get_task_execution(task_ex.id) task_ex = db_api.get_task_execution(task_ex.id)
self.assert_capacity(1, task_ex) self.assert_capacity(1, task_ex)
self._await( self._await(lambda: self.is_execution_success(wf_ex.id))
lambda: self.is_execution_success(wf_ex.id),
)
task_ex = db_api.get_task_execution(task_ex.id) task_ex = db_api.get_task_execution(task_ex.id)
# Since we know that we can receive results in random order, # Since we know that we can receive results in random order,
# check is not depend on order of items. # check is not depend on order of items.
result = data_flow.get_task_execution_result(task_ex) result = data_flow.get_task_execution_result(task_ex)
@ -629,7 +629,7 @@ class WithItemsEngineTest(base.EngineTestCase):
self.assertEqual(states.SUCCESS, task_ex.state) self.assertEqual(states.SUCCESS, task_ex.state)
def test_with_items_concurrency_yaql(self): def test_with_items_concurrency_yaql(self):
workflow_with_concurrency_yaql = """--- wf_with_concurrency_yaql = """---
version: "2.0" version: "2.0"
concurrency_test: concurrency_test:
@ -644,9 +644,9 @@ class WithItemsEngineTest(base.EngineTestCase):
action: std.echo output=<% $.name %> action: std.echo output=<% $.name %>
with-items: name in <% $.names %> with-items: name in <% $.names %>
concurrency: <% $.concurrency %> concurrency: <% $.concurrency %>
""" """
wf_service.create_workflows(workflow_with_concurrency_yaql)
wf_service.create_workflows(wf_with_concurrency_yaql)
# Start workflow. # Start workflow.
wf_ex = self.engine.start_workflow( wf_ex = self.engine.start_workflow(
@ -654,13 +654,14 @@ class WithItemsEngineTest(base.EngineTestCase):
{'concurrency': 2} {'concurrency': 2}
) )
self._await( self._await(lambda: self.is_execution_success(wf_ex.id))
lambda: self.is_execution_success(wf_ex.id),
)
wf_ex = db_api.get_workflow_execution(wf_ex.id) wf_ex = db_api.get_workflow_execution(wf_ex.id)
task_ex = wf_ex.task_executions[0] task_ex = wf_ex.task_executions[0]
self.assertEqual(states.SUCCESS, task_ex.state)
# Since we know that we can receive results in random order, # Since we know that we can receive results in random order,
# check is not depend on order of items. # check is not depend on order of items.
result = data_flow.get_task_execution_result(task_ex) result = data_flow.get_task_execution_result(task_ex)
@ -671,10 +672,8 @@ class WithItemsEngineTest(base.EngineTestCase):
self.assertIn('Ivan', result) self.assertIn('Ivan', result)
self.assertIn('Mistral', result) self.assertIn('Mistral', result)
self.assertEqual(states.SUCCESS, task_ex.state)
def test_with_items_concurrency_yaql_wrong_type(self): def test_with_items_concurrency_yaql_wrong_type(self):
workflow_with_concurrency_yaql = """--- wf_with_concurrency_yaql = """---
version: "2.0" version: "2.0"
concurrency_test: concurrency_test:
@ -689,9 +688,9 @@ class WithItemsEngineTest(base.EngineTestCase):
action: std.echo output=<% $.name %> action: std.echo output=<% $.name %>
with-items: name in <% $.names %> with-items: name in <% $.names %>
concurrency: <% $.concurrency %> concurrency: <% $.concurrency %>
""" """
wf_service.create_workflows(workflow_with_concurrency_yaql)
wf_service.create_workflows(wf_with_concurrency_yaql)
# Start workflow. # Start workflow.
wf_ex = self.engine.start_workflow( wf_ex = self.engine.start_workflow(
@ -700,13 +699,13 @@ class WithItemsEngineTest(base.EngineTestCase):
) )
self.assertIn( self.assertIn(
"Invalid data type in ConcurrencyPolicy", 'Invalid data type in ConcurrencyPolicy',
wf_ex.state_info wf_ex.state_info
) )
self.assertEqual(states.ERROR, wf_ex.state) self.assertEqual(states.ERROR, wf_ex.state)
def test_with_items_concurrency_2(self): def test_with_items_concurrency_2(self):
workflow_with_concurrency_2 = """--- wf_with_concurrency_2 = """---
version: "2.0" version: "2.0"
concurrency_test: concurrency_test:
@ -722,10 +721,11 @@ class WithItemsEngineTest(base.EngineTestCase):
concurrency: 2 concurrency: 2
""" """
wf_service.create_workflows(workflow_with_concurrency_2) wf_service.create_workflows(wf_with_concurrency_2)
# Start workflow. # Start workflow.
wf_ex = self.engine.start_workflow('concurrency_test', {}) wf_ex = self.engine.start_workflow('concurrency_test', {})
wf_ex = db_api.get_execution(wf_ex.id) wf_ex = db_api.get_execution(wf_ex.id)
task_ex = wf_ex.task_executions[0] task_ex = wf_ex.task_executions[0]
@ -739,6 +739,7 @@ class WithItemsEngineTest(base.EngineTestCase):
) )
task_ex = db_api.get_task_execution(task_ex.id) task_ex = db_api.get_task_execution(task_ex.id)
self.assert_capacity(0, task_ex) self.assert_capacity(0, task_ex)
self.assertEqual(2, self.get_running_action_exs_number(task_ex)) self.assertEqual(2, self.get_running_action_exs_number(task_ex))
@ -749,6 +750,7 @@ class WithItemsEngineTest(base.EngineTestCase):
) )
task_ex = db_api.get_task_execution(task_ex.id) task_ex = db_api.get_task_execution(task_ex.id)
self.assert_capacity(0, task_ex) self.assert_capacity(0, task_ex)
self.assertEqual(2, self.get_running_action_exs_number(task_ex)) self.assertEqual(2, self.get_running_action_exs_number(task_ex))
@ -759,6 +761,7 @@ class WithItemsEngineTest(base.EngineTestCase):
) )
task_ex = db_api.get_task_execution(task_ex.id) task_ex = db_api.get_task_execution(task_ex.id)
self.assert_capacity(1, task_ex) self.assert_capacity(1, task_ex)
# 4th iteration complete. # 4th iteration complete.
@ -768,16 +771,17 @@ class WithItemsEngineTest(base.EngineTestCase):
) )
task_ex = db_api.get_task_execution(task_ex.id) task_ex = db_api.get_task_execution(task_ex.id)
self.assert_capacity(2, task_ex) self.assert_capacity(2, task_ex)
self._await( self._await(lambda: self.is_execution_success(wf_ex.id))
lambda: self.is_execution_success(wf_ex.id),
)
task_ex = db_api.get_task_execution(task_ex.id) task_ex = db_api.get_task_execution(task_ex.id)
# Since we know that we can receive results in random order, # Since we know that we can receive results in random order,
# check is not depend on order of items. # check is not depend on order of items.
result = data_flow.get_task_execution_result(task_ex) result = data_flow.get_task_execution_result(task_ex)
self.assertIsInstance(result, list) self.assertIsInstance(result, list)
self.assertIn('John', result) self.assertIn('John', result)
@ -788,7 +792,7 @@ class WithItemsEngineTest(base.EngineTestCase):
self.assertEqual(states.SUCCESS, task_ex.state) self.assertEqual(states.SUCCESS, task_ex.state)
def test_with_items_concurrency_2_fail(self): def test_with_items_concurrency_2_fail(self):
workflow_with_concurrency_2_fail = """--- wf_with_concurrency_2_fail = """---
version: "2.0" version: "2.0"
concurrency_test_fail: concurrency_test_fail:
@ -805,14 +809,13 @@ class WithItemsEngineTest(base.EngineTestCase):
action: std.echo output="With-items failed" action: std.echo output="With-items failed"
""" """
wf_service.create_workflows(workflow_with_concurrency_2_fail) wf_service.create_workflows(wf_with_concurrency_2_fail)
# Start workflow. # Start workflow.
wf_ex = self.engine.start_workflow('concurrency_test_fail', {}) wf_ex = self.engine.start_workflow('concurrency_test_fail', {})
self._await( self._await(lambda: self.is_execution_success(wf_ex.id))
lambda: self.is_execution_success(wf_ex.id),
)
wf_ex = db_api.get_execution(wf_ex.id) wf_ex = db_api.get_execution(wf_ex.id)
task_exs = wf_ex.task_executions task_exs = wf_ex.task_executions
@ -822,12 +825,12 @@ class WithItemsEngineTest(base.EngineTestCase):
task_2 = self._assert_single_item(task_exs, name='task2') task_2 = self._assert_single_item(task_exs, name='task2')
self.assertEqual( self.assertEqual(
"With-items failed", 'With-items failed',
data_flow.get_task_execution_result(task_2) data_flow.get_task_execution_result(task_2)
) )
def test_with_items_concurrency_3(self): def test_with_items_concurrency_3(self):
workflow_with_concurrency_3 = """--- wf_with_concurrency_3 = """---
version: "2.0" version: "2.0"
concurrency_test: concurrency_test:
@ -843,10 +846,12 @@ class WithItemsEngineTest(base.EngineTestCase):
concurrency: 3 concurrency: 3
""" """
wf_service.create_workflows(workflow_with_concurrency_3)
wf_service.create_workflows(wf_with_concurrency_3)
# Start workflow. # Start workflow.
wf_ex = self.engine.start_workflow('concurrency_test', {}) wf_ex = self.engine.start_workflow('concurrency_test', {})
wf_ex = db_api.get_execution(wf_ex.id) wf_ex = db_api.get_execution(wf_ex.id)
task_ex = wf_ex.task_executions[0] task_ex = wf_ex.task_executions[0]
@ -860,6 +865,7 @@ class WithItemsEngineTest(base.EngineTestCase):
) )
task_ex = db_api.get_task_execution(task_ex.id) task_ex = db_api.get_task_execution(task_ex.id)
self.assert_capacity(1, task_ex) self.assert_capacity(1, task_ex)
# 2nd iteration complete. # 2nd iteration complete.
@ -869,6 +875,7 @@ class WithItemsEngineTest(base.EngineTestCase):
) )
task_ex = db_api.get_task_execution(task_ex.id) task_ex = db_api.get_task_execution(task_ex.id)
self.assert_capacity(2, task_ex) self.assert_capacity(2, task_ex)
# 3rd iteration complete. # 3rd iteration complete.
@ -878,26 +885,27 @@ class WithItemsEngineTest(base.EngineTestCase):
) )
task_ex = db_api.get_task_execution(task_ex.id) task_ex = db_api.get_task_execution(task_ex.id)
self.assert_capacity(3, task_ex) self.assert_capacity(3, task_ex)
self._await( self._await(lambda: self.is_execution_success(wf_ex.id))
lambda: self.is_execution_success(wf_ex.id),
)
task_ex = db_api.get_task_execution(task_ex.id) task_ex = db_api.get_task_execution(task_ex.id)
self.assertEqual(states.SUCCESS, task_ex.state)
# Since we know that we can receive results in random order, # Since we know that we can receive results in random order,
# check is not depend on order of items. # check is not depend on order of items.
result = data_flow.get_task_execution_result(task_ex) result = data_flow.get_task_execution_result(task_ex)
self.assertIsInstance(result, list) self.assertIsInstance(result, list)
self.assertIn('John', result) self.assertIn('John', result)
self.assertIn('Ivan', result) self.assertIn('Ivan', result)
self.assertIn('Mistral', result) self.assertIn('Mistral', result)
self.assertEqual(states.SUCCESS, task_ex.state)
def test_with_items_concurrency_gt_list_length(self): def test_with_items_concurrency_gt_list_length(self):
workflow_definition = """--- wf_definition = """---
version: "2.0" version: "2.0"
concurrency_test: concurrency_test:
@ -913,26 +921,29 @@ class WithItemsEngineTest(base.EngineTestCase):
concurrency: 3 concurrency: 3
""" """
wf_service.create_workflows(workflow_definition) wf_service.create_workflows(wf_definition)
# Start workflow. # Start workflow.
wf_ex = self.engine.start_workflow('concurrency_test', {}) wf_ex = self.engine.start_workflow('concurrency_test', {})
self._await( self._await(lambda: self.is_execution_success(wf_ex.id))
lambda: self.is_execution_success(wf_ex.id),
)
wf_ex = db_api.get_execution(wf_ex.id) wf_ex = db_api.get_execution(wf_ex.id)
task_ex = self._assert_single_item(wf_ex.task_executions, name='task1')
task_ex = self._assert_single_item(
wf_ex.task_executions,
name='task1',
state=states.SUCCESS
)
result = data_flow.get_task_execution_result(task_ex) result = data_flow.get_task_execution_result(task_ex)
self.assertEqual(states.SUCCESS, task_ex.state)
self.assertIsInstance(result, list) self.assertIsInstance(result, list)
self.assertIn('John', result) self.assertIn('John', result)
self.assertIn('Ivan', result) self.assertIn('Ivan', result)
def test_with_items_retry_policy(self): def test_with_items_retry_policy(self):
workflow = """--- wf_text = """---
version: "2.0" version: "2.0"
with_items_retry: with_items_retry:
@ -948,32 +959,32 @@ class WithItemsEngineTest(base.EngineTestCase):
task2: task2:
action: std.echo output="With-items failed" action: std.echo output="With-items failed"
""" """
wf_service.create_workflows(workflow)
wf_service.create_workflows(wf_text)
# Start workflow. # Start workflow.
wf_ex = self.engine.start_workflow('with_items_retry', {}) wf_ex = self.engine.start_workflow('with_items_retry', {})
self._await( self._await(lambda: self.is_execution_success(wf_ex.id))
lambda: self.is_execution_success(wf_ex.id)
)
# Note: We need to reread execution to access related tasks. # Note: We need to reread execution to access related tasks.
wf_ex = db_api.get_workflow_execution(wf_ex.id) wf_ex = db_api.get_workflow_execution(wf_ex.id)
tasks = wf_ex.task_executions task_execs = wf_ex.task_executions
self.assertEqual(2, len(tasks))
task1 = self._assert_single_item(tasks, name='task1') self.assertEqual(2, len(task_execs))
task1_ex = self._assert_single_item(task_execs, name='task1')
self.assertEqual( self.assertEqual(
2, 2,
task1.runtime_context['retry_task_policy']['retry_no'] task1_ex.runtime_context['retry_task_policy']['retry_no']
) )
self.assertEqual(9, len(task1.executions)) self.assertEqual(9, len(task1_ex.executions))
self._assert_multiple_items(task1.executions, 3, accepted=True) self._assert_multiple_items(task1_ex.executions, 3, accepted=True)
def test_with_items_retry_policy_concurrency(self): def test_with_items_retry_policy_concurrency(self):
workflow = """--- wf_text = """---
version: "2.0" version: "2.0"
with_items_retry_concurrency: with_items_retry_concurrency:
@ -990,31 +1001,28 @@ class WithItemsEngineTest(base.EngineTestCase):
task2: task2:
action: std.echo output="With-items failed" action: std.echo output="With-items failed"
""" """
wf_service.create_workflows(workflow)
wf_service.create_workflows(wf_text)
# Start workflow. # Start workflow.
wf_ex = self.engine.start_workflow( wf_ex = self.engine.start_workflow('with_items_retry_concurrency', {})
'with_items_retry_concurrency',
{}
)
self._await( self._await(lambda: self.is_execution_success(wf_ex.id))
lambda: self.is_execution_success(wf_ex.id),
)
# Note: We need to reread execution to access related tasks. # Note: We need to reread execution to access related tasks.
wf_ex = db_api.get_workflow_execution(wf_ex.id) wf_ex = db_api.get_workflow_execution(wf_ex.id)
tasks = wf_ex.task_executions task_execs = wf_ex.task_executions
self.assertEqual(2, len(tasks))
task1 = self._assert_single_item(tasks, name='task1') self.assertEqual(2, len(task_execs))
self.assertEqual(12, len(task1.executions)) task1_ex = self._assert_single_item(task_execs, name='task1')
self._assert_multiple_items(task1.executions, 4, accepted=True)
self.assertEqual(12, len(task1_ex.executions))
self._assert_multiple_items(task1_ex.executions, 4, accepted=True)
def test_with_items_env(self): def test_with_items_env(self):
workflow = """--- wf_text = """---
version: "2.0" version: "2.0"
with_items_env: with_items_env:
@ -1023,19 +1031,17 @@ class WithItemsEngineTest(base.EngineTestCase):
with-items: i in [1, 2, 3, 4] with-items: i in [1, 2, 3, 4]
action: std.echo output="<% $.i %>.<% env().name %>" action: std.echo output="<% $.i %>.<% env().name %>"
""" """
wf_service.create_workflows(workflow)
env = {'name': 'Mistral'} wf_service.create_workflows(wf_text)
# Start workflow. # Start workflow.
wf_ex = self.engine.start_workflow( wf_ex = self.engine.start_workflow(
'with_items_env', 'with_items_env',
{}, {},
env=env env={'name': 'Mistral'}
) )
self._await( self._await(lambda: self.is_execution_success(wf_ex.id))
lambda: self.is_execution_success(wf_ex.id),
)
# Note: We need to reread execution to access related tasks. # Note: We need to reread execution to access related tasks.
wf_ex = db_api.get_workflow_execution(wf_ex.id) wf_ex = db_api.get_workflow_execution(wf_ex.id)
@ -1058,7 +1064,7 @@ class WithItemsEngineTest(base.EngineTestCase):
self.assertEqual(states.SUCCESS, task1.state) self.assertEqual(states.SUCCESS, task1.state)
def test_with_items_two_tasks_second_starts_on_success(self): def test_with_items_two_tasks_second_starts_on_success(self):
workbook = """--- wb_text = """---
version: "2.0" version: "2.0"
name: wb1 name: wb1
@ -1076,7 +1082,8 @@ class WithItemsEngineTest(base.EngineTestCase):
with-items: i in [3, 4] with-items: i in [3, 4]
action: std.echo output=<% $.i %> action: std.echo output=<% $.i %>
""" """
wb_service.create_workbook_v2(workbook)
wb_service.create_workbook_v2(wb_text)
# Start workflow. # Start workflow.
wf_ex = self.engine.start_workflow('wb1.with_items', {}) wf_ex = self.engine.start_workflow('wb1.with_items', {})
@ -1086,14 +1093,21 @@ class WithItemsEngineTest(base.EngineTestCase):
# Note: We need to reread execution to access related tasks. # Note: We need to reread execution to access related tasks.
wf_ex = db_api.get_workflow_execution(wf_ex.id) wf_ex = db_api.get_workflow_execution(wf_ex.id)
tasks = wf_ex.task_executions task_execs = wf_ex.task_executions
task1 = self._assert_single_item(tasks, name='task1')
task2 = self._assert_single_item(tasks, name='task2')
self.assertEqual(states.SUCCESS, task1.state)
self.assertEqual(states.SUCCESS, task2.state)
result_task1 = data_flow.get_task_execution_result(task1) task1_ex = self._assert_single_item(
result_task2 = data_flow.get_task_execution_result(task2) task_execs,
name='task1',
state=states.SUCCESS
)
task2_ex = self._assert_single_item(
task_execs,
name='task2',
state=states.SUCCESS
)
result_task1 = data_flow.get_task_execution_result(task1_ex)
result_task2 = data_flow.get_task_execution_result(task2_ex)
# Since we know that we can receive results in random order, # Since we know that we can receive results in random order,
# check is not depend on order of items. # check is not depend on order of items.
@ -1103,45 +1117,51 @@ class WithItemsEngineTest(base.EngineTestCase):
self.assertIn(4, result_task2) self.assertIn(4, result_task2)
def test_with_items_subflow_concurrency_gt_list_length(self): def test_with_items_subflow_concurrency_gt_list_length(self):
workbook_definition = """--- wb_text = """---
version: "2.0" version: "2.0"
name: wb1 name: wb1
workflows: workflows:
main: main:
type: direct type: direct
input: input:
- names - names
tasks: tasks:
task1: task1:
with-items: name in <% $.names %> with-items: name in <% $.names %>
workflow: subflow1 name=<% $.name %> workflow: subflow1 name=<% $.name %>
concurrency: 3 concurrency: 3
subflow1: subflow1:
type: direct type: direct
input: input:
- name - name
output: output:
result: <% task(task1).result %> result: <% task(task1).result %>
tasks: tasks:
task1: task1:
action: std.echo output=<% $.name %> action: std.echo output=<% $.name %>
""" """
wb_service.create_workbook_v2(workbook_definition) wb_service.create_workbook_v2(wb_text)
# Start workflow. # Start workflow.
names = ["Peter", "Susan", "Edmund", "Lucy", "Aslan", "Caspian"] names = ["Peter", "Susan", "Edmund", "Lucy", "Aslan", "Caspian"]
wf_ex = self.engine.start_workflow('wb1.main', {'names': names}) wf_ex = self.engine.start_workflow('wb1.main', {'names': names})
self._await( self._await(lambda: self.is_execution_success(wf_ex.id))
lambda: self.is_execution_success(wf_ex.id),
)
wf_ex = db_api.get_execution(wf_ex.id) wf_ex = db_api.get_execution(wf_ex.id)
task_ex = self._assert_single_item(wf_ex.task_executions, name='task1')
self.assertEqual(states.SUCCESS, task_ex.state) task_ex = self._assert_single_item(
wf_ex.task_executions,
name='task1',
state=states.SUCCESS
)
result = [ result = [
item['result'] item['result']

View File

@ -63,15 +63,15 @@ class RunExistingTask(WorkflowCommand):
"""Command for running already existent task.""" """Command for running already existent task."""
def __init__(self, task_ex, reset=True): def __init__(self, task_ex, reset=True):
wf_ex = task_ex.workflow_execution super(RunExistingTask, self).__init__(
task_spec = spec_parser.get_task_spec(task_ex.spec) task_ex.workflow_execution,
spec_parser.get_task_spec(task_ex.spec),
task_ex.in_context
)
self.task_ex = task_ex self.task_ex = task_ex
self.reset = reset self.reset = reset
super(RunExistingTask, self).__init__(
wf_ex, task_spec, task_ex.in_context
)
class SetWorkflowState(WorkflowCommand): class SetWorkflowState(WorkflowCommand):
"""Instruction to change a workflow state.""" """Instruction to change a workflow state."""