Add started_at and finished_at to task execution.
Sometimes it is very important to know exact time of task execution, but using fields created_at and updated_at is incorrect, because in this case duration will consist of not only Running time. That new fields solve this problem. Change-Id: I15be0648a0346f5b3dc9ef4a1b330a6c0e818385 Implements: blueprint mistral-add-started-finished-at Signed-off-by: Oleg Ovcharuk <vgvoleg@gmail.com>
This commit is contained in:
parent
c9e08a8839
commit
ea7fa0e4a6
@ -404,6 +404,9 @@ class Task(resource.Resource):
|
|||||||
created_at = wtypes.text
|
created_at = wtypes.text
|
||||||
updated_at = wtypes.text
|
updated_at = wtypes.text
|
||||||
|
|
||||||
|
started_at = wtypes.text
|
||||||
|
finished_at = wtypes.text
|
||||||
|
|
||||||
# Add this param to make Mistral API work with WSME 0.8.0 or higher version
|
# Add this param to make Mistral API work with WSME 0.8.0 or higher version
|
||||||
reset = wsme.wsattr(bool, mandatory=True)
|
reset = wsme.wsattr(bool, mandatory=True)
|
||||||
|
|
||||||
|
@ -0,0 +1,40 @@
|
|||||||
|
# Copyright 2018 OpenStack Foundation.
|
||||||
|
#
|
||||||
|
# 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.
|
||||||
|
|
||||||
|
"""Add started_at and finished_at to task execution
|
||||||
|
|
||||||
|
Revision ID: 031
|
||||||
|
Revises: 030
|
||||||
|
Create Date: 2018-10-03 20:09:45.582597
|
||||||
|
|
||||||
|
"""
|
||||||
|
|
||||||
|
# revision identifiers, used by Alembic.
|
||||||
|
revision = '031'
|
||||||
|
down_revision = '030'
|
||||||
|
|
||||||
|
from alembic import op
|
||||||
|
import sqlalchemy as sa
|
||||||
|
|
||||||
|
|
||||||
|
def upgrade():
|
||||||
|
op.add_column(
|
||||||
|
'task_executions_v2',
|
||||||
|
sa.Column('started_at', sa.DateTime(), nullable=True)
|
||||||
|
)
|
||||||
|
op.add_column(
|
||||||
|
'task_executions_v2',
|
||||||
|
sa.Column('finished_at', sa.DateTime(), nullable=True)
|
||||||
|
)
|
@ -255,6 +255,8 @@ class TaskExecution(Execution):
|
|||||||
action_spec = sa.Column(st.JsonLongDictType())
|
action_spec = sa.Column(st.JsonLongDictType())
|
||||||
unique_key = sa.Column(sa.String(255), nullable=True)
|
unique_key = sa.Column(sa.String(255), nullable=True)
|
||||||
type = sa.Column(sa.String(10))
|
type = sa.Column(sa.String(10))
|
||||||
|
started_at = sa.Column(sa.DateTime, nullable=True)
|
||||||
|
finished_at = sa.Column(sa.DateTime, nullable=True)
|
||||||
|
|
||||||
# Whether the task is fully processed (publishing and calculating commands
|
# Whether the task is fully processed (publishing and calculating commands
|
||||||
# after it). It allows to simplify workflow controller implementations
|
# after it). It allows to simplify workflow controller implementations
|
||||||
@ -273,6 +275,14 @@ class TaskExecution(Execution):
|
|||||||
else self.workflow_executions
|
else self.workflow_executions
|
||||||
)
|
)
|
||||||
|
|
||||||
|
def to_dict(self):
|
||||||
|
d = super(TaskExecution, self).to_dict()
|
||||||
|
|
||||||
|
utils.datetime_to_str_in_dict(d, 'started_at')
|
||||||
|
utils.datetime_to_str_in_dict(d, 'finished_at')
|
||||||
|
|
||||||
|
return d
|
||||||
|
|
||||||
|
|
||||||
for cls in utils.iter_subclasses(Execution):
|
for cls in utils.iter_subclasses(Execution):
|
||||||
event.listen(
|
event.listen(
|
||||||
|
@ -73,6 +73,7 @@ def run_task(wf_cmd):
|
|||||||
LOG.error(msg)
|
LOG.error(msg)
|
||||||
|
|
||||||
task.set_state(states.ERROR, msg)
|
task.set_state(states.ERROR, msg)
|
||||||
|
task.save_finished_time()
|
||||||
|
|
||||||
wf_handler.force_fail_workflow(wf_ex, msg)
|
wf_handler.force_fail_workflow(wf_ex, msg)
|
||||||
|
|
||||||
@ -125,6 +126,7 @@ def _on_action_complete(action_ex):
|
|||||||
LOG.error(msg)
|
LOG.error(msg)
|
||||||
|
|
||||||
task.set_state(states.ERROR, msg)
|
task.set_state(states.ERROR, msg)
|
||||||
|
task.save_finished_time()
|
||||||
|
|
||||||
wf_handler.force_fail_workflow(wf_ex, msg)
|
wf_handler.force_fail_workflow(wf_ex, msg)
|
||||||
|
|
||||||
@ -184,6 +186,7 @@ def _on_action_update(action_ex):
|
|||||||
LOG.error(msg)
|
LOG.error(msg)
|
||||||
|
|
||||||
task.set_state(states.ERROR, msg)
|
task.set_state(states.ERROR, msg)
|
||||||
|
task.save_finished_time()
|
||||||
|
|
||||||
wf_handler.force_fail_workflow(wf_ex, msg)
|
wf_handler.force_fail_workflow(wf_ex, msg)
|
||||||
|
|
||||||
@ -212,6 +215,7 @@ def force_fail_task(task_ex, msg):
|
|||||||
task = _build_task_from_execution(wf_spec, task_ex)
|
task = _build_task_from_execution(wf_spec, task_ex)
|
||||||
|
|
||||||
task.set_state(states.ERROR, msg)
|
task.set_state(states.ERROR, msg)
|
||||||
|
task.save_finished_time()
|
||||||
|
|
||||||
wf_handler.force_fail_workflow(task_ex.workflow_execution, msg)
|
wf_handler.force_fail_workflow(task_ex.workflow_execution, msg)
|
||||||
|
|
||||||
@ -238,6 +242,7 @@ def continue_task(task_ex):
|
|||||||
LOG.error(msg)
|
LOG.error(msg)
|
||||||
|
|
||||||
task.set_state(states.ERROR, msg)
|
task.set_state(states.ERROR, msg)
|
||||||
|
task.save_finished_time()
|
||||||
|
|
||||||
wf_handler.force_fail_workflow(wf_ex, msg)
|
wf_handler.force_fail_workflow(wf_ex, msg)
|
||||||
|
|
||||||
@ -266,6 +271,7 @@ def complete_task(task_ex, state, state_info):
|
|||||||
LOG.error(msg)
|
LOG.error(msg)
|
||||||
|
|
||||||
task.set_state(states.ERROR, msg)
|
task.set_state(states.ERROR, msg)
|
||||||
|
task.save_finished_time()
|
||||||
|
|
||||||
wf_handler.force_fail_workflow(wf_ex, msg)
|
wf_handler.force_fail_workflow(wf_ex, msg)
|
||||||
|
|
||||||
@ -341,7 +347,8 @@ def _build_task_from_command(cmd):
|
|||||||
task_ex=cmd.task_ex,
|
task_ex=cmd.task_ex,
|
||||||
unique_key=cmd.task_ex.unique_key,
|
unique_key=cmd.task_ex.unique_key,
|
||||||
waiting=cmd.task_ex.state == states.WAITING,
|
waiting=cmd.task_ex.state == states.WAITING,
|
||||||
triggered_by=cmd.triggered_by
|
triggered_by=cmd.triggered_by,
|
||||||
|
rerun=cmd.rerun
|
||||||
)
|
)
|
||||||
|
|
||||||
if cmd.reset:
|
if cmd.reset:
|
||||||
@ -366,7 +373,8 @@ def _build_task_from_command(cmd):
|
|||||||
|
|
||||||
|
|
||||||
def _create_task(wf_ex, wf_spec, task_spec, ctx, task_ex=None,
|
def _create_task(wf_ex, wf_spec, task_spec, ctx, task_ex=None,
|
||||||
unique_key=None, waiting=False, triggered_by=None):
|
unique_key=None, waiting=False, triggered_by=None,
|
||||||
|
rerun=False):
|
||||||
if task_spec.get_with_items():
|
if task_spec.get_with_items():
|
||||||
cls = tasks.WithItemsTask
|
cls = tasks.WithItemsTask
|
||||||
else:
|
else:
|
||||||
@ -380,7 +388,8 @@ def _create_task(wf_ex, wf_spec, task_spec, ctx, task_ex=None,
|
|||||||
task_ex=task_ex,
|
task_ex=task_ex,
|
||||||
unique_key=unique_key,
|
unique_key=unique_key,
|
||||||
waiting=waiting,
|
waiting=waiting,
|
||||||
triggered_by=triggered_by
|
triggered_by=triggered_by,
|
||||||
|
rerun=rerun
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
@ -51,7 +51,8 @@ class Task(object):
|
|||||||
"""
|
"""
|
||||||
|
|
||||||
def __init__(self, wf_ex, wf_spec, task_spec, ctx, task_ex=None,
|
def __init__(self, wf_ex, wf_spec, task_spec, ctx, task_ex=None,
|
||||||
unique_key=None, waiting=False, triggered_by=None):
|
unique_key=None, waiting=False, triggered_by=None,
|
||||||
|
rerun=False):
|
||||||
self.wf_ex = wf_ex
|
self.wf_ex = wf_ex
|
||||||
self.task_spec = task_spec
|
self.task_spec = task_spec
|
||||||
self.ctx = ctx
|
self.ctx = ctx
|
||||||
@ -60,6 +61,7 @@ class Task(object):
|
|||||||
self.unique_key = unique_key
|
self.unique_key = unique_key
|
||||||
self.waiting = waiting
|
self.waiting = waiting
|
||||||
self.triggered_by = triggered_by
|
self.triggered_by = triggered_by
|
||||||
|
self.rerun = rerun
|
||||||
self.reset_flag = False
|
self.reset_flag = False
|
||||||
self.created = False
|
self.created = False
|
||||||
self.state_changed = False
|
self.state_changed = False
|
||||||
@ -174,6 +176,12 @@ class Task(object):
|
|||||||
|
|
||||||
cur_state = self.task_ex.state
|
cur_state = self.task_ex.state
|
||||||
|
|
||||||
|
# Set initial started_at in case of waiting => running.
|
||||||
|
# We can't set this just in run_existing, because task retries
|
||||||
|
# will update started_at, which is incorrect.
|
||||||
|
if cur_state == states.WAITING and state == states.RUNNING:
|
||||||
|
self.save_started_time()
|
||||||
|
|
||||||
if cur_state != state or self.task_ex.state_info != state_info:
|
if cur_state != state or self.task_ex.state_info != state_info:
|
||||||
task_ex = db_api.update_task_execution_state(
|
task_ex = db_api.update_task_execution_state(
|
||||||
id=self.task_ex.id,
|
id=self.task_ex.id,
|
||||||
@ -270,6 +278,8 @@ class Task(object):
|
|||||||
|
|
||||||
self.register_workflow_completion_check()
|
self.register_workflow_completion_check()
|
||||||
|
|
||||||
|
self.save_finished_time()
|
||||||
|
|
||||||
# Publish task event.
|
# Publish task event.
|
||||||
self.notify(old_task_state, self.task_ex.state)
|
self.notify(old_task_state, self.task_ex.state)
|
||||||
|
|
||||||
@ -398,6 +408,18 @@ class Task(object):
|
|||||||
|
|
||||||
return env.get('__actions', {}).get(action_name, {})
|
return env.get('__actions', {}).get(action_name, {})
|
||||||
|
|
||||||
|
def save_started_time(self, value='default'):
|
||||||
|
if not self.task_ex:
|
||||||
|
return
|
||||||
|
time = value if value is not 'default' else utils.utc_now_sec()
|
||||||
|
self.task_ex.started_at = time
|
||||||
|
|
||||||
|
def save_finished_time(self, value='default'):
|
||||||
|
if not self.task_ex:
|
||||||
|
return
|
||||||
|
time = value if value is not 'default' else utils.utc_now_sec()
|
||||||
|
self.task_ex.finished_at = time
|
||||||
|
|
||||||
|
|
||||||
class RegularTask(Task):
|
class RegularTask(Task):
|
||||||
"""Regular task.
|
"""Regular task.
|
||||||
@ -440,6 +462,7 @@ class RegularTask(Task):
|
|||||||
return
|
return
|
||||||
|
|
||||||
self._create_task_execution()
|
self._create_task_execution()
|
||||||
|
self.save_started_time()
|
||||||
|
|
||||||
# Publish event.
|
# Publish event.
|
||||||
self.notify(None, self.task_ex.state)
|
self.notify(None, self.task_ex.state)
|
||||||
@ -481,6 +504,15 @@ class RegularTask(Task):
|
|||||||
# Publish event.
|
# Publish event.
|
||||||
self.notify(old_task_state, self.task_ex.state)
|
self.notify(old_task_state, self.task_ex.state)
|
||||||
|
|
||||||
|
if self.rerun:
|
||||||
|
self.save_started_time()
|
||||||
|
self.save_finished_time(value=None)
|
||||||
|
self._before_task_start()
|
||||||
|
|
||||||
|
# Policies could possibly change task state.
|
||||||
|
if self.task_ex.state != states.RUNNING:
|
||||||
|
return
|
||||||
|
|
||||||
self._update_inbound_context()
|
self._update_inbound_context()
|
||||||
self._update_triggered_by()
|
self._update_triggered_by()
|
||||||
self._reset_actions()
|
self._reset_actions()
|
||||||
|
@ -73,6 +73,8 @@ TASK_EX = models.TaskExecution(
|
|||||||
workflow_execution_id=WF_EX.id,
|
workflow_execution_id=WF_EX.id,
|
||||||
created_at=datetime.datetime(1970, 1, 1),
|
created_at=datetime.datetime(1970, 1, 1),
|
||||||
updated_at=datetime.datetime(1970, 1, 1),
|
updated_at=datetime.datetime(1970, 1, 1),
|
||||||
|
started_at=datetime.datetime(1970, 1, 1),
|
||||||
|
finished_at=datetime.datetime(1970, 1, 1),
|
||||||
published=PUBLISHED,
|
published=PUBLISHED,
|
||||||
processed=True
|
processed=True
|
||||||
)
|
)
|
||||||
@ -96,6 +98,8 @@ WITH_ITEMS_TASK_EX = models.TaskExecution(
|
|||||||
workflow_execution_id=WF_EX.id,
|
workflow_execution_id=WF_EX.id,
|
||||||
created_at=datetime.datetime(1970, 1, 1),
|
created_at=datetime.datetime(1970, 1, 1),
|
||||||
updated_at=datetime.datetime(1970, 1, 1),
|
updated_at=datetime.datetime(1970, 1, 1),
|
||||||
|
started_at=datetime.datetime(1970, 1, 1),
|
||||||
|
finished_at=datetime.datetime(1970, 1, 1),
|
||||||
published=PUBLISHED,
|
published=PUBLISHED,
|
||||||
processed=True
|
processed=True
|
||||||
)
|
)
|
||||||
@ -109,6 +113,8 @@ TASK = {
|
|||||||
'workflow_execution_id': WF_EX.id,
|
'workflow_execution_id': WF_EX.id,
|
||||||
'created_at': '1970-01-01 00:00:00',
|
'created_at': '1970-01-01 00:00:00',
|
||||||
'updated_at': '1970-01-01 00:00:00',
|
'updated_at': '1970-01-01 00:00:00',
|
||||||
|
'started_at': '1970-01-01 00:00:00',
|
||||||
|
'finished_at': '1970-01-01 00:00:00',
|
||||||
'result': json.dumps(RESULT),
|
'result': json.dumps(RESULT),
|
||||||
'published': json.dumps(PUBLISHED),
|
'published': json.dumps(PUBLISHED),
|
||||||
'runtime_context': json.dumps(RUNTIME_CONTEXT),
|
'runtime_context': json.dumps(RUNTIME_CONTEXT),
|
||||||
|
163
mistral/tests/unit/engine/test_task_started_finished_at.py
Normal file
163
mistral/tests/unit/engine/test_task_started_finished_at.py
Normal file
@ -0,0 +1,163 @@
|
|||||||
|
# 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 mistral.tests.unit.engine import base
|
||||||
|
|
||||||
|
from mistral.db.v2 import api as db_api
|
||||||
|
from mistral.services import workflows as wf_service
|
||||||
|
|
||||||
|
WF_FAIL_WITH_WAIT = """
|
||||||
|
version: "2.0"
|
||||||
|
|
||||||
|
wf_fail:
|
||||||
|
tasks:
|
||||||
|
task1:
|
||||||
|
action: std.fail
|
||||||
|
wait-before: 2
|
||||||
|
"""
|
||||||
|
|
||||||
|
WF_JOIN_ALL = """
|
||||||
|
version: "2.0"
|
||||||
|
|
||||||
|
wf_join:
|
||||||
|
tasks:
|
||||||
|
wait_2:
|
||||||
|
action: std.sleep seconds=2
|
||||||
|
on-success: finish_wf
|
||||||
|
wait_4:
|
||||||
|
action: std.sleep seconds=4
|
||||||
|
on-success: finish_wf
|
||||||
|
wait_6:
|
||||||
|
action: std.sleep seconds=6
|
||||||
|
on-success: finish_wf
|
||||||
|
finish_wf:
|
||||||
|
join: all
|
||||||
|
action: std.sleep seconds=2
|
||||||
|
"""
|
||||||
|
|
||||||
|
WF_WITH_RETRIES = """
|
||||||
|
version: "2.0"
|
||||||
|
|
||||||
|
wf_retry:
|
||||||
|
tasks:
|
||||||
|
task1:
|
||||||
|
action: std.fail
|
||||||
|
retry:
|
||||||
|
delay: 1
|
||||||
|
count: 5
|
||||||
|
"""
|
||||||
|
|
||||||
|
WF_WAIT_BEFORE_AFTER = """
|
||||||
|
version: "2.0"
|
||||||
|
|
||||||
|
wf_wait:
|
||||||
|
tasks:
|
||||||
|
task1:
|
||||||
|
action: std.noop
|
||||||
|
wait-before: 3
|
||||||
|
wait-after: 3
|
||||||
|
"""
|
||||||
|
|
||||||
|
|
||||||
|
class TaskStartedFinishedAtTest(base.EngineTestCase):
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
super(TaskStartedFinishedAtTest, self).setUp()
|
||||||
|
|
||||||
|
def test_started_finished_fields_updated_after_rerun(self):
|
||||||
|
wf_service.create_workflows(WF_FAIL_WITH_WAIT)
|
||||||
|
wf_ex = self.engine.start_workflow('wf_fail')
|
||||||
|
self.await_workflow_error(wf_ex.id)
|
||||||
|
|
||||||
|
task_ex = self._get_task_from_wf(wf_ex.id)
|
||||||
|
started_1st, finished_1st = self._get_started_finished(task_ex)
|
||||||
|
|
||||||
|
wf_ex = self.engine.rerun_workflow(task_ex.id)
|
||||||
|
|
||||||
|
task_ex = self._get_task_from_wf(wf_ex.id)
|
||||||
|
self.assertIsNone(task_ex.finished_at)
|
||||||
|
self.await_workflow_error(wf_ex.id)
|
||||||
|
|
||||||
|
task_ex = self._get_task_from_wf(wf_ex.id)
|
||||||
|
started_2nd, finished_2nd = self._get_started_finished(task_ex)
|
||||||
|
|
||||||
|
self.assertNotEqual(started_1st, started_2nd)
|
||||||
|
self.assertNotEqual(finished_1st, finished_2nd)
|
||||||
|
|
||||||
|
def test_correct_duration_in_case_of_join_all(self):
|
||||||
|
wf_service.create_workflows(WF_JOIN_ALL)
|
||||||
|
wf_ex = self.engine.start_workflow('wf_join')
|
||||||
|
self.await_workflow_success(wf_ex.id)
|
||||||
|
|
||||||
|
wait_2 = self._get_task_from_wf(wf_ex.id, 'wait_2')
|
||||||
|
wait_4 = self._get_task_from_wf(wf_ex.id, 'wait_4')
|
||||||
|
wait_6 = self._get_task_from_wf(wf_ex.id, 'wait_6')
|
||||||
|
finish_wf = self._get_task_from_wf(wf_ex.id, 'finish_wf')
|
||||||
|
|
||||||
|
self._check_was_started_after(finish_wf, wait_2)
|
||||||
|
self._check_was_started_after(finish_wf, wait_4)
|
||||||
|
self._check_was_started_after(finish_wf, wait_6)
|
||||||
|
|
||||||
|
def test_retries_do_not_update_created_at(self):
|
||||||
|
wf_service.create_workflows(WF_WITH_RETRIES)
|
||||||
|
wf_ex = self.engine.start_workflow('wf_retry')
|
||||||
|
self.await_workflow_error(wf_ex.id)
|
||||||
|
|
||||||
|
task_ex = self._get_task_from_wf(wf_ex.id)
|
||||||
|
created_at = task_ex.created_at
|
||||||
|
started_at = self._get_started_finished(task_ex)[0]
|
||||||
|
|
||||||
|
self.assertEqual(created_at, started_at)
|
||||||
|
|
||||||
|
def test_wait_before_after_are_included_to_duration(self):
|
||||||
|
wf_service.create_workflows(WF_WAIT_BEFORE_AFTER)
|
||||||
|
wf_ex = self.engine.start_workflow('wf_wait')
|
||||||
|
self.await_workflow_success(wf_ex.id)
|
||||||
|
|
||||||
|
task_ex = self._get_task_from_wf(wf_ex.id)
|
||||||
|
started, finished = self._get_started_finished(task_ex)
|
||||||
|
duration = self._get_task_duration(started, finished)
|
||||||
|
|
||||||
|
self._check_duration_more_than(duration, 1)
|
||||||
|
|
||||||
|
def _get_task_from_wf(self, wf_ex_id, name='task1'):
|
||||||
|
with db_api.transaction():
|
||||||
|
wf_ex = db_api.get_workflow_execution(wf_ex_id)
|
||||||
|
task_execs = wf_ex.task_executions
|
||||||
|
return self._assert_single_item(task_execs, name=name)
|
||||||
|
|
||||||
|
def _get_started_finished(self, task_ex):
|
||||||
|
started_at = task_ex.started_at
|
||||||
|
finished_at = task_ex.finished_at
|
||||||
|
self.assertIsNotNone(started_at)
|
||||||
|
self.assertIsNotNone(finished_at)
|
||||||
|
return started_at, finished_at
|
||||||
|
|
||||||
|
def _get_task_duration(self, start_time, finish_time):
|
||||||
|
return (finish_time - start_time).total_seconds()
|
||||||
|
|
||||||
|
def _check_was_started_after(self, task_started, task_finished):
|
||||||
|
first_finished = self._get_started_finished(task_finished)[1]
|
||||||
|
second_started = self._get_started_finished(task_started)[0]
|
||||||
|
delta = self._get_task_duration(first_finished, second_started)
|
||||||
|
|
||||||
|
self.assertTrue(
|
||||||
|
delta >= 0,
|
||||||
|
"Expected {} was started after {} was finished".format(
|
||||||
|
task_started.name, task_finished.name)
|
||||||
|
)
|
||||||
|
|
||||||
|
def _check_duration_more_than(self, duration, time):
|
||||||
|
self.assertTrue(
|
||||||
|
time < duration,
|
||||||
|
"Expected duration {} was more than {}".format(duration, time)
|
||||||
|
)
|
@ -144,7 +144,8 @@ class WorkflowController(object):
|
|||||||
return []
|
return []
|
||||||
|
|
||||||
cmds = [
|
cmds = [
|
||||||
commands.RunExistingTask(self.wf_ex, self.wf_spec, t_e, reset)
|
commands.RunExistingTask(self.wf_ex, self.wf_spec, t_e, reset,
|
||||||
|
rerun=True)
|
||||||
for t_e in task_execs
|
for t_e in task_execs
|
||||||
]
|
]
|
||||||
|
|
||||||
|
@ -101,7 +101,8 @@ class RunTask(WorkflowCommand):
|
|||||||
class RunExistingTask(WorkflowCommand):
|
class RunExistingTask(WorkflowCommand):
|
||||||
"""Command to run an existing workflow task."""
|
"""Command to run an existing workflow task."""
|
||||||
|
|
||||||
def __init__(self, wf_ex, wf_spec, task_ex, reset=True, triggered_by=None):
|
def __init__(self, wf_ex, wf_spec, task_ex, reset=True, triggered_by=None,
|
||||||
|
rerun=False):
|
||||||
super(RunExistingTask, self).__init__(
|
super(RunExistingTask, self).__init__(
|
||||||
wf_ex,
|
wf_ex,
|
||||||
wf_spec,
|
wf_spec,
|
||||||
@ -113,6 +114,7 @@ class RunExistingTask(WorkflowCommand):
|
|||||||
self.task_ex = task_ex
|
self.task_ex = task_ex
|
||||||
self.reset = reset
|
self.reset = reset
|
||||||
self.unique_key = task_ex.unique_key
|
self.unique_key = task_ex.unique_key
|
||||||
|
self.rerun = rerun
|
||||||
|
|
||||||
def to_dict(self):
|
def to_dict(self):
|
||||||
d = super(RunExistingTask, self).to_dict()
|
d = super(RunExistingTask, self).to_dict()
|
||||||
|
Loading…
x
Reference in New Issue
Block a user