Merge remote-tracking branch 'origin/main' into feature/remove_spif_step_details

This commit is contained in:
jasquat 2023-03-20 11:06:02 -04:00
commit 50e4d27dbd
7 changed files with 96 additions and 32 deletions

View File

@ -1538,6 +1538,25 @@ paths:
items: items:
$ref: "#/components/schemas/User" $ref: "#/components/schemas/User"
/users/exists/by-username:
post:
tags:
- Users
operationId: spiffworkflow_backend.routes.users_controller.user_exists_by_username
summary: Returns a true if user exists by username.
requestBody:
content:
application/json:
schema:
$ref: "#/components/schemas/User"
responses:
"200":
description: true if user exists
content:
application/json:
schema:
$ref: "#/components/schemas/OkTrue"
/user-groups/for-current-user: /user-groups/for-current-user:
get: get:
tags: tags:

View File

@ -1,13 +1,29 @@
"""Users_controller.""" """Users_controller."""
from typing import Any
from typing import Dict
import flask import flask
from flask import current_app from flask import current_app
from flask import g from flask import g
from flask import jsonify from flask import jsonify
from flask import make_response from flask import make_response
from spiffworkflow_backend.exceptions.api_error import ApiError
from spiffworkflow_backend.models.user import UserModel from spiffworkflow_backend.models.user import UserModel
def user_exists_by_username(body: Dict[str, Any]) -> flask.wrappers.Response:
if "username" not in body:
raise ApiError(
error_code="username_not_given",
message="Username could not be found in post body.",
status_code=400,
)
username = body["username"]
found_user = UserModel.query.filter_by(username=username).first()
return make_response(jsonify({"user_found": found_user is not None}), 200)
def user_search(username_prefix: str) -> flask.wrappers.Response: def user_search(username_prefix: str) -> flask.wrappers.Response:
"""User_search.""" """User_search."""
found_users = UserModel.query.filter(UserModel.username.like(f"{username_prefix}%")).all() # type: ignore found_users = UserModel.query.filter(UserModel.username.like(f"{username_prefix}%")).all() # type: ignore

View File

@ -565,6 +565,7 @@ class AuthorizationService:
permissions_to_assign.append(PermissionToAssign(permission="read", target_uri="/processes")) permissions_to_assign.append(PermissionToAssign(permission="read", target_uri="/processes"))
permissions_to_assign.append(PermissionToAssign(permission="read", target_uri="/service-tasks")) permissions_to_assign.append(PermissionToAssign(permission="read", target_uri="/service-tasks"))
permissions_to_assign.append(PermissionToAssign(permission="read", target_uri="/user-groups/for-current-user")) permissions_to_assign.append(PermissionToAssign(permission="read", target_uri="/user-groups/for-current-user"))
permissions_to_assign.append(PermissionToAssign(permission="create", target_uri="/users/exists/by-username"))
permissions_to_assign.append( permissions_to_assign.append(
PermissionToAssign(permission="read", target_uri="/process-instances/find-by-id/*") PermissionToAssign(permission="read", target_uri="/process-instances/find-by-id/*")
) )

View File

