Пример #1
0
    def test_task_sequence(self):
        """Tasks in a sequence are called in order"""
        class Task(tasks.WorkflowTask):
            name = 'task'

            def apply_async(self):
                record.append(self.i)
                self.set_state(tasks.TASK_SUCCEEDED)

        task_count = 10

        # prepare the task seuqence
        seq_tasks = []
        for i in range(task_count):
            t = Task(None)
            seq_tasks.append(t)
            t.i = i
        g = TaskDependencyGraph(MockWorkflowContext())
        seq = g.sequence()
        seq.add(*seq_tasks)

        record = []

        with limited_sleep_mock():
            g.execute()

        expected = list(range(task_count))
        self.assertEqual(expected, record)
Пример #2
0
    def test_task_failed(self):
        """Execution is stopped when a task failed. The next task is not
        executed"""
        class FailedTask(tasks.WorkflowTask):
            name = 'failtask'

            def apply_async(self):
                self.set_state(tasks.TASK_FAILED)

        task1 = FailedTask(mock.Mock())
        task2 = mock.Mock()

        g = TaskDependencyGraph(MockWorkflowContext())
        seq = g.sequence()
        seq.add(task1, task2)

        with limited_sleep_mock():
            try:
                g.execute()
            except RuntimeError as e:
                self.assertIn('Workflow failed', e.message)
            else:
                self.fail('Expected task to fail')

        self.assertTrue(task1.is_terminated)
        self.assertFalse(task2.apply_async.called)
Пример #3
0
    def test_task_on_success(self):
        """If a task has a success callback, dependent tasks still run"""
        wctx = MockWorkflowContext()
        g = TaskDependencyGraph(wctx)
        record = []

        class Task(tasks.WorkflowTask):
            name = 'task'

            def apply_async(self):
                record.append(self.i)
                self.set_state(tasks.TASK_SUCCEEDED)
                self.async_result.result = None
                return self.async_result

        def on_success(task):
            record.append('success')
            return tasks.HandlerResult.cont()

        t1 = Task(wctx)
        t1.i = 1
        t2 = Task(wctx)
        t1.on_success = on_success
        t2.i = 2
        g.add_task(t1)
        g.add_task(t2)
        g.add_dependency(t2, t1)
        g.execute()
        assert record == [1, 'success', 2]
Пример #4
0
 def test_executes_single_task(self):
     """A single NOP task is executed within a single iteration of the
     tasks graph loop"""
     g = TaskDependencyGraph(MockWorkflowContext())
     task = tasks.NOPLocalWorkflowTask(None)
     g.add_task(task)
     with limited_sleep_mock(limit=1):
         g.execute()
     self.assertTrue(task.is_terminated)
Пример #5
0
 def test_executes_multiple_concurrent(self):
     """Independent tasks will be executed concurrently within the same
     iteration of the graph loop.
     """
     g = TaskDependencyGraph(MockWorkflowContext())
     task1 = tasks.NOPLocalWorkflowTask(None)
     task2 = tasks.NOPLocalWorkflowTask(None)
     g.add_task(task1)
     g.add_task(task2)
     with limited_sleep_mock(limit=1):
         g.execute()
     self.assertTrue(task1.is_terminated)
     self.assertTrue(task2.is_terminated)
Пример #6
0
 def test_cancel(self):
     """When execution is cancelled, an error is thrown and tasks are
     not executed.
     """
     g = TaskDependencyGraph(MockWorkflowContext())
     task = mock.Mock()
     g.add_task(task)
     with mock.patch('cloudify.workflows.api.cancel_request', True):
         try:
             g.execute()
         except api.ExecutionCancelled:
             pass
         else:
             self.fail('Execution should have been cancelled')
     self.assertFalse(task.apply_async.called)
     self.assertFalse(task.cancel.called)
Пример #7
0
    def test_wait_after_fail(self):
        """When a task fails, the already-running tasks are waited for"""
        class FailedTask(tasks.WorkflowTask):
            """Task that fails 1 second after starting"""
            name = 'failtask'

            def get_state(self):
                if time.time() > start_time + 1:
                    return tasks.TASK_FAILED
                else:
                    return tasks.TASK_SENT

        class DelayedTask(tasks.WorkflowTask):
            """Task that succeeds 3 seconds after starting"""
            name = 'delayedtask'

            def get_state(self):
                if time.time() > start_time + 3:
                    return tasks.TASK_SUCCEEDED
                else:
                    return tasks.TASK_SENT

            handle_task_terminated = mock.Mock()

        task1 = FailedTask(mock.Mock())
        task2 = DelayedTask(mock.Mock())

        g = TaskDependencyGraph(MockWorkflowContext())
        seq = g.sequence()
        seq.add(task1, task2)
        with limited_sleep_mock():
            start_time = time.time()
            try:
                g.execute()
            except RuntimeError as e:
                self.assertIn('Workflow failed', e.message)
            else:
                self.fail('Expected task to fail')

        # even though the workflow failed 1 second in, the other task was
        # still waited for and completed
        task2.handle_task_terminated.assert_called()