added new notification component that allows links based on carbons w/ burnettk cullerton
This commit is contained in:
parent
1ccdd7887b
commit
675c776a5a
|
@ -459,7 +459,7 @@ paths:
|
||||||
description: the name of the branch we want to merge into
|
description: the name of the branch we want to merge into
|
||||||
schema:
|
schema:
|
||||||
type: string
|
type: string
|
||||||
put:
|
post:
|
||||||
operationId: spiffworkflow_backend.routes.process_api_blueprint.process_model_publish
|
operationId: spiffworkflow_backend.routes.process_api_blueprint.process_model_publish
|
||||||
summary: Merge changes from this model to another branch.
|
summary: Merge changes from this model to another branch.
|
||||||
tags:
|
tags:
|
||||||
|
|
|
@ -96,8 +96,8 @@ class GitService:
|
||||||
os.system("git push")
|
os.system("git push")
|
||||||
|
|
||||||
# build url for github page to open PR
|
# build url for github page to open PR
|
||||||
output = os.popen("git remote -v").read()
|
output = os.popen("git config --get remote.origin.url").read()
|
||||||
remote_url = output.strip().split("\n")[0].split("\t")[1].split(" ")[0].replace(".git", "")
|
remote_url = output.strip().replace(".git", "")
|
||||||
pr_url = f"{remote_url}/compare/{publish_branch}?expand=1"
|
pr_url = f"{remote_url}/compare/{publish_branch}?expand=1"
|
||||||
|
|
||||||
# try to clean up
|
# try to clean up
|
||||||
|
|
|
@ -2665,7 +2665,7 @@ class TestProcessApi(BaseTest):
|
||||||
assert "new_file.txt" in listing
|
assert "new_file.txt" in listing
|
||||||
|
|
||||||
modified_process_model_id = process_model_identifier.replace("/", ":")
|
modified_process_model_id = process_model_identifier.replace("/", ":")
|
||||||
# response = client.put(
|
# response = client.post(
|
||||||
# f"/v1.0/process-models/{modified_process_model_id}/publish?branch_to_update=staging",
|
# f"/v1.0/process-models/{modified_process_model_id}/publish?branch_to_update=staging",
|
||||||
# headers=self.logged_in_headers(with_super_admin_user),
|
# headers=self.logged_in_headers(with_super_admin_user),
|
||||||
# )
|
# )
|
||||||
|
|
|
@ -0,0 +1,48 @@
|
||||||
|
import React, { useState } from 'react';
|
||||||
|
// @ts-ignore
|
||||||
|
import { Close, CheckmarkFilled } from '@carbon/icons-react';
|
||||||
|
// @ts-ignore
|
||||||
|
import { Button } from '@carbon/react';
|
||||||
|
|
||||||
|
type OwnProps = {
|
||||||
|
title: string;
|
||||||
|
children: React.ReactNode;
|
||||||
|
onClose: (..._args: any[]) => any;
|
||||||
|
type?: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
export function Notification({
|
||||||
|
title,
|
||||||
|
children,
|
||||||
|
onClose,
|
||||||
|
type = 'success',
|
||||||
|
}: OwnProps) {
|
||||||
|
let iconClassName = 'green-icon';
|
||||||
|
if (type === 'error') {
|
||||||
|
iconClassName = 'red-icon';
|
||||||
|
}
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
role="status"
|
||||||
|
className={`with-bottom-margin cds--inline-notification cds--inline-notification--low-contrast cds--inline-notification--${type}`}
|
||||||
|
>
|
||||||
|
<div className="cds--inline-notification__details">
|
||||||
|
<div className="cds--inline-notification__text-wrapper">
|
||||||
|
<CheckmarkFilled className={`${iconClassName} notification-icon`} />
|
||||||
|
<div className="cds--inline-notification__title">{title}</div>
|
||||||
|
<div className="cds--inline-notification__subtitle">{children}</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<Button
|
||||||
|
data-qa="close-publish-notification"
|
||||||
|
renderIcon={Close}
|
||||||
|
iconDescription="Close Notification"
|
||||||
|
className="cds--inline-notification__close-button"
|
||||||
|
hasIconOnly
|
||||||
|
size="sm"
|
||||||
|
kind=""
|
||||||
|
onClick={onClose}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
|
@ -7,7 +7,7 @@ import {
|
||||||
} from 'react-router-dom';
|
} from 'react-router-dom';
|
||||||
|
|
||||||
// @ts-ignore
|
// @ts-ignore
|
||||||
import { Filter, Close, AddAlt } from '@carbon/icons-react';
|
import { Filter, Close, AddAlt, CheckmarkFilled } from '@carbon/icons-react';
|
||||||
import {
|
import {
|
||||||
Button,
|
Button,
|
||||||
ButtonSet,
|
ButtonSet,
|
||||||
|
@ -65,6 +65,7 @@ import ProcessModelSearch from './ProcessModelSearch';
|
||||||
import ProcessInstanceReportSearch from './ProcessInstanceReportSearch';
|
import ProcessInstanceReportSearch from './ProcessInstanceReportSearch';
|
||||||
import ProcessInstanceListSaveAsReport from './ProcessInstanceListSaveAsReport';
|
import ProcessInstanceListSaveAsReport from './ProcessInstanceListSaveAsReport';
|
||||||
import { FormatProcessModelDisplayName } from './MiniComponents';
|
import { FormatProcessModelDisplayName } from './MiniComponents';
|
||||||
|
import { Notification } from './Notification';
|
||||||
|
|
||||||
const REFRESH_INTERVAL = 5;
|
const REFRESH_INTERVAL = 5;
|
||||||
const REFRESH_TIMEOUT = 600;
|
const REFRESH_TIMEOUT = 600;
|
||||||
|
@ -372,18 +373,13 @@ export default function ProcessInstanceListTable({
|
||||||
titleOperation = 'Created';
|
titleOperation = 'Created';
|
||||||
}
|
}
|
||||||
return (
|
return (
|
||||||
<>
|
<Notification title={`Perspective: ${titleOperation}`} onClose={() => setProcessInstanceReportJustSaved(null)}>
|
||||||
<InlineNotification
|
<span>{`'${
|
||||||
title={`Perspective ${titleOperation}:`}
|
|
||||||
subtitle={`'${
|
|
||||||
processInstanceReportSelection
|
processInstanceReportSelection
|
||||||
? processInstanceReportSelection.identifier
|
? processInstanceReportSelection.identifier
|
||||||
: ''
|
: ''
|
||||||
}'`}
|
}'`}</span>
|
||||||
kind="success"
|
</Notification>
|
||||||
/>
|
|
||||||
<br />
|
|
||||||
</>
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
return null;
|
return null;
|
||||||
|
|
|
@ -18,6 +18,7 @@ export const useUriListForPermissions = () => {
|
||||||
processModelCreatePath: `/v1.0/process-models/${params.process_group_id}`,
|
processModelCreatePath: `/v1.0/process-models/${params.process_group_id}`,
|
||||||
processModelFileCreatePath: `/v1.0/process-models/${params.process_model_id}/files`,
|
processModelFileCreatePath: `/v1.0/process-models/${params.process_model_id}/files`,
|
||||||
processModelFileShowPath: `/v1.0/process-models/${params.process_model_id}/files/${params.file_name}`,
|
processModelFileShowPath: `/v1.0/process-models/${params.process_model_id}/files/${params.file_name}`,
|
||||||
|
processModelPublishPath: `/v1.0/process-models/${params.process_model_id}/publish`,
|
||||||
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`,
|
||||||
};
|
};
|
||||||
|
|
|
@ -332,6 +332,14 @@ td.actions-cell {
|
||||||
fill: red;
|
fill: red;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
svg.green-icon {
|
||||||
|
fill: #198038;
|
||||||
|
}
|
||||||
|
|
||||||
|
svg.notification-icon {
|
||||||
|
margin-right: 1rem;
|
||||||
|
}
|
||||||
|
|
||||||
.failure-string {
|
.failure-string {
|
||||||
color: red;
|
color: red;
|
||||||
}
|
}
|
||||||
|
@ -358,7 +366,8 @@ td.actions-cell {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* lime green */
|
||||||
.tag-type-green:hover {
|
.tag-type-green:hover {
|
||||||
background-color: #00FF00;
|
background-color: #80ee90;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -49,6 +49,7 @@ import ProcessInstanceListTable from '../components/ProcessInstanceListTable';
|
||||||
import { usePermissionFetcher } from '../hooks/PermissionService';
|
import { usePermissionFetcher } from '../hooks/PermissionService';
|
||||||
import { useUriListForPermissions } from '../hooks/UriListForPermissions';
|
import { useUriListForPermissions } from '../hooks/UriListForPermissions';
|
||||||
import ProcessInstanceRun from '../components/ProcessInstanceRun';
|
import ProcessInstanceRun from '../components/ProcessInstanceRun';
|
||||||
|
import { Notification } from '../components/Notification';
|
||||||
|
|
||||||
export default function ProcessModelShow() {
|
export default function ProcessModelShow() {
|
||||||
const params = useParams();
|
const params = useParams();
|
||||||
|
@ -62,11 +63,13 @@ export default function ProcessModelShow() {
|
||||||
const [showFileUploadModal, setShowFileUploadModal] =
|
const [showFileUploadModal, setShowFileUploadModal] =
|
||||||
useState<boolean>(false);
|
useState<boolean>(false);
|
||||||
const [processModelPublished, setProcessModelPublished] = useState<any>(null);
|
const [processModelPublished, setProcessModelPublished] = useState<any>(null);
|
||||||
|
const [publishDisabled, setPublishDisabled] = useState<boolean>(false);
|
||||||
const navigate = useNavigate();
|
const navigate = useNavigate();
|
||||||
|
|
||||||
const { targetUris } = useUriListForPermissions();
|
const { targetUris } = useUriListForPermissions();
|
||||||
const permissionRequestData: PermissionsToCheck = {
|
const permissionRequestData: PermissionsToCheck = {
|
||||||
[targetUris.processModelShowPath]: ['PUT', 'DELETE'],
|
[targetUris.processModelShowPath]: ['PUT', 'DELETE'],
|
||||||
|
[targetUris.processModelPublishPath]: ['POST'],
|
||||||
[targetUris.processInstanceListPath]: ['GET'],
|
[targetUris.processInstanceListPath]: ['GET'],
|
||||||
[targetUris.processInstanceCreatePath]: ['POST'],
|
[targetUris.processInstanceCreatePath]: ['POST'],
|
||||||
[targetUris.processModelFileCreatePath]: ['POST', 'PUT', 'GET', 'DELETE'],
|
[targetUris.processModelFileCreatePath]: ['POST', 'PUT', 'GET', 'DELETE'],
|
||||||
|
@ -93,19 +96,17 @@ export default function ProcessModelShow() {
|
||||||
const processInstanceRunResultTag = () => {
|
const processInstanceRunResultTag = () => {
|
||||||
if (processInstance) {
|
if (processInstance) {
|
||||||
return (
|
return (
|
||||||
<div className="alert alert-success with-top-margin" role="alert">
|
<Notification
|
||||||
<p>
|
title="Process Instance Kicked Off:"
|
||||||
Process Instance {processInstance.id} kicked off (
|
onClose={() => setProcessInstance(null)}
|
||||||
|
>
|
||||||
<Link
|
<Link
|
||||||
to={`/admin/process-instances/${modifiedProcessModelId}/${processInstance.id}`}
|
to={`/admin/process-instances/${modifiedProcessModelId}/${processInstance.id}`}
|
||||||
data-qa="process-instance-show-link"
|
data-qa="process-instance-show-link"
|
||||||
>
|
>
|
||||||
view
|
view
|
||||||
</Link>
|
</Link>
|
||||||
).
|
</Notification>
|
||||||
</p>
|
|
||||||
<br />
|
|
||||||
</div>
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
return null;
|
return null;
|
||||||
|
@ -206,15 +207,17 @@ export default function ProcessModelShow() {
|
||||||
};
|
};
|
||||||
|
|
||||||
const postPublish = (value: any) => {
|
const postPublish = (value: any) => {
|
||||||
|
setPublishDisabled(false);
|
||||||
setProcessModelPublished(value);
|
setProcessModelPublished(value);
|
||||||
};
|
};
|
||||||
|
|
||||||
const publishProcessModel = () => {
|
const publishProcessModel = () => {
|
||||||
|
setPublishDisabled(true);
|
||||||
setProcessModelPublished(null);
|
setProcessModelPublished(null);
|
||||||
HttpService.makeCallToBackend({
|
HttpService.makeCallToBackend({
|
||||||
path: `/process-models/${modifiedProcessModelId}/publish`,
|
path: `/process-models/${modifiedProcessModelId}/publish`,
|
||||||
successCallback: postPublish,
|
successCallback: postPublish,
|
||||||
httpMethod: 'PUT',
|
httpMethod: 'POST',
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -529,16 +532,14 @@ export default function ProcessModelShow() {
|
||||||
if (processModelPublished) {
|
if (processModelPublished) {
|
||||||
const prUrl: string = processModelPublished.pr_url;
|
const prUrl: string = processModelPublished.pr_url;
|
||||||
return (
|
return (
|
||||||
<>
|
<Notification
|
||||||
<InlineNotification
|
|
||||||
title="Model Published:"
|
title="Model Published:"
|
||||||
subtitle={`You can view the changes and create a Pull Request at ${prUrl}`}
|
onClose={() => setProcessModelPublished(false)}
|
||||||
kind="success"
|
>
|
||||||
type="banner"
|
<a href={prUrl} target="_void()">
|
||||||
links={prUrl}
|
view the changes and create a Pull Request
|
||||||
/>
|
</a>
|
||||||
<br />
|
</Notification>
|
||||||
</>
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
return null;
|
return null;
|
||||||
|
@ -548,7 +549,6 @@ export default function ProcessModelShow() {
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
{fileUploadModal()}
|
{fileUploadModal()}
|
||||||
{processModelPublishMessage()}
|
|
||||||
<ProcessBreadcrumb
|
<ProcessBreadcrumb
|
||||||
hotCrumbs={[
|
hotCrumbs={[
|
||||||
['Process Groups', '/admin'],
|
['Process Groups', '/admin'],
|
||||||
|
@ -558,6 +558,8 @@ export default function ProcessModelShow() {
|
||||||
},
|
},
|
||||||
]}
|
]}
|
||||||
/>
|
/>
|
||||||
|
{processModelPublishMessage()}
|
||||||
|
{processInstanceRunResultTag()}
|
||||||
<Stack orientation="horizontal" gap={1}>
|
<Stack orientation="horizontal" gap={1}>
|
||||||
<h1 className="with-icons">
|
<h1 className="with-icons">
|
||||||
Process Model: {processModel.display_name}
|
Process Model: {processModel.display_name}
|
||||||
|
@ -603,11 +605,16 @@ export default function ProcessModelShow() {
|
||||||
<br />
|
<br />
|
||||||
</>
|
</>
|
||||||
</Can>
|
</Can>
|
||||||
<div>
|
<Can
|
||||||
<Button onClick={publishProcessModel}>Publish Changes</Button>
|
I="POST"
|
||||||
</div>
|
a={targetUris.processModelPublishPath}
|
||||||
|
ability={ability}
|
||||||
|
>
|
||||||
|
<Button disabled={publishDisabled} onClick={publishProcessModel}>
|
||||||
|
Publish Changes
|
||||||
|
</Button>
|
||||||
|
</Can>
|
||||||
</Stack>
|
</Stack>
|
||||||
{processInstanceRunResultTag()}
|
|
||||||
{processModelFilesSection()}
|
{processModelFilesSection()}
|
||||||
<Can I="GET" a={targetUris.processInstanceListPath} ability={ability}>
|
<Can I="GET" a={targetUris.processInstanceListPath} ability={ability}>
|
||||||
{processInstanceListTableButton()}
|
{processInstanceListTableButton()}
|
||||||
|
|
Loading…
Reference in New Issue