@strapi/review-workflows 0.0.0-next.d2d15ef227d67cce89c2673764c0555c841cd29c → 0.0.0-next.d65d44102fd32871728c0d74ec4f2519b7cc0a16

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (73) hide show
  1. package/dist/_chunks/{Layout-BJOxxAeM.mjs → Layout-DU1Rt4Mu.mjs} +3 -3
  2. package/dist/_chunks/Layout-DU1Rt4Mu.mjs.map +1 -0
  3. package/dist/_chunks/{Layout-DnEAvmeU.js → Layout-ywt0cSLq.js} +3 -3
  4. package/dist/_chunks/Layout-ywt0cSLq.js.map +1 -0
  5. package/dist/_chunks/{en-xcewH2pC.js → en-CYgjfSep.js} +5 -2
  6. package/dist/_chunks/en-CYgjfSep.js.map +1 -0
  7. package/dist/_chunks/{en-D9ZrQAV6.mjs → en-D9dxziEb.mjs} +5 -2
  8. package/dist/_chunks/en-D9dxziEb.mjs.map +1 -0
  9. package/dist/_chunks/{_id-6LK95-rZ.mjs → id-MDQVDdXD.mjs} +73 -14
  10. package/dist/_chunks/id-MDQVDdXD.mjs.map +1 -0
  11. package/dist/_chunks/{_id-DSDzUpwX.js → id-StiN3nkI.js} +73 -14
  12. package/dist/_chunks/id-StiN3nkI.js.map +1 -0
  13. package/dist/_chunks/{index-lJqpw8bs.mjs → index-BRmc4tbQ.mjs} +3 -9
  14. package/dist/_chunks/index-BRmc4tbQ.mjs.map +1 -0
  15. package/dist/_chunks/{index-DZSLya4b.js → index-BSsDJNiQ.js} +3 -9
  16. package/dist/_chunks/index-BSsDJNiQ.js.map +1 -0
  17. package/dist/_chunks/{index-Df1alkCk.mjs → index-CsOMMJ--.mjs} +4 -4
  18. package/dist/_chunks/index-CsOMMJ--.mjs.map +1 -0
  19. package/dist/_chunks/{index-6FZL88pd.js → index-dc2WfugK.js} +4 -4
  20. package/dist/_chunks/index-dc2WfugK.js.map +1 -0
  21. package/dist/_chunks/{router-CZqe-02r.js → router-C5QvEzrI.js} +3 -3
  22. package/dist/_chunks/router-C5QvEzrI.js.map +1 -0
  23. package/dist/_chunks/{router-DeSH-NeW.mjs → router-CZH4yq9s.mjs} +3 -3
  24. package/dist/_chunks/router-CZH4yq9s.mjs.map +1 -0
  25. package/dist/admin/index.js +1 -1
  26. package/dist/admin/index.mjs +1 -1
  27. package/dist/admin/src/services/settings.d.ts +7 -3
  28. package/dist/server/index.js +154 -56
  29. package/dist/server/index.js.map +1 -1
  30. package/dist/server/index.mjs +154 -56
  31. package/dist/server/index.mjs.map +1 -1
  32. package/dist/server/src/bootstrap.d.ts.map +1 -1
  33. package/dist/server/src/constants/workflows.d.ts +1 -0
  34. package/dist/server/src/constants/workflows.d.ts.map +1 -1
  35. package/dist/server/src/content-types/index.d.ts +6 -0
  36. package/dist/server/src/content-types/index.d.ts.map +1 -1
  37. package/dist/server/src/content-types/workflow/index.d.ts +6 -0
  38. package/dist/server/src/content-types/workflow/index.d.ts.map +1 -1
  39. package/dist/server/src/controllers/assignees.d.ts.map +1 -1
  40. package/dist/server/src/controllers/stages.d.ts.map +1 -1
  41. package/dist/server/src/index.d.ts +28 -6
  42. package/dist/server/src/index.d.ts.map +1 -1
  43. package/dist/server/src/register.d.ts.map +1 -1
  44. package/dist/server/src/services/assignees.d.ts +8 -4
  45. package/dist/server/src/services/assignees.d.ts.map +1 -1
  46. package/dist/server/src/services/document-service-middleware.d.ts +1 -0
  47. package/dist/server/src/services/document-service-middleware.d.ts.map +1 -1
  48. package/dist/server/src/services/index.d.ts +16 -6
  49. package/dist/server/src/services/index.d.ts.map +1 -1
  50. package/dist/server/src/services/metrics/index.d.ts +4 -4
  51. package/dist/server/src/services/metrics/index.d.ts.map +1 -1
  52. package/dist/server/src/services/metrics/weekly-metrics.d.ts.map +1 -1
  53. package/dist/server/src/services/stages.d.ts +6 -1
  54. package/dist/server/src/services/stages.d.ts.map +1 -1
  55. package/dist/server/src/services/workflows.d.ts.map +1 -1
  56. package/dist/server/src/validation/review-workflows.d.ts +4 -0
  57. package/dist/server/src/validation/review-workflows.d.ts.map +1 -1
  58. package/dist/shared/contracts/review-workflows.d.ts +8 -3
  59. package/dist/shared/contracts/review-workflows.d.ts.map +1 -1
  60. package/package.json +11 -11
  61. package/dist/_chunks/Layout-BJOxxAeM.mjs.map +0 -1
  62. package/dist/_chunks/Layout-DnEAvmeU.js.map +0 -1
  63. package/dist/_chunks/_id-6LK95-rZ.mjs.map +0 -1
  64. package/dist/_chunks/_id-DSDzUpwX.js.map +0 -1
  65. package/dist/_chunks/en-D9ZrQAV6.mjs.map +0 -1
  66. package/dist/_chunks/en-xcewH2pC.js.map +0 -1
  67. package/dist/_chunks/index-6FZL88pd.js.map +0 -1
  68. package/dist/_chunks/index-DZSLya4b.js.map +0 -1
  69. package/dist/_chunks/index-Df1alkCk.mjs.map +0 -1
  70. package/dist/_chunks/index-lJqpw8bs.mjs.map +0 -1
  71. package/dist/_chunks/router-CZqe-02r.js.map +0 -1
  72. package/dist/_chunks/router-DeSH-NeW.mjs.map +0 -1
  73. /package/dist/admin/src/routes/settings/{:id.d.ts → id.d.ts} +0 -0
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index-dc2WfugK.js","sources":["../../admin/src/routes/settings/index.tsx"],"sourcesContent":["/* eslint-disable check-file/no-index */\n/* eslint-disable check-file/filename-naming-convention */\nimport * as React from 'react';\n\nimport { Page, useTracking, ConfirmDialog, useRBAC, Table } from '@strapi/admin/strapi-admin';\nimport { useLicenseLimits } from '@strapi/admin/strapi-admin/ee';\nimport { Flex, IconButton, TFooter, Typography, LinkButton, Dialog } from '@strapi/design-system';\nimport { Pencil, Plus, Trash } from '@strapi/icons';\nimport { useIntl } from 'react-intl';\nimport { NavLink, Link, useNavigate } from 'react-router-dom';\n\nimport { LimitsModal } from '../../components/LimitsModal';\nimport { CHARGEBEE_WORKFLOW_ENTITLEMENT_NAME } from '../../constants';\nimport { useTypedSelector } from '../../modules/hooks';\nimport { ContentType, useGetContentTypesQuery } from '../../services/content-manager';\n\nimport * as Layout from './components/Layout';\nimport { useReviewWorkflows } from './hooks/useReviewWorkflows';\n\nexport const ReviewWorkflowsListView = () => {\n const { formatMessage } = useIntl();\n const navigate = useNavigate();\n const { trackUsage } = useTracking();\n const [workflowToDelete, setWorkflowToDelete] = React.useState<string | null>(null);\n const [showLimitModal, setShowLimitModal] = React.useState<boolean>(false);\n const { data, isLoading: isLoadingModels } = useGetContentTypesQuery();\n const { meta, workflows, isLoading, delete: deleteAction } = useReviewWorkflows();\n const { getFeature, isLoading: isLicenseLoading } = useLicenseLimits();\n const permissions = useTypedSelector(\n (state) => state.admin_app.permissions.settings?.['review-workflows']\n );\n const {\n allowedActions: { canCreate, canRead, canUpdate, canDelete },\n } = useRBAC(permissions);\n\n const limits = getFeature('review-workflows');\n const numberOfWorkflows = limits?.[CHARGEBEE_WORKFLOW_ENTITLEMENT_NAME] as string;\n\n const handleDeleteWorkflow = (workflowId: string) => {\n setWorkflowToDelete(workflowId);\n };\n\n const toggleConfirmDeleteDialog = () => {\n setWorkflowToDelete(null);\n };\n\n const handleConfirmDeleteDialog = async () => {\n if (!workflowToDelete) return;\n\n await deleteAction(workflowToDelete);\n\n setWorkflowToDelete(null);\n };\n\n const handleCreateClick: React.MouseEventHandler<HTMLAnchorElement> &\n ((event: React.MouseEvent<HTMLButtonElement, MouseEvent>) => void) = (event) => {\n event.preventDefault();\n /**\n * If the current license has a workflow limit:\n * check if the total count of workflows exceeds that limit. If so,\n * prevent the navigation and show the limits overlay.\n *\n * If the current license does not have a limit (e.g. offline license):\n * allow the user to navigate to the create-view. In case they exceed the\n * current hard-limit of 200 they will see an error thrown by the API.\n */\n\n if (numberOfWorkflows && meta && meta?.workflowCount >= parseInt(numberOfWorkflows, 10)) {\n event.preventDefault();\n setShowLimitModal(true);\n } else {\n navigate('create');\n trackUsage('willCreateWorkflow');\n }\n };\n\n /**\n * If the current license has a limit:\n * check if the total count of workflows or stages exceeds that limit and display\n * the limits modal on page load. It can be closed by the user, but the\n * API will throw an error in case they try to create a new workflow or update the\n * stages.\n *\n * If the current license does not have a limit (e.g. offline license):\n * do nothing (for now). In case they are trying to create the 201st workflow/ stage\n * the API will throw an error.\n *\n */\n React.useEffect(() => {\n if (!isLoading && !isLicenseLoading) {\n if (numberOfWorkflows && meta && meta?.workflowCount > parseInt(numberOfWorkflows, 10)) {\n setShowLimitModal(true);\n }\n }\n }, [isLicenseLoading, isLoading, meta, meta?.workflowCount, numberOfWorkflows]);\n\n const headers = [\n {\n label: formatMessage({\n id: 'Settings.review-workflows.list.page.list.column.name.title',\n defaultMessage: 'Name',\n }),\n name: 'name',\n },\n {\n label: formatMessage({\n id: 'Settings.review-workflows.list.page.list.column.stages.title',\n defaultMessage: 'Stages',\n }),\n name: 'stages',\n },\n {\n label: formatMessage({\n id: 'Settings.review-workflows.list.page.list.column.contentTypes.title',\n defaultMessage: 'Content Types',\n }),\n name: 'content-types',\n },\n ];\n\n if (isLoading || isLoadingModels) {\n return <Page.Loading />;\n }\n\n const contentTypes = Object.values(data ?? {}).reduce<ContentType[]>((acc, curr) => {\n acc.push(...curr);\n return acc;\n }, []);\n\n return (\n <>\n <Layout.Header\n primaryAction={\n canCreate ? (\n <LinkButton\n startIcon={<Plus />}\n size=\"S\"\n tag={NavLink}\n to=\"create\"\n onClick={handleCreateClick}\n >\n {formatMessage({\n id: 'Settings.review-workflows.list.page.create',\n defaultMessage: 'Create new workflow',\n })}\n </LinkButton>\n ) : null\n }\n subtitle={formatMessage({\n id: 'Settings.review-workflows.list.page.subtitle',\n defaultMessage: 'Manage your content review process',\n })}\n title={formatMessage({\n id: 'Settings.review-workflows.list.page.title',\n defaultMessage: 'Review Workflows',\n })}\n />\n\n <Layout.Root>\n <Table.Root\n isLoading={isLoading}\n rows={workflows}\n footer={\n canCreate ? (\n <TFooter cursor=\"pointer\" icon={<Plus />} onClick={handleCreateClick}>\n {formatMessage({\n id: 'Settings.review-workflows.list.page.create',\n defaultMessage: 'Create new workflow',\n })}\n </TFooter>\n ) : null\n }\n headers={headers}\n >\n <Table.Content>\n <Table.Head>\n {headers.map((head) => (\n <Table.HeaderCell key={head.name} {...head} />\n ))}\n </Table.Head>\n\n <Table.Body>\n {workflows.map((workflow) => (\n <Table.Row\n onClick={() => {\n navigate(`${workflow.id}`);\n }}\n key={workflow.id}\n >\n <Table.Cell width=\"25rem\">\n <Typography textColor=\"neutral800\" fontWeight=\"bold\" ellipsis>\n {workflow.name}\n </Typography>\n </Table.Cell>\n <Table.Cell>\n <Typography textColor=\"neutral800\">{workflow.stages.length}</Typography>\n </Table.Cell>\n <Table.Cell>\n <Typography textColor=\"neutral800\">\n {workflow.contentTypes\n .map((uid: string) => {\n const contentType = contentTypes.find(\n (contentType) => contentType.uid === uid\n );\n\n return contentType?.info.displayName ?? '';\n })\n .join(', ')}\n </Typography>\n </Table.Cell>\n <Table.Cell>\n <Flex alignItems=\"center\" justifyContent=\"end\">\n {canRead || canUpdate ? (\n <IconButton\n tag={Link}\n to={workflow.id.toString()}\n label={formatMessage(\n {\n id: 'Settings.review-workflows.list.page.list.column.actions.edit.label',\n defaultMessage: 'Edit {name}',\n },\n { name: workflow.name }\n )}\n variant=\"ghost\"\n >\n <Pencil />\n </IconButton>\n ) : null}\n {workflows.length > 1 && canDelete ? (\n <IconButton\n withTooltip={false}\n label={formatMessage(\n {\n id: 'Settings.review-workflows.list.page.list.column.actions.delete.label',\n defaultMessage: 'Delete {name}',\n },\n { name: 'Default workflow' }\n )}\n variant=\"ghost\"\n onClick={(e) => {\n e.stopPropagation();\n handleDeleteWorkflow(String(workflow.id));\n }}\n >\n <Trash />\n </IconButton>\n ) : null}\n </Flex>\n </Table.Cell>\n </Table.Row>\n ))}\n </Table.Body>\n </Table.Content>\n </Table.Root>\n\n <Dialog.Root open={!!workflowToDelete} onOpenChange={toggleConfirmDeleteDialog}>\n <ConfirmDialog onConfirm={handleConfirmDeleteDialog}>\n {formatMessage({\n id: 'Settings.review-workflows.list.page.delete.confirm.body',\n defaultMessage:\n 'If you remove this worfklow, all stage-related information will be removed for this content-type. Are you sure you want to remove it?',\n })}\n </ConfirmDialog>\n </Dialog.Root>\n\n <LimitsModal.Root open={showLimitModal} onOpenChange={() => setShowLimitModal(false)}>\n <LimitsModal.Title>\n {formatMessage({\n id: 'Settings.review-workflows.list.page.workflows.limit.title',\n defaultMessage: 'You’ve reached the limit of workflows in your plan',\n })}\n </LimitsModal.Title>\n\n <LimitsModal.Body>\n {formatMessage({\n id: 'Settings.review-workflows.list.page.workflows.limit.body',\n defaultMessage: 'Delete a workflow or contact Sales to enable more workflows.',\n })}\n </LimitsModal.Body>\n </LimitsModal.Root>\n </Layout.Root>\n </>\n );\n};\n\nconst ProtectedListPage = () => {\n const permissions = useTypedSelector(\n (state) => state.admin_app.permissions.settings?.['review-workflows']?.main\n );\n\n return (\n <Page.Protect permissions={permissions}>\n <ReviewWorkflowsListView />\n </Page.Protect>\n );\n};\n\nexport { ProtectedListPage };\n"],"names":["useIntl","useNavigate","useTracking","React","useGetContentTypesQuery","useReviewWorkflows","useLicenseLimits","useTypedSelector","useRBAC","CHARGEBEE_WORKFLOW_ENTITLEMENT_NAME","jsx","Page","jsxs","Fragment","Layout.Header","LinkButton","Plus","NavLink","Layout.Root","Table","TFooter","Typography","contentType","Flex","IconButton","Link","Pencil","Trash","Dialog","ConfirmDialog","LimitsModal"],"mappings":"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"}
@@ -4,10 +4,10 @@ const jsxRuntime = require("react/jsx-runtime");
4
4
  const React = require("react");
5
5
  const reactRouterDom = require("react-router-dom");
6
6
  const ProtectedListPage = React.lazy(
7
- () => Promise.resolve().then(() => require("./index-6FZL88pd.js")).then((mod) => ({ default: mod.ProtectedListPage }))
7
+ () => Promise.resolve().then(() => require("./index-dc2WfugK.js")).then((mod) => ({ default: mod.ProtectedListPage }))
8
8
  );
9
9
  const ProtectedEditPage = React.lazy(
10
- () => Promise.resolve().then(() => require("./_id-DSDzUpwX.js")).then((mod) => ({ default: mod.ProtectedEditPage }))
10
+ () => Promise.resolve().then(() => require("./id-StiN3nkI.js")).then((mod) => ({ default: mod.ProtectedEditPage }))
11
11
  );
12
12
  const routes = [
13
13
  {
@@ -21,4 +21,4 @@ const routes = [
21
21
  ];
22
22
  const Router = () => /* @__PURE__ */ jsxRuntime.jsx(reactRouterDom.Routes, { children: routes.map((route) => /* @__PURE__ */ jsxRuntime.jsx(reactRouterDom.Route, { ...route }, route.path)) });
23
23
  exports.Router = Router;
24
- //# sourceMappingURL=router-CZqe-02r.js.map
24
+ //# sourceMappingURL=router-C5QvEzrI.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"router-C5QvEzrI.js","sources":["../../admin/src/router.tsx"],"sourcesContent":["/* eslint-disable check-file/filename-naming-convention */\nimport { lazy } from 'react';\n\nimport { Routes, Route, PathRouteProps } from 'react-router-dom';\n\nconst ProtectedListPage = lazy(() =>\n import('./routes/settings').then((mod) => ({ default: mod.ProtectedListPage }))\n);\nconst ProtectedEditPage = lazy(() =>\n import('./routes/settings/id').then((mod) => ({ default: mod.ProtectedEditPage }))\n);\n\nconst routes: PathRouteProps[] = [\n {\n path: '/',\n Component: ProtectedListPage,\n },\n {\n path: ':id',\n Component: ProtectedEditPage,\n },\n];\n\nconst Router = () => (\n <Routes>\n {routes.map((route) => (\n <Route key={route.path} {...route} />\n ))}\n </Routes>\n);\n\nexport { Router };\n"],"names":["lazy","jsx","Routes","Route"],"mappings":";;;;;AAKA,MAAM,oBAAoBA,MAAA;AAAA,EAAK,MAC7B,QAAO,QAAA,EAAA,KAAA,MAAA,QAAA,qBAAmB,GAAE,KAAK,CAAC,SAAS,EAAE,SAAS,IAAI,kBAAA,EAAoB;AAChF;AACA,MAAM,oBAAoBA,MAAA;AAAA,EAAK,MAC7B,QAAO,QAAA,EAAA,KAAA,MAAA,QAAA,kBAAsB,GAAE,KAAK,CAAC,SAAS,EAAE,SAAS,IAAI,kBAAA,EAAoB;AACnF;AAEA,MAAM,SAA2B;AAAA,EAC/B;AAAA,IACE,MAAM;AAAA,IACN,WAAW;AAAA,EACb;AAAA,EACA;AAAA,IACE,MAAM;AAAA,IACN,WAAW;AAAA,EACb;AACF;AAEA,MAAM,SAAS,MACZC,2BAAAA,IAAAC,eAAAA,QAAA,EACE,iBAAO,IAAI,CAAC,UACXD,2BAAA,IAACE,wBAAwB,GAAG,MAAA,GAAhB,MAAM,IAAiB,CACpC,EACH,CAAA;;"}
@@ -2,10 +2,10 @@ import { jsx } from "react/jsx-runtime";
2
2
  import { lazy } from "react";
3
3
  import { Routes, Route } from "react-router-dom";
4
4
  const ProtectedListPage = lazy(
5
- () => import("./index-Df1alkCk.mjs").then((mod) => ({ default: mod.ProtectedListPage }))
5
+ () => import("./index-CsOMMJ--.mjs").then((mod) => ({ default: mod.ProtectedListPage }))
6
6
  );
7
7
  const ProtectedEditPage = lazy(
8
- () => import("./_id-6LK95-rZ.mjs").then((mod) => ({ default: mod.ProtectedEditPage }))
8
+ () => import("./id-MDQVDdXD.mjs").then((mod) => ({ default: mod.ProtectedEditPage }))
9
9
  );
10
10
  const routes = [
11
11
  {
@@ -21,4 +21,4 @@ const Router = () => /* @__PURE__ */ jsx(Routes, { children: routes.map((route)
21
21
  export {
22
22
  Router
23
23
  };
24
- //# sourceMappingURL=router-DeSH-NeW.mjs.map
24
+ //# sourceMappingURL=router-CZH4yq9s.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"router-CZH4yq9s.mjs","sources":["../../admin/src/router.tsx"],"sourcesContent":["/* eslint-disable check-file/filename-naming-convention */\nimport { lazy } from 'react';\n\nimport { Routes, Route, PathRouteProps } from 'react-router-dom';\n\nconst ProtectedListPage = lazy(() =>\n import('./routes/settings').then((mod) => ({ default: mod.ProtectedListPage }))\n);\nconst ProtectedEditPage = lazy(() =>\n import('./routes/settings/id').then((mod) => ({ default: mod.ProtectedEditPage }))\n);\n\nconst routes: PathRouteProps[] = [\n {\n path: '/',\n Component: ProtectedListPage,\n },\n {\n path: ':id',\n Component: ProtectedEditPage,\n },\n];\n\nconst Router = () => (\n <Routes>\n {routes.map((route) => (\n <Route key={route.path} {...route} />\n ))}\n </Routes>\n);\n\nexport { Router };\n"],"names":[],"mappings":";;;AAKA,MAAM,oBAAoB;AAAA,EAAK,MAC7B,OAAO,sBAAmB,EAAE,KAAK,CAAC,SAAS,EAAE,SAAS,IAAI,kBAAA,EAAoB;AAChF;AACA,MAAM,oBAAoB;AAAA,EAAK,MAC7B,OAAO,mBAAsB,EAAE,KAAK,CAAC,SAAS,EAAE,SAAS,IAAI,kBAAA,EAAoB;AACnF;AAEA,MAAM,SAA2B;AAAA,EAC/B;AAAA,IACE,MAAM;AAAA,IACN,WAAW;AAAA,EACb;AAAA,EACA;AAAA,IACE,MAAM;AAAA,IACN,WAAW;AAAA,EACb;AACF;AAEA,MAAM,SAAS,MACZ,oBAAA,QAAA,EACE,iBAAO,IAAI,CAAC,UACX,oBAAC,SAAwB,GAAG,MAAA,GAAhB,MAAM,IAAiB,CACpC,EACH,CAAA;"}
@@ -1,4 +1,4 @@
1
1
  "use strict";
2
- const index = require("../_chunks/index-DZSLya4b.js");
2
+ const index = require("../_chunks/index-BSsDJNiQ.js");
3
3
  module.exports = index.admin;
4
4
  //# sourceMappingURL=index.js.map
@@ -1,4 +1,4 @@
1
- import { h } from "../_chunks/index-lJqpw8bs.mjs";
1
+ import { h } from "../_chunks/index-BRmc4tbQ.mjs";
2
2
  export {
3
3
  h as default
4
4
  };
@@ -1727,13 +1727,17 @@ declare const useGetWorkflowsQuery: import("@reduxjs/toolkit/dist/query/react/bu
1727
1727
  }) | undefined;
1728
1728
  } & import("@strapi/types/dist/modules/entity-service/params/filters").AbstractAttributesFiltering<"admin::review-workflow">) | undefined;
1729
1729
  } & {
1730
- populate?: string | undefined;
1730
+ populate?: import("@strapi/types/dist/modules/entity-service/params/populate").ArrayNotation<"admin::review-workflow"> | undefined;
1731
1731
  }), import("@reduxjs/toolkit/query").BaseQueryFn<string | import("@strapi/admin/strapi-admin").QueryArguments, unknown, import("@strapi/admin/strapi-admin").BaseQueryError, {}, {}>, "ReviewWorkflow" | "ReviewWorkflowStages" | "Document" | "ContentTypeSettings", {
1732
1732
  workflows: GetAll.Response['data'];
1733
1733
  meta?: GetAll.Response['meta'];
1734
1734
  }, "adminApi">>, useCreateWorkflowMutation: import("@reduxjs/toolkit/dist/query/react/buildHooks").UseMutation<import("@reduxjs/toolkit/query").MutationDefinition<{
1735
- data: Omit<import("../../../shared/contracts/review-workflows").Workflow, "id" | "createdAt" | "updatedAt">;
1735
+ data: Omit<import("../../../shared/contracts/review-workflows").Workflow, "id" | "createdAt" | "updatedAt"> & {
1736
+ stageRequiredToPublishName?: string | null | undefined;
1737
+ };
1736
1738
  }, import("@reduxjs/toolkit/query").BaseQueryFn<string | import("@strapi/admin/strapi-admin").QueryArguments, unknown, import("@strapi/admin/strapi-admin").BaseQueryError, {}, {}>, "ReviewWorkflow" | "ReviewWorkflowStages" | "Document" | "ContentTypeSettings", import("../../../shared/contracts/review-workflows").Workflow, "adminApi">>, useDeleteWorkflowMutation: import("@reduxjs/toolkit/dist/query/react/buildHooks").UseMutation<import("@reduxjs/toolkit/query").MutationDefinition<Delete.Params, import("@reduxjs/toolkit/query").BaseQueryFn<string | import("@strapi/admin/strapi-admin").QueryArguments, unknown, import("@strapi/admin/strapi-admin").BaseQueryError, {}, {}>, "ReviewWorkflow" | "ReviewWorkflowStages" | "Document" | "ContentTypeSettings", import("../../../shared/contracts/review-workflows").Workflow, "adminApi">>, useUpdateWorkflowMutation: import("@reduxjs/toolkit/dist/query/react/buildHooks").UseMutation<import("@reduxjs/toolkit/query").MutationDefinition<{
1737
- data: Partial<import("../../../shared/contracts/review-workflows").Workflow>;
1739
+ data: Partial<Omit<import("../../../shared/contracts/review-workflows").Workflow, "stageRequiredToPublish">> & {
1740
+ stageRequiredToPublishName?: string | null | undefined;
1741
+ };
1738
1742
  } & Update.Params, import("@reduxjs/toolkit/query").BaseQueryFn<string | import("@strapi/admin/strapi-admin").QueryArguments, unknown, import("@strapi/admin/strapi-admin").BaseQueryError, {}, {}>, "ReviewWorkflow" | "ReviewWorkflowStages" | "Document" | "ContentTypeSettings", import("../../../shared/contracts/review-workflows").Workflow, "adminApi">>;
1739
1743
  export { useGetWorkflowsQuery, useCreateWorkflowMutation, useDeleteWorkflowMutation, useUpdateWorkflowMutation, type GetWorkflowsParams, };
@@ -6387,7 +6387,8 @@ const WORKFLOW_POPULATE = {
6387
6387
  }
6388
6388
  }
6389
6389
  }
6390
- }
6390
+ },
6391
+ stageRequiredToPublish: true
6391
6392
  };
