mirror of
https://github.com/sartography/spiff-arena.git
synced 2025-01-14 11:34:29 +00:00
wip for get_last_user_completing_task script task
This commit is contained in:
parent
f7dc076f75
commit
e9b3ababc1
28
spiffworkflow-backend/migrations/versions/b91143f4e414_.py
Normal file
28
spiffworkflow-backend/migrations/versions/b91143f4e414_.py
Normal file
@ -0,0 +1,28 @@
|
|||||||
|
"""empty message
|
||||||
|
|
||||||
|
Revision ID: b91143f4e414
|
||||||
|
Revises: 63fc8d693b9f
|
||||||
|
Create Date: 2023-02-25 22:46:03.533624
|
||||||
|
|
||||||
|
"""
|
||||||
|
from alembic import op
|
||||||
|
import sqlalchemy as sa
|
||||||
|
|
||||||
|
|
||||||
|
# revision identifiers, used by Alembic.
|
||||||
|
revision = 'b91143f4e414'
|
||||||
|
down_revision = '63fc8d693b9f'
|
||||||
|
branch_labels = None
|
||||||
|
depends_on = None
|
||||||
|
|
||||||
|
|
||||||
|
def upgrade():
|
||||||
|
# ### commands auto generated by Alembic - please adjust! ###
|
||||||
|
op.add_column('human_task', sa.Column('process_model_identifier', sa.String(length=255), nullable=True))
|
||||||
|
# ### end Alembic commands ###
|
||||||
|
|
||||||
|
|
||||||
|
def downgrade():
|
||||||
|
# ### commands auto generated by Alembic - please adjust! ###
|
||||||
|
op.drop_column('human_task', 'process_model_identifier')
|
||||||
|
# ### end Alembic commands ###
|
@ -49,6 +49,7 @@ class HumanTaskModel(SpiffworkflowBaseDBModel):
|
|||||||
task_type: str = db.Column(db.String(50))
|
task_type: str = db.Column(db.String(50))
|
||||||
task_status: str = db.Column(db.String(50))
|
task_status: str = db.Column(db.String(50))
|
||||||
process_model_display_name: str = db.Column(db.String(255))
|
process_model_display_name: str = db.Column(db.String(255))
|
||||||
|
process_model_identifier: str = db.Column(db.String(255))
|
||||||
completed: bool = db.Column(db.Boolean, default=False, nullable=False, index=True)
|
completed: bool = db.Column(db.Boolean, default=False, nullable=False, index=True)
|
||||||
|
|
||||||
human_task_users = relationship("HumanTaskUserModel", cascade="delete")
|
human_task_users = relationship("HumanTaskUserModel", cascade="delete")
|
||||||
|
@ -0,0 +1,46 @@
|
|||||||
|
"""Get current user."""
|
||||||
|
from typing import Any
|
||||||
|
from spiffworkflow_backend.models.user import UserModel
|
||||||
|
from spiffworkflow_backend.models.human_task import HumanTaskModel
|
||||||
|
|
||||||
|
from flask import current_app
|
||||||
|
from flask import g
|
||||||
|
|
||||||
|
from spiffworkflow_backend.models.script_attributes_context import (
|
||||||
|
ScriptAttributesContext,
|
||||||
|
)
|
||||||
|
from spiffworkflow_backend.scripts.script import Script
|
||||||
|
|
||||||
|
|
||||||
|
class GetLastUserCompletingTask(Script):
|
||||||
|
@staticmethod
|
||||||
|
def requires_privileged_permissions() -> bool:
|
||||||
|
"""We have deemed this function safe to run without elevated permissions."""
|
||||||
|
return False
|
||||||
|
|
||||||
|
def get_description(self) -> str:
|
||||||
|
return """Return the last user who completed the given task."""
|
||||||
|
|
||||||
|
def run(
|
||||||
|
self,
|
||||||
|
script_attributes_context: ScriptAttributesContext,
|
||||||
|
*_args: Any,
|
||||||
|
**kwargs: Any
|
||||||
|
) -> Any:
|
||||||
|
"""Run."""
|
||||||
|
# dump the user using our json encoder and then load it back up as a dict
|
||||||
|
# to remove unwanted field types
|
||||||
|
if len(_args) == 2:
|
||||||
|
process_model_identifier = _args[0]
|
||||||
|
task_bpmn_identifier = _args[1]
|
||||||
|
else:
|
||||||
|
process_model_identifier = kwargs["process_model_identifier"]
|
||||||
|
task_bpmn_identifier = kwargs["task_bpmn_identifier"]
|
||||||
|
process_model_identifier = _args[0] or kwargs["process_model_identifier"]
|
||||||
|
print(f"process_model_identifier: {process_model_identifier}")
|
||||||
|
import pdb; pdb.set_trace()
|
||||||
|
# human_task = HumanTaskModel.query.filter_by(process_model_identifier=process_model_identifier).order_by(HumanTaskModel.id.desc()).first()
|
||||||
|
human_task = HumanTaskModel.query.filter_by(process_model_identifier=process_model_identifier).order_by(HumanTaskModel.id.desc()).join(UserModel, UserModel.id == HumanTaskModel.completed_by_user_id).first()
|
||||||
|
return human_task.completed_by_user
|
||||||
|
# user_as_json_string = current_app.json.dumps(g.user)
|
||||||
|
# return current_app.json.loads(user_as_json_string)
|
@ -875,11 +875,13 @@ class ProcessInstanceProcessor:
|
|||||||
).all()
|
).all()
|
||||||
ready_or_waiting_tasks = self.get_all_ready_or_waiting_tasks()
|
ready_or_waiting_tasks = self.get_all_ready_or_waiting_tasks()
|
||||||
process_model_display_name = ""
|
process_model_display_name = ""
|
||||||
|
process_model_identifier = ""
|
||||||
process_model_info = self.process_model_service.get_process_model(
|
process_model_info = self.process_model_service.get_process_model(
|
||||||
self.process_instance_model.process_model_identifier
|
self.process_instance_model.process_model_identifier
|
||||||
)
|
)
|
||||||
if process_model_info is not None:
|
if process_model_info is not None:
|
||||||
process_model_display_name = process_model_info.display_name
|
process_model_display_name = process_model_info.display_name
|
||||||
|
process_model_identifier = process_model_info.id
|
||||||
|
|
||||||
self.extract_metadata(process_model_info)
|
self.extract_metadata(process_model_info)
|
||||||
|
|
||||||
@ -911,6 +913,7 @@ class ProcessInstanceProcessor:
|
|||||||
human_task = HumanTaskModel(
|
human_task = HumanTaskModel(
|
||||||
process_instance_id=self.process_instance_model.id,
|
process_instance_id=self.process_instance_model.id,
|
||||||
process_model_display_name=process_model_display_name,
|
process_model_display_name=process_model_display_name,
|
||||||
|
process_model_identifier=process_model_identifier,
|
||||||
form_file_name=form_file_name,
|
form_file_name=form_file_name,
|
||||||
ui_form_file_name=ui_form_file_name,
|
ui_form_file_name=ui_form_file_name,
|
||||||
task_id=str(ready_or_waiting_task.id),
|
task_id=str(ready_or_waiting_task.id),
|
||||||
|
@ -7,7 +7,7 @@
|
|||||||
<bpmn:laneSet id="LaneSet_17rankp">
|
<bpmn:laneSet id="LaneSet_17rankp">
|
||||||
<bpmn:lane id="process_initiator" name="Process Initiator">
|
<bpmn:lane id="process_initiator" name="Process Initiator">
|
||||||
<bpmn:flowNodeRef>StartEvent_1</bpmn:flowNodeRef>
|
<bpmn:flowNodeRef>StartEvent_1</bpmn:flowNodeRef>
|
||||||
<bpmn:flowNodeRef>initator_one</bpmn:flowNodeRef>
|
<bpmn:flowNodeRef>initiator_one</bpmn:flowNodeRef>
|
||||||
<bpmn:flowNodeRef>Event_06f4e68</bpmn:flowNodeRef>
|
<bpmn:flowNodeRef>Event_06f4e68</bpmn:flowNodeRef>
|
||||||
<bpmn:flowNodeRef>initiator_two</bpmn:flowNodeRef>
|
<bpmn:flowNodeRef>initiator_two</bpmn:flowNodeRef>
|
||||||
</bpmn:lane>
|
</bpmn:lane>
|
||||||
@ -18,18 +18,19 @@
|
|||||||
<bpmn:startEvent id="StartEvent_1">
|
<bpmn:startEvent id="StartEvent_1">
|
||||||
<bpmn:outgoing>Flow_1tbyols</bpmn:outgoing>
|
<bpmn:outgoing>Flow_1tbyols</bpmn:outgoing>
|
||||||
</bpmn:startEvent>
|
</bpmn:startEvent>
|
||||||
<bpmn:sequenceFlow id="Flow_1tbyols" sourceRef="StartEvent_1" targetRef="initator_one" />
|
<bpmn:sequenceFlow id="Flow_1tbyols" sourceRef="StartEvent_1" targetRef="initiator_one" />
|
||||||
<bpmn:sequenceFlow id="Flow_16ppta1" sourceRef="initator_one" targetRef="finance_approval" />
|
<bpmn:sequenceFlow id="Flow_16ppta1" sourceRef="initiator_one" targetRef="finance_approval" />
|
||||||
<bpmn:manualTask id="initator_one" name="Initiator One">
|
<bpmn:manualTask id="initiator_one" name="Initiator One">
|
||||||
<bpmn:extensionElements>
|
<bpmn:extensionElements>
|
||||||
<spiffworkflow:instructionsForEndUser>This is initiator user?</spiffworkflow:instructionsForEndUser>
|
<spiffworkflow:instructionsForEndUser>This is for the initiator user</spiffworkflow:instructionsForEndUser>
|
||||||
|
<spiffworkflow:postScript>user_completing_task = get_last_user_completing_task("misc/category_number_one/lanes", "initiator_one")</spiffworkflow:postScript>
|
||||||
</bpmn:extensionElements>
|
</bpmn:extensionElements>
|
||||||
<bpmn:incoming>Flow_1tbyols</bpmn:incoming>
|
<bpmn:incoming>Flow_1tbyols</bpmn:incoming>
|
||||||
<bpmn:outgoing>Flow_16ppta1</bpmn:outgoing>
|
<bpmn:outgoing>Flow_16ppta1</bpmn:outgoing>
|
||||||
</bpmn:manualTask>
|
</bpmn:manualTask>
|
||||||
<bpmn:manualTask id="finance_approval" name="Finance Approval">
|
<bpmn:manualTask id="finance_approval" name="Finance Approval">
|
||||||
<bpmn:extensionElements>
|
<bpmn:extensionElements>
|
||||||
<spiffworkflow:instructionsForEndUser>This is finance user?</spiffworkflow:instructionsForEndUser>
|
<spiffworkflow:instructionsForEndUser>This is for a Finance Team user</spiffworkflow:instructionsForEndUser>
|
||||||
</bpmn:extensionElements>
|
</bpmn:extensionElements>
|
||||||
<bpmn:incoming>Flow_16ppta1</bpmn:incoming>
|
<bpmn:incoming>Flow_16ppta1</bpmn:incoming>
|
||||||
<bpmn:outgoing>Flow_1cfcauf</bpmn:outgoing>
|
<bpmn:outgoing>Flow_1cfcauf</bpmn:outgoing>
|
||||||
@ -41,7 +42,8 @@
|
|||||||
<bpmn:sequenceFlow id="Flow_0x92f7d" sourceRef="initiator_two" targetRef="Event_06f4e68" />
|
<bpmn:sequenceFlow id="Flow_0x92f7d" sourceRef="initiator_two" targetRef="Event_06f4e68" />
|
||||||
<bpmn:manualTask id="initiator_two" name="Initiator Two">
|
<bpmn:manualTask id="initiator_two" name="Initiator Two">
|
||||||
<bpmn:extensionElements>
|
<bpmn:extensionElements>
|
||||||
<spiffworkflow:instructionsForEndUser>This is initiator again?</spiffworkflow:instructionsForEndUser>
|
<spiffworkflow:instructionsForEndUser>This is initiator again</spiffworkflow:instructionsForEndUser>
|
||||||
|
<spiffworkflow:postScript />
|
||||||
</bpmn:extensionElements>
|
</bpmn:extensionElements>
|
||||||
<bpmn:incoming>Flow_1cfcauf</bpmn:incoming>
|
<bpmn:incoming>Flow_1cfcauf</bpmn:incoming>
|
||||||
<bpmn:outgoing>Flow_0x92f7d</bpmn:outgoing>
|
<bpmn:outgoing>Flow_0x92f7d</bpmn:outgoing>
|
||||||
@ -63,7 +65,7 @@
|
|||||||
<bpmndi:BPMNShape id="_BPMNShape_StartEvent_2" bpmnElement="StartEvent_1">
|
<bpmndi:BPMNShape id="_BPMNShape_StartEvent_2" bpmnElement="StartEvent_1">
|
||||||
<dc:Bounds x="179" y="159" width="36" height="36" />
|
<dc:Bounds x="179" y="159" width="36" height="36" />
|
||||||
</bpmndi:BPMNShape>
|
</bpmndi:BPMNShape>
|
||||||
<bpmndi:BPMNShape id="Activity_1lm1ald_di" bpmnElement="initator_one">
|
<bpmndi:BPMNShape id="Activity_1lm1ald_di" bpmnElement="initiator_one">
|
||||||
<dc:Bounds x="270" y="137" width="100" height="80" />
|
<dc:Bounds x="270" y="137" width="100" height="80" />
|
||||||
<bpmndi:BPMNLabel />
|
<bpmndi:BPMNLabel />
|
||||||
</bpmndi:BPMNShape>
|
</bpmndi:BPMNShape>
|
||||||
|
@ -65,6 +65,54 @@ class TestProcessInstanceProcessor(BaseTest):
|
|||||||
app.config["THREAD_LOCAL_DATA"].process_model_identifier = None
|
app.config["THREAD_LOCAL_DATA"].process_model_identifier = None
|
||||||
app.config["THREAD_LOCAL_DATA"].process_instance_id = None
|
app.config["THREAD_LOCAL_DATA"].process_instance_id = None
|
||||||
|
|
||||||
|
def test_get_last_user_completing_task_script_works(
|
||||||
|
self,
|
||||||
|
app: Flask,
|
||||||
|
client: FlaskClient,
|
||||||
|
with_db_and_bpmn_file_cleanup: None,
|
||||||
|
with_super_admin_user: UserModel,
|
||||||
|
) -> None:
|
||||||
|
"""Test_sets_permission_correctly_on_human_task."""
|
||||||
|
self.create_process_group(
|
||||||
|
client, with_super_admin_user, "test_group", "test_group"
|
||||||
|
)
|
||||||
|
initiator_user = self.find_or_create_user("initiator_user")
|
||||||
|
finance_user = self.find_or_create_user("testuser2")
|
||||||
|
assert initiator_user.principal is not None
|
||||||
|
assert finance_user.principal is not None
|
||||||
|
AuthorizationService.import_permissions_from_yaml_file()
|
||||||
|
|
||||||
|
finance_group = GroupModel.query.filter_by(identifier="Finance Team").first()
|
||||||
|
assert finance_group is not None
|
||||||
|
|
||||||
|
process_model = load_test_spec(
|
||||||
|
process_model_id="misc/category_number_one/lanes",
|
||||||
|
bpmn_file_name="lanes.bpmn",
|
||||||
|
process_model_source_directory="model_with_lanes",
|
||||||
|
)
|
||||||
|
process_instance = self.create_process_instance_from_process_model(
|
||||||
|
process_model=process_model, user=initiator_user
|
||||||
|
)
|
||||||
|
processor = ProcessInstanceProcessor(process_instance)
|
||||||
|
processor.do_engine_steps(save=True)
|
||||||
|
|
||||||
|
assert len(process_instance.active_human_tasks) == 1
|
||||||
|
human_task = process_instance.active_human_tasks[0]
|
||||||
|
assert human_task.lane_assignment_id is None
|
||||||
|
assert len(human_task.potential_owners) == 1
|
||||||
|
assert human_task.potential_owners[0] == initiator_user
|
||||||
|
|
||||||
|
spiff_task = processor.__class__.get_task_by_bpmn_identifier(
|
||||||
|
human_task.task_name, processor.bpmn_process_instance
|
||||||
|
)
|
||||||
|
ProcessInstanceService.complete_form_task(
|
||||||
|
processor, spiff_task, {}, initiator_user, human_task
|
||||||
|
)
|
||||||
|
print(f"initiator_user.username: {initiator_user.username}")
|
||||||
|
print(f"data: {processor.get_data()}")
|
||||||
|
print(f"task_data: {spiff_task.data}")
|
||||||
|
assert initiator_user.username == spiff_task.get_data("user_completing_task")["username"]
|
||||||
|
|
||||||
def test_sets_permission_correctly_on_human_task(
|
def test_sets_permission_correctly_on_human_task(
|
||||||
self,
|
self,
|
||||||
app: Flask,
|
app: Flask,
|
||||||
|
Loading…
x
Reference in New Issue
Block a user