wip for get_last_user_completing_task script task

This commit is contained in:
burnettk 2023-02-25 23:31:23 -05:00
parent f7dc076f75
commit e9b3ababc1
6 changed files with 136 additions and 8 deletions

View 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 ###

View File

@ -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")

View File

@ -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)

View File

@ -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),

View File

@ -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>

View File

@ -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,