6392
6393
  function checkVersionThreshold(startVersion, currentVersion, thresholdVersion) {
6393
6394
  return semver$1.gte(currentVersion, thresholdVersion) && semver$1.lt(startVersion, thresholdVersion);
@@ -6562,8 +6563,40 @@ function extendReviewWorkflowContentTypes({ strapi: strapi2 }) {
6562
6563
  });
6563
6564
  }
6564
6565
  }
6566
+ function persistRWOnDowngrade({ strapi: strapi2 }) {
6567
+ const { removePersistedTablesWithSuffix, persistTables } = getAdminService("persist-tables");
6568
+ return async ({ contentTypes: contentTypes2 }) => {
6569
+ const getStageTableToPersist = (contentTypeUID) => {
6570
+ const { attributes, tableName } = strapi2.db.metadata.get(contentTypeUID);
6571
+ const joinTableName = attributes[ENTITY_STAGE_ATTRIBUTE].joinTable.name;
6572
+ return {
6573
+ name: joinTableName,
6574
+ dependsOn: [{ name: tableName }]
6575
+ };
6576
+ };
6577
+ const getAssigneeTableToPersist = (contentTypeUID) => {
6578
+ const { attributes, tableName } = strapi2.db.metadata.get(contentTypeUID);
6579
+ const joinTableName = attributes[ENTITY_ASSIGNEE_ATTRIBUTE].joinTable.name;
6580
+ return {
6581
+ name: joinTableName,
6582
+ dependsOn: [{ name: tableName }]
6583
+ };
6584
+ };
6585
+ const enabledRWContentTypes = fp.pipe([
6586
+ getVisibleContentTypesUID,
6587
+ fp.filter((uid) => hasStageAttribute(contentTypes2[uid]))
6588
+ ])(contentTypes2);
6589
+ const stageJoinTablesToPersist = enabledRWContentTypes.map(getStageTableToPersist);
6590
+ await removePersistedTablesWithSuffix("_strapi_stage_lnk");
6591
+ await persistTables(stageJoinTablesToPersist);
6592
+ const assigneeJoinTablesToPersist = enabledRWContentTypes.map(getAssigneeTableToPersist);
6593
+ await removePersistedTablesWithSuffix("_strapi_assignee_lnk");
6594
+ await persistTables(assigneeJoinTablesToPersist);
6595
+ };
6596
+ }
6565
6597
  const register = async ({ strapi: strapi2 }) => {
6566
6598
  strapi2.hook("strapi::content-types.beforeSync").register(migrateStageAttribute);
6599
+ strapi2.hook("strapi::content-types.afterSync").register(persistRWOnDowngrade({ strapi: strapi2 }));
6567
6600
  strapi2.hook("strapi::content-types.afterSync").register(migrateReviewWorkflowStagesColor).register(migrateReviewWorkflowStagesRoles).register(migrateReviewWorkflowName).register(migrateWorkflowsContentTypes).register(migrateDeletedCTInWorkflows);
6568
6601
  reviewWorkflowsMiddlewares.contentTypeMiddleware(strapi2);
6569
6602
  extendReviewWorkflowContentTypes({ strapi: strapi2 });
@@ -6608,6 +6641,12 @@ const workflow = {
6608
6641
  relation: "oneToMany",
6609
6642
  mappedBy: "workflow"
6610
6643
  },
6644
+ stageRequiredToPublish: {
6645
+ type: "relation",
6646
+ target: "plugin::review-workflows.workflow-stage",
6647
+ relation: "oneToOne",
6648
+ required: false
6649
+ },
6611
6650
  contentTypes: {
6612
6651
  type: "json",
6613
6652
  required: true,
@@ -6757,6 +6796,7 @@ const bootstrap = async (args) => {
6757
6796
  const docsMiddlewares = getService("document-service-middlewares");
6758
6797
  strapi.documents.use(docsMiddlewares.assignStageOnCreate);
6759
6798
  strapi.documents.use(docsMiddlewares.handleStageOnUpdate);
6799
+ strapi.documents.use(docsMiddlewares.checkStageBeforePublish);
6760
6800
  };
6761
6801
  const destroy = async ({ strapi: strapi2 }) => {
6762
6802
  };
@@ -6989,9 +7029,9 @@ const processFilters = ({ strapi: strapi2 }, filters = {}) => {
6989
7029
  };
6990
7030
  const processPopulate = (populate) => {
6991
7031
  if (!populate) {
6992
- return populate;
7032
+ return WORKFLOW_POPULATE;
6993
7033
  }
6994
- return WORKFLOW_POPULATE;
7034
+ return populate;
6995
7035
  };
6996
7036
  const workflows$1 = ({ strapi: strapi2 }) => {
6997
7037
  const workflowsContentTypes = workflowsContentTypesFactory({ strapi: strapi2 });
@@ -7042,14 +7082,27 @@ const workflows$1 = ({ strapi: strapi2 }) => {
7042
7082
  const stages2 = await getService("stages", { strapi: strapi2 }).createMany(opts.data.stages);
7043
7083
  const mapIds = fp.map(fp.get("id"));
7044
7084
  createOpts = fp.set("data.stages", mapIds(stages2), createOpts);
7085
+ if (opts.data.stageRequiredToPublishName) {
7086
+ const stageRequiredToPublish = stages2.find(
7087
+ (stage) => stage.name === opts.data.stageRequiredToPublishName
7088
+ );
7089
+ if (!stageRequiredToPublish) {
7090
+ throw new utils.errors.ApplicationError("Stage required to publish does not exist");
7091
+ }
7092
+ createOpts = fp.set("data.stageRequiredToPublish", stageRequiredToPublish.id, createOpts);
7093
+ }
7045
7094
  if (opts.data.contentTypes) {
7046
7095
  await workflowsContentTypes.migrate({
7047
7096
  destContentTypes: opts.data.contentTypes,
7048
7097
  stageId: stages2[0].id
7049
7098
  });
7050
7099
  }
7051
- metrics.sendDidCreateWorkflow();
7052
- return strapi2.db.query(WORKFLOW_MODEL_UID).create(strapi2.get("query-params").transform(WORKFLOW_MODEL_UID, createOpts));
7100
+ const createdWorkflow = await strapi2.db.query(WORKFLOW_MODEL_UID).create(strapi2.get("query-params").transform(WORKFLOW_MODEL_UID, createOpts));
7101
+ metrics.sendDidCreateWorkflow(createdWorkflow.id, !!opts.data.stageRequiredToPublishName);
7102
+ if (opts.data.stageRequiredToPublishName) {
7103
+ await strapi2.plugin("content-releases").service("release-action").validateActionsByContentTypes(opts.data.contentTypes);
7104
+ }
7105
+ return createdWorkflow;
7053
7106
  });
7054
7107
  },
7055
7108
  /**
@@ -7062,6 +7115,7 @@ const workflows$1 = ({ strapi: strapi2 }) => {
7062
7115
  async update(workflow2, opts) {
7063
7116
  const stageService = getService("stages", { strapi: strapi2 });
7064
7117
  let updateOpts = { ...opts, populate: { ...WORKFLOW_POPULATE } };
7118
+ let updatedStages = [];
7065
7119
  let updatedStageIds;
7066
7120
  await workflowValidator.validateWorkflowCount();
7067
7121
  return strapi2.db.transaction(async () => {
@@ -7070,9 +7124,28 @@ const workflows$1 = ({ strapi: strapi2 }) => {
7070
7124
  opts.data.stages.forEach(
7071
7125
  (stage) => this.assertStageBelongsToWorkflow(stage.id, workflow2)
7072
7126
  );
7073
- updatedStageIds = await stageService.replaceStages(workflow2.stages, opts.data.stages, workflow2.contentTypes).then((stages2) => stages2.map((stage) => stage.id));
7127
+ updatedStages = await stageService.replaceStages(
7128
+ workflow2.stages,
7129
+ opts.data.stages,
7130
+ workflow2.contentTypes
7131
+ );
7132
+ updatedStageIds = updatedStages.map((stage) => stage.id);
7074
7133
  updateOpts = fp.set("data.stages", updatedStageIds, updateOpts);
7075
7134
  }
7135
+ if (opts.data.stageRequiredToPublishName !== void 0) {
7136
+ const stages2 = updatedStages ?? workflow2.stages;
7137
+ if (opts.data.stageRequiredToPublishName === null) {
7138
+ updateOpts = fp.set("data.stageRequiredToPublish", null, updateOpts);
7139
+ } else {
7140
+ const stageRequiredToPublish = stages2.find(
7141
+ (stage) => stage.name === opts.data.stageRequiredToPublishName
7142
+ );
7143
+ if (!stageRequiredToPublish) {
7144
+ throw new utils.errors.ApplicationError("Stage required to publish does not exist");
7145
+ }
7146
+ updateOpts = fp.set("data.stageRequiredToPublish", stageRequiredToPublish.id, updateOpts);
7147
+ }
7148
+ }
7076
7149
  if (opts.data.contentTypes) {
7077
7150
  await workflowsContentTypes.migrate({
7078
7151
  srcContentTypes: workflow2.contentTypes,
@@ -7080,12 +7153,17 @@ const workflows$1 = ({ strapi: strapi2 }) => {
7080
7153
  stageId: updatedStageIds ? updatedStageIds[0] : workflow2.stages[0].id
7081
7154
  });
7082
7155
  }
7083
- metrics.sendDidEditWorkflow();
7156
+ metrics.sendDidEditWorkflow(workflow2.id, !!opts.data.stageRequiredToPublishName);
7084
7157
  const query = strapi2.get("query-params").transform(WORKFLOW_MODEL_UID, updateOpts);
7085
- return strapi2.db.query(WORKFLOW_MODEL_UID).update({
7158
+ const updatedWorkflow = await strapi2.db.query(WORKFLOW_MODEL_UID).update({
7086
7159
  ...query,
7087
7160
  where: { id: workflow2.id }
7088
7161
  });
7162
+ await strapi2.plugin("content-releases").service("release-action").validateActionsByContentTypes([
7163
+ ...workflow2.contentTypes,
7164
+ ...opts.data.contentTypes || []
7165
+ ]);
7166
+ return updatedWorkflow;
7089
7167
  });
7090
7168
  },
7091
7169
  /**
@@ -7108,10 +7186,12 @@ const workflows$1 = ({ strapi: strapi2 }) => {
7108
7186
  destContentTypes: []
7109
7187
  });
7110
7188
  const query = strapi2.get("query-params").transform(WORKFLOW_MODEL_UID, opts);
7111
- return strapi2.db.query(WORKFLOW_MODEL_UID).delete({
7189
+ const deletedWorkflow = await strapi2.db.query(WORKFLOW_MODEL_UID).delete({
7112
7190
  ...query,
7113
7191
  where: { id: workflow2.id }
7114
7192
  });
7193
+ await strapi2.plugin("content-releases").service("release-action").validateActionsByContentTypes(workflow2.contentTypes);
7194
+ return deletedWorkflow;
7115
7195
  });
7116
7196
  },
7117
7197
  /**
@@ -7323,8 +7403,9 @@ const stages$1 = ({ strapi: strapi2 }) => {
7323
7403
  /**
7324
7404
  * Update the stage of an entity
7325
7405
  */
7326
- async updateEntity(documentId, locale, model, stageId) {
7406
+ async updateEntity(entityToUpdate, model, stageId) {
7327
7407
  const stage = await this.findById(stageId);
7408
+ const { documentId, locale } = entityToUpdate;
7328
7409
  await workflowValidator.validateWorkflowCount();
7329
7410
  if (!stage) {
7330
7411
  throw new ApplicationError$2(`Selected stage does not exist`);
@@ -7332,9 +7413,15 @@ const stages$1 = ({ strapi: strapi2 }) => {
7332
7413
  const entity = await strapi2.documents(model).update({
7333
7414
  documentId,
7334
7415
  locale,
7335
- data: { [ENTITY_STAGE_ATTRIBUTE]: stage },
7416
+ // Stage doesn't have DP or i18n enabled, connecting it through the `id`
7417
+ // will be safer than relying on the `documentId` + `locale` + `status` transformation
7418
+ data: { [ENTITY_STAGE_ATTRIBUTE]: fp.pick(["id"], stage) },
7336
7419
  populate: [ENTITY_STAGE_ATTRIBUTE]
7337
7420
  });
7421
+ const { tableName } = strapi2.db.metadata.get(model);
7422
+ await strapi2.db.connection(tableName).where({ id: entityToUpdate.id }).update({
7423
+ updated_at: new Date(entityToUpdate.updatedAt)
7424
+ });
7338
7425
  metrics.sendDidChangeEntryStage();
7339
7426
  return entity;
7340
7427
  },
@@ -7481,32 +7568,28 @@ const assignees$1 = ({ strapi: strapi2 }) => {
7481
7568
  /**
7482
7569
  * Update the assignee of an entity
7483
7570
  */
7484
- async updateEntityAssignee(documentId, locale, model, assigneeId) {
7485
- if (fp.isNil(assigneeId)) {
7486
- return this.deleteEntityAssignee(documentId, locale, model);
7487
- }
7488
- const userExists = await getAdminService("user", { strapi: strapi2 }).exists({ id: assigneeId });
7489
- if (!userExists) {
7490
- throw new ApplicationError(`Selected user does not exist`);
7571
+ async updateEntityAssignee(entityToUpdate, model, assigneeId) {
7572
+ const { documentId, locale } = entityToUpdate;
7573
+ if (!fp.isNil(assigneeId)) {
7574
+ const userExists = await getAdminService("user", { strapi: strapi2 }).exists({ id: assigneeId });
7575
+ if (!userExists) {
7576
+ throw new ApplicationError(`Selected user does not exist`);
7577
+ }
7491
7578
  }
7492
- metrics.sendDidEditAssignee(await this.findEntityAssigneeId(documentId, model), assigneeId);
7493
- return strapi2.documents(model).update({
7579
+ const oldAssigneeId = await this.findEntityAssigneeId(entityToUpdate.id, model);
7580
+ metrics.sendDidEditAssignee(oldAssigneeId, assigneeId || null);
7581
+ const entity = await strapi2.documents(model).update({
7494
7582
  documentId,
7495
7583
  locale,
7496
- data: { [ENTITY_ASSIGNEE_ATTRIBUTE]: assigneeId },
7584
+ data: { [ENTITY_ASSIGNEE_ATTRIBUTE]: assigneeId || null },
7497
7585
  populate: [ENTITY_ASSIGNEE_ATTRIBUTE],
7498
7586
  fields: []
7499
7587
  });
7500
- },
7501
- async deleteEntityAssignee(documentId, locale, model) {
7502
- metrics.sendDidEditAssignee(await this.findEntityAssigneeId(documentId, model), null);
7503
- return strapi2.documents(model).update({
7504
- documentId,
7505
- locale,
7506
- data: { [ENTITY_ASSIGNEE_ATTRIBUTE]: null },
7507
- populate: [ENTITY_ASSIGNEE_ATTRIBUTE],
7508
- fields: []
7588
+ const { tableName } = strapi2.db.metadata.get(model);
7589
+ await strapi2.db.connection(tableName).where({ id: entityToUpdate.id }).update({
7590
+ updated_at: new Date(entityToUpdate.updatedAt)
7509
7591
  });
7592
+ return entity;
7510
7593
  }
7511
7594
  };
7512
7595
  };
@@ -7579,11 +7662,11 @@ const sendDidDeleteStage = async () => {
7579
7662
  const sendDidChangeEntryStage = async () => {
7580
7663
  strapi.telemetry.send("didChangeEntryStage", {});
7581
7664
  };
7582
- const sendDidCreateWorkflow = async () => {
7583
- strapi.telemetry.send("didCreateWorkflow", {});
7665
+ const sendDidCreateWorkflow = async (workflowId, hasRequiredStageToPublish) => {
7666
+ strapi.telemetry.send("didCreateWorkflow", { workflowId, hasRequiredStageToPublish });
7584
7667
  };
7585
- const sendDidEditWorkflow = async () => {
7586
- strapi.telemetry.send("didEditWorkflow", {});
7668
+ const sendDidEditWorkflow = async (workflowId, hasRequiredStageToPublish) => {
7669
+ strapi.telemetry.send("didEditWorkflow", { workflowId, hasRequiredStageToPublish });
7587
7670
  };
7588
7671
  const sendDidEditAssignee = async (fromId, toId) => {
7589
7672
  strapi.telemetry.send("didEditAssignee", { from: fromId, to: toId });
@@ -7608,13 +7691,13 @@ const reviewWorkflowsMetrics = {
7608
7691
  sendDidSendReviewWorkflowPropertiesOnceAWeek,
7609
7692
  sendDidEditAssignee
7610
7693
  };
7611
- function _typeof(obj) {
7694
+ function _typeof(o) {
7612
7695
  "@babel/helpers - typeof";
7613
- return _typeof = "function" == typeof Symbol && "symbol" == typeof Symbol.iterator ? function(obj2) {
7614
- return typeof obj2;
7615
- } : function(obj2) {
7616
- return obj2 && "function" == typeof Symbol && obj2.constructor === Symbol && obj2 !== Symbol.prototype ? "symbol" : typeof obj2;
7617
- }, _typeof(obj);
7696
+ return _typeof = "function" == typeof Symbol && "symbol" == typeof Symbol.iterator ? function(o2) {
7697
+ return typeof o2;
7698
+ } : function(o2) {
7699
+ return o2 && "function" == typeof Symbol && o2.constructor === Symbol && o2 !== Symbol.prototype ? "symbol" : typeof o2;
7700
+ }, _typeof(o);
7618
7701
  }
7619
7702
  function toInteger(dirtyNumber) {
7620
7703
  if (dirtyNumber === null || dirtyNumber === true || dirtyNumber === false) {
@@ -7749,7 +7832,12 @@ const reviewWorkflowsWeeklyMetrics = ({ strapi: strapi2 }) => {
7749
7832
  },
7750
7833
  async registerCron() {
7751
7834
  const weeklySchedule = await this.ensureWeeklyStoredCronSchedule();
7752
- strapi2.cron.add({ [weeklySchedule]: this.sendMetrics.bind(this) });
7835
+ strapi2.cron.add({
7836
+ reviewWorkflowsWeekly: {
7837
+ task: this.sendMetrics.bind(this),
7838
+ options: weeklySchedule
7839
+ }
7840
+ });
7753
7841
  }
7754
7842
  };
7755
7843
  };
@@ -7831,9 +7919,27 @@ const handleStageOnUpdate = async (ctx, next) => {
7831
7919
  }
7832
7920
  return next();
7833
7921
  };
7922
+ const checkStageBeforePublish = async (ctx, next) => {
7923
+ if (ctx.action !== "publish") {
7924
+ return next();
7925
+ }
7926
+ const workflow2 = await getService("workflows").getAssignedWorkflow(ctx.contentType.uid, {
7927
+ populate: "stageRequiredToPublish"
7928
+ });
7929
+ if (!workflow2 || !workflow2.stageRequiredToPublish) {
7930
+ return next();
7931
+ }
7932
+ const { documentId } = ctx.params;
7933
+ const entryStage = await getEntityStage(ctx.contentType.uid, documentId, ctx.params);
7934
+ if (entryStage.id !== workflow2.stageRequiredToPublish.id) {
7935
+ throw new utils.errors.ValidationError("Entry is not at the required stage to publish");
7936
+ }
7937
+ return next();
7938
+ };
7834
7939
  const documentServiceMiddleware = () => ({
7835
7940
  assignStageOnCreate,
7836
- handleStageOnUpdate
7941
+ handleStageOnUpdate,
7942
+ checkStageBeforePublish
7837
7943
  });
7838
7944
  const services = {
7839
7945
  workflows: workflows$1,
@@ -7883,12 +7989,14 @@ const validateContentTypes = utils.yup.array().of(
7883
7989
  const validateWorkflowCreateSchema = utils.yup.object().shape({
7884
7990
  name: utils.yup.string().max(255).min(1, "Workflow name can not be empty").required(),
7885
7991
  stages: utils.yup.array().of(stageObject).uniqueProperty("name", "Stage name must be unique").min(1, "Can not create a workflow without stages").max(200, "Can not have more than 200 stages").required("Can not create a workflow without stages"),
7886
- contentTypes: validateContentTypes
7992
+ contentTypes: validateContentTypes,
7993
+ stageRequiredToPublishName: utils.yup.string().min(1).nullable()
7887
7994
  });
7888
7995
  const validateWorkflowUpdateSchema = utils.yup.object().shape({
7889
7996
  name: utils.yup.string().max(255).min(1, "Workflow name can not be empty"),
7890
7997
  stages: utils.yup.array().of(stageObject).uniqueProperty("name", "Stage name must be unique").min(1, "Can not update a workflow without stages").max(200, "Can not have more than 200 stages"),
7891
- contentTypes: validateContentTypes
7998
+ contentTypes: validateContentTypes,
7999
+ stageRequiredToPublishName: utils.yup.string().min(1).nullable()
7892
8000
  });
7893
8001
  const validateUpdateAssigneeOnEntitySchema = utils.yup.object().shape({
7894
8002
  id: utils.yup.number().integer().min(1).nullable()
@@ -8091,12 +8199,7 @@ const stages = {
8091
8199
  );
8092
8200
  const workflow2 = await workflowService.assertContentTypeBelongsToWorkflow(modelUID);
8093
8201
  workflowService.assertStageBelongsToWorkflow(stageId, workflow2);
8094
- const updatedEntity = await stagesService.updateEntity(
8095
- entity.documentId,
8096
- entity.locale,
8097
- modelUID,
8098
- stageId
8099
- );
8202
+ const updatedEntity = await stagesService.updateEntity(entity, modelUID, stageId);
8100
8203
  ctx.body = { data: await sanitizeOutput(updatedEntity) };
8101
8204
  },
8102
8205
  /**
@@ -8193,12 +8296,7 @@ const assignees = {
8193
8296
  "You should pass a valid id to the body of the put request."
8194
8297
  );
8195
8298
  await workflowService.assertContentTypeBelongsToWorkflow(model);
8196
- const updatedEntity = await assigneeService.updateEntityAssignee(
8197
- documentId,
8198
- locale || null,
8199
- model,
8200
- assigneeId
8201
- );
8299
+ const updatedEntity = await assigneeService.updateEntityAssignee(entity, model, assigneeId);
8202
8300
  ctx.body = { data: await sanitizeOutput(updatedEntity) };
8203
8301
  }
8204
8302
  };