Merge branch 'main' of github.com:sartography/spiff-arena into send_filters
This commit is contained in:
commit
1584f52a18
41
bin/run_pyl
41
bin/run_pyl
|
@ -12,25 +12,54 @@ python_projects=(
|
||||||
spiffworkflow-backend
|
spiffworkflow-backend
|
||||||
)
|
)
|
||||||
|
|
||||||
function run_fix_docstrings() {
|
react_projects=(
|
||||||
fix_python_docstrings $(get_top_level_directories_containing_python_files)
|
spiffworkflow-frontend
|
||||||
|
)
|
||||||
|
|
||||||
|
function get_python_dirs() {
|
||||||
|
(git ls-tree -r HEAD --name-only | grep -E '\.py$' | awk -F '/' '{print $1}' | sort | uniq | grep -v '\.' | grep -Ev '^(bin|migrations)$') || echo ''
|
||||||
|
}
|
||||||
|
|
||||||
|
function run_autoflake() {
|
||||||
|
if ! command -v autoflake8 >/dev/null ; then
|
||||||
|
pip install autoflake8
|
||||||
|
asdf reshim python
|
||||||
|
fi
|
||||||
|
|
||||||
|
if ! command -v autopep8 >/dev/null ; then
|
||||||
|
pip install autopep8
|
||||||
|
asdf reshim python
|
||||||
|
fi
|
||||||
|
|
||||||
|
python_dirs=$(get_python_dirs)
|
||||||
|
python_files=$(find $python_dirs -type f -name "*.py" ! -name '.null-ls*' ! -name '_null-ls*')
|
||||||
|
|
||||||
|
autoflake8 --in-place --remove-unused-variables --remove-duplicate-keys --expand-star-imports --exit-zero-even-if-changed $python_files
|
||||||
|
autoflake --in-place --remove-all-unused-imports $python_files
|
||||||
|
autopep8 --in-place $python_files
|
||||||
}
|
}
|
||||||
|
|
||||||
function run_pre_commmit() {
|
function run_pre_commmit() {
|
||||||
poetry run pre-commit run --verbose --all-files
|
poetry run pre-commit run --verbose --all-files
|
||||||
}
|
}
|
||||||
|
|
||||||
|
for react_project in "${react_projects[@]}" ; do
|
||||||
|
pushd "$react_project"
|
||||||
|
npm run lint:fix
|
||||||
|
popd
|
||||||
|
done
|
||||||
|
|
||||||
for python_project in "${python_projects[@]}" ; do
|
for python_project in "${python_projects[@]}" ; do
|
||||||
pushd "$python_project"
|
pushd "$python_project"
|
||||||
run_fix_docstrings || run_fix_docstrings
|
run_autoflake || run_autoflake
|
||||||
popd
|
popd
|
||||||
done
|
done
|
||||||
run_pre_commmit || run_pre_commmit
|
run_pre_commmit || run_pre_commmit
|
||||||
|
|
||||||
for python_project in "${python_projects[@]}"; do
|
for python_project in "${python_projects[@]}"; do
|
||||||
pushd "$python_project"
|
pushd "$python_project"
|
||||||
poet i
|
poetry install
|
||||||
poet mypy
|
poetry run mypy $(get_python_dirs)
|
||||||
poet test
|
poetry run coverage run --parallel -m pytest
|
||||||
popd
|
popd
|
||||||
done
|
done
|
||||||
|
|
|
@ -98,7 +98,7 @@ python-versions = ">=3.5"
|
||||||
dev = ["cloudpickle", "coverage[toml] (>=5.0.2)", "furo", "hypothesis", "mypy (>=0.900,!=0.940)", "pre-commit", "pympler", "pytest (>=4.3.0)", "pytest-mypy-plugins", "sphinx", "sphinx-notfound-page", "zope.interface"]
|
dev = ["cloudpickle", "coverage[toml] (>=5.0.2)", "furo", "hypothesis", "mypy (>=0.900,!=0.940)", "pre-commit", "pympler", "pytest (>=4.3.0)", "pytest-mypy-plugins", "sphinx", "sphinx-notfound-page", "zope.interface"]
|
||||||
docs = ["furo", "sphinx", "sphinx-notfound-page", "zope.interface"]
|
docs = ["furo", "sphinx", "sphinx-notfound-page", "zope.interface"]
|
||||||
tests = ["cloudpickle", "coverage[toml] (>=5.0.2)", "hypothesis", "mypy (>=0.900,!=0.940)", "pympler", "pytest (>=4.3.0)", "pytest-mypy-plugins", "zope.interface"]
|
tests = ["cloudpickle", "coverage[toml] (>=5.0.2)", "hypothesis", "mypy (>=0.900,!=0.940)", "pympler", "pytest (>=4.3.0)", "pytest-mypy-plugins", "zope.interface"]
|
||||||
tests_no_zope = ["cloudpickle", "coverage[toml] (>=5.0.2)", "hypothesis", "mypy (>=0.900,!=0.940)", "pympler", "pytest (>=4.3.0)", "pytest-mypy-plugins"]
|
tests-no-zope = ["cloudpickle", "coverage[toml] (>=5.0.2)", "hypothesis", "mypy (>=0.900,!=0.940)", "pympler", "pytest (>=4.3.0)", "pytest-mypy-plugins"]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "Babel"
|
name = "Babel"
|
||||||
|
@ -271,7 +271,7 @@ optional = false
|
||||||
python-versions = ">=3.6.0"
|
python-versions = ">=3.6.0"
|
||||||
|
|
||||||
[package.extras]
|
[package.extras]
|
||||||
unicode_backport = ["unicodedata2"]
|
unicode-backport = ["unicodedata2"]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "classify-imports"
|
name = "classify-imports"
|
||||||
|
@ -643,7 +643,7 @@ werkzeug = "*"
|
||||||
type = "git"
|
type = "git"
|
||||||
url = "https://github.com/sartography/flask-bpmn"
|
url = "https://github.com/sartography/flask-bpmn"
|
||||||
reference = "main"
|
reference = "main"
|
||||||
resolved_reference = "6f6762ec83bb6eec24f7cc799d4d5fa7867c7474"
|
resolved_reference = "860f2387bebdaa9220e9fbf6f8fa7f74e805d0d4"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "Flask-Cors"
|
name = "Flask-Cors"
|
||||||
|
@ -1517,7 +1517,7 @@ urllib3 = ">=1.21.1,<1.27"
|
||||||
|
|
||||||
[package.extras]
|
[package.extras]
|
||||||
socks = ["PySocks (>=1.5.6,!=1.5.7)"]
|
socks = ["PySocks (>=1.5.6,!=1.5.7)"]
|
||||||
use_chardet_on_py3 = ["chardet (>=3.0.2,<6)"]
|
use-chardet-on-py3 = ["chardet (>=3.0.2,<6)"]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "requests-toolbelt"
|
name = "requests-toolbelt"
|
||||||
|
@ -1630,7 +1630,7 @@ falcon = ["falcon (>=1.4)"]
|
||||||
fastapi = ["fastapi (>=0.79.0)"]
|
fastapi = ["fastapi (>=0.79.0)"]
|
||||||
flask = ["blinker (>=1.1)", "flask (>=0.11)"]
|
flask = ["blinker (>=1.1)", "flask (>=0.11)"]
|
||||||
httpx = ["httpx (>=0.16.0)"]
|
httpx = ["httpx (>=0.16.0)"]
|
||||||
pure_eval = ["asttokens", "executing", "pure-eval"]
|
pure-eval = ["asttokens", "executing", "pure-eval"]
|
||||||
pyspark = ["pyspark (>=2.4.4)"]
|
pyspark = ["pyspark (>=2.4.4)"]
|
||||||
quart = ["blinker (>=1.1)", "quart (>=0.16.1)"]
|
quart = ["blinker (>=1.1)", "quart (>=0.16.1)"]
|
||||||
rq = ["rq (>=0.6)"]
|
rq = ["rq (>=0.6)"]
|
||||||
|
@ -1876,7 +1876,7 @@ lxml = "*"
|
||||||
type = "git"
|
type = "git"
|
||||||
url = "https://github.com/sartography/SpiffWorkflow"
|
url = "https://github.com/sartography/SpiffWorkflow"
|
||||||
reference = "main"
|
reference = "main"
|
||||||
resolved_reference = "025bc30f27366e06dd1286b7563e4b1cb04c1c46"
|
resolved_reference = "eea53c912984d21a064330c3b3334ac219cb8e18"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "SQLAlchemy"
|
name = "SQLAlchemy"
|
||||||
|
@ -1894,19 +1894,19 @@ aiomysql = ["aiomysql", "greenlet (!=0.4.17)"]
|
||||||
aiosqlite = ["aiosqlite", "greenlet (!=0.4.17)", "typing_extensions (!=3.10.0.1)"]
|
aiosqlite = ["aiosqlite", "greenlet (!=0.4.17)", "typing_extensions (!=3.10.0.1)"]
|
||||||
asyncio = ["greenlet (!=0.4.17)"]
|
asyncio = ["greenlet (!=0.4.17)"]
|
||||||
asyncmy = ["asyncmy (>=0.2.3,!=0.2.4)", "greenlet (!=0.4.17)"]
|
asyncmy = ["asyncmy (>=0.2.3,!=0.2.4)", "greenlet (!=0.4.17)"]
|
||||||
mariadb_connector = ["mariadb (>=1.0.1,!=1.1.2)"]
|
mariadb-connector = ["mariadb (>=1.0.1,!=1.1.2)"]
|
||||||
mssql = ["pyodbc"]
|
mssql = ["pyodbc"]
|
||||||
mssql_pymssql = ["pymssql"]
|
mssql-pymssql = ["pymssql"]
|
||||||
mssql_pyodbc = ["pyodbc"]
|
mssql-pyodbc = ["pyodbc"]
|
||||||
mypy = ["mypy (>=0.910)", "sqlalchemy2-stubs"]
|
mypy = ["mypy (>=0.910)", "sqlalchemy2-stubs"]
|
||||||
mysql = ["mysqlclient (>=1.4.0)", "mysqlclient (>=1.4.0,<2)"]
|
mysql = ["mysqlclient (>=1.4.0)", "mysqlclient (>=1.4.0,<2)"]
|
||||||
mysql_connector = ["mysql-connector-python"]
|
mysql-connector = ["mysql-connector-python"]
|
||||||
oracle = ["cx_oracle (>=7)", "cx_oracle (>=7,<8)"]
|
oracle = ["cx_oracle (>=7)", "cx_oracle (>=7,<8)"]
|
||||||
postgresql = ["psycopg2 (>=2.7)"]
|
postgresql = ["psycopg2 (>=2.7)"]
|
||||||
postgresql_asyncpg = ["asyncpg", "greenlet (!=0.4.17)"]
|
postgresql-asyncpg = ["asyncpg", "greenlet (!=0.4.17)"]
|
||||||
postgresql_pg8000 = ["pg8000 (>=1.16.6,!=1.29.0)"]
|
postgresql-pg8000 = ["pg8000 (>=1.16.6,!=1.29.0)"]
|
||||||
postgresql_psycopg2binary = ["psycopg2-binary"]
|
postgresql-psycopg2binary = ["psycopg2-binary"]
|
||||||
postgresql_psycopg2cffi = ["psycopg2cffi"]
|
postgresql-psycopg2cffi = ["psycopg2cffi"]
|
||||||
pymysql = ["pymysql", "pymysql (<1)"]
|
pymysql = ["pymysql", "pymysql (<1)"]
|
||||||
sqlcipher = ["sqlcipher3_binary"]
|
sqlcipher = ["sqlcipher3_binary"]
|
||||||
|
|
||||||
|
@ -3056,7 +3056,18 @@ py = [
|
||||||
{file = "py-1.11.0.tar.gz", hash = "sha256:51c75c4126074b472f746a24399ad32f6053d1b34b68d2fa41e558e6f4a98719"},
|
{file = "py-1.11.0.tar.gz", hash = "sha256:51c75c4126074b472f746a24399ad32f6053d1b34b68d2fa41e558e6f4a98719"},
|
||||||
]
|
]
|
||||||
pyasn1 = [
|
pyasn1 = [
|
||||||
|
{file = "pyasn1-0.4.8-py2.4.egg", hash = "sha256:fec3e9d8e36808a28efb59b489e4528c10ad0f480e57dcc32b4de5c9d8c9fdf3"},
|
||||||
|
{file = "pyasn1-0.4.8-py2.5.egg", hash = "sha256:0458773cfe65b153891ac249bcf1b5f8f320b7c2ce462151f8fa74de8934becf"},
|
||||||
|
{file = "pyasn1-0.4.8-py2.6.egg", hash = "sha256:5c9414dcfede6e441f7e8f81b43b34e834731003427e5b09e4e00e3172a10f00"},
|
||||||
|
{file = "pyasn1-0.4.8-py2.7.egg", hash = "sha256:6e7545f1a61025a4e58bb336952c5061697da694db1cae97b116e9c46abcf7c8"},
|
||||||
{file = "pyasn1-0.4.8-py2.py3-none-any.whl", hash = "sha256:39c7e2ec30515947ff4e87fb6f456dfc6e84857d34be479c9d4a4ba4bf46aa5d"},
|
{file = "pyasn1-0.4.8-py2.py3-none-any.whl", hash = "sha256:39c7e2ec30515947ff4e87fb6f456dfc6e84857d34be479c9d4a4ba4bf46aa5d"},
|
||||||
|
{file = "pyasn1-0.4.8-py3.1.egg", hash = "sha256:78fa6da68ed2727915c4767bb386ab32cdba863caa7dbe473eaae45f9959da86"},
|
||||||
|
{file = "pyasn1-0.4.8-py3.2.egg", hash = "sha256:08c3c53b75eaa48d71cf8c710312316392ed40899cb34710d092e96745a358b7"},
|
||||||
|
{file = "pyasn1-0.4.8-py3.3.egg", hash = "sha256:03840c999ba71680a131cfaee6fab142e1ed9bbd9c693e285cc6aca0d555e576"},
|
||||||
|
{file = "pyasn1-0.4.8-py3.4.egg", hash = "sha256:7ab8a544af125fb704feadb008c99a88805126fb525280b2270bb25cc1d78a12"},
|
||||||
|
{file = "pyasn1-0.4.8-py3.5.egg", hash = "sha256:e89bf84b5437b532b0803ba5c9a5e054d21fec423a89952a74f87fa2c9b7bce2"},
|
||||||
|
{file = "pyasn1-0.4.8-py3.6.egg", hash = "sha256:014c0e9976956a08139dc0712ae195324a75e142284d5f87f1a87ee1b068a359"},
|
||||||
|
{file = "pyasn1-0.4.8-py3.7.egg", hash = "sha256:99fcc3c8d804d1bc6d9a099921e39d827026409a58f2a720dcdb89374ea0c776"},
|
||||||
{file = "pyasn1-0.4.8.tar.gz", hash = "sha256:aef77c9fb94a3ac588e87841208bdec464471d9871bd5050a287cc9a475cd0ba"},
|
{file = "pyasn1-0.4.8.tar.gz", hash = "sha256:aef77c9fb94a3ac588e87841208bdec464471d9871bd5050a287cc9a475cd0ba"},
|
||||||
]
|
]
|
||||||
pycodestyle = [
|
pycodestyle = [
|
||||||
|
|
|
@ -258,7 +258,6 @@ paths:
|
||||||
description: The number of models to show per page. Defaults to page 10.
|
description: The number of models to show per page. Defaults to page 10.
|
||||||
schema:
|
schema:
|
||||||
type: integer
|
type: integer
|
||||||
# process_model_list
|
|
||||||
get:
|
get:
|
||||||
operationId: spiffworkflow_backend.routes.process_api_blueprint.process_model_list
|
operationId: spiffworkflow_backend.routes.process_api_blueprint.process_model_list
|
||||||
summary: Return a list of process models for a given process group
|
summary: Return a list of process models for a given process group
|
||||||
|
@ -273,9 +272,10 @@ paths:
|
||||||
type: array
|
type: array
|
||||||
items:
|
items:
|
||||||
$ref: "#/components/schemas/ProcessModel"
|
$ref: "#/components/schemas/ProcessModel"
|
||||||
# process_model_add
|
|
||||||
|
/process-models/{modified_process_group_id}:
|
||||||
post:
|
post:
|
||||||
operationId: spiffworkflow_backend.routes.process_api_blueprint.process_model_add
|
operationId: spiffworkflow_backend.routes.process_api_blueprint.process_model_create
|
||||||
summary: Creates a new process model with the given parameters.
|
summary: Creates a new process model with the given parameters.
|
||||||
tags:
|
tags:
|
||||||
- Process Models
|
- Process Models
|
||||||
|
@ -371,7 +371,7 @@ paths:
|
||||||
application/json:
|
application/json:
|
||||||
schema:
|
schema:
|
||||||
$ref: "#/components/schemas/OkTrue"
|
$ref: "#/components/schemas/OkTrue"
|
||||||
# process_model_list
|
|
||||||
/processes:
|
/processes:
|
||||||
get:
|
get:
|
||||||
operationId: spiffworkflow_backend.routes.process_api_blueprint.process_list
|
operationId: spiffworkflow_backend.routes.process_api_blueprint.process_list
|
||||||
|
@ -976,7 +976,7 @@ paths:
|
||||||
items:
|
items:
|
||||||
$ref: "#/components/schemas/Task"
|
$ref: "#/components/schemas/Task"
|
||||||
|
|
||||||
/process-instance/{process_instance_id}/tasks:
|
/process-instances/{modified_process_model_id}/{process_instance_id}/tasks:
|
||||||
parameters:
|
parameters:
|
||||||
- name: process_instance_id
|
- name: process_instance_id
|
||||||
in: path
|
in: path
|
||||||
|
|
|
@ -63,6 +63,12 @@ permissions:
|
||||||
allowed_permissions: [read]
|
allowed_permissions: [read]
|
||||||
uri: /v1.0/process-groups/*
|
uri: /v1.0/process-groups/*
|
||||||
|
|
||||||
|
process-instance-list:
|
||||||
|
groups: [everybody]
|
||||||
|
users: []
|
||||||
|
allowed_permissions: [read]
|
||||||
|
uri: /v1.0/process-instances
|
||||||
|
|
||||||
# TODO: all uris should really have the same structure
|
# TODO: all uris should really have the same structure
|
||||||
finance-admin-group:
|
finance-admin-group:
|
||||||
groups: ["Finance Team"]
|
groups: ["Finance Team"]
|
||||||
|
@ -81,3 +87,9 @@ permissions:
|
||||||
users: []
|
users: []
|
||||||
allowed_permissions: [read]
|
allowed_permissions: [read]
|
||||||
uri: /*
|
uri: /*
|
||||||
|
|
||||||
|
invoice-approval-tasks-read:
|
||||||
|
groups: ["Finance Team"]
|
||||||
|
users: []
|
||||||
|
allowed_permissions: [read]
|
||||||
|
uri: /v1.0/process-instances/category_number_one:lanes/*
|
||||||
|
|
|
@ -47,7 +47,7 @@ class SpecReferenceCache(SpiffworkflowBaseDBModel):
|
||||||
file_name = db.Column(db.String(255))
|
file_name = db.Column(db.String(255))
|
||||||
relative_path = db.Column(db.String(255))
|
relative_path = db.Column(db.String(255))
|
||||||
has_lanes = db.Column(db.Boolean())
|
has_lanes = db.Column(db.Boolean())
|
||||||
is_executable = db.Column(db.Boolean()) # either 'process' or 'decision'
|
is_executable = db.Column(db.Boolean())
|
||||||
is_primary = db.Column(db.Boolean())
|
is_primary = db.Column(db.Boolean())
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
|
|
|
@ -232,11 +232,17 @@ def process_group_show(
|
||||||
return make_response(jsonify(process_group), 200)
|
return make_response(jsonify(process_group), 200)
|
||||||
|
|
||||||
|
|
||||||
def process_model_add(
|
def process_model_create(
|
||||||
body: Dict[str, Union[str, bool, int]]
|
modified_process_group_id: str, body: Dict[str, Union[str, bool, int]]
|
||||||
) -> flask.wrappers.Response:
|
) -> flask.wrappers.Response:
|
||||||
"""Add_process_model."""
|
"""Process_model_create."""
|
||||||
process_model_info = ProcessModelInfoSchema().load(body)
|
process_model_info = ProcessModelInfoSchema().load(body)
|
||||||
|
if modified_process_group_id is None:
|
||||||
|
raise ApiError(
|
||||||
|
error_code="process_group_id_not_specified",
|
||||||
|
message="Process Model could not be created when process_group_id path param is unspecified",
|
||||||
|
status_code=400,
|
||||||
|
)
|
||||||
if process_model_info is None:
|
if process_model_info is None:
|
||||||
raise ApiError(
|
raise ApiError(
|
||||||
error_code="process_model_could_not_be_created",
|
error_code="process_model_could_not_be_created",
|
||||||
|
@ -1136,7 +1142,10 @@ def get_tasks(
|
||||||
|
|
||||||
|
|
||||||
def process_instance_task_list(
|
def process_instance_task_list(
|
||||||
process_instance_id: int, all_tasks: bool = False, spiff_step: int = 0
|
modified_process_model_id: str,
|
||||||
|
process_instance_id: int,
|
||||||
|
all_tasks: bool = False,
|
||||||
|
spiff_step: int = 0,
|
||||||
) -> flask.wrappers.Response:
|
) -> flask.wrappers.Response:
|
||||||
"""Process_instance_task_list."""
|
"""Process_instance_task_list."""
|
||||||
process_instance = find_process_instance_by_id_or_raise(process_instance_id)
|
process_instance = find_process_instance_by_id_or_raise(process_instance_id)
|
||||||
|
@ -1199,6 +1208,7 @@ def task_show(process_instance_id: int, task_id: str) -> flask.wrappers.Response
|
||||||
task = ProcessInstanceService.spiff_task_to_api_task(spiff_task)
|
task = ProcessInstanceService.spiff_task_to_api_task(spiff_task)
|
||||||
task.data = spiff_task.data
|
task.data = spiff_task.data
|
||||||
task.process_model_display_name = process_model.display_name
|
task.process_model_display_name = process_model.display_name
|
||||||
|
task.process_model_identifier = process_model.id
|
||||||
process_model_with_form = process_model
|
process_model_with_form = process_model
|
||||||
|
|
||||||
if task.type == "User Task":
|
if task.type == "User Task":
|
||||||
|
|
|
@ -54,7 +54,8 @@ class FileSystemService:
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def process_group_path_for_spec(spec: ProcessModelInfo) -> str:
|
def process_group_path_for_spec(spec: ProcessModelInfo) -> str:
|
||||||
"""Category_path_for_spec."""
|
"""Category_path_for_spec."""
|
||||||
process_group_id, _ = os.path.split(spec.id)
|
# os.path.split apparently returns 2 element tulple like: (first/path, last_item)
|
||||||
|
process_group_id, _ = os.path.split(spec.id_for_file_path())
|
||||||
return FileSystemService.process_group_path(process_group_id)
|
return FileSystemService.process_group_path(process_group_id)
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
|
|
|
@ -701,9 +701,11 @@ class ProcessInstanceProcessor:
|
||||||
"bpmn_file_full_path_from_bpmn_process_identifier: bpmn_process_identifier is unexpectedly None"
|
"bpmn_file_full_path_from_bpmn_process_identifier: bpmn_process_identifier is unexpectedly None"
|
||||||
)
|
)
|
||||||
|
|
||||||
spec_reference = SpecReferenceCache.query.filter_by(
|
spec_reference = (
|
||||||
identifier=bpmn_process_identifier
|
SpecReferenceCache.query.filter_by(identifier=bpmn_process_identifier)
|
||||||
).first()
|
.filter_by(type="process")
|
||||||
|
.first()
|
||||||
|
)
|
||||||
bpmn_file_full_path = None
|
bpmn_file_full_path = None
|
||||||
if spec_reference is None:
|
if spec_reference is None:
|
||||||
bpmn_file_full_path = (
|
bpmn_file_full_path = (
|
||||||
|
|
|
@ -45,7 +45,9 @@ class SpecFileService(FileSystemService):
|
||||||
) -> List[File]:
|
) -> List[File]:
|
||||||
"""Return all files associated with a workflow specification."""
|
"""Return all files associated with a workflow specification."""
|
||||||
# path = SpecFileService.workflow_path(process_model_info)
|
# path = SpecFileService.workflow_path(process_model_info)
|
||||||
path = os.path.join(FileSystemService.root_path(), process_model_info.id)
|
path = os.path.join(
|
||||||
|
FileSystemService.root_path(), process_model_info.id_for_file_path()
|
||||||
|
)
|
||||||
files = SpecFileService._get_files(path, file_name)
|
files = SpecFileService._get_files(path, file_name)
|
||||||
if extension_filter != "":
|
if extension_filter != "":
|
||||||
files = list(
|
files = list(
|
||||||
|
@ -88,7 +90,7 @@ class SpecFileService(FileSystemService):
|
||||||
"""
|
"""
|
||||||
references: list[SpecReference] = []
|
references: list[SpecReference] = []
|
||||||
full_file_path = SpecFileService.full_file_path(process_model_info, file.name)
|
full_file_path = SpecFileService.full_file_path(process_model_info, file.name)
|
||||||
file_path = os.path.join(process_model_info.id, file.name)
|
file_path = os.path.join(process_model_info.id_for_file_path(), file.name)
|
||||||
parser = MyCustomParser()
|
parser = MyCustomParser()
|
||||||
parser_type = None
|
parser_type = None
|
||||||
sub_parser = None
|
sub_parser = None
|
||||||
|
@ -160,6 +162,8 @@ class SpecFileService(FileSystemService):
|
||||||
(ref for ref in references if ref.is_primary and ref.is_executable), None
|
(ref for ref in references if ref.is_primary and ref.is_executable), None
|
||||||
)
|
)
|
||||||
|
|
||||||
|
SpecFileService.clear_caches_for_file(file_name, process_model_info)
|
||||||
|
|
||||||
for ref in references:
|
for ref in references:
|
||||||
# If no valid primary process is defined, default to the first process in the
|
# If no valid primary process is defined, default to the first process in the
|
||||||
# updated file.
|
# updated file.
|
||||||
|
@ -235,6 +239,16 @@ class SpecFileService(FileSystemService):
|
||||||
SpecFileService.update_message_trigger_cache(ref)
|
SpecFileService.update_message_trigger_cache(ref)
|
||||||
SpecFileService.update_correlation_cache(ref)
|
SpecFileService.update_correlation_cache(ref)
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def clear_caches_for_file(
|
||||||
|
file_name: str, process_model_info: ProcessModelInfo
|
||||||
|
) -> None:
|
||||||
|
"""Clear all caches related to a file."""
|
||||||
|
db.session.query(SpecReferenceCache).filter(
|
||||||
|
SpecReferenceCache.file_name == file_name
|
||||||
|
).filter(SpecReferenceCache.process_model_id == process_model_info.id).delete()
|
||||||
|
# fixme: likely the other caches should be cleared as well, but we don't have a clean way to do so yet.
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def clear_caches() -> None:
|
def clear_caches() -> None:
|
||||||
"""Clear_caches."""
|
"""Clear_caches."""
|
||||||
|
@ -254,6 +268,7 @@ class SpecFileService(FileSystemService):
|
||||||
if process_id_lookup is None:
|
if process_id_lookup is None:
|
||||||
process_id_lookup = SpecReferenceCache.from_spec_reference(ref)
|
process_id_lookup = SpecReferenceCache.from_spec_reference(ref)
|
||||||
db.session.add(process_id_lookup)
|
db.session.add(process_id_lookup)
|
||||||
|
db.session.commit()
|
||||||
else:
|
else:
|
||||||
if ref.relative_path != process_id_lookup.relative_path:
|
if ref.relative_path != process_id_lookup.relative_path:
|
||||||
full_bpmn_file_path = SpecFileService.full_path_from_relative_path(
|
full_bpmn_file_path = SpecFileService.full_path_from_relative_path(
|
||||||
|
|
|
@ -136,6 +136,7 @@ class BaseTest:
|
||||||
|
|
||||||
# make sure we have a group
|
# make sure we have a group
|
||||||
process_group_id, _ = os.path.split(process_model_id)
|
process_group_id, _ = os.path.split(process_model_id)
|
||||||
|
modified_process_group_id = process_group_id.replace("/", ":")
|
||||||
process_group_path = f"{FileSystemService.root_path()}/{process_group_id}"
|
process_group_path = f"{FileSystemService.root_path()}/{process_group_id}"
|
||||||
if ProcessModelService().is_group(process_group_path):
|
if ProcessModelService().is_group(process_group_path):
|
||||||
|
|
||||||
|
@ -156,11 +157,12 @@ class BaseTest:
|
||||||
user = self.find_or_create_user()
|
user = self.find_or_create_user()
|
||||||
|
|
||||||
response = client.post(
|
response = client.post(
|
||||||
"/v1.0/process-models",
|
f"/v1.0/process-models/{modified_process_group_id}",
|
||||||
content_type="application/json",
|
content_type="application/json",
|
||||||
data=json.dumps(ProcessModelInfoSchema().dump(model)),
|
data=json.dumps(ProcessModelInfoSchema().dump(model)),
|
||||||
headers=self.logged_in_headers(user),
|
headers=self.logged_in_headers(user),
|
||||||
)
|
)
|
||||||
|
|
||||||
assert response.status_code == 201
|
assert response.status_code == 201
|
||||||
return response
|
return response
|
||||||
|
|
||||||
|
|
|
@ -143,7 +143,6 @@ class TestNestedGroups(BaseTest):
|
||||||
response = client.get( # noqa: F841
|
response = client.get( # noqa: F841
|
||||||
target_uri, headers=self.logged_in_headers(user)
|
target_uri, headers=self.logged_in_headers(user)
|
||||||
)
|
)
|
||||||
print("test_nested_groups")
|
|
||||||
|
|
||||||
def test_add_nested_group(
|
def test_add_nested_group(
|
||||||
self,
|
self,
|
||||||
|
@ -153,10 +152,6 @@ class TestNestedGroups(BaseTest):
|
||||||
with_super_admin_user: UserModel,
|
with_super_admin_user: UserModel,
|
||||||
) -> None:
|
) -> None:
|
||||||
"""Test_add_nested_group."""
|
"""Test_add_nested_group."""
|
||||||
# user = self.find_or_create_user()
|
|
||||||
# self.add_permissions_to_user(
|
|
||||||
# user, target_uri=target_uri, permission_names=["read", "create"]
|
|
||||||
# )
|
|
||||||
process_group_a = ProcessGroup(
|
process_group_a = ProcessGroup(
|
||||||
id="group_a",
|
id="group_a",
|
||||||
display_name="Group A",
|
display_name="Group A",
|
||||||
|
@ -194,16 +189,14 @@ class TestNestedGroups(BaseTest):
|
||||||
data=json.dumps(ProcessGroupSchema().dump(process_group_c)),
|
data=json.dumps(ProcessGroupSchema().dump(process_group_c)),
|
||||||
)
|
)
|
||||||
|
|
||||||
print("test_add_nested_group")
|
def test_process_model_create(
|
||||||
|
|
||||||
def test_process_model_add(
|
|
||||||
self,
|
self,
|
||||||
app: Flask,
|
app: Flask,
|
||||||
client: FlaskClient,
|
client: FlaskClient,
|
||||||
with_db_and_bpmn_file_cleanup: None,
|
with_db_and_bpmn_file_cleanup: None,
|
||||||
with_super_admin_user: UserModel,
|
with_super_admin_user: UserModel,
|
||||||
) -> None:
|
) -> None:
|
||||||
"""Test_process_model_add."""
|
"""Test_process_model_create."""
|
||||||
process_group_a = ProcessGroup(
|
process_group_a = ProcessGroup(
|
||||||
id="group_a",
|
id="group_a",
|
||||||
display_name="Group A",
|
display_name="Group A",
|
||||||
|
@ -242,7 +235,6 @@ class TestNestedGroups(BaseTest):
|
||||||
content_type="application/json",
|
content_type="application/json",
|
||||||
data=json.dumps(ProcessModelInfoSchema().dump(process_model)),
|
data=json.dumps(ProcessModelInfoSchema().dump(process_model)),
|
||||||
)
|
)
|
||||||
print("test_process_model_add")
|
|
||||||
|
|
||||||
def test_process_group_show(
|
def test_process_group_show(
|
||||||
self,
|
self,
|
||||||
|
|
|
@ -105,14 +105,14 @@ class TestProcessApi(BaseTest):
|
||||||
assert response.json is not None
|
assert response.json is not None
|
||||||
assert response.json == expected_response_body
|
assert response.json == expected_response_body
|
||||||
|
|
||||||
def test_process_model_add(
|
def test_process_model_create(
|
||||||
self,
|
self,
|
||||||
app: Flask,
|
app: Flask,
|
||||||
client: FlaskClient,
|
client: FlaskClient,
|
||||||
with_db_and_bpmn_file_cleanup: None,
|
with_db_and_bpmn_file_cleanup: None,
|
||||||
with_super_admin_user: UserModel,
|
with_super_admin_user: UserModel,
|
||||||
) -> None:
|
) -> None:
|
||||||
"""Test_add_new_process_model."""
|
"""Test_process_model_create."""
|
||||||
process_group_id = "test_process_group"
|
process_group_id = "test_process_group"
|
||||||
process_group_display_name = "Test Process Group"
|
process_group_display_name = "Test Process Group"
|
||||||
# creates the group directory, and the json file
|
# creates the group directory, and the json file
|
||||||
|
|
|
@ -119,6 +119,43 @@ class TestSpecFileService(BaseTest):
|
||||||
== self.call_activity_nested_relative_file_path
|
== self.call_activity_nested_relative_file_path
|
||||||
)
|
)
|
||||||
|
|
||||||
|
def test_change_the_identifier_cleans_up_cache(
|
||||||
|
self,
|
||||||
|
app: Flask,
|
||||||
|
client: FlaskClient,
|
||||||
|
with_db_and_bpmn_file_cleanup: None,
|
||||||
|
with_super_admin_user: UserModel,
|
||||||
|
) -> None:
|
||||||
|
"""When a BPMN processes identifier is changed in a file, the old id is removed from the cache."""
|
||||||
|
old_identifier = "ye_old_identifier"
|
||||||
|
process_id_lookup = SpecReferenceCache(
|
||||||
|
identifier=old_identifier,
|
||||||
|
relative_path=self.call_activity_nested_relative_file_path,
|
||||||
|
file_name=self.bpmn_file_name,
|
||||||
|
process_model_id=f"{self.process_group_id}/{self.process_model_id}",
|
||||||
|
type="process",
|
||||||
|
)
|
||||||
|
db.session.add(process_id_lookup)
|
||||||
|
db.session.commit()
|
||||||
|
|
||||||
|
self.create_group_and_model_with_bpmn(
|
||||||
|
client=client,
|
||||||
|
user=with_super_admin_user,
|
||||||
|
process_group_id=self.process_group_id,
|
||||||
|
process_model_id=self.process_model_id,
|
||||||
|
bpmn_file_name=self.bpmn_file_name,
|
||||||
|
bpmn_file_location=self.process_model_id,
|
||||||
|
)
|
||||||
|
|
||||||
|
bpmn_process_id_lookups = SpecReferenceCache.query.all()
|
||||||
|
assert len(bpmn_process_id_lookups) == 1
|
||||||
|
assert bpmn_process_id_lookups[0].identifier != old_identifier
|
||||||
|
assert bpmn_process_id_lookups[0].identifier == "Level1"
|
||||||
|
assert (
|
||||||
|
bpmn_process_id_lookups[0].relative_path
|
||||||
|
== self.call_activity_nested_relative_file_path
|
||||||
|
)
|
||||||
|
|
||||||
def test_load_reference_information(
|
def test_load_reference_information(
|
||||||
self,
|
self,
|
||||||
app: Flask,
|
app: Flask,
|
||||||
|
|
|
@ -36,6 +36,7 @@ module.exports = {
|
||||||
],
|
],
|
||||||
'react/react-in-jsx-scope': 'off',
|
'react/react-in-jsx-scope': 'off',
|
||||||
'react/require-default-props': 'off',
|
'react/require-default-props': 'off',
|
||||||
|
'import/prefer-default-export': 'off',
|
||||||
'no-unused-vars': [
|
'no-unused-vars': [
|
||||||
'error',
|
'error',
|
||||||
{
|
{
|
||||||
|
|
|
@ -7,6 +7,9 @@ function error_handler() {
|
||||||
trap 'error_handler ${LINENO} $?' ERR
|
trap 'error_handler ${LINENO} $?' ERR
|
||||||
set -o errtrace -o errexit -o nounset -o pipefail
|
set -o errtrace -o errexit -o nounset -o pipefail
|
||||||
|
|
||||||
|
# see also: npx cypress run --env grep="can filter",grepFilterSpecs=true
|
||||||
|
# https://github.com/cypress-io/cypress/tree/develop/npm/grep#pre-filter-specs-grepfilterspecs
|
||||||
|
|
||||||
command="${1:-}"
|
command="${1:-}"
|
||||||
if [[ -z "$command" ]]; then
|
if [[ -z "$command" ]]; then
|
||||||
command=open
|
command=open
|
||||||
|
|
|
@ -6,8 +6,9 @@ module.exports = defineConfig({
|
||||||
chromeWebSecurity: false,
|
chromeWebSecurity: false,
|
||||||
e2e: {
|
e2e: {
|
||||||
baseUrl: 'http://localhost:7001',
|
baseUrl: 'http://localhost:7001',
|
||||||
setupNodeEvents(_on, _config) {
|
setupNodeEvents(_on, config) {
|
||||||
// implement node event listeners here
|
require('@cypress/grep/src/plugin')(config);
|
||||||
|
return config;
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
|
||||||
|
|
|
@ -16,5 +16,9 @@
|
||||||
// Import commands.js using ES2015 syntax:
|
// Import commands.js using ES2015 syntax:
|
||||||
import './commands';
|
import './commands';
|
||||||
|
|
||||||
|
import registerCypressGrep from '@cypress/grep';
|
||||||
|
|
||||||
|
registerCypressGrep();
|
||||||
|
|
||||||
// Alternatively you can use CommonJS syntax:
|
// Alternatively you can use CommonJS syntax:
|
||||||
// require('./commands')
|
// require('./commands')
|
||||||
|
|
File diff suppressed because it is too large
Load Diff
|
@ -96,6 +96,7 @@
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
|
"@cypress/grep": "^3.1.0",
|
||||||
"@typescript-eslint/eslint-plugin": "^5.30.5",
|
"@typescript-eslint/eslint-plugin": "^5.30.5",
|
||||||
"@typescript-eslint/parser": "^5.30.6",
|
"@typescript-eslint/parser": "^5.30.6",
|
||||||
"cypress": "^10.8.0",
|
"cypress": "^10.8.0",
|
||||||
|
|
|
@ -24,7 +24,7 @@ export default function App() {
|
||||||
[errorMessage]
|
[errorMessage]
|
||||||
);
|
);
|
||||||
|
|
||||||
const ability = defineAbility((can: any) => {});
|
const ability = defineAbility(() => {});
|
||||||
|
|
||||||
let errorTag = null;
|
let errorTag = null;
|
||||||
if (errorMessage) {
|
if (errorMessage) {
|
||||||
|
@ -60,7 +60,7 @@ export default function App() {
|
||||||
{errorTag}
|
{errorTag}
|
||||||
<ErrorBoundary>
|
<ErrorBoundary>
|
||||||
<Routes>
|
<Routes>
|
||||||
<Route path="/" element={<HomePageRoutes />} />
|
<Route path="/*" element={<HomePageRoutes />} />
|
||||||
<Route path="/tasks/*" element={<HomePageRoutes />} />
|
<Route path="/tasks/*" element={<HomePageRoutes />} />
|
||||||
<Route path="/admin/*" element={<AdminRoutes />} />
|
<Route path="/admin/*" element={<AdminRoutes />} />
|
||||||
</Routes>
|
</Routes>
|
||||||
|
|
|
@ -74,10 +74,7 @@ export default function ProcessModelForm({
|
||||||
if (hasErrors) {
|
if (hasErrors) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
let path = `/process-models`;
|
const path = `/process-models/${modifiedProcessModelPath}`;
|
||||||
if (mode === 'edit') {
|
|
||||||
path = `/process-models/${modifiedProcessModelPath}`;
|
|
||||||
}
|
|
||||||
let httpMethod = 'POST';
|
let httpMethod = 'POST';
|
||||||
if (mode === 'edit') {
|
if (mode === 'edit') {
|
||||||
httpMethod = 'PUT';
|
httpMethod = 'PUT';
|
||||||
|
|
|
@ -52,10 +52,14 @@ import TouchModule from 'diagram-js/lib/navigation/touch';
|
||||||
// @ts-expect-error TS(7016) FIXME
|
// @ts-expect-error TS(7016) FIXME
|
||||||
import ZoomScrollModule from 'diagram-js/lib/navigation/zoomscroll';
|
import ZoomScrollModule from 'diagram-js/lib/navigation/zoomscroll';
|
||||||
|
|
||||||
|
import { Can } from '@casl/react';
|
||||||
import HttpService from '../services/HttpService';
|
import HttpService from '../services/HttpService';
|
||||||
|
|
||||||
import ButtonWithConfirmation from './ButtonWithConfirmation';
|
import ButtonWithConfirmation from './ButtonWithConfirmation';
|
||||||
import { makeid } from '../helpers';
|
import { makeid } from '../helpers';
|
||||||
|
import { useUriListForPermissions } from '../hooks/UriListForPermissions';
|
||||||
|
import { PermissionsToCheck } from '../interfaces';
|
||||||
|
import { usePermissionFetcher } from '../hooks/PermissionService';
|
||||||
|
|
||||||
type OwnProps = {
|
type OwnProps = {
|
||||||
processModelId: string;
|
processModelId: string;
|
||||||
|
@ -107,6 +111,13 @@ export default function ReactDiagramEditor({
|
||||||
|
|
||||||
const alreadyImportedXmlRef = useRef(false);
|
const alreadyImportedXmlRef = useRef(false);
|
||||||
|
|
||||||
|
const { targetUris } = useUriListForPermissions();
|
||||||
|
const permissionRequestData: PermissionsToCheck = {
|
||||||
|
[targetUris.processModelShowPath]: ['PUT'],
|
||||||
|
[targetUris.processModelFileShowPath]: ['POST', 'GET', 'PUT', 'DELETE'],
|
||||||
|
};
|
||||||
|
const { ability } = usePermissionFetcher(permissionRequestData);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (diagramModelerState) {
|
if (diagramModelerState) {
|
||||||
return;
|
return;
|
||||||
|
@ -517,9 +528,18 @@ export default function ReactDiagramEditor({
|
||||||
if (diagramType !== 'readonly') {
|
if (diagramType !== 'readonly') {
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<Button onClick={handleSave} variant="danger">
|
<Can
|
||||||
Save
|
I="PUT"
|
||||||
</Button>
|
a={targetUris.processModelFileShowPath}
|
||||||
|
ability={ability}
|
||||||
|
>
|
||||||
|
<Button onClick={handleSave}>Save</Button>
|
||||||
|
</Can>
|
||||||
|
<Can
|
||||||
|
I="DELETE"
|
||||||
|
a={targetUris.processModelFileShowPath}
|
||||||
|
ability={ability}
|
||||||
|
>
|
||||||
{fileName && (
|
{fileName && (
|
||||||
<ButtonWithConfirmation
|
<ButtonWithConfirmation
|
||||||
description={`Delete file ${fileName}?`}
|
description={`Delete file ${fileName}?`}
|
||||||
|
@ -527,10 +547,21 @@ export default function ReactDiagramEditor({
|
||||||
buttonLabel="Delete"
|
buttonLabel="Delete"
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
|
</Can>
|
||||||
|
<Can I="PUT" a={targetUris.processModelShowPath} ability={ability}>
|
||||||
{onSetPrimaryFile && (
|
{onSetPrimaryFile && (
|
||||||
<Button onClick={handleSetPrimaryFile}>Set as primary file</Button>
|
<Button onClick={handleSetPrimaryFile}>
|
||||||
|
Set as primary file
|
||||||
|
</Button>
|
||||||
)}
|
)}
|
||||||
|
</Can>
|
||||||
|
<Can
|
||||||
|
I="GET"
|
||||||
|
a={targetUris.processModelFileShowPath}
|
||||||
|
ability={ability}
|
||||||
|
>
|
||||||
<Button onClick={downloadXmlFile}>Download xml</Button>
|
<Button onClick={downloadXmlFile}>Download xml</Button>
|
||||||
|
</Can>
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
import { createContext } from 'react';
|
import { createContext } from 'react';
|
||||||
import { AbilityBuilder, Ability } from '@casl/ability';
|
import { Ability } from '@casl/ability';
|
||||||
import { createContextualCan } from '@casl/react';
|
import { createContextualCan } from '@casl/react';
|
||||||
|
|
||||||
export const AbilityContext = createContext(new Ability());
|
export const AbilityContext = createContext(new Ability());
|
||||||
|
|
|
@ -1,3 +1,5 @@
|
||||||
|
// We may need to update usage of Ability when we update.
|
||||||
|
// They say they are going to rename PureAbility to Ability and remove the old class.
|
||||||
import { AbilityBuilder, Ability } from '@casl/ability';
|
import { AbilityBuilder, Ability } from '@casl/ability';
|
||||||
import { useContext, useEffect } from 'react';
|
import { useContext, useEffect } from 'react';
|
||||||
import { AbilityContext } from '../contexts/Can';
|
import { AbilityContext } from '../contexts/Can';
|
||||||
|
@ -11,28 +13,34 @@ export const usePermissionFetcher = (
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
const processPermissionResult = (result: PermissionCheckResponseBody) => {
|
const processPermissionResult = (result: PermissionCheckResponseBody) => {
|
||||||
|
const oldRules = ability.rules;
|
||||||
const { can, cannot, rules } = new AbilityBuilder(Ability);
|
const { can, cannot, rules } = new AbilityBuilder(Ability);
|
||||||
for (const [url, permissionVerbResults] of Object.entries(
|
Object.keys(result.results).forEach((url: string) => {
|
||||||
result.results
|
const permissionVerbResults = result.results[url];
|
||||||
)) {
|
Object.keys(permissionVerbResults).forEach((permissionVerb: string) => {
|
||||||
for (const [permissionVerb, hasPermission] of Object.entries(
|
const hasPermission = permissionVerbResults[permissionVerb];
|
||||||
permissionVerbResults
|
|
||||||
)) {
|
|
||||||
if (hasPermission) {
|
if (hasPermission) {
|
||||||
can(permissionVerb, url);
|
can(permissionVerb, url);
|
||||||
} else {
|
} else {
|
||||||
cannot(permissionVerb, url);
|
cannot(permissionVerb, url);
|
||||||
}
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
oldRules.forEach((oldRule: any) => {
|
||||||
|
if (oldRule.inverted) {
|
||||||
|
cannot(oldRule.action, oldRule.subject);
|
||||||
|
} else {
|
||||||
|
can(oldRule.action, oldRule.subject);
|
||||||
}
|
}
|
||||||
}
|
});
|
||||||
ability.update(rules);
|
ability.update(rules);
|
||||||
};
|
};
|
||||||
|
|
||||||
HttpService.makeCallToBackend({
|
HttpService.makeCallToBackend({
|
||||||
path: `/permissions-check`,
|
path: `/permissions-check`,
|
||||||
httpMethod: 'POST',
|
httpMethod: 'POST',
|
||||||
successCallback: processPermissionResult,
|
successCallback: processPermissionResult,
|
||||||
postBody: { requests_to_check: permissionsToCheck },
|
postBody: { requests_to_check: permissionsToCheck },
|
||||||
// failureCallback: setErrorMessage,
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
|
@ -0,0 +1,18 @@
|
||||||
|
import { useParams } from 'react-router-dom';
|
||||||
|
|
||||||
|
export const useUriListForPermissions = () => {
|
||||||
|
const params = useParams();
|
||||||
|
const targetUris = {
|
||||||
|
messageInstanceListPath: '/v1.0/messages',
|
||||||
|
processGroupListPath: '/v1.0/process-groups',
|
||||||
|
processGroupShowPath: `/v1.0/process-groups/${params.process_group_id}`,
|
||||||
|
processInstanceActionPath: `/v1.0/process-models/${params.process_model_id}/process-instances`,
|
||||||
|
processInstanceListPath: '/v1.0/process-instances',
|
||||||
|
processModelCreatePath: `/v1.0/process-models/${params.process_group_id}`,
|
||||||
|
processModelFileCreatePath: `/v1.0/process-models/${params.process_model_id}/files`,
|
||||||
|
processModelFileShowPath: `/v1.0/process-models/${params.process_model_id}/files/${params.file_name}`,
|
||||||
|
processModelShowPath: `/v1.0/process-models/${params.process_model_id}`,
|
||||||
|
};
|
||||||
|
|
||||||
|
return { targetUris };
|
||||||
|
};
|
|
@ -74,12 +74,12 @@ export interface CarbonComboBoxSelection {
|
||||||
export interface PermissionsToCheck {
|
export interface PermissionsToCheck {
|
||||||
[key: string]: string[];
|
[key: string]: string[];
|
||||||
}
|
}
|
||||||
export interface PermissionCheckResponseBody {
|
export interface PermissionVerbResults {
|
||||||
results: PermissionCheckResult;
|
[key: string]: boolean;
|
||||||
}
|
}
|
||||||
export interface PermissionCheckResult {
|
export interface PermissionCheckResult {
|
||||||
[key: string]: PermissionVerbResults;
|
[key: string]: PermissionVerbResults;
|
||||||
}
|
}
|
||||||
export interface PermissionVerbResults {
|
export interface PermissionCheckResponseBody {
|
||||||
[key: string]: boolean;
|
results: PermissionCheckResult;
|
||||||
}
|
}
|
||||||
|
|
|
@ -10,6 +10,7 @@ import {
|
||||||
// ClickableTile,
|
// ClickableTile,
|
||||||
// @ts-ignore
|
// @ts-ignore
|
||||||
} from '@carbon/react';
|
} from '@carbon/react';
|
||||||
|
import { Can } from '@casl/react';
|
||||||
import ProcessBreadcrumb from '../components/ProcessBreadcrumb';
|
import ProcessBreadcrumb from '../components/ProcessBreadcrumb';
|
||||||
import PaginationForTable from '../components/PaginationForTable';
|
import PaginationForTable from '../components/PaginationForTable';
|
||||||
import HttpService from '../services/HttpService';
|
import HttpService from '../services/HttpService';
|
||||||
|
@ -17,8 +18,14 @@ import {
|
||||||
getPageInfoFromSearchParams,
|
getPageInfoFromSearchParams,
|
||||||
modifyProcessModelPath,
|
modifyProcessModelPath,
|
||||||
} from '../helpers';
|
} from '../helpers';
|
||||||
import { CarbonComboBoxSelection, ProcessGroup } from '../interfaces';
|
import {
|
||||||
|
CarbonComboBoxSelection,
|
||||||
|
PermissionsToCheck,
|
||||||
|
ProcessGroup,
|
||||||
|
} from '../interfaces';
|
||||||
import ProcessModelSearch from '../components/ProcessModelSearch';
|
import ProcessModelSearch from '../components/ProcessModelSearch';
|
||||||
|
import { useUriListForPermissions } from '../hooks/UriListForPermissions';
|
||||||
|
import { usePermissionFetcher } from '../hooks/PermissionService';
|
||||||
|
|
||||||
// Example process group json
|
// Example process group json
|
||||||
// {'process_group_id': 'sure', 'display_name': 'Test Workflows', 'id': 'test_process_group'}
|
// {'process_group_id': 'sure', 'display_name': 'Test Workflows', 'id': 'test_process_group'}
|
||||||
|
@ -32,6 +39,12 @@ export default function ProcessGroupList() {
|
||||||
[]
|
[]
|
||||||
);
|
);
|
||||||
|
|
||||||
|
const { targetUris } = useUriListForPermissions();
|
||||||
|
const permissionRequestData: PermissionsToCheck = {
|
||||||
|
[targetUris.processGroupListPath]: ['POST'],
|
||||||
|
};
|
||||||
|
const { ability } = usePermissionFetcher(permissionRequestData);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
const setProcessGroupsFromResult = (result: any) => {
|
const setProcessGroupsFromResult = (result: any) => {
|
||||||
setProcessGroups(result.results);
|
setProcessGroups(result.results);
|
||||||
|
@ -84,17 +97,6 @@ export default function ProcessGroupList() {
|
||||||
<tbody>{rows}</tbody>
|
<tbody>{rows}</tbody>
|
||||||
</Table>
|
</Table>
|
||||||
);
|
);
|
||||||
// const rows = processGroups.map((row: ProcessGroup) => {
|
|
||||||
// return (
|
|
||||||
// <span>
|
|
||||||
// <ClickableTile href={`/admin/process-groups/${row.id}`}>
|
|
||||||
// {row.display_name}
|
|
||||||
// </ClickableTile>
|
|
||||||
// </span>
|
|
||||||
// );
|
|
||||||
// });
|
|
||||||
//
|
|
||||||
// return <div style={{ width: '400px' }}>{rows}</div>;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
const processGroupsDisplayArea = () => {
|
const processGroupsDisplayArea = () => {
|
||||||
|
@ -138,11 +140,13 @@ export default function ProcessGroupList() {
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<ProcessBreadcrumb hotCrumbs={[['Process Groups']]} />
|
<ProcessBreadcrumb hotCrumbs={[['Process Groups']]} />
|
||||||
|
<Can I="POST" a={targetUris.processGroupListPath} ability={ability}>
|
||||||
<Button kind="secondary" href="/admin/process-groups/new">
|
<Button kind="secondary" href="/admin/process-groups/new">
|
||||||
Add a process group
|
Add a process group
|
||||||
</Button>
|
</Button>
|
||||||
<br />
|
<br />
|
||||||
<br />
|
<br />
|
||||||
|
</Can>
|
||||||
{processModelSearchArea()}
|
{processModelSearchArea()}
|
||||||
<br />
|
<br />
|
||||||
{processGroupsDisplayArea()}
|
{processGroupsDisplayArea()}
|
||||||
|
|
|
@ -2,6 +2,7 @@ import { useEffect, useState } from 'react';
|
||||||
import { Link, useSearchParams, useParams } from 'react-router-dom';
|
import { Link, useSearchParams, useParams } from 'react-router-dom';
|
||||||
// @ts-ignore
|
// @ts-ignore
|
||||||
import { Button, Table, Stack } from '@carbon/react';
|
import { Button, Table, Stack } from '@carbon/react';
|
||||||
|
import { Can } from '@casl/react';
|
||||||
import ProcessBreadcrumb from '../components/ProcessBreadcrumb';
|
import ProcessBreadcrumb from '../components/ProcessBreadcrumb';
|
||||||
import PaginationForTable from '../components/PaginationForTable';
|
import PaginationForTable from '../components/PaginationForTable';
|
||||||
import HttpService from '../services/HttpService';
|
import HttpService from '../services/HttpService';
|
||||||
|
@ -10,7 +11,14 @@ import {
|
||||||
modifyProcessModelPath,
|
modifyProcessModelPath,
|
||||||
unModifyProcessModelPath,
|
unModifyProcessModelPath,
|
||||||
} from '../helpers';
|
} from '../helpers';
|
||||||
import { PaginationObject, ProcessGroup, ProcessModel } from '../interfaces';
|
import {
|
||||||
|
PaginationObject,
|
||||||
|
PermissionsToCheck,
|
||||||
|
ProcessGroup,
|
||||||
|
ProcessModel,
|
||||||
|
} from '../interfaces';
|
||||||
|
import { useUriListForPermissions } from '../hooks/UriListForPermissions';
|
||||||
|
import { usePermissionFetcher } from '../hooks/PermissionService';
|
||||||
|
|
||||||
export default function ProcessGroupShow() {
|
export default function ProcessGroupShow() {
|
||||||
const params = useParams();
|
const params = useParams();
|
||||||
|
@ -24,6 +32,14 @@ export default function ProcessGroupShow() {
|
||||||
const [groupPagination, setGroupPagination] =
|
const [groupPagination, setGroupPagination] =
|
||||||
useState<PaginationObject | null>(null);
|
useState<PaginationObject | null>(null);
|
||||||
|
|
||||||
|
const { targetUris } = useUriListForPermissions();
|
||||||
|
const permissionRequestData: PermissionsToCheck = {
|
||||||
|
[targetUris.processGroupListPath]: ['POST'],
|
||||||
|
[targetUris.processGroupShowPath]: ['PUT'],
|
||||||
|
[targetUris.processModelCreatePath]: ['POST'],
|
||||||
|
};
|
||||||
|
const { ability } = usePermissionFetcher(permissionRequestData);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
const { page, perPage } = getPageInfoFromSearchParams(searchParams);
|
const { page, perPage } = getPageInfoFromSearchParams(searchParams);
|
||||||
|
|
||||||
|
@ -143,23 +159,31 @@ export default function ProcessGroupShow() {
|
||||||
<h1>Process Group: {processGroup.display_name}</h1>
|
<h1>Process Group: {processGroup.display_name}</h1>
|
||||||
<ul>
|
<ul>
|
||||||
<Stack orientation="horizontal" gap={3}>
|
<Stack orientation="horizontal" gap={3}>
|
||||||
|
<Can I="POST" a={targetUris.processGroupListPath} ability={ability}>
|
||||||
<Button
|
<Button
|
||||||
kind="secondary"
|
|
||||||
href={`/admin/process-groups/new?parentGroupId=${processGroup.id}`}
|
href={`/admin/process-groups/new?parentGroupId=${processGroup.id}`}
|
||||||
>
|
>
|
||||||
Add a process group
|
Add a process group
|
||||||
</Button>
|
</Button>
|
||||||
|
</Can>
|
||||||
|
<Can
|
||||||
|
I="POST"
|
||||||
|
a={targetUris.processModelCreatePath}
|
||||||
|
ability={ability}
|
||||||
|
>
|
||||||
<Button
|
<Button
|
||||||
href={`/admin/process-models/${modifiedProcessGroupId}/new`}
|
href={`/admin/process-models/${modifiedProcessGroupId}/new`}
|
||||||
>
|
>
|
||||||
Add a process model
|
Add a process model
|
||||||
</Button>
|
</Button>
|
||||||
|
</Can>
|
||||||
|
<Can I="PUT" a={targetUris.processGroupShowPath} ability={ability}>
|
||||||
<Button
|
<Button
|
||||||
href={`/admin/process-groups/${modifiedProcessGroupId}/edit`}
|
href={`/admin/process-groups/${modifiedProcessGroupId}/edit`}
|
||||||
variant="secondary"
|
|
||||||
>
|
>
|
||||||
Edit process group
|
Edit process group
|
||||||
</Button>
|
</Button>
|
||||||
|
</Can>
|
||||||
</Stack>
|
</Stack>
|
||||||
<br />
|
<br />
|
||||||
<br />
|
<br />
|
||||||
|
|
|
@ -23,6 +23,7 @@ import {
|
||||||
Stack,
|
Stack,
|
||||||
// @ts-ignore
|
// @ts-ignore
|
||||||
} from '@carbon/react';
|
} from '@carbon/react';
|
||||||
|
import { Can } from '@casl/react';
|
||||||
import ProcessBreadcrumb from '../components/ProcessBreadcrumb';
|
import ProcessBreadcrumb from '../components/ProcessBreadcrumb';
|
||||||
import HttpService from '../services/HttpService';
|
import HttpService from '../services/HttpService';
|
||||||
import ReactDiagramEditor from '../components/ReactDiagramEditor';
|
import ReactDiagramEditor from '../components/ReactDiagramEditor';
|
||||||
|
@ -32,6 +33,9 @@ import {
|
||||||
} from '../helpers';
|
} from '../helpers';
|
||||||
import ButtonWithConfirmation from '../components/ButtonWithConfirmation';
|
import ButtonWithConfirmation from '../components/ButtonWithConfirmation';
|
||||||
import ErrorContext from '../contexts/ErrorContext';
|
import ErrorContext from '../contexts/ErrorContext';
|
||||||
|
import { useUriListForPermissions } from '../hooks/UriListForPermissions';
|
||||||
|
import { PermissionsToCheck } from '../interfaces';
|
||||||
|
import { usePermissionFetcher } from '../hooks/PermissionService';
|
||||||
|
|
||||||
export default function ProcessInstanceShow() {
|
export default function ProcessInstanceShow() {
|
||||||
const navigate = useNavigate();
|
const navigate = useNavigate();
|
||||||
|
@ -50,6 +54,12 @@ export default function ProcessInstanceShow() {
|
||||||
);
|
);
|
||||||
const modifiedProcessModelId = params.process_model_id;
|
const modifiedProcessModelId = params.process_model_id;
|
||||||
|
|
||||||
|
const { targetUris } = useUriListForPermissions();
|
||||||
|
const permissionRequestData: PermissionsToCheck = {
|
||||||
|
[targetUris.messageInstanceListPath]: ['GET'],
|
||||||
|
};
|
||||||
|
const { ability } = usePermissionFetcher(permissionRequestData);
|
||||||
|
|
||||||
const navigateToProcessInstances = (_result: any) => {
|
const navigateToProcessInstances = (_result: any) => {
|
||||||
navigate(
|
navigate(
|
||||||
`/admin/process-instances?process_model_identifier=${unModifiedProcessModelId}`
|
`/admin/process-instances?process_model_identifier=${unModifiedProcessModelId}`
|
||||||
|
@ -63,12 +73,12 @@ export default function ProcessInstanceShow() {
|
||||||
});
|
});
|
||||||
if (typeof params.spiff_step === 'undefined')
|
if (typeof params.spiff_step === 'undefined')
|
||||||
HttpService.makeCallToBackend({
|
HttpService.makeCallToBackend({
|
||||||
path: `/process-instance/${params.process_instance_id}/tasks?all_tasks=true`,
|
path: `/process-instances/${modifiedProcessModelId}/${params.process_instance_id}/tasks?all_tasks=true`,
|
||||||
successCallback: setTasks,
|
successCallback: setTasks,
|
||||||
});
|
});
|
||||||
else
|
else
|
||||||
HttpService.makeCallToBackend({
|
HttpService.makeCallToBackend({
|
||||||
path: `/process-instance/${params.process_instance_id}/tasks?all_tasks=true&spiff_step=${params.spiff_step}`,
|
path: `/process-instances/${modifiedProcessModelId}/${params.process_instance_id}/tasks?all_tasks=true&spiff_step=${params.spiff_step}`,
|
||||||
successCallback: setTasks,
|
successCallback: setTasks,
|
||||||
});
|
});
|
||||||
}, [params, modifiedProcessModelId]);
|
}, [params, modifiedProcessModelId]);
|
||||||
|
@ -245,6 +255,11 @@ export default function ProcessInstanceShow() {
|
||||||
>
|
>
|
||||||
Logs
|
Logs
|
||||||
</Button>
|
</Button>
|
||||||
|
<Can
|
||||||
|
I="GET"
|
||||||
|
a={targetUris.messageInstanceListPath}
|
||||||
|
ability={ability}
|
||||||
|
>
|
||||||
<Button
|
<Button
|
||||||
size="sm"
|
size="sm"
|
||||||
className="button-white-background"
|
className="button-white-background"
|
||||||
|
@ -253,6 +268,7 @@ export default function ProcessInstanceShow() {
|
||||||
>
|
>
|
||||||
Messages
|
Messages
|
||||||
</Button>
|
</Button>
|
||||||
|
</Can>
|
||||||
</ButtonSet>
|
</ButtonSet>
|
||||||
</Column>
|
</Column>
|
||||||
</Grid>
|
</Grid>
|
||||||
|
|
|
@ -40,7 +40,8 @@ import {
|
||||||
} from '../interfaces';
|
} from '../interfaces';
|
||||||
import ButtonWithConfirmation from '../components/ButtonWithConfirmation';
|
import ButtonWithConfirmation from '../components/ButtonWithConfirmation';
|
||||||
import ProcessInstanceListTable from '../components/ProcessInstanceListTable';
|
import ProcessInstanceListTable from '../components/ProcessInstanceListTable';
|
||||||
import { usePermissionFetcher } from '../components/PermissionService';
|
import { usePermissionFetcher } from '../hooks/PermissionService';
|
||||||
|
import { useUriListForPermissions } from '../hooks/UriListForPermissions';
|
||||||
|
|
||||||
const storeRecentProcessModelInLocalStorage = (
|
const storeRecentProcessModelInLocalStorage = (
|
||||||
processModelForStorage: ProcessModel
|
processModelForStorage: ProcessModel
|
||||||
|
@ -103,16 +104,12 @@ export default function ProcessModelShow() {
|
||||||
useState<boolean>(false);
|
useState<boolean>(false);
|
||||||
const navigate = useNavigate();
|
const navigate = useNavigate();
|
||||||
|
|
||||||
const targetUris = {
|
const { targetUris } = useUriListForPermissions();
|
||||||
processModelPath: `/process-models/${params.process_model_id}`,
|
|
||||||
processInstancesPath: `/process-instances`,
|
|
||||||
};
|
|
||||||
const permissionRequestData: PermissionsToCheck = {
|
const permissionRequestData: PermissionsToCheck = {
|
||||||
[`/v1.0${targetUris.processModelPath}`]: ['GET', 'PUT'],
|
[targetUris.processModelShowPath]: ['PUT'],
|
||||||
[`/v1.0${targetUris.processInstancesPath}`]: ['GET'],
|
[targetUris.processInstanceListPath]: ['GET'],
|
||||||
[`/v1.0${targetUris.processModelPath}${targetUris.processInstancesPath}`]: [
|
[targetUris.processInstanceActionPath]: ['POST'],
|
||||||
'POST',
|
[targetUris.processModelFileCreatePath]: ['POST', 'GET', 'DELETE'],
|
||||||
],
|
|
||||||
};
|
};
|
||||||
const { ability } = usePermissionFetcher(permissionRequestData);
|
const { ability } = usePermissionFetcher(permissionRequestData);
|
||||||
|
|
||||||
|
@ -267,6 +264,7 @@ export default function ProcessModelShow() {
|
||||||
) => {
|
) => {
|
||||||
const elements = [];
|
const elements = [];
|
||||||
elements.push(
|
elements.push(
|
||||||
|
<Can I="GET" a={targetUris.processModelFileCreatePath} ability={ability}>
|
||||||
<Button
|
<Button
|
||||||
kind="ghost"
|
kind="ghost"
|
||||||
renderIcon={Edit}
|
renderIcon={Edit}
|
||||||
|
@ -276,8 +274,10 @@ export default function ProcessModelShow() {
|
||||||
data-qa={`edit-file-${processModelFile.name.replace('.', '-')}`}
|
data-qa={`edit-file-${processModelFile.name.replace('.', '-')}`}
|
||||||
onClick={() => navigateToFileEdit(processModelFile)}
|
onClick={() => navigateToFileEdit(processModelFile)}
|
||||||
/>
|
/>
|
||||||
|
</Can>
|
||||||
);
|
);
|
||||||
elements.push(
|
elements.push(
|
||||||
|
<Can I="GET" a={targetUris.processModelFileCreatePath} ability={ability}>
|
||||||
<Button
|
<Button
|
||||||
kind="ghost"
|
kind="ghost"
|
||||||
renderIcon={Download}
|
renderIcon={Download}
|
||||||
|
@ -286,9 +286,15 @@ export default function ProcessModelShow() {
|
||||||
size="lg"
|
size="lg"
|
||||||
onClick={() => downloadFile(processModelFile.name)}
|
onClick={() => downloadFile(processModelFile.name)}
|
||||||
/>
|
/>
|
||||||
|
</Can>
|
||||||
);
|
);
|
||||||
|
|
||||||
elements.push(
|
elements.push(
|
||||||
|
<Can
|
||||||
|
I="DELETE"
|
||||||
|
a={targetUris.processModelFileCreatePath}
|
||||||
|
ability={ability}
|
||||||
|
>
|
||||||
<ButtonWithConfirmation
|
<ButtonWithConfirmation
|
||||||
kind="ghost"
|
kind="ghost"
|
||||||
renderIcon={TrashCan}
|
renderIcon={TrashCan}
|
||||||
|
@ -300,9 +306,11 @@ export default function ProcessModelShow() {
|
||||||
}}
|
}}
|
||||||
confirmButtonLabel="Delete"
|
confirmButtonLabel="Delete"
|
||||||
/>
|
/>
|
||||||
|
</Can>
|
||||||
);
|
);
|
||||||
if (processModelFile.name.match(/\.bpmn$/) && !isPrimaryBpmnFile) {
|
if (processModelFile.name.match(/\.bpmn$/) && !isPrimaryBpmnFile) {
|
||||||
elements.push(
|
elements.push(
|
||||||
|
<Can I="PUT" a={targetUris.processModelShowPath} ability={ability}>
|
||||||
<Button
|
<Button
|
||||||
kind="ghost"
|
kind="ghost"
|
||||||
renderIcon={Favorite}
|
renderIcon={Favorite}
|
||||||
|
@ -311,6 +319,7 @@ export default function ProcessModelShow() {
|
||||||
size="lg"
|
size="lg"
|
||||||
onClick={() => onSetPrimaryFile(processModelFile.name)}
|
onClick={() => onSetPrimaryFile(processModelFile.name)}
|
||||||
/>
|
/>
|
||||||
|
</Can>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
return elements;
|
return elements;
|
||||||
|
@ -341,7 +350,11 @@ export default function ProcessModelShow() {
|
||||||
let fileLink = null;
|
let fileLink = null;
|
||||||
const fileUrl = profileModelFileEditUrl(processModelFile);
|
const fileUrl = profileModelFileEditUrl(processModelFile);
|
||||||
if (fileUrl) {
|
if (fileUrl) {
|
||||||
|
if (ability.can('GET', targetUris.processModelFileCreatePath)) {
|
||||||
fileLink = <Link to={fileUrl}>{processModelFile.name}</Link>;
|
fileLink = <Link to={fileUrl}>{processModelFile.name}</Link>;
|
||||||
|
} else {
|
||||||
|
fileLink = <span>{processModelFile.name}</span>;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
constructedTag = (
|
constructedTag = (
|
||||||
<TableRow key={processModelFile.name}>
|
<TableRow key={processModelFile.name}>
|
||||||
|
@ -445,6 +458,11 @@ export default function ProcessModelShow() {
|
||||||
</span>
|
</span>
|
||||||
</Stack>
|
</Stack>
|
||||||
}
|
}
|
||||||
|
>
|
||||||
|
<Can
|
||||||
|
I="POST"
|
||||||
|
a={targetUris.processModelFileCreatePath}
|
||||||
|
ability={ability}
|
||||||
>
|
>
|
||||||
<ButtonSet>
|
<ButtonSet>
|
||||||
<Button
|
<Button
|
||||||
|
@ -487,6 +505,7 @@ export default function ProcessModelShow() {
|
||||||
</Button>
|
</Button>
|
||||||
</ButtonSet>
|
</ButtonSet>
|
||||||
<br />
|
<br />
|
||||||
|
</Can>
|
||||||
{processModelFileList()}
|
{processModelFileList()}
|
||||||
</AccordionItem>
|
</AccordionItem>
|
||||||
</Accordion>
|
</Accordion>
|
||||||
|
@ -513,18 +532,14 @@ export default function ProcessModelShow() {
|
||||||
<Stack orientation="horizontal" gap={3}>
|
<Stack orientation="horizontal" gap={3}>
|
||||||
<Can
|
<Can
|
||||||
I="POST"
|
I="POST"
|
||||||
a={`/v1.0${targetUris.processModelPath}${targetUris.processInstancesPath}`}
|
a={targetUris.processInstanceActionPath}
|
||||||
ability={ability}
|
ability={ability}
|
||||||
>
|
>
|
||||||
<Button onClick={processInstanceCreateAndRun} variant="primary">
|
<Button onClick={processInstanceCreateAndRun} variant="primary">
|
||||||
Run
|
Run
|
||||||
</Button>
|
</Button>
|
||||||
</Can>
|
</Can>
|
||||||
<Can
|
<Can I="PUT" a={targetUris.processModelShowPath} ability={ability}>
|
||||||
I="PUT"
|
|
||||||
a={`/v1.0${targetUris.processModelPath}`}
|
|
||||||
ability={ability}
|
|
||||||
>
|
|
||||||
<Button
|
<Button
|
||||||
href={`/admin/process-models/${modifiedProcessModelId}/edit`}
|
href={`/admin/process-models/${modifiedProcessModelId}/edit`}
|
||||||
variant="secondary"
|
variant="secondary"
|
||||||
|
@ -537,11 +552,7 @@ export default function ProcessModelShow() {
|
||||||
<br />
|
<br />
|
||||||
{processInstanceRunResultTag()}
|
{processInstanceRunResultTag()}
|
||||||
<br />
|
<br />
|
||||||
<Can
|
<Can I="GET" a={targetUris.processInstanceListPath} ability={ability}>
|
||||||
I="GET"
|
|
||||||
a={`/v1.0${targetUris.processInstancesPath}`}
|
|
||||||
ability={ability}
|
|
||||||
>
|
|
||||||
<ProcessInstanceListTable
|
<ProcessInstanceListTable
|
||||||
filtersEnabled={false}
|
filtersEnabled={false}
|
||||||
processModelFullIdentifier={processModel.id}
|
processModelFullIdentifier={processModel.id}
|
||||||
|
|
|
@ -8,6 +8,7 @@ import ReactMarkdown from 'react-markdown';
|
||||||
import remarkGfm from 'remark-gfm';
|
import remarkGfm from 'remark-gfm';
|
||||||
import HttpService from '../services/HttpService';
|
import HttpService from '../services/HttpService';
|
||||||
import ErrorContext from '../contexts/ErrorContext';
|
import ErrorContext from '../contexts/ErrorContext';
|
||||||
|
import { modifyProcessModelPath } from '../helpers';
|
||||||
|
|
||||||
export default function TaskShow() {
|
export default function TaskShow() {
|
||||||
const [task, setTask] = useState(null);
|
const [task, setTask] = useState(null);
|
||||||
|
@ -18,16 +19,22 @@ export default function TaskShow() {
|
||||||
const setErrorMessage = (useContext as any)(ErrorContext)[1];
|
const setErrorMessage = (useContext as any)(ErrorContext)[1];
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
|
const processResult = (result: any) => {
|
||||||
|
setTask(result);
|
||||||
|
HttpService.makeCallToBackend({
|
||||||
|
path: `/process-instances/${modifyProcessModelPath(
|
||||||
|
result.process_model_identifier
|
||||||
|
)}/${params.process_instance_id}/tasks`,
|
||||||
|
successCallback: setUserTasks,
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
HttpService.makeCallToBackend({
|
HttpService.makeCallToBackend({
|
||||||
path: `/tasks/${params.process_instance_id}/${params.task_id}`,
|
path: `/tasks/${params.process_instance_id}/${params.task_id}`,
|
||||||
successCallback: setTask,
|
successCallback: processResult,
|
||||||
// This causes the page to continuously reload
|
// This causes the page to continuously reload
|
||||||
// failureCallback: setErrorMessage,
|
// failureCallback: setErrorMessage,
|
||||||
});
|
});
|
||||||
HttpService.makeCallToBackend({
|
|
||||||
path: `/process-instance/${params.process_instance_id}/tasks`,
|
|
||||||
successCallback: setUserTasks,
|
|
||||||
});
|
|
||||||
}, [params]);
|
}, [params]);
|
||||||
|
|
||||||
const processSubmitResult = (result: any) => {
|
const processSubmitResult = (result: any) => {
|
||||||
|
|
Loading…
Reference in New Issue