mirror of
https://github.com/sartography/spiff-arena.git
synced 2025-01-18 13:31:08 +00:00
1bed0fb3ee
cd4da465 Merge pull request #264 from sartography/bugfix/dmn-equality-with-boolean 414a59eb disambiguate DMN expressions eea53c91 Merge pull request #263 from sartography/feature/cleanup-task-completion d248d5b1 execute postscript before other complete hook tasks c09f1a90 streamline predict & remove some duplicated calls to it 64c21791 remove duplicate calls to update 4ca1076d move task update to _on_complete to ensure data is copied consistently after task related activities are done d037a7eb small changes for readability 025bc30f Quick patch -- is_executable needs to be accurate immediately. 14d3d8c3 Merge pull request #262 from sartography/feature/parser_info_features 849c223e We are jumping through a lot of complex xml parsing in SpiffWorkflow-Backend because we need to know some basic information about a BPMN process at the moment it is saved. Rather than do that work in the backend, it seems better to have SpiffWorkflow handle parsing the xml and providing a bit of metadata, including: git-subtree-dir: SpiffWorkflow git-subtree-split: cd4da465e125ca1ae1b57d227bfa324d9d4c507c
146 lines
5.6 KiB
Python
146 lines
5.6 KiB
Python
# -*- coding: utf-8 -*-
|
|
|
|
import json
|
|
import os
|
|
import unittest
|
|
from SpiffWorkflow.bpmn.parser.BpmnParser import BpmnValidator
|
|
|
|
from SpiffWorkflow.task import TaskState
|
|
|
|
from SpiffWorkflow.bpmn.serializer.workflow import BpmnWorkflowSerializer
|
|
from .BpmnLoaderForTests import TestUserTaskConverter, TestBpmnParser
|
|
|
|
__author__ = 'matth'
|
|
|
|
|
|
wf_spec_converter = BpmnWorkflowSerializer.configure_workflow_spec_converter([TestUserTaskConverter])
|
|
|
|
class BpmnWorkflowTestCase(unittest.TestCase):
|
|
|
|
serializer = BpmnWorkflowSerializer(wf_spec_converter)
|
|
|
|
def get_parser(self, filename, validate=True):
|
|
f = os.path.join(os.path.dirname(__file__), 'data', filename)
|
|
validator = BpmnValidator() if validate else None
|
|
parser = TestBpmnParser(validator=validator)
|
|
parser.add_bpmn_files_by_glob(f)
|
|
return parser
|
|
|
|
def load_workflow_spec(self, filename, process_name, validate=True):
|
|
parser = self.get_parser(filename, validate)
|
|
top_level_spec = parser.get_spec(process_name)
|
|
subprocesses = parser.get_subprocess_specs(process_name)
|
|
return top_level_spec, subprocesses
|
|
|
|
def load_collaboration(self, filename, collaboration_name):
|
|
f = os.path.join(os.path.dirname(__file__), 'data', filename)
|
|
parser = TestBpmnParser()
|
|
parser.add_bpmn_files_by_glob(f)
|
|
return parser.get_collaboration(collaboration_name)
|
|
|
|
def get_all_specs(self, filename):
|
|
f = os.path.join(os.path.dirname(__file__), 'data', filename)
|
|
parser = TestBpmnParser()
|
|
parser.add_bpmn_files_by_glob(f)
|
|
return parser.find_all_specs()
|
|
|
|
def do_next_exclusive_step(self, step_name, with_save_load=False, set_attribs=None, choice=None):
|
|
if with_save_load:
|
|
self.save_restore_all()
|
|
|
|
self.workflow.do_engine_steps()
|
|
tasks = self.workflow.get_tasks(TaskState.READY)
|
|
self._do_single_step(step_name, tasks, set_attribs, choice)
|
|
|
|
def do_next_named_step(self, step_name, with_save_load=False, set_attribs=None, choice=None, only_one_instance=True):
|
|
if with_save_load:
|
|
self.save_restore()
|
|
|
|
self.workflow.do_engine_steps()
|
|
step_name_path = step_name.split("|")
|
|
|
|
def switch_workflow(p):
|
|
for task_id, sp in p.workflow._get_outermost_workflow().subprocesses.items():
|
|
if p in sp.get_tasks(workflow=sp):
|
|
return p.workflow.get_task(task_id)
|
|
|
|
def is_match(t):
|
|
if not (t.task_spec.name == step_name_path[-1] or t.task_spec.description == step_name_path[-1]):
|
|
return False
|
|
for parent_name in step_name_path[:-1]:
|
|
p = t.parent
|
|
found = False
|
|
while (p and p != p.parent):
|
|
if (p.task_spec.name == parent_name or p.task_spec.description == parent_name):
|
|
found = True
|
|
break
|
|
if p.parent is None and p.workflow != p.workflow.outer_workflow:
|
|
p = switch_workflow(p)
|
|
else:
|
|
p = p.parent
|
|
if not found:
|
|
return False
|
|
return True
|
|
|
|
tasks = list(
|
|
[t for t in self.workflow.get_tasks(TaskState.READY) if is_match(t)])
|
|
|
|
self._do_single_step(
|
|
step_name_path[-1], tasks, set_attribs, choice, only_one_instance=only_one_instance)
|
|
|
|
def assertTaskNotReady(self, step_name):
|
|
tasks = list([t for t in self.workflow.get_tasks(TaskState.READY)
|
|
if t.task_spec.name == step_name or t.task_spec.description == step_name])
|
|
self.assertEqual([], tasks)
|
|
|
|
def _do_single_step(self, step_name, tasks, set_attribs=None, choice=None, only_one_instance=True):
|
|
|
|
if only_one_instance:
|
|
self.assertEqual(
|
|
len(tasks), 1, 'Did not find one task for \'%s\' (got %d)' % (step_name, len(tasks)))
|
|
else:
|
|
self.assertNotEqual(
|
|
len(tasks), 0, 'Did not find any tasks for \'%s\'' % (step_name))
|
|
|
|
self.assertTrue(
|
|
tasks[0].task_spec.name == step_name or tasks[
|
|
0].task_spec.description == step_name,
|
|
'Expected step %s, got %s (%s)' % (step_name, tasks[0].task_spec.description, tasks[0].task_spec.name))
|
|
if not set_attribs:
|
|
set_attribs = {}
|
|
|
|
if choice:
|
|
set_attribs['choice'] = choice
|
|
|
|
if set_attribs:
|
|
tasks[0].set_data(**set_attribs)
|
|
tasks[0].complete()
|
|
|
|
def save_restore(self):
|
|
|
|
before_state = self._get_workflow_state(do_steps=False)
|
|
before_dump = self.workflow.get_dump()
|
|
# Check that we can actully convert this to JSON
|
|
json_str = json.dumps(before_state)
|
|
after = self.serializer.workflow_from_dict(json.loads(json_str), read_only=False)
|
|
# Check that serializing and deserializing results in the same workflow
|
|
after_state = self.serializer.workflow_to_dict(after)
|
|
after_dump = after.get_dump()
|
|
self.maxDiff = None
|
|
self.assertEqual(before_dump, after_dump)
|
|
self.assertEqual(before_state, after_state)
|
|
self.workflow = after
|
|
|
|
def restore(self, state):
|
|
self.workflow = self.serializer.workflow_from_dict(state, read_only=False)
|
|
|
|
def get_read_only_workflow(self):
|
|
state = self._get_workflow_state()
|
|
return self.serializer.workflow_from_dict(state, read_only=True)
|
|
|
|
def _get_workflow_state(self, do_steps=True):
|
|
if do_steps:
|
|
self.workflow.do_engine_steps()
|
|
self.workflow.refresh_waiting_tasks()
|
|
return self.serializer.workflow_to_dict(self.workflow)
|