From ba634fe29b04391a701819a05ad70f6fef850a47 Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Fri, 21 Apr 2023 15:02:55 +0000 Subject: [PATCH 01/15] Bump codecov/codecov-action from 3.1.0 to 3.1.3 Bumps [codecov/codecov-action](https://github.com/codecov/codecov-action) from 3.1.0 to 3.1.3. - [Release notes](https://github.com/codecov/codecov-action/releases) - [Changelog](https://github.com/codecov/codecov-action/blob/main/CHANGELOG.md) - [Commits](https://github.com/codecov/codecov-action/compare/v3.1.0...v3.1.3) --- updated-dependencies: - dependency-name: codecov/codecov-action dependency-type: direct:production update-type: version-update:semver-patch ... Signed-off-by: dependabot[bot] --- .github/workflows/backend_tests.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/backend_tests.yml b/.github/workflows/backend_tests.yml index 979c37164..66fbb2573 100644 --- a/.github/workflows/backend_tests.yml +++ b/.github/workflows/backend_tests.yml @@ -273,7 +273,7 @@ jobs: nox --force-color --session=coverage -- xml - name: Upload coverage report - uses: codecov/codecov-action@v3.1.0 + uses: codecov/codecov-action@v3.1.3 - name: SonarCloud Scan uses: sonarsource/sonarcloud-github-action@master From f000f47794ca3377b2ed207e8b294faeb90a4af8 Mon Sep 17 00:00:00 2001 From: jasquat Date: Fri, 21 Apr 2023 15:54:31 -0400 Subject: [PATCH 02/15] some updates so task_show no longer needs the processor. i think it mostly works w/ burnettk --- .../src/spiffworkflow_backend/models/task.py | 11 ++ .../routes/tasks_controller.py | 145 ++++++++++------- .../services/authorization_service.py | 10 +- .../services/process_instance_processor.py | 6 +- .../services/process_instance_service.py | 6 +- .../services/task_service.py | 55 +++++-- .../call_activity_level_2b.bpmn | 70 ++++++-- .../call_activity_level_3.bpmn | 8 +- .../call_activity_nested/process_model.json | 11 ++ .../data/call_activity_nested/schema.json | 1 - .../unit/test_process_model.py | 11 -- .../unit/test_process_model_service.py | 4 - .../unit/test_task_service.py | 153 ++++++++++++++++++ .../src/components/InstructionsForEndUser.tsx | 5 +- .../src/routes/HomePageRoutes.tsx | 5 - .../src/routes/ProcessInterstitial.tsx | 7 +- .../src/routes/ProcessRoutes.tsx | 2 +- 17 files changed, 392 insertions(+), 118 deletions(-) create mode 100644 spiffworkflow-backend/tests/data/call_activity_nested/process_model.json delete mode 100644 spiffworkflow-backend/tests/data/call_activity_nested/schema.json create mode 100644 spiffworkflow-backend/tests/spiffworkflow_backend/unit/test_task_service.py diff --git a/spiffworkflow-backend/src/spiffworkflow_backend/models/task.py b/spiffworkflow-backend/src/spiffworkflow_backend/models/task.py index c9bf311b4..3b67883cf 100644 --- a/spiffworkflow-backend/src/spiffworkflow_backend/models/task.py +++ b/spiffworkflow-backend/src/spiffworkflow_backend/models/task.py @@ -69,6 +69,17 @@ class TaskModel(SpiffworkflowBaseDBModel): data: Optional[dict] = None + # these are here to be compatible with task api + form_schema: Optional[dict] = None + form_ui_schema: Optional[dict] = None + process_model_display_name: Optional[str] = None + process_model_identifier: Optional[str] = None + type: Optional[str] = None + can_complete: Optional[bool] = None + + def get_data(self) -> dict: + return {**self.python_env_data(), **self.json_data()} + def python_env_data(self) -> dict: return JsonDataModel.find_data_dict_by_hash(self.python_env_data_hash) diff --git a/spiffworkflow-backend/src/spiffworkflow_backend/routes/tasks_controller.py b/spiffworkflow-backend/src/spiffworkflow_backend/routes/tasks_controller.py index 4adc95364..b1ed17c0f 100644 --- a/spiffworkflow-backend/src/spiffworkflow_backend/routes/tasks_controller.py +++ b/spiffworkflow-backend/src/spiffworkflow_backend/routes/tasks_controller.py @@ -1,5 +1,7 @@ """APIs for dealing with process groups, process models, and process instances.""" import json +from spiffworkflow_backend.services.authorization_service import UserDoesNotHaveAccessToTaskError +from spiffworkflow_backend.services.authorization_service import HumanTaskNotFoundError import os import uuid from sys import exc_info @@ -283,39 +285,67 @@ def task_show(process_instance_id: int, task_guid: str = "next") -> flask.wrappe form_schema_file_name = "" form_ui_schema_file_name = "" + processor = ProcessInstanceProcessor(process_instance) - if task_guid == "next": - spiff_task = processor.next_task() - task_guid = spiff_task.id - else: - spiff_task = _get_spiff_task_from_process_instance(task_guid, process_instance, processor=processor) + spiff_task = _get_spiff_task_from_process_instance(task_guid, process_instance, processor=processor) extensions = spiff_task.task_spec.extensions + task_model = _get_task_model_from_guid_or_raise(task_guid, process_instance_id) + # extensions = task_model.properties_json['extensions'] if 'extensions' in task_model.properties_json else {} + if "properties" in extensions: properties = extensions["properties"] if "formJsonSchemaFilename" in properties: form_schema_file_name = properties["formJsonSchemaFilename"] if "formUiSchemaFilename" in properties: form_ui_schema_file_name = properties["formUiSchemaFilename"] + + can_complete = False + try: + AuthorizationService.assert_user_can_complete_task( + process_instance.id, task_model.task_definition.bpmn_identifier, g.user + ) + can_complete = True + except HumanTaskNotFoundError: + can_complete = False + except UserDoesNotHaveAccessToTaskError: + can_complete = False + task = ProcessInstanceService.spiff_task_to_api_task(processor, spiff_task) task.data = spiff_task.data task.process_model_display_name = process_model.display_name task.process_model_identifier = process_model.id + # task.data + # task.form_schema + # task.form_ui_schema + # task.id + # task.process_model_display_name + # task.process_model_identifier + # task.state + # task.type + + task_model.data = task_model.get_data() + task_model.process_model_display_name = process_model.display_name + task_model.process_model_identifier = process_model.id + task_model.type = task_model.task_definition.typename + task_model.can_complete = can_complete + task_process_identifier = task_model.bpmn_process.bpmn_process_definition.bpmn_identifier + process_model_with_form = process_model refs = SpecFileService.get_references_for_process(process_model_with_form) all_processes = [i.identifier for i in refs] - if task.process_identifier not in all_processes: - top_process_name = processor.find_process_model_process_name_by_task_name(task.process_identifier) + if task_process_identifier not in all_processes: + top_bpmn_process = TaskService.bpmn_process_for_called_activity_or_top_level_process(task_model) bpmn_file_full_path = ProcessInstanceProcessor.bpmn_file_full_path_from_bpmn_process_identifier( - top_process_name + top_bpmn_process.bpmn_process_definition.bpmn_identifier ) relative_path = os.path.relpath(bpmn_file_full_path, start=FileSystemService.root_path()) process_model_relative_path = os.path.dirname(relative_path) process_model_with_form = ProcessModelService.get_process_model_from_relative_path(process_model_relative_path) - if task.type == "User Task": + if task_model.task_definition.typename == "UserTask": if not form_schema_file_name: raise ( ApiError( @@ -330,37 +360,38 @@ def task_show(process_instance_id: int, task_guid: str = "next") -> flask.wrappe form_dict = _prepare_form_data( form_schema_file_name, - spiff_task, + task_model, process_model_with_form, ) - if task.data: - _update_form_schema_with_task_data_as_needed(form_dict, task, spiff_task) + if task_model.data: + _update_form_schema_with_task_data_as_needed(form_dict, task_model) if form_dict: - task.form_schema = form_dict + task_model.form_schema = form_dict if form_ui_schema_file_name: ui_form_contents = _prepare_form_data( form_ui_schema_file_name, - task, + task_model, process_model_with_form, ) if ui_form_contents: - task.form_ui_schema = ui_form_contents + task_model.form_ui_schema = ui_form_contents - _munge_form_ui_schema_based_on_hidden_fields_in_task_data(task) - _render_instructions_for_end_user(spiff_task, task) - return make_response(jsonify(task), 200) + _munge_form_ui_schema_based_on_hidden_fields_in_task_data(task_model) + _render_instructions_for_end_user(task_model) + return make_response(jsonify(task_model), 200) -def _render_instructions_for_end_user(spiff_task: SpiffTask, task: Task) -> str: +def _render_instructions_for_end_user(task_model: TaskModel) -> str: """Assure any instructions for end user are processed for jinja syntax.""" - if task.properties and "instructionsForEndUser" in task.properties: - if task.properties["instructionsForEndUser"]: + extensions = task_model.properties_json['extensions'] if 'extensions' in task_model.properties_json else {} + if extensions and "instructionsForEndUser" in extensions: + if extensions["instructionsForEndUser"]: try: - instructions = _render_jinja_template(task.properties["instructionsForEndUser"], spiff_task) - task.properties["instructionsForEndUser"] = instructions + instructions = _render_jinja_template(extensions["instructionsForEndUser"], task_model) + extensions["instructionsForEndUser"] = instructions return instructions except WorkflowTaskException as wfe: wfe.add_note("Failed to render instructions for end user.") @@ -397,10 +428,12 @@ def _interstitial_stream(process_instance_id: int) -> Generator[str, Optional[st processor = ProcessInstanceProcessor(process_instance) reported_ids = [] # bit of an issue with end tasks showing as getting completed twice. spiff_task = processor.next_task() + task_model = TaskModel.query.filter_by(guid=str(spiff_task.id)).first() last_task = None while last_task != spiff_task: + # import pdb; pdb.set_trace() task = ProcessInstanceService.spiff_task_to_api_task(processor, processor.next_task()) - instructions = _render_instructions_for_end_user(spiff_task, task) + instructions = _render_instructions_for_end_user(task_model) if instructions and spiff_task.id not in reported_ids: reported_ids.append(spiff_task.id) yield f"data: {current_app.json.dumps(task)} \n\n" @@ -408,6 +441,9 @@ def _interstitial_stream(process_instance_id: int) -> Generator[str, Optional[st processor.do_engine_steps(execution_strategy_name="run_until_user_message") processor.do_engine_steps(execution_strategy_name="one_at_a_time") spiff_task = processor.next_task() + task_model = TaskModel.query.filter_by(guid=str(spiff_task.id)).first() + # print(f"spiff_task: {spiff_task}") + # print(f"last_task: {last_task}") # Note, this has to be done in case someone leaves the page, # which can otherwise cancel this function and leave completed tasks un-registered. processor.save() # Fixme - maybe find a way not to do this on every loop? @@ -446,7 +482,7 @@ def _task_submit_shared( processor = ProcessInstanceProcessor(process_instance) spiff_task = _get_spiff_task_from_process_instance(task_guid, process_instance, processor=processor) - AuthorizationService.assert_user_can_complete_spiff_task(process_instance.id, spiff_task, principal.user) + AuthorizationService.assert_user_can_complete_task(process_instance.id, spiff_task.task_spec.name, principal.user) if spiff_task.state != TaskState.READY: raise ( @@ -634,14 +670,14 @@ def _get_tasks( return make_response(jsonify(response_json), 200) -def _prepare_form_data(form_file: str, spiff_task: SpiffTask, process_model: ProcessModelInfo) -> dict: +def _prepare_form_data(form_file: str, task_model: TaskModel, process_model: ProcessModelInfo) -> dict: """Prepare_form_data.""" - if spiff_task.data is None: + if task_model.data is None: return {} file_contents = SpecFileService.get_data(process_model, form_file).decode("utf-8") try: - form_contents = _render_jinja_template(file_contents, spiff_task) + form_contents = _render_jinja_template(file_contents, task_model) try: # form_contents is a str hot_dict: dict = json.loads(form_contents) @@ -661,14 +697,14 @@ def _prepare_form_data(form_file: str, spiff_task: SpiffTask, process_model: Pro raise api_error -def _render_jinja_template(unprocessed_template: str, spiff_task: SpiffTask) -> str: +def _render_jinja_template(unprocessed_template: str, task_model: TaskModel) -> str: """Render_jinja_template.""" jinja_environment = jinja2.Environment(autoescape=True, lstrip_blocks=True, trim_blocks=True) try: template = jinja_environment.from_string(unprocessed_template) - return template.render(**spiff_task.data) + return template.render(**(task_model.data or {})) except jinja2.exceptions.TemplateError as template_error: - wfe = WorkflowTaskException(str(template_error), task=spiff_task, exception=template_error) + wfe = WorkflowTaskException(str(template_error), task=task_model, exception=template_error) if isinstance(template_error, TemplateSyntaxError): wfe.line_number = template_error.lineno wfe.error_line = template_error.source.split("\n")[template_error.lineno - 1] @@ -676,7 +712,7 @@ def _render_jinja_template(unprocessed_template: str, spiff_task: SpiffTask) -> raise wfe from template_error except Exception as error: _type, _value, tb = exc_info() - wfe = WorkflowTaskException(str(error), task=spiff_task, exception=error) + wfe = WorkflowTaskException(str(error), task=task_model, exception=error) while tb: if tb.tb_frame.f_code.co_filename == "