@ -25,7 +25,6 @@ from spiffworkflow_backend.models.process_instance_file_data import (
from spiffworkflow_backend.models.process_model import ProcessModelInfo from spiffworkflow_backend.models.process_model import ProcessModelInfo
from spiffworkflow_backend.models.task import Task from spiffworkflow_backend.models.task import Task
from spiffworkflow_backend.models.user import UserModel from spiffworkflow_backend.models.user import UserModel
from spiffworkflow_backend.services.assertion_service import safe_assertion
from spiffworkflow_backend.services.authorization_service import AuthorizationService from spiffworkflow_backend.services.authorization_service import AuthorizationService
from spiffworkflow_backend.services.git_service import GitCommandError from spiffworkflow_backend.services.git_service import GitCommandError
from spiffworkflow_backend.services.git_service import GitService from spiffworkflow_backend.services.git_service import GitService
@ -96,13 +95,6 @@ class ProcessInstanceService:
) )
process_instance_lock_prefix = "Background" process_instance_lock_prefix = "Background"
for process_instance in records: for process_instance in records:
with safe_assertion(process_instance.status == status_value) as false_assumption:
if false_assumption:
raise AssertionError(
f"Queue assumed process instance {process_instance.id} has status of {status_value} "
f"when it really is {process_instance.status}"
)
locked = False locked = False
processor = None processor = None
try: try:
@ -110,6 +102,8 @@ class ProcessInstanceService:
processor = ProcessInstanceProcessor(process_instance) processor = ProcessInstanceProcessor(process_instance)
processor.lock_process_instance(process_instance_lock_prefix) processor.lock_process_instance(process_instance_lock_prefix)
locked = True locked = True
db.session.refresh(process_instance)
if process_instance.status == status_value:
execution_strategy_name = current_app.config[ execution_strategy_name = current_app.config[
"SPIFFWORKFLOW_BACKEND_ENGINE_STEP_DEFAULT_STRATEGY_BACKGROUND" "SPIFFWORKFLOW_BACKEND_ENGINE_STEP_DEFAULT_STRATEGY_BACKGROUND"
] ]

View File

@ -288,6 +288,7 @@ class TestAuthorizationService(BaseTest):
("/tasks/*", "read"), ("/tasks/*", "read"),
("/tasks/*", "update"), ("/tasks/*", "update"),
("/user-groups/for-current-user", "read"), ("/user-groups/for-current-user", "read"),
("/users/exists/by-username", "create"),
] ]
permissions_to_assign = AuthorizationService.explode_permissions("all", "BASIC") permissions_to_assign = AuthorizationService.explode_permissions("all", "BASIC")
permissions_to_assign_tuples = sorted([(p.target_uri, p.permission) for p in permissions_to_assign]) permissions_to_assign_tuples = sorted([(p.target_uri, p.permission) for p in permissions_to_assign])

View File

