Add option to run actions locally on the engine
Make executor pluggable and allow option to run the executor locally on the engine or remotely over RPC. Change-Id: I7cfb13068aa1d1f88136eaa092e629c34b78adf2 Implements: blueprint mistral-actions-run-by-engine
This commit is contained in:
parent
a753759cad
commit
62d8c5edaf
@ -44,9 +44,9 @@ from oslo_service import service
|
|||||||
from mistral.api import service as api_service
|
from mistral.api import service as api_service
|
||||||
from mistral import config
|
from mistral import config
|
||||||
from mistral.engine import engine_server
|
from mistral.engine import engine_server
|
||||||
from mistral.engine import executor_server
|
|
||||||
from mistral.engine.rpc_backend import rpc
|
from mistral.engine.rpc_backend import rpc
|
||||||
from mistral.event_engine import event_engine_server
|
from mistral.event_engine import event_engine_server
|
||||||
|
from mistral.executors import executor_server
|
||||||
from mistral import version
|
from mistral import version
|
||||||
|
|
||||||
|
|
||||||
|
@ -169,6 +169,16 @@ engine_opts = [
|
|||||||
]
|
]
|
||||||
|
|
||||||
executor_opts = [
|
executor_opts = [
|
||||||
|
cfg.StrOpt(
|
||||||
|
'type',
|
||||||
|
choices=['local', 'remote'],
|
||||||
|
default='remote',
|
||||||
|
help=(
|
||||||
|
'Type of executor. Use local to run the executor within the '
|
||||||
|
'engine server. Use remote if the executor is launched as '
|
||||||
|
'a separate server to run action executions.'
|
||||||
|
)
|
||||||
|
),
|
||||||
cfg.StrOpt(
|
cfg.StrOpt(
|
||||||
'host',
|
'host',
|
||||||
default='0.0.0.0',
|
default='0.0.0.0',
|
||||||
|
@ -15,7 +15,9 @@
|
|||||||
|
|
||||||
import functools
|
import functools
|
||||||
|
|
||||||
from mistral.engine.rpc_backend import rpc
|
from oslo_config import cfg
|
||||||
|
|
||||||
|
from mistral.executors import base as exe
|
||||||
from mistral import utils
|
from mistral import utils
|
||||||
|
|
||||||
|
|
||||||
@ -44,14 +46,16 @@ def _get_queue():
|
|||||||
|
|
||||||
|
|
||||||
def _run_actions():
|
def _run_actions():
|
||||||
|
executor = exe.get_executor(cfg.CONF.executor.type)
|
||||||
|
|
||||||
for action_ex, action_def, target in _get_queue():
|
for action_ex, action_def, target in _get_queue():
|
||||||
rpc.get_executor_client().run_action(
|
executor.run_action(
|
||||||
action_ex.id,
|
action_ex.id,
|
||||||
action_def.action_class,
|
action_def.action_class,
|
||||||
action_def.attributes or {},
|
action_def.attributes or {},
|
||||||
action_ex.input,
|
action_ex.input,
|
||||||
target,
|
action_ex.runtime_context.get('safe_rerun', False),
|
||||||
safe_rerun=action_ex.runtime_context.get('safe_rerun', False)
|
target=target
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
@ -15,15 +15,16 @@
|
|||||||
|
|
||||||
import abc
|
import abc
|
||||||
from oslo_config import cfg
|
from oslo_config import cfg
|
||||||
|
from oslo_log import log as logging
|
||||||
from osprofiler import profiler
|
from osprofiler import profiler
|
||||||
import six
|
import six
|
||||||
|
|
||||||
from mistral.db.v2 import api as db_api
|
from mistral.db.v2 import api as db_api
|
||||||
from mistral.engine import action_queue
|
from mistral.engine import action_queue
|
||||||
from mistral.engine.rpc_backend import rpc
|
|
||||||
from mistral.engine import utils as engine_utils
|
from mistral.engine import utils as engine_utils
|
||||||
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 import exceptions as exc
|
||||||
|
from mistral.executors import base as exe
|
||||||
from mistral import expressions as expr
|
from mistral import expressions as expr
|
||||||
from mistral.lang import parser as spec_parser
|
from mistral.lang import parser as spec_parser
|
||||||
from mistral.services import action_manager as a_m
|
from mistral.services import action_manager as a_m
|
||||||
@ -34,6 +35,9 @@ from mistral.workflow import states
|
|||||||
from mistral.workflow import utils as wf_utils
|
from mistral.workflow import utils as wf_utils
|
||||||
|
|
||||||
|
|
||||||
|
LOG = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
@six.add_metaclass(abc.ABCMeta)
|
@six.add_metaclass(abc.ABCMeta)
|
||||||
class Action(object):
|
class Action(object):
|
||||||
"""Action.
|
"""Action.
|
||||||
@ -251,14 +255,16 @@ class PythonAction(Action):
|
|||||||
action_ex_id=action_ex_id
|
action_ex_id=action_ex_id
|
||||||
)
|
)
|
||||||
|
|
||||||
result = rpc.get_executor_client().run_action(
|
executor = exe.get_executor(cfg.CONF.executor.type)
|
||||||
|
|
||||||
|
result = executor.run_action(
|
||||||
self.action_ex.id if self.action_ex else None,
|
self.action_ex.id if self.action_ex else None,
|
||||||
self.action_def.action_class,
|
self.action_def.action_class,
|
||||||
self.action_def.attributes or {},
|
self.action_def.attributes or {},
|
||||||
input_dict,
|
input_dict,
|
||||||
target,
|
safe_rerun=safe_rerun,
|
||||||
async_=False,
|
target=target,
|
||||||
safe_rerun=safe_rerun
|
async_=False
|
||||||
)
|
)
|
||||||
|
|
||||||
return self._prepare_output(result)
|
return self._prepare_output(result)
|
||||||
@ -528,6 +534,7 @@ def resolve_action_definition(action_spec_name, wf_name=None,
|
|||||||
:param wf_spec_name: Workflow name according to a spec.
|
:param wf_spec_name: Workflow name according to a spec.
|
||||||
:return: Action definition (python or ad-hoc).
|
:return: Action definition (python or ad-hoc).
|
||||||
"""
|
"""
|
||||||
|
|
||||||
action_db = None
|
action_db = None
|
||||||
|
|
||||||
if wf_name and wf_name != wf_spec_name:
|
if wf_name and wf_name != wf_spec_name:
|
||||||
|
@ -128,26 +128,6 @@ class Engine(object):
|
|||||||
raise NotImplementedError
|
raise NotImplementedError
|
||||||
|
|
||||||
|
|
||||||
@six.add_metaclass(abc.ABCMeta)
|
|
||||||
class Executor(object):
|
|
||||||
"""Action executor interface."""
|
|
||||||
|
|
||||||
@abc.abstractmethod
|
|
||||||
def run_action(self, action_ex_id, action_class_str, attributes,
|
|
||||||
action_params, safe_rerun, redelivered=False):
|
|
||||||
"""Runs action.
|
|
||||||
|
|
||||||
:param action_ex_id: Corresponding action execution id.
|
|
||||||
:param action_class_str: Path to action class in dot notation.
|
|
||||||
:param attributes: Attributes of action class which will be set to.
|
|
||||||
:param action_params: Action parameters.
|
|
||||||
:param safe_rerun: Tells if given action can be safely rerun.
|
|
||||||
:param redelivered: Tells if given action was run before on another
|
|
||||||
executor.
|
|
||||||
"""
|
|
||||||
raise NotImplementedError()
|
|
||||||
|
|
||||||
|
|
||||||
@six.add_metaclass(abc.ABCMeta)
|
@six.add_metaclass(abc.ABCMeta)
|
||||||
class EventEngine(object):
|
class EventEngine(object):
|
||||||
"""Action event trigger interface."""
|
"""Action event trigger interface."""
|
||||||
|
@ -1,5 +1,6 @@
|
|||||||
# Copyright 2014 - Mirantis, Inc.
|
# Copyright 2014 - Mirantis, Inc.
|
||||||
# Copyright 2015 - StackStorm, Inc.
|
# Copyright 2015 - StackStorm, Inc.
|
||||||
|
# Copyright 2017 - Brocade Communications Systems, Inc.
|
||||||
#
|
#
|
||||||
# Licensed under the Apache License, Version 2.0 (the "License");
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
# you may not use this file except in compliance with the License.
|
# you may not use this file except in compliance with the License.
|
||||||
@ -21,8 +22,9 @@ from osprofiler import profiler
|
|||||||
from stevedore import driver
|
from stevedore import driver
|
||||||
|
|
||||||
from mistral import context as auth_ctx
|
from mistral import context as auth_ctx
|
||||||
from mistral.engine import base
|
from mistral.engine import base as eng
|
||||||
from mistral import exceptions as exc
|
from mistral import exceptions as exc
|
||||||
|
from mistral.executors import base as exe
|
||||||
|
|
||||||
|
|
||||||
LOG = logging.getLogger(__name__)
|
LOG = logging.getLogger(__name__)
|
||||||
@ -149,7 +151,7 @@ def wrap_messaging_exception(method):
|
|||||||
return decorator
|
return decorator
|
||||||
|
|
||||||
|
|
||||||
class EngineClient(base.Engine):
|
class EngineClient(eng.Engine):
|
||||||
"""RPC Engine client."""
|
"""RPC Engine client."""
|
||||||
|
|
||||||
def __init__(self, rpc_conf_dict):
|
def __init__(self, rpc_conf_dict):
|
||||||
@ -317,50 +319,50 @@ class EngineClient(base.Engine):
|
|||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
class ExecutorClient(base.Executor):
|
class ExecutorClient(exe.Executor):
|
||||||
"""RPC Executor client."""
|
"""RPC Executor client."""
|
||||||
|
|
||||||
def __init__(self, rpc_conf_dict):
|
def __init__(self, rpc_conf_dict):
|
||||||
"""Constructs an RPC client for the Executor.
|
"""Constructs an RPC client for the Executor."""
|
||||||
|
|
||||||
:param rpc_conf_dict: Dict containing RPC configuration.
|
|
||||||
"""
|
|
||||||
|
|
||||||
self.topic = cfg.CONF.executor.topic
|
self.topic = cfg.CONF.executor.topic
|
||||||
self._client = get_rpc_client_driver()(rpc_conf_dict)
|
self._client = get_rpc_client_driver()(rpc_conf_dict)
|
||||||
|
|
||||||
@profiler.trace('executor-client-run-action')
|
@profiler.trace('executor-client-run-action')
|
||||||
def run_action(self, action_ex_id, action_class_str, attributes,
|
def run_action(self, action_ex_id, action_cls_str, action_cls_attrs,
|
||||||
action_params, target=None, async_=True, safe_rerun=False):
|
params, safe_rerun, redelivered=False,
|
||||||
|
target=None, async_=True):
|
||||||
"""Sends a request to run action to executor.
|
"""Sends a request to run action to executor.
|
||||||
|
|
||||||
:param action_ex_id: Action execution id.
|
:param action_ex_id: Action execution id.
|
||||||
:param action_class_str: Action class name.
|
:param action_cls_str: Action class name.
|
||||||
:param attributes: Action class attributes.
|
:param action_cls_attrs: Action class attributes.
|
||||||
:param action_params: Action input parameters.
|
:param params: Action input parameters.
|
||||||
:param target: Target (group of action executors).
|
|
||||||
:param async: If True, run action in asynchronous mode (w/o waiting
|
|
||||||
for completion).
|
|
||||||
:param safe_rerun: If true, action would be re-run if executor dies
|
:param safe_rerun: If true, action would be re-run if executor dies
|
||||||
during execution.
|
during execution.
|
||||||
|
:param redelivered: Tells if given action was run before on another
|
||||||
|
executor.
|
||||||
|
:param target: Target (group of action executors).
|
||||||
|
:param async_: If True, run action in asynchronous mode (w/o waiting
|
||||||
|
for completion).
|
||||||
:return: Action result.
|
:return: Action result.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
kwargs = {
|
rpc_kwargs = {
|
||||||
'action_ex_id': action_ex_id,
|
'action_ex_id': action_ex_id,
|
||||||
'action_class_str': action_class_str,
|
'action_cls_str': action_cls_str,
|
||||||
'attributes': attributes,
|
'action_cls_attrs': action_cls_attrs,
|
||||||
'params': action_params,
|
'params': params,
|
||||||
'safe_rerun': safe_rerun
|
'safe_rerun': safe_rerun
|
||||||
}
|
}
|
||||||
|
|
||||||
rpc_client_method = (self._client.async_call
|
rpc_client_method = (self._client.async_call
|
||||||
if async_ else self._client.sync_call)
|
if async_ else self._client.sync_call)
|
||||||
|
|
||||||
return rpc_client_method(auth_ctx.ctx(), 'run_action', **kwargs)
|
return rpc_client_method(auth_ctx.ctx(), 'run_action', **rpc_kwargs)
|
||||||
|
|
||||||
|
|
||||||
class EventEngineClient(base.EventEngine):
|
class EventEngineClient(eng.EventEngine):
|
||||||
"""RPC EventEngine client."""
|
"""RPC EventEngine client."""
|
||||||
|
|
||||||
def __init__(self, rpc_conf_dict):
|
def __init__(self, rpc_conf_dict):
|
||||||
|
0
mistral/executors/__init__.py
Normal file
0
mistral/executors/__init__.py
Normal file
67
mistral/executors/base.py
Normal file
67
mistral/executors/base.py
Normal file
@ -0,0 +1,67 @@
|
|||||||
|
# Copyright 2017 - Brocade Communications Systems, Inc.
|
||||||
|
#
|
||||||
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
# you may not use this file except in compliance with the License.
|
||||||
|
# You may obtain a copy of the License at
|
||||||
|
#
|
||||||
|
# http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
#
|
||||||
|
# Unless required by applicable law or agreed to in writing, software
|
||||||
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
# See the License for the specific language governing permissions and
|
||||||
|
# limitations under the License.
|
||||||
|
|
||||||
|
import abc
|
||||||
|
import six
|
||||||
|
|
||||||
|
from stevedore import driver
|
||||||
|
|
||||||
|
|
||||||
|
_EXECUTORS = {}
|
||||||
|
|
||||||
|
|
||||||
|
def cleanup():
|
||||||
|
global _EXECUTORS
|
||||||
|
_EXECUTORS = {}
|
||||||
|
|
||||||
|
|
||||||
|
def get_executor(exec_type):
|
||||||
|
global _EXECUTORS
|
||||||
|
|
||||||
|
if not _EXECUTORS.get(exec_type):
|
||||||
|
mgr = driver.DriverManager(
|
||||||
|
'mistral.executors',
|
||||||
|
exec_type,
|
||||||
|
invoke_on_load=True
|
||||||
|
)
|
||||||
|
|
||||||
|
_EXECUTORS[exec_type] = mgr.driver
|
||||||
|
|
||||||
|
return _EXECUTORS[exec_type]
|
||||||
|
|
||||||
|
|
||||||
|
@six.add_metaclass(abc.ABCMeta)
|
||||||
|
class Executor(object):
|
||||||
|
"""Action executor interface."""
|
||||||
|
|
||||||
|
@abc.abstractmethod
|
||||||
|
def run_action(self, action_ex_id, action_cls_str, action_cls_attrs,
|
||||||
|
params, safe_rerun, redelivered=False,
|
||||||
|
target=None, async_=True):
|
||||||
|
"""Runs action.
|
||||||
|
|
||||||
|
:param action_ex_id: Corresponding action execution id.
|
||||||
|
:param action_cls_str: Path to action class in dot notation.
|
||||||
|
:param action_cls_attrs: Attributes of action class which
|
||||||
|
will be set to.
|
||||||
|
:param params: Action parameters.
|
||||||
|
:param safe_rerun: Tells if given action can be safely rerun.
|
||||||
|
:param redelivered: Tells if given action was run before on another
|
||||||
|
executor.
|
||||||
|
:param target: Target (group of action executors).
|
||||||
|
:param async_: If True, run action in asynchronous mode (w/o waiting
|
||||||
|
for completion).
|
||||||
|
:return: Action result.
|
||||||
|
"""
|
||||||
|
raise NotImplementedError()
|
@ -17,9 +17,9 @@ from oslo_log import log as logging
|
|||||||
from osprofiler import profiler
|
from osprofiler import profiler
|
||||||
|
|
||||||
from mistral.actions import action_factory as a_f
|
from mistral.actions import action_factory as a_f
|
||||||
from mistral.engine import base
|
|
||||||
from mistral.engine.rpc_backend import rpc
|
from mistral.engine.rpc_backend import rpc
|
||||||
from mistral import exceptions as exc
|
from mistral import exceptions as exc
|
||||||
|
from mistral.executors import base
|
||||||
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
|
||||||
|
|
||||||
@ -31,18 +31,24 @@ class DefaultExecutor(base.Executor):
|
|||||||
def __init__(self):
|
def __init__(self):
|
||||||
self._engine_client = rpc.get_engine_client()
|
self._engine_client = rpc.get_engine_client()
|
||||||
|
|
||||||
@profiler.trace('executor-run-action', hide_args=True)
|
@profiler.trace('default-executor-run-action', hide_args=True)
|
||||||
def run_action(self, action_ex_id, action_class_str, attributes,
|
def run_action(self, action_ex_id, action_cls_str, action_cls_attrs,
|
||||||
action_params, safe_rerun, redelivered=False):
|
params, safe_rerun, redelivered=False,
|
||||||
|
target=None, async_=True):
|
||||||
"""Runs action.
|
"""Runs action.
|
||||||
|
|
||||||
:param action_ex_id: Action execution id.
|
:param action_ex_id: Action execution id.
|
||||||
:param action_class_str: Path to action class in dot notation.
|
:param action_cls_str: Path to action class in dot notation.
|
||||||
:param attributes: Attributes of action class which will be set to.
|
:param action_cls_attrs: Attributes of action class which
|
||||||
:param action_params: Action parameters.
|
will be set to.
|
||||||
|
:param params: Action parameters.
|
||||||
:param safe_rerun: Tells if given action can be safely rerun.
|
:param safe_rerun: Tells if given action can be safely rerun.
|
||||||
:param redelivered: Tells if given action was run before on another
|
:param redelivered: Tells if given action was run before on another
|
||||||
executor.
|
executor.
|
||||||
|
:param target: Target (group of action executors).
|
||||||
|
:param async_: If True, run action in asynchronous mode (w/o waiting
|
||||||
|
for completion).
|
||||||
|
:return: Action result.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def send_error_back(error_msg):
|
def send_error_back(error_msg):
|
||||||
@ -60,31 +66,38 @@ class DefaultExecutor(base.Executor):
|
|||||||
|
|
||||||
if redelivered and not safe_rerun:
|
if redelivered and not safe_rerun:
|
||||||
msg = (
|
msg = (
|
||||||
"Request to run action %s was redelivered, but action %s"
|
"Request to run action %s was redelivered, but action %s "
|
||||||
" cannot be re-run safely. The only safe thing to do is fail"
|
"cannot be re-run safely. The only safe thing to do is fail "
|
||||||
" action."
|
"action." % (action_cls_str, action_cls_str)
|
||||||
% (action_class_str, action_class_str)
|
|
||||||
)
|
)
|
||||||
|
|
||||||
return send_error_back(msg)
|
return send_error_back(msg)
|
||||||
|
|
||||||
action_cls = a_f.construct_action_class(action_class_str, attributes)
|
# Load action module.
|
||||||
|
action_cls = a_f.construct_action_class(
|
||||||
|
action_cls_str,
|
||||||
|
action_cls_attrs
|
||||||
|
)
|
||||||
|
|
||||||
# Instantiate action.
|
# Instantiate action.
|
||||||
|
|
||||||
try:
|
try:
|
||||||
action = action_cls(**action_params)
|
action = action_cls(**params)
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
msg = ("Failed to initialize action %s. Action init params = %s."
|
msg = (
|
||||||
" Actual init params = %s. More info: %s"
|
"Failed to initialize action %s. Action init params = %s. "
|
||||||
% (action_class_str, i_u.get_arg_list(action_cls.__init__),
|
"Actual init params = %s. More info: %s" % (
|
||||||
action_params.keys(), e))
|
action_cls_str,
|
||||||
LOG.exception(msg)
|
i_u.get_arg_list(action_cls.__init__),
|
||||||
|
params.keys(),
|
||||||
|
e
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
LOG.warning(msg)
|
||||||
|
|
||||||
return send_error_back(msg)
|
return send_error_back(msg)
|
||||||
|
|
||||||
# Run action.
|
# Run action.
|
||||||
|
|
||||||
try:
|
try:
|
||||||
result = action.run()
|
result = action.run()
|
||||||
|
|
||||||
@ -95,15 +108,22 @@ class DefaultExecutor(base.Executor):
|
|||||||
result = wf_utils.Result(data=result)
|
result = wf_utils.Result(data=result)
|
||||||
|
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
msg = ("Failed to run action [action_ex_id=%s, action_cls='%s',"
|
msg = (
|
||||||
" attributes='%s', params='%s']\n %s"
|
"Failed to run action [action_ex_id=%s, action_cls='%s', "
|
||||||
% (action_ex_id, action_cls, attributes, action_params, e))
|
"attributes='%s', params='%s']\n %s" % (
|
||||||
|
action_ex_id,
|
||||||
|
action_cls,
|
||||||
|
action_cls_attrs,
|
||||||
|
params,
|
||||||
|
e
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
LOG.exception(msg)
|
LOG.exception(msg)
|
||||||
|
|
||||||
return send_error_back(msg)
|
return send_error_back(msg)
|
||||||
|
|
||||||
# Send action result.
|
# Send action result.
|
||||||
|
|
||||||
try:
|
try:
|
||||||
if action_ex_id and (action.is_sync() or result.is_error()):
|
if action_ex_id and (action.is_sync() or result.is_error()):
|
||||||
self._engine_client.on_action_complete(
|
self._engine_client.on_action_complete(
|
||||||
@ -118,22 +138,36 @@ class DefaultExecutor(base.Executor):
|
|||||||
# such as message bus or network. One known case is when the action
|
# such as message bus or network. One known case is when the action
|
||||||
# returns a bad result (e.g. invalid unicode) which can't be
|
# returns a bad result (e.g. invalid unicode) which can't be
|
||||||
# serialized.
|
# serialized.
|
||||||
msg = ("Failed to call engine's on_action_complete() method due"
|
msg = (
|
||||||
" to a Mistral exception"
|
"Failed to complete action due to a Mistral exception "
|
||||||
" [action_ex_id=%s, action_cls='%s',"
|
"[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,
|
||||||
|
action_cls_attrs,
|
||||||
|
params,
|
||||||
|
e
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
LOG.exception(msg)
|
LOG.exception(msg)
|
||||||
|
|
||||||
return send_error_back(msg)
|
return send_error_back(msg)
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
# If it's not a Mistral exception all we can do is only
|
# If it's not a Mistral exception all we can do is only
|
||||||
# log the error.
|
# log the error.
|
||||||
msg = ("Failed to call engine's on_action_complete() method due"
|
msg = (
|
||||||
" to an unexpected exception"
|
"Failed to complete action due to an unexpected exception "
|
||||||
" [action_ex_id=%s, action_cls='%s',"
|
"[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,
|
||||||
|
action_cls_attrs,
|
||||||
|
params,
|
||||||
|
e
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
LOG.exception(msg)
|
LOG.exception(msg)
|
||||||
|
|
||||||
return result
|
return result
|
@ -15,8 +15,8 @@
|
|||||||
from oslo_log import log as logging
|
from oslo_log import log as logging
|
||||||
|
|
||||||
from mistral import config as cfg
|
from mistral import config as cfg
|
||||||
from mistral.engine import default_executor
|
|
||||||
from mistral.engine.rpc_backend import rpc
|
from mistral.engine.rpc_backend import rpc
|
||||||
|
from mistral.executors import default_executor as exe
|
||||||
from mistral.service import base as service_base
|
from mistral.service import base as service_base
|
||||||
from mistral import utils
|
from mistral import utils
|
||||||
from mistral.utils import profiler as profiler_utils
|
from mistral.utils import profiler as profiler_utils
|
||||||
@ -59,14 +59,14 @@ class ExecutorServer(service_base.MistralService):
|
|||||||
if self._rpc_server:
|
if self._rpc_server:
|
||||||
self._rpc_server.stop(graceful)
|
self._rpc_server.stop(graceful)
|
||||||
|
|
||||||
def run_action(self, rpc_ctx, action_ex_id, action_class_str,
|
def run_action(self, rpc_ctx, action_ex_id, action_cls_str,
|
||||||
attributes, params, safe_rerun):
|
action_cls_attrs, params, safe_rerun):
|
||||||
"""Receives calls over RPC to run action on executor.
|
"""Receives calls over RPC to run action on executor.
|
||||||
|
|
||||||
:param rpc_ctx: RPC request context dictionary.
|
:param rpc_ctx: RPC request context dictionary.
|
||||||
:param action_ex_id: Action execution id.
|
:param action_ex_id: Action execution id.
|
||||||
:param action_class_str: Action class name.
|
:param action_cls_str: Action class name.
|
||||||
:param attributes: Action class attributes.
|
:param action_cls_attrs: Action class attributes.
|
||||||
:param params: Action input parameters.
|
:param params: Action input parameters.
|
||||||
:param safe_rerun: Tells if given action can be safely rerun.
|
:param safe_rerun: Tells if given action can be safely rerun.
|
||||||
:return: Action result.
|
:return: Action result.
|
||||||
@ -74,17 +74,20 @@ class ExecutorServer(service_base.MistralService):
|
|||||||
|
|
||||||
LOG.info(
|
LOG.info(
|
||||||
"Received RPC request 'run_action'[action_ex_id=%s, "
|
"Received RPC request 'run_action'[action_ex_id=%s, "
|
||||||
"action_class=%s, attributes=%s, params=%s]"
|
"action_cls_str=%s, action_cls_attrs=%s, params=%s]" % (
|
||||||
% (action_ex_id, action_class_str, attributes,
|
action_ex_id,
|
||||||
utils.cut(params))
|
action_cls_str,
|
||||||
|
action_cls_attrs,
|
||||||
|
utils.cut(params)
|
||||||
|
)
|
||||||
)
|
)
|
||||||
|
|
||||||
redelivered = rpc_ctx.redelivered or False
|
redelivered = rpc_ctx.redelivered or False
|
||||||
|
|
||||||
return self.executor.run_action(
|
return self.executor.run_action(
|
||||||
action_ex_id,
|
action_ex_id,
|
||||||
action_class_str,
|
action_cls_str,
|
||||||
attributes,
|
action_cls_attrs,
|
||||||
params,
|
params,
|
||||||
safe_rerun,
|
safe_rerun,
|
||||||
redelivered
|
redelivered
|
||||||
@ -93,6 +96,6 @@ class ExecutorServer(service_base.MistralService):
|
|||||||
|
|
||||||
def get_oslo_service(setup_profiler=True):
|
def get_oslo_service(setup_profiler=True):
|
||||||
return ExecutorServer(
|
return ExecutorServer(
|
||||||
default_executor.DefaultExecutor(),
|
exe.DefaultExecutor(),
|
||||||
setup_profiler=setup_profiler
|
setup_profiler=setup_profiler
|
||||||
)
|
)
|
29
mistral/executors/remote_executor.py
Normal file
29
mistral/executors/remote_executor.py
Normal file
@ -0,0 +1,29 @@
|
|||||||
|
# Copyright 2017 - Brocade Communications Systems, Inc.
|
||||||
|
#
|
||||||
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
# you may not use this file except in compliance with the License.
|
||||||
|
# You may obtain a copy of the License at
|
||||||
|
#
|
||||||
|
# http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
#
|
||||||
|
# Unless required by applicable law or agreed to in writing, software
|
||||||
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
# See the License for the specific language governing permissions and
|
||||||
|
# limitations under the License.
|
||||||
|
|
||||||
|
from oslo_config import cfg
|
||||||
|
from oslo_log import log as logging
|
||||||
|
|
||||||
|
from mistral.engine.rpc_backend import rpc
|
||||||
|
|
||||||
|
|
||||||
|
LOG = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
|
class RemoteExecutor(rpc.ExecutorClient):
|
||||||
|
"""Executor that passes execution request to a remote executor."""
|
||||||
|
|
||||||
|
def __init__(self):
|
||||||
|
self.topic = cfg.CONF.executor.topic
|
||||||
|
self._client = rpc.get_rpc_client_driver()(cfg.CONF.executor)
|
@ -22,8 +22,9 @@ from oslo_service import service
|
|||||||
|
|
||||||
from mistral.db.v2 import api as db_api
|
from mistral.db.v2 import api as db_api
|
||||||
from mistral.engine import engine_server
|
from mistral.engine import engine_server
|
||||||
from mistral.engine import executor_server
|
|
||||||
from mistral.engine.rpc_backend import rpc
|
from mistral.engine.rpc_backend import rpc
|
||||||
|
from mistral.executors import base as exe
|
||||||
|
from mistral.executors import executor_server
|
||||||
from mistral.tests.unit import base
|
from mistral.tests.unit import base
|
||||||
from mistral.workflow import states
|
from mistral.workflow import states
|
||||||
|
|
||||||
@ -57,40 +58,42 @@ class EngineTestCase(base.DbTestCase):
|
|||||||
|
|
||||||
# Drop all RPC objects (transport, clients).
|
# Drop all RPC objects (transport, clients).
|
||||||
rpc.cleanup()
|
rpc.cleanup()
|
||||||
|
exe.cleanup()
|
||||||
|
|
||||||
|
self.threads = []
|
||||||
|
|
||||||
|
# Start remote executor.
|
||||||
|
if cfg.CONF.executor.type == 'remote':
|
||||||
|
LOG.info("Starting remote executor threads...")
|
||||||
|
self.executor_client = rpc.get_executor_client()
|
||||||
|
exe_svc = executor_server.get_oslo_service(setup_profiler=False)
|
||||||
|
self.executor = exe_svc.executor
|
||||||
|
self.threads.append(eventlet.spawn(launch_service, exe_svc))
|
||||||
|
self.addCleanup(exe_svc.stop, True)
|
||||||
|
|
||||||
|
# Start engine.
|
||||||
|
LOG.info("Starting engine threads...")
|
||||||
self.engine_client = rpc.get_engine_client()
|
self.engine_client = rpc.get_engine_client()
|
||||||
self.executor_client = rpc.get_executor_client()
|
eng_svc = engine_server.get_oslo_service(setup_profiler=False)
|
||||||
|
self.engine = eng_svc.engine
|
||||||
LOG.info("Starting engine and executor threads...")
|
self.threads.append(eventlet.spawn(launch_service, eng_svc))
|
||||||
|
self.addCleanup(eng_svc.stop, True)
|
||||||
engine_service = engine_server.get_oslo_service(setup_profiler=False)
|
|
||||||
executor_service = executor_server.get_oslo_service(
|
|
||||||
setup_profiler=False
|
|
||||||
)
|
|
||||||
|
|
||||||
self.engine = engine_service.engine
|
|
||||||
self.executor = executor_service.executor
|
|
||||||
|
|
||||||
self.threads = [
|
|
||||||
eventlet.spawn(launch_service, executor_service),
|
|
||||||
eventlet.spawn(launch_service, engine_service)
|
|
||||||
]
|
|
||||||
|
|
||||||
self.addOnException(self.print_executions)
|
self.addOnException(self.print_executions)
|
||||||
|
|
||||||
self.addCleanup(executor_service.stop, True)
|
|
||||||
self.addCleanup(engine_service.stop, True)
|
|
||||||
self.addCleanup(self.kill_threads)
|
self.addCleanup(self.kill_threads)
|
||||||
|
|
||||||
# Make sure that both services fully started, otherwise
|
# Make sure that both services fully started, otherwise
|
||||||
# the test may run too early.
|
# the test may run too early.
|
||||||
executor_service.wait_started()
|
if cfg.CONF.executor.type == 'remote':
|
||||||
engine_service.wait_started()
|
exe_svc.wait_started()
|
||||||
|
|
||||||
|
eng_svc.wait_started()
|
||||||
|
|
||||||
def kill_threads(self):
|
def kill_threads(self):
|
||||||
LOG.info("Finishing engine and executor threads...")
|
LOG.info("Finishing engine and executor threads...")
|
||||||
|
|
||||||
[thread.kill() for thread in self.threads]
|
for thread in self.threads:
|
||||||
|
thread.kill()
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def print_executions(exc_info=None):
|
def print_executions(exc_info=None):
|
||||||
|
@ -23,8 +23,8 @@ from oslo_utils import uuidutils
|
|||||||
from mistral.db.v2 import api as db_api
|
from mistral.db.v2 import api as db_api
|
||||||
from mistral.db.v2.sqlalchemy import models
|
from mistral.db.v2.sqlalchemy import models
|
||||||
from mistral.engine import default_engine as d_eng
|
from mistral.engine import default_engine as d_eng
|
||||||
from mistral.engine.rpc_backend import rpc
|
|
||||||
from mistral import exceptions as exc
|
from mistral import exceptions as exc
|
||||||
|
from mistral.executors import base as exe
|
||||||
from mistral.services import workbooks as wb_service
|
from mistral.services import workbooks as wb_service
|
||||||
from mistral.tests.unit import base
|
from mistral.tests.unit import base
|
||||||
from mistral.tests.unit.engine import base as eng_test_base
|
from mistral.tests.unit.engine import base as eng_test_base
|
||||||
@ -93,7 +93,7 @@ MOCK_ENVIRONMENT = mock.MagicMock(return_value=ENVIRONMENT_DB)
|
|||||||
MOCK_NOT_FOUND = mock.MagicMock(side_effect=exc.DBEntityNotFoundError())
|
MOCK_NOT_FOUND = mock.MagicMock(side_effect=exc.DBEntityNotFoundError())
|
||||||
|
|
||||||
|
|
||||||
@mock.patch.object(rpc, 'get_executor_client', mock.Mock())
|
@mock.patch.object(exe, 'get_executor', mock.Mock())
|
||||||
class DefaultEngineTest(base.DbTestCase):
|
class DefaultEngineTest(base.DbTestCase):
|
||||||
def setUp(self):
|
def setUp(self):
|
||||||
super(DefaultEngineTest, self).setUp()
|
super(DefaultEngineTest, self).setUp()
|
||||||
|
@ -16,8 +16,8 @@ import mock
|
|||||||
from oslo_config import cfg
|
from oslo_config import cfg
|
||||||
|
|
||||||
from mistral.db.v2 import api as db_api
|
from mistral.db.v2 import api as db_api
|
||||||
from mistral.engine import default_executor
|
from mistral.executors import default_executor as d_exe
|
||||||
from mistral.engine.rpc_backend import rpc
|
from mistral.executors import remote_executor as r_exe
|
||||||
from mistral.services import workbooks as wb_service
|
from mistral.services import workbooks as wb_service
|
||||||
from mistral.tests.unit.engine import base
|
from mistral.tests.unit.engine import base
|
||||||
|
|
||||||
@ -77,16 +77,17 @@ workflows:
|
|||||||
"""
|
"""
|
||||||
|
|
||||||
|
|
||||||
def _run_at_target(action_ex_id, action_class_str, attributes,
|
def _run_at_target(action_ex_id, action_cls_str, action_cls_attrs,
|
||||||
action_params, target=None, async_=True, safe_rerun=False):
|
params, safe_rerun, target=None, async_=True):
|
||||||
|
|
||||||
# We'll just call executor directly for testing purposes.
|
# We'll just call executor directly for testing purposes.
|
||||||
executor = default_executor.DefaultExecutor()
|
executor = d_exe.DefaultExecutor()
|
||||||
|
|
||||||
executor.run_action(
|
executor.run_action(
|
||||||
action_ex_id,
|
action_ex_id,
|
||||||
action_class_str,
|
action_cls_str,
|
||||||
attributes,
|
action_cls_attrs,
|
||||||
action_params,
|
params,
|
||||||
safe_rerun
|
safe_rerun
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -100,7 +101,7 @@ class EnvironmentTest(base.EngineTestCase):
|
|||||||
|
|
||||||
wb_service.create_workbook_v2(WORKBOOK)
|
wb_service.create_workbook_v2(WORKBOOK)
|
||||||
|
|
||||||
@mock.patch.object(rpc.ExecutorClient, 'run_action', MOCK_RUN_AT_TARGET)
|
@mock.patch.object(r_exe.RemoteExecutor, 'run_action', MOCK_RUN_AT_TARGET)
|
||||||
def _test_subworkflow(self, env):
|
def _test_subworkflow(self, env):
|
||||||
wf2_ex = self.engine.start_workflow('my_wb.wf2', {}, env=env)
|
wf2_ex = self.engine.start_workflow('my_wb.wf2', {}, env=env)
|
||||||
|
|
||||||
@ -169,13 +170,13 @@ class EnvironmentTest(base.EngineTestCase):
|
|||||||
for t_ex in wf1_task_execs:
|
for t_ex in wf1_task_execs:
|
||||||
a_ex = t_ex.action_executions[0]
|
a_ex = t_ex.action_executions[0]
|
||||||
|
|
||||||
rpc.ExecutorClient.run_action.assert_any_call(
|
r_exe.RemoteExecutor.run_action.assert_any_call(
|
||||||
a_ex.id,
|
a_ex.id,
|
||||||
'mistral.actions.std_actions.EchoAction',
|
'mistral.actions.std_actions.EchoAction',
|
||||||
{},
|
{},
|
||||||
a_ex.input,
|
a_ex.input,
|
||||||
TARGET,
|
False,
|
||||||
safe_rerun=False
|
target=TARGET
|
||||||
)
|
)
|
||||||
|
|
||||||
def test_subworkflow_env_task_input(self):
|
def test_subworkflow_env_task_input(self):
|
||||||
|
@ -16,8 +16,8 @@
|
|||||||
import mock
|
import mock
|
||||||
|
|
||||||
from mistral.db.v2 import api as db_api
|
from mistral.db.v2 import api as db_api
|
||||||
from mistral.engine import default_executor
|
from mistral.executors import default_executor as d_exe
|
||||||
from mistral.engine.rpc_backend import rpc
|
from mistral.executors import remote_executor as r_exe
|
||||||
from mistral.services import workflows as wf_service
|
from mistral.services import workflows as wf_service
|
||||||
from mistral.tests.unit.engine import base
|
from mistral.tests.unit.engine import base
|
||||||
from mistral.workflow import data_flow
|
from mistral.workflow import data_flow
|
||||||
@ -25,9 +25,9 @@ from mistral.workflow import states
|
|||||||
|
|
||||||
|
|
||||||
def _run_at_target(action_ex_id, action_class_str, attributes,
|
def _run_at_target(action_ex_id, action_class_str, attributes,
|
||||||
action_params, target=None, async_=True, safe_rerun=False):
|
action_params, safe_rerun, target=None, async_=True):
|
||||||
# We'll just call executor directly for testing purposes.
|
# We'll just call executor directly for testing purposes.
|
||||||
executor = default_executor.DefaultExecutor()
|
executor = d_exe.DefaultExecutor()
|
||||||
|
|
||||||
executor.run_action(
|
executor.run_action(
|
||||||
action_ex_id,
|
action_ex_id,
|
||||||
@ -43,7 +43,8 @@ MOCK_RUN_AT_TARGET = mock.MagicMock(side_effect=_run_at_target)
|
|||||||
|
|
||||||
|
|
||||||
class TestSafeRerun(base.EngineTestCase):
|
class TestSafeRerun(base.EngineTestCase):
|
||||||
@mock.patch.object(rpc.ExecutorClient, 'run_action', MOCK_RUN_AT_TARGET)
|
|
||||||
|
@mock.patch.object(r_exe.RemoteExecutor, 'run_action', MOCK_RUN_AT_TARGET)
|
||||||
def test_safe_rerun_true(self):
|
def test_safe_rerun_true(self):
|
||||||
wf_text = """---
|
wf_text = """---
|
||||||
version: '2.0'
|
version: '2.0'
|
||||||
@ -89,7 +90,7 @@ class TestSafeRerun(base.EngineTestCase):
|
|||||||
self.assertEqual(task1.state, states.SUCCESS)
|
self.assertEqual(task1.state, states.SUCCESS)
|
||||||
self.assertEqual(task2.state, states.SUCCESS)
|
self.assertEqual(task2.state, states.SUCCESS)
|
||||||
|
|
||||||
@mock.patch.object(rpc.ExecutorClient, 'run_action', MOCK_RUN_AT_TARGET)
|
@mock.patch.object(r_exe.RemoteExecutor, 'run_action', MOCK_RUN_AT_TARGET)
|
||||||
def test_safe_rerun_false(self):
|
def test_safe_rerun_false(self):
|
||||||
wf_text = """---
|
wf_text = """---
|
||||||
version: '2.0'
|
version: '2.0'
|
||||||
@ -135,7 +136,7 @@ class TestSafeRerun(base.EngineTestCase):
|
|||||||
self.assertEqual(task1.state, states.ERROR)
|
self.assertEqual(task1.state, states.ERROR)
|
||||||
self.assertEqual(task3.state, states.SUCCESS)
|
self.assertEqual(task3.state, states.SUCCESS)
|
||||||
|
|
||||||
@mock.patch.object(rpc.ExecutorClient, 'run_action', MOCK_RUN_AT_TARGET)
|
@mock.patch.object(r_exe.RemoteExecutor, 'run_action', MOCK_RUN_AT_TARGET)
|
||||||
def test_safe_rerun_with_items(self):
|
def test_safe_rerun_with_items(self):
|
||||||
wf_text = """---
|
wf_text = """---
|
||||||
version: '2.0'
|
version: '2.0'
|
||||||
|
0
mistral/tests/unit/executors/__init__.py
Normal file
0
mistral/tests/unit/executors/__init__.py
Normal file
24
mistral/tests/unit/executors/base.py
Normal file
24
mistral/tests/unit/executors/base.py
Normal file
@ -0,0 +1,24 @@
|
|||||||
|
# Copyright 2017 - Brocade Communications Systems, Inc.
|
||||||
|
#
|
||||||
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
# you may not use this file except in compliance with the License.
|
||||||
|
# You may obtain a copy of the License at
|
||||||
|
#
|
||||||
|
# http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
#
|
||||||
|
# Unless required by applicable law or agreed to in writing, software
|
||||||
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
# See the License for the specific language governing permissions and
|
||||||
|
# limitations under the License.
|
||||||
|
|
||||||
|
from oslo_log import log as logging
|
||||||
|
|
||||||
|
from mistral.tests.unit.engine import base as engine_test_base
|
||||||
|
|
||||||
|
|
||||||
|
LOG = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
|
class ExecutorTestCase(engine_test_base.EngineTestCase):
|
||||||
|
pass
|
171
mistral/tests/unit/executors/test_local_executor.py
Normal file
171
mistral/tests/unit/executors/test_local_executor.py
Normal file
@ -0,0 +1,171 @@
|
|||||||
|
# Copyright 2017 - Brocade Communications Systems, Inc.
|
||||||
|
#
|
||||||
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
# you may not use this file except in compliance with the License.
|
||||||
|
# You may obtain a copy of the License at
|
||||||
|
#
|
||||||
|
# http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
#
|
||||||
|
# Unless required by applicable law or agreed to in writing, software
|
||||||
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
# See the License for the specific language governing permissions and
|
||||||
|
# limitations under the License.
|
||||||
|
|
||||||
|
import mock
|
||||||
|
|
||||||
|
from oslo_config import cfg
|
||||||
|
from oslo_log import log as logging
|
||||||
|
|
||||||
|
from mistral.actions import std_actions
|
||||||
|
from mistral.db.v2 import api as db_api
|
||||||
|
from mistral.executors import base as exe
|
||||||
|
from mistral.executors import remote_executor as r_exe
|
||||||
|
from mistral.services import workbooks as wb_svc
|
||||||
|
from mistral.tests.unit.executors import base
|
||||||
|
from mistral.workflow import states
|
||||||
|
|
||||||
|
|
||||||
|
LOG = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
|
# Use the set_default method to set value otherwise in certain test cases
|
||||||
|
# the change in value is not permanent.
|
||||||
|
cfg.CONF.set_default('auth_enable', False, group='pecan')
|
||||||
|
|
||||||
|
|
||||||
|
@mock.patch.object(
|
||||||
|
r_exe.RemoteExecutor,
|
||||||
|
'run_action',
|
||||||
|
mock.MagicMock(return_value=None)
|
||||||
|
)
|
||||||
|
class LocalExecutorTestCase(base.ExecutorTestCase):
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def setUpClass(cls):
|
||||||
|
super(LocalExecutorTestCase, cls).setUpClass()
|
||||||
|
cfg.CONF.set_default('type', 'local', group='executor')
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def tearDownClass(cls):
|
||||||
|
exe.cleanup()
|
||||||
|
cfg.CONF.set_default('type', 'remote', group='executor')
|
||||||
|
super(LocalExecutorTestCase, cls).tearDownClass()
|
||||||
|
|
||||||
|
@mock.patch.object(
|
||||||
|
std_actions.EchoAction,
|
||||||
|
'run',
|
||||||
|
mock.MagicMock(
|
||||||
|
side_effect=[
|
||||||
|
'Task 1', # Mock task1 success.
|
||||||
|
'Task 2', # Mock task2 success.
|
||||||
|
'Task 3' # Mock task3 success.
|
||||||
|
]
|
||||||
|
)
|
||||||
|
)
|
||||||
|
def test_run(self):
|
||||||
|
wb_def = """
|
||||||
|
version: '2.0'
|
||||||
|
|
||||||
|
name: wb1
|
||||||
|
|
||||||
|
workflows:
|
||||||
|
wf1:
|
||||||
|
type: direct
|
||||||
|
|
||||||
|
tasks:
|
||||||
|
t1:
|
||||||
|
action: std.echo output="Task 1"
|
||||||
|
on-success:
|
||||||
|
- t2
|
||||||
|
t2:
|
||||||
|
action: std.echo output="Task 2"
|
||||||
|
on-success:
|
||||||
|
- t3
|
||||||
|
t3:
|
||||||
|
action: std.echo output="Task 3"
|
||||||
|
"""
|
||||||
|
|
||||||
|
wb_svc.create_workbook_v2(wb_def)
|
||||||
|
wf_ex = self.engine.start_workflow('wb1.wf1', {})
|
||||||
|
self.await_workflow_success(wf_ex.id)
|
||||||
|
|
||||||
|
with db_api.transaction():
|
||||||
|
wf_ex = db_api.get_workflow_execution(wf_ex.id)
|
||||||
|
task_execs = wf_ex.task_executions
|
||||||
|
|
||||||
|
self.assertEqual(states.SUCCESS, wf_ex.state)
|
||||||
|
self.assertIsNone(wf_ex.state_info)
|
||||||
|
self.assertEqual(3, len(task_execs))
|
||||||
|
|
||||||
|
task_1_ex = self._assert_single_item(task_execs, name='t1')
|
||||||
|
task_2_ex = self._assert_single_item(task_execs, name='t2')
|
||||||
|
task_3_ex = self._assert_single_item(task_execs, name='t3')
|
||||||
|
|
||||||
|
self.assertEqual(states.SUCCESS, task_1_ex.state)
|
||||||
|
self.assertEqual(states.SUCCESS, task_2_ex.state)
|
||||||
|
self.assertEqual(states.SUCCESS, task_3_ex.state)
|
||||||
|
|
||||||
|
# Make sure the remote executor is not called.
|
||||||
|
self.assertFalse(r_exe.RemoteExecutor.run_action.called)
|
||||||
|
|
||||||
|
@mock.patch.object(
|
||||||
|
std_actions.EchoAction,
|
||||||
|
'run',
|
||||||
|
mock.MagicMock(
|
||||||
|
side_effect=[
|
||||||
|
'Task 1.0', # Mock task1 success.
|
||||||
|
'Task 1.1', # Mock task1 success.
|
||||||
|
'Task 1.2', # Mock task1 success.
|
||||||
|
'Task 2' # Mock task2 success.
|
||||||
|
]
|
||||||
|
)
|
||||||
|
)
|
||||||
|
def test_run_with_items(self):
|
||||||
|
wb_def = """
|
||||||
|
version: '2.0'
|
||||||
|
|
||||||
|
name: wb1
|
||||||
|
|
||||||
|
workflows:
|
||||||
|
wf1:
|
||||||
|
type: direct
|
||||||
|
|
||||||
|
tasks:
|
||||||
|
t1:
|
||||||
|
with-items: i in <% list(range(0, 3)) %>
|
||||||
|
action: std.echo output="Task 1.<% $.i %>"
|
||||||
|
publish:
|
||||||
|
v1: <% task(t1).result %>
|
||||||
|
on-success:
|
||||||
|
- t2
|
||||||
|
t2:
|
||||||
|
action: std.echo output="Task 2"
|
||||||
|
"""
|
||||||
|
|
||||||
|
wb_svc.create_workbook_v2(wb_def)
|
||||||
|
wf_ex = self.engine.start_workflow('wb1.wf1', {})
|
||||||
|
self.await_workflow_success(wf_ex.id)
|
||||||
|
|
||||||
|
with db_api.transaction():
|
||||||
|
wf_ex = db_api.get_workflow_execution(wf_ex.id)
|
||||||
|
task_execs = wf_ex.task_executions
|
||||||
|
|
||||||
|
self.assertEqual(states.SUCCESS, wf_ex.state)
|
||||||
|
self.assertEqual(2, len(wf_ex.task_executions))
|
||||||
|
|
||||||
|
task_1_ex = self._assert_single_item(task_execs, name='t1')
|
||||||
|
task_2_ex = self._assert_single_item(task_execs, name='t2')
|
||||||
|
|
||||||
|
self.assertEqual(states.SUCCESS, task_1_ex.state)
|
||||||
|
self.assertEqual(states.SUCCESS, task_2_ex.state)
|
||||||
|
|
||||||
|
with db_api.transaction():
|
||||||
|
task_1_action_exs = db_api.get_action_executions(
|
||||||
|
task_execution_id=task_1_ex.id
|
||||||
|
)
|
||||||
|
|
||||||
|
self.assertEqual(3, len(task_1_action_exs))
|
||||||
|
|
||||||
|
# Make sure the remote executor is not called.
|
||||||
|
self.assertFalse(r_exe.RemoteExecutor.run_action.called)
|
40
mistral/tests/unit/executors/test_plugins.py
Normal file
40
mistral/tests/unit/executors/test_plugins.py
Normal file
@ -0,0 +1,40 @@
|
|||||||
|
# Copyright 2017 - Brocade Communications Systems, Inc.
|
||||||
|
#
|
||||||
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
# you may not use this file except in compliance with the License.
|
||||||
|
# You may obtain a copy of the License at
|
||||||
|
#
|
||||||
|
# http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
#
|
||||||
|
# Unless required by applicable law or agreed to in writing, software
|
||||||
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
# See the License for the specific language governing permissions and
|
||||||
|
# limitations under the License.
|
||||||
|
|
||||||
|
from oslo_log import log as logging
|
||||||
|
|
||||||
|
from mistral.executors import base as exe
|
||||||
|
from mistral.executors import default_executor as d_exe
|
||||||
|
from mistral.executors import remote_executor as r_exe
|
||||||
|
from mistral.tests.unit.executors import base
|
||||||
|
|
||||||
|
|
||||||
|
LOG = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
|
class PluginTestCase(base.ExecutorTestCase):
|
||||||
|
|
||||||
|
def tearDown(self):
|
||||||
|
exe.cleanup()
|
||||||
|
super(PluginTestCase, self).tearDown()
|
||||||
|
|
||||||
|
def test_get_local_executor(self):
|
||||||
|
executor = exe.get_executor('local')
|
||||||
|
|
||||||
|
self.assertIsInstance(executor, d_exe.DefaultExecutor)
|
||||||
|
|
||||||
|
def test_get_remote_executor(self):
|
||||||
|
executor = exe.get_executor('remote')
|
||||||
|
|
||||||
|
self.assertIsInstance(executor, r_exe.RemoteExecutor)
|
@ -70,6 +70,10 @@ mistral.actions =
|
|||||||
std.sleep = mistral.actions.std_actions:SleepAction
|
std.sleep = mistral.actions.std_actions:SleepAction
|
||||||
std.test_dict = mistral.actions.std_actions:TestDictAction
|
std.test_dict = mistral.actions.std_actions:TestDictAction
|
||||||
|
|
||||||
|
mistral.executors =
|
||||||
|
local = mistral.executors.default_executor:DefaultExecutor
|
||||||
|
remote = mistral.executors.remote_executor:RemoteExecutor
|
||||||
|
|
||||||
mistral.expression.functions =
|
mistral.expression.functions =
|
||||||
global = mistral.utils.expression_utils:global_
|
global = mistral.utils.expression_utils:global_
|
||||||
json_pp = mistral.utils.expression_utils:json_pp_
|
json_pp = mistral.utils.expression_utils:json_pp_
|
||||||
|
Loading…
Reference in New Issue
Block a user