spiff-arena/SpiffWorkflow/bpmn/specs/InclusiveGateway.py
Dan ba67d7ad34 Squashed 'SpiffWorkflow/' changes from 450ef3bcd..98c6294f1
98c6294f1 Merge pull request #287 from sartography/feature/workflow_data_exceptions
d40a1da59 Workflow Data Exceptions were broken in the previous error refactor.  This assures we are getting good messages from these errors.
a156378e1 Merge pull request #286 from sartography/feature/inclusive-gateway-support
7f6e398c2 bypass unnecessary checks in gateway joins
ade21a894 revert a few things
e1cf75202 Merge branch 'main' into feature/inclusive-gateway-support
15a0a4414 revert change to MultiChoice and handle no defaults in BPMN specs
e1469e6bb add support for diverging inclusive gateways
71fd86386 really prevent non-default flows without conditions
924759d9b clean up join specs
7378639d3 Merge pull request #284 from sartography/feature/improved-timer-events
dc8d139d2 remove useless method
530f23697 Merge branch 'main' into feature/improved-timer-events
307cca9c5 partially clean up existing gateways
0a344285e clean up task parsers
2cef997d1 add waiting_events method to bpmn workflow
48091c407 serializer migration script and miscellaneous fixes to serialization
61316854b store internal timer data as string/float
389c14c4c add some tests for parsing durations
582bc9482 convert timers to iso 8601
6dfd7ebe9 remove extraneous calls to update
6bd429529 clean up tests
d56e9912f remove useless method

git-subtree-dir: SpiffWorkflow
git-subtree-split: 98c6294f1240aee599cd98bcee58d121cb57b331
2023-01-26 18:17:35 -05:00

121 lines
5.0 KiB
Python

# -*- coding: utf-8 -*-
# Copyright (C) 2012 Matthew Hampton
#
# This library is free software; you can redistribute it and/or
# modify it under the terms of the GNU Lesser General Public
# License as published by the Free Software Foundation; either
# version 2.1 of the License, or (at your option) any later version.
#
# This library is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
# Lesser General Public License for more details.
#
# You should have received a copy of the GNU Lesser General Public
# License along with this library; if not, write to the Free Software
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
# 02110-1301 USA
from SpiffWorkflow.exceptions import WorkflowTaskException
from ...task import TaskState
from .UnstructuredJoin import UnstructuredJoin
from ...specs.MultiChoice import MultiChoice
class InclusiveGateway(MultiChoice, UnstructuredJoin):
"""
Task Spec for a bpmn:parallelGateway node. From the specification of BPMN
(http://www.omg.org/spec/BPMN/2.0/PDF - document number:formal/2011-01-03):
The Inclusive Gateway is activated if
* At least one incoming Sequence Flow has at least one token and
* For every directed path formed by sequence flow that
* starts with a Sequence Flow f of the diagram that has a token,
* ends with an incoming Sequence Flow of the inclusive gateway that has
no token, and
* does not visit the Inclusive Gateway.
* There is also a directed path formed by Sequence Flow that
* starts with f,
* ends with an incoming Sequence Flow of the inclusive gateway that has
a token, and
* does not visit the Inclusive Gateway.
Upon execution, a token is consumed from each incoming Sequence Flow that
has a token. A token will be produced on some of the outgoing Sequence
Flows.
TODO: Not implemented: At the moment, we can't handle having more than one
token at a single incoming sequence
TODO: At the moment only converging Inclusive Gateways are supported.
In order to determine the outgoing Sequence Flows that receive a token, all
conditions on the outgoing Sequence Flows are evaluated. The evaluation
does not have to respect a certain order.
For every condition which evaluates to true, a token MUST be passed on the
respective Sequence Flow.
If and only if none of the conditions evaluates to true, the token is
passed on the default Sequence Flow.
In case all conditions evaluate to false and a default flow has not been
specified, the Inclusive Gateway throws an exception.
"""
def test(self):
MultiChoice.test(self)
UnstructuredJoin.test(self)
def _check_threshold_unstructured(self, my_task, force=False):
completed_inputs, waiting_tasks = self._get_inputs_with_tokens(my_task)
uncompleted_inputs = [i for i in self.inputs if i not in completed_inputs]
# We only have to complete a task once for it to count, even if's on multiple paths
for task in waiting_tasks:
if task.task_spec in completed_inputs:
waiting_tasks.remove(task)
if force:
# If force is true, complete the task
complete = True
elif len(waiting_tasks) > 0:
# If we have waiting tasks, we're obviously not done
complete = False
else:
# Handle the case where there are paths from active tasks that must go through uncompleted inputs
tasks = my_task.workflow.get_tasks(TaskState.READY | TaskState.WAITING, workflow=my_task.workflow)
sources = [t.task_spec for t in tasks]
# This will go back through a task spec's ancestors and return the source, if applicable
def check(spec):
for parent in spec.inputs:
return parent if parent in sources else check(parent)
# If we can get to a completed input from this task, we don't have to wait for it
for spec in completed_inputs:
source = check(spec)
if source is not None:
sources.remove(source)
# Now check the rest of the uncompleted inputs and see if they can be reached from any of the remaining tasks
unfinished_paths = []
for spec in uncompleted_inputs:
if check(spec) is not None:
unfinished_paths.append(spec)
break
complete = len(unfinished_paths) == 0
return complete, waiting_tasks
def _on_complete_hook(self, my_task):
outputs = self._get_matching_outputs(my_task)
if len(outputs) == 0:
raise WorkflowTaskException(f'No conditions satisfied on gateway', task=my_task)
my_task._sync_children(outputs, TaskState.FUTURE)
@property
def spec_type(self):
return 'Inclusive Gateway'