@ -182,6 +182,10 @@ export default function ProcessInstanceListTable({
const [processInitiatorText, setProcessInitiatorText] = useState< const [processInitiatorText, setProcessInitiatorText] = useState<
string | null string | null
>(null); >(null);
const [
processInitiatorNotFoundErrorText,
setProcessInitiatorNotFoundErrorText,
] = useState<string>('');
const lastRequestedInitatorSearchTerm = useRef<string>(); const lastRequestedInitatorSearchTerm = useRef<string>();
@ -560,8 +564,17 @@ export default function ProcessInstanceListTable({
return (reportMetadata as any).filter_by; return (reportMetadata as any).filter_by;
}; };
const navigateToNewReport = (queryParamString: string) => {
removeError();
setProcessInstanceReportJustSaved(null);
setProcessInstanceFilters({});
navigate(`${processInstanceListPathPrefix}?${queryParamString}`);
};
const applyFilter = (event: any) => { const applyFilter = (event: any) => {
event.preventDefault(); event.preventDefault();
setProcessInitiatorNotFoundErrorText('');
const { page, perPage } = getPageInfoFromSearchParams( const { page, perPage } = getPageInfoFromSearchParams(
searchParams, searchParams,
undefined, undefined,
@ -605,21 +618,33 @@ export default function ProcessInstanceListTable({
queryParamString += `&report_id=${processInstanceReportSelection.id}`; queryParamString += `&report_id=${processInstanceReportSelection.id}`;
} }
if (processInitiatorSelection) {
queryParamString += `&process_initiator_username=${processInitiatorSelection.username}`;
} else if (processInitiatorText) {
queryParamString += `&process_initiator_username=${processInitiatorText}`;
}
const reportColumnsBase64 = encodeBase64(JSON.stringify(reportColumns())); const reportColumnsBase64 = encodeBase64(JSON.stringify(reportColumns()));
queryParamString += `&report_columns=${reportColumnsBase64}`; queryParamString += `&report_columns=${reportColumnsBase64}`;
const reportFilterByBase64 = encodeBase64(JSON.stringify(reportFilterBy())); const reportFilterByBase64 = encodeBase64(JSON.stringify(reportFilterBy()));
queryParamString += `&report_filter_by=${reportFilterByBase64}`; queryParamString += `&report_filter_by=${reportFilterByBase64}`;
removeError(); if (processInitiatorSelection) {
setProcessInstanceReportJustSaved(null); queryParamString += `&process_initiator_username=${processInitiatorSelection.username}`;
setProcessInstanceFilters({}); navigateToNewReport(queryParamString);
navigate(`${processInstanceListPathPrefix}?${queryParamString}`); } else if (processInitiatorText) {
HttpService.makeCallToBackend({
path: targetUris.userExists,
httpMethod: 'POST',
postBody: { username: processInitiatorText },
successCallback: (result: any) => {
if (result.user_found) {
queryParamString += `&process_initiator_username=${processInitiatorText}`;
navigateToNewReport(queryParamString);
} else {
setProcessInitiatorNotFoundErrorText(
`The provided username is invalid. Please type the exact username.`
);
}
},
});
} else {
navigateToNewReport(queryParamString);
}
}; };
const dateComponent = ( const dateComponent = (
@ -683,6 +708,7 @@ export default function ProcessInstanceListTable({
items={processStatusAllOptions} items={processStatusAllOptions}
onChange={(selection: any) => { onChange={(selection: any) => {
setProcessStatusSelection(selection.selectedItems); setProcessStatusSelection(selection.selectedItems);
setRequiresRefilter(true);
}} }}
itemToString={(item: any) => { itemToString={(item: any) => {
return item || ''; return item || '';
@ -1142,6 +1168,8 @@ export default function ProcessInstanceListTable({
id="process-instance-initiator-search" id="process-instance-initiator-search"
placeholder="Enter username" placeholder="Enter username"
labelText="Process Initiator" labelText="Process Initiator"
invalid={processInitiatorNotFoundErrorText !== ''}
invalidText={processInitiatorNotFoundErrorText}
onChange={(event: any) => { onChange={(event: any) => {
setProcessInitiatorText(event.target.value); setProcessInitiatorText(event.target.value);
setRequiresRefilter(true); setRequiresRefilter(true);
@ -1450,14 +1478,18 @@ export default function ProcessInstanceListTable({
// eslint-disable-next-line prefer-destructuring // eslint-disable-next-line prefer-destructuring
perPage = perPageOptions[1]; perPage = perPageOptions[1];
} }
let resultsTable = ( let refilterTextComponent = null;
if (requiresRefilter) {
refilterTextComponent = (
<h2> <h2>
Please press the filter button when you have completed updating the Please press the filter button when you have completed updating the
filters. filters.
</h2> </h2>
); );
if (!requiresRefilter) { }
resultsTable = ( const resultsTable = (
<>
{refilterTextComponent}
<PaginationForTable <PaginationForTable
page={page} page={page}
perPage={perPage} perPage={perPage}
@ -1467,8 +1499,8 @@ export default function ProcessInstanceListTable({
perPageOptions={perPageOptions} perPageOptions={perPageOptions}
paginationClassName={paginationClassName} paginationClassName={paginationClassName}
/> />
</>
); );
}
return ( return (
<> <>
{reportColumnForm()} {reportColumnForm()}

View File

@ -30,6 +30,7 @@ export const useUriListForPermissions = () => {
processModelShowPath: `/v1.0/process-models/${params.process_model_id}`, processModelShowPath: `/v1.0/process-models/${params.process_model_id}`,
secretListPath: `/v1.0/secrets`, secretListPath: `/v1.0/secrets`,
userSearch: `/v1.0/users/search`, userSearch: `/v1.0/users/search`,
userExists: `/v1.0/users/exists/by-username`,
}; };
}, [params]); }, [params]);