@strapi/plugin-users-permissions 0.0.0-next.f5107c72369ee86a8ec1b6782f2e890925033ca3 → 0.0.0-next.f6dca5adf05ef6bed9605a1535999ab0bbbf063e

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 (218) hide show
  1. package/.eslintignore +1 -2
  2. package/.eslintrc +17 -0
  3. package/LICENSE +18 -3
  4. package/admin/src/components/BoundRoute/{index.js → index.jsx} +2 -2
  5. package/admin/src/components/FormModal/Input/{index.js → index.jsx} +32 -31
  6. package/admin/src/components/FormModal/index.jsx +115 -0
  7. package/admin/src/components/Permissions/PermissionRow/{CheckboxWrapper.js → CheckboxWrapper.jsx} +4 -3
  8. package/admin/src/components/Permissions/PermissionRow/{SubCategory.js → SubCategory.jsx} +13 -22
  9. package/admin/src/components/Permissions/index.jsx +47 -0
  10. package/admin/src/components/Permissions/reducer.js +1 -1
  11. package/admin/src/components/Policies/{index.js → index.jsx} +7 -5
  12. package/admin/src/components/UsersPermissions/{index.js → index.jsx} +15 -7
  13. package/admin/src/components/UsersPermissions/reducer.js +1 -1
  14. package/admin/src/index.js +19 -47
  15. package/admin/src/pages/AdvancedSettings/index.jsx +214 -0
  16. package/admin/src/pages/AdvancedSettings/utils/layout.js +20 -35
  17. package/admin/src/pages/AdvancedSettings/utils/schema.js +5 -2
  18. package/admin/src/pages/EmailTemplates/components/EmailForm.jsx +156 -0
  19. package/admin/src/pages/EmailTemplates/components/{EmailTable.js → EmailTable.jsx} +20 -17
  20. package/admin/src/pages/EmailTemplates/index.jsx +148 -0
  21. package/admin/src/pages/EmailTemplates/utils/schema.js +18 -6
  22. package/admin/src/pages/Providers/index.jsx +262 -0
  23. package/admin/src/pages/Providers/utils/forms.js +23 -11
  24. package/admin/src/pages/Roles/constants.js +3 -3
  25. package/admin/src/{hooks → pages/Roles/hooks}/usePlugins.js +19 -12
  26. package/admin/src/pages/Roles/index.jsx +24 -0
  27. package/admin/src/pages/Roles/pages/CreatePage.jsx +194 -0
  28. package/admin/src/pages/Roles/pages/EditPage.jsx +215 -0
  29. package/admin/src/pages/Roles/pages/ListPage/components/TableBody.jsx +119 -0
  30. package/admin/src/pages/Roles/{ListPage/index.js → pages/ListPage/index.jsx} +93 -65
  31. package/admin/src/translations/en.json +1 -1
  32. package/admin/src/translations/zh-Hans.json +80 -80
  33. package/admin/src/utils/prefixPluginTranslations.js +13 -0
  34. package/dist/_chunks/ar-BguGUqwK.js +44 -0
  35. package/dist/_chunks/ar-BguGUqwK.js.map +1 -0
  36. package/dist/_chunks/ar-CK8BRRXB.mjs +44 -0
  37. package/dist/_chunks/ar-CK8BRRXB.mjs.map +1 -0
  38. package/dist/_chunks/cs-BVigMk0l.mjs +50 -0
  39. package/dist/_chunks/cs-BVigMk0l.mjs.map +1 -0
  40. package/dist/_chunks/cs-BW8-K_GY.js +50 -0
  41. package/dist/_chunks/cs-BW8-K_GY.js.map +1 -0
  42. package/dist/_chunks/de-BKUdRFI4.mjs +62 -0
  43. package/dist/_chunks/de-BKUdRFI4.mjs.map +1 -0
  44. package/dist/_chunks/de-owXpVluI.js +62 -0
  45. package/dist/_chunks/de-owXpVluI.js.map +1 -0
  46. package/dist/_chunks/dk-BQiTK50l.mjs +86 -0
  47. package/dist/_chunks/dk-BQiTK50l.mjs.map +1 -0
  48. package/dist/_chunks/dk-LXAnbuBk.js +86 -0
  49. package/dist/_chunks/dk-LXAnbuBk.js.map +1 -0
  50. package/dist/_chunks/en-DOHtPf-2.mjs +86 -0
  51. package/dist/_chunks/en-DOHtPf-2.mjs.map +1 -0
  52. package/dist/_chunks/en-MHo5mcsU.js +86 -0
  53. package/dist/_chunks/en-MHo5mcsU.js.map +1 -0
  54. package/dist/_chunks/es-BwLCLXAQ.js +86 -0
  55. package/dist/_chunks/es-BwLCLXAQ.js.map +1 -0
  56. package/dist/_chunks/es-DNgOVMjD.mjs +86 -0
  57. package/dist/_chunks/es-DNgOVMjD.mjs.map +1 -0
  58. package/dist/_chunks/fr-DkgRugiU.mjs +50 -0
  59. package/dist/_chunks/fr-DkgRugiU.mjs.map +1 -0
  60. package/dist/_chunks/fr-DkhpSjjm.js +50 -0
  61. package/dist/_chunks/fr-DkhpSjjm.js.map +1 -0
  62. package/dist/_chunks/id-BTemOeTZ.js +62 -0
  63. package/dist/_chunks/id-BTemOeTZ.js.map +1 -0
  64. package/dist/_chunks/id-BdEsvnaF.mjs +62 -0
  65. package/dist/_chunks/id-BdEsvnaF.mjs.map +1 -0
  66. package/dist/_chunks/index-BH6NqE8I.js +245 -0
  67. package/dist/_chunks/index-BH6NqE8I.js.map +1 -0
  68. package/dist/_chunks/index-C3i__jX-.js +281 -0
  69. package/dist/_chunks/index-C3i__jX-.js.map +1 -0
  70. package/dist/_chunks/index-CFzpNgIL.js +640 -0
  71. package/dist/_chunks/index-CFzpNgIL.js.map +1 -0
  72. package/dist/_chunks/index-CJUWqtrZ.mjs +344 -0
  73. package/dist/_chunks/index-CJUWqtrZ.mjs.map +1 -0
  74. package/dist/_chunks/index-CcWTKtCY.js +366 -0
  75. package/dist/_chunks/index-CcWTKtCY.js.map +1 -0
  76. package/dist/_chunks/index-CnX2wiLY.mjs +1142 -0
  77. package/dist/_chunks/index-CnX2wiLY.mjs.map +1 -0
  78. package/dist/_chunks/index-DUWx-KG9-C1E7ADbJ.mjs +11984 -0
  79. package/dist/_chunks/index-DUWx-KG9-C1E7ADbJ.mjs.map +1 -0
  80. package/dist/_chunks/index-DUWx-KG9-fMZ4nmrb.js +12008 -0
  81. package/dist/_chunks/index-DUWx-KG9-fMZ4nmrb.js.map +1 -0
  82. package/dist/_chunks/index-DruWuFWL.js +1172 -0
  83. package/dist/_chunks/index-DruWuFWL.js.map +1 -0
  84. package/dist/_chunks/index-PJoz36kJ.mjs +262 -0
  85. package/dist/_chunks/index-PJoz36kJ.mjs.map +1 -0
  86. package/dist/_chunks/index-R4R0dEOv.mjs +246 -0
  87. package/dist/_chunks/index-R4R0dEOv.mjs.map +1 -0
  88. package/dist/_chunks/index-XOcQhJMB.mjs +617 -0
  89. package/dist/_chunks/index-XOcQhJMB.mjs.map +1 -0
  90. package/dist/_chunks/it-B-rv0E24.mjs +62 -0
  91. package/dist/_chunks/it-B-rv0E24.mjs.map +1 -0
  92. package/dist/_chunks/it-D1rH6V6_.js +62 -0
  93. package/dist/_chunks/it-D1rH6V6_.js.map +1 -0
  94. package/dist/_chunks/ja-C8K-VBPD.mjs +48 -0
  95. package/dist/_chunks/ja-C8K-VBPD.mjs.map +1 -0
  96. package/dist/_chunks/ja-DqShgTMf.js +48 -0
  97. package/dist/_chunks/ja-DqShgTMf.js.map +1 -0
  98. package/dist/_chunks/ko-B9DGEPWH.js +86 -0
  99. package/dist/_chunks/ko-B9DGEPWH.js.map +1 -0
  100. package/dist/_chunks/ko-Busb0wIY.mjs +86 -0
  101. package/dist/_chunks/ko-Busb0wIY.mjs.map +1 -0
  102. package/dist/_chunks/ms-ByvsQjRt.mjs +49 -0
  103. package/dist/_chunks/ms-ByvsQjRt.mjs.map +1 -0
  104. package/dist/_chunks/ms-CPBU3LWf.js +49 -0
  105. package/dist/_chunks/ms-CPBU3LWf.js.map +1 -0
  106. package/dist/_chunks/nl-5qO8Rpcy.mjs +48 -0
  107. package/dist/_chunks/nl-5qO8Rpcy.mjs.map +1 -0
  108. package/dist/_chunks/nl-CwNB6YoO.js +48 -0
  109. package/dist/_chunks/nl-CwNB6YoO.js.map +1 -0
  110. package/dist/_chunks/pl-BdIzifBE.mjs +86 -0
  111. package/dist/_chunks/pl-BdIzifBE.mjs.map +1 -0
  112. package/dist/_chunks/pl-Do9UD69f.js +86 -0
  113. package/dist/_chunks/pl-Do9UD69f.js.map +1 -0
  114. package/dist/_chunks/pt-BIO24ioG.mjs +48 -0
  115. package/dist/_chunks/pt-BIO24ioG.mjs.map +1 -0
  116. package/dist/_chunks/pt-BR-D7dZhxuP.js +44 -0
  117. package/dist/_chunks/pt-BR-D7dZhxuP.js.map +1 -0
  118. package/dist/_chunks/pt-BR-f0p23AQZ.mjs +44 -0
  119. package/dist/_chunks/pt-BR-f0p23AQZ.mjs.map +1 -0
  120. package/dist/_chunks/pt-fdvyOnUp.js +48 -0
  121. package/dist/_chunks/pt-fdvyOnUp.js.map +1 -0
  122. package/dist/_chunks/ru-C94rjPGA.js +86 -0
  123. package/dist/_chunks/ru-C94rjPGA.js.map +1 -0
  124. package/dist/_chunks/ru-VWy-IB7K.mjs +86 -0
  125. package/dist/_chunks/ru-VWy-IB7K.mjs.map +1 -0
  126. package/dist/_chunks/sk-BABEhykl.js +50 -0
  127. package/dist/_chunks/sk-BABEhykl.js.map +1 -0
  128. package/dist/_chunks/sk-B_LIcepm.mjs +50 -0
  129. package/dist/_chunks/sk-B_LIcepm.mjs.map +1 -0
  130. package/dist/_chunks/sv-ABLKOokl.mjs +86 -0
  131. package/dist/_chunks/sv-ABLKOokl.mjs.map +1 -0
  132. package/dist/_chunks/sv-Be43LhA9.js +86 -0
  133. package/dist/_chunks/sv-Be43LhA9.js.map +1 -0
  134. package/dist/_chunks/th-DKyP7ueR.mjs +60 -0
  135. package/dist/_chunks/th-DKyP7ueR.mjs.map +1 -0
  136. package/dist/_chunks/th-DgVhVLhL.js +60 -0
  137. package/dist/_chunks/th-DgVhVLhL.js.map +1 -0
  138. package/dist/_chunks/tr-B_idhkEs.js +85 -0
  139. package/dist/_chunks/tr-B_idhkEs.js.map +1 -0
  140. package/dist/_chunks/tr-qa1Q5UjC.mjs +85 -0
  141. package/dist/_chunks/tr-qa1Q5UjC.mjs.map +1 -0
  142. package/dist/_chunks/uk-BmRqbeQc.mjs +49 -0
  143. package/dist/_chunks/uk-BmRqbeQc.mjs.map +1 -0
  144. package/dist/_chunks/uk-LHOivnhP.js +49 -0
  145. package/dist/_chunks/uk-LHOivnhP.js.map +1 -0
  146. package/dist/_chunks/vi-CdVRdKDw.js +50 -0
  147. package/dist/_chunks/vi-CdVRdKDw.js.map +1 -0
  148. package/dist/_chunks/vi-HW-EdMea.mjs +50 -0
  149. package/dist/_chunks/vi-HW-EdMea.mjs.map +1 -0
  150. package/dist/_chunks/zh-5hKkVPA4.mjs +86 -0
  151. package/dist/_chunks/zh-5hKkVPA4.mjs.map +1 -0
  152. package/dist/_chunks/zh-Cuq8gMnF.js +86 -0
  153. package/dist/_chunks/zh-Cuq8gMnF.js.map +1 -0
  154. package/dist/_chunks/zh-Hans-BHilK-yc.mjs +86 -0
  155. package/dist/_chunks/zh-Hans-BHilK-yc.mjs.map +1 -0
  156. package/dist/_chunks/zh-Hans-GQDMKtY4.js +86 -0
  157. package/dist/_chunks/zh-Hans-GQDMKtY4.js.map +1 -0
  158. package/dist/admin/index.js +4 -0
  159. package/dist/admin/index.js.map +1 -0
  160. package/dist/admin/index.mjs +5 -0
  161. package/dist/admin/index.mjs.map +1 -0
  162. package/jest.config.front.js +1 -1
  163. package/package.json +46 -27
  164. package/packup.config.ts +22 -0
  165. package/server/bootstrap/index.js +18 -15
  166. package/server/bootstrap/users-permissions-actions.js +6 -0
  167. package/server/config.js +29 -0
  168. package/server/content-types/user/index.js +0 -1
  169. package/server/controllers/auth.js +74 -38
  170. package/server/controllers/content-manager-user.js +28 -30
  171. package/server/controllers/role.js +17 -4
  172. package/server/controllers/user.js +18 -8
  173. package/server/controllers/validation/auth.js +81 -25
  174. package/server/middlewares/rateLimit.js +1 -1
  175. package/server/register.js +1 -1
  176. package/server/services/jwt.js +3 -3
  177. package/server/services/permission.js +3 -7
  178. package/server/services/providers-registry.js +469 -261
  179. package/server/services/providers.js +10 -5
  180. package/server/services/role.js +15 -13
  181. package/server/services/user.js +56 -19
  182. package/server/services/users-permissions.js +15 -13
  183. package/server/utils/index.d.ts +2 -1
  184. package/server/utils/sanitize/sanitizers.js +7 -3
  185. package/server/utils/sanitize/visitors/remove-user-relation-from-role-entities.js +2 -2
  186. package/.eslintrc.js +0 -14
  187. package/admin/src/components/FormModal/index.js +0 -126
  188. package/admin/src/components/Permissions/index.js +0 -57
  189. package/admin/src/hooks/index.js +0 -5
  190. package/admin/src/hooks/useFetchRole/index.js +0 -67
  191. package/admin/src/hooks/useFetchRole/reducer.js +0 -31
  192. package/admin/src/hooks/useForm/index.js +0 -68
  193. package/admin/src/hooks/useForm/reducer.js +0 -40
  194. package/admin/src/hooks/useRolesList/index.js +0 -65
  195. package/admin/src/hooks/useRolesList/init.js +0 -5
  196. package/admin/src/hooks/useRolesList/reducer.js +0 -31
  197. package/admin/src/pages/AdvancedSettings/index.js +0 -242
  198. package/admin/src/pages/AdvancedSettings/utils/api.js +0 -16
  199. package/admin/src/pages/EmailTemplates/components/EmailForm.js +0 -176
  200. package/admin/src/pages/EmailTemplates/index.js +0 -159
  201. package/admin/src/pages/EmailTemplates/utils/api.js +0 -16
  202. package/admin/src/pages/Providers/index.js +0 -271
  203. package/admin/src/pages/Providers/reducer.js +0 -54
  204. package/admin/src/pages/Providers/utils/api.js +0 -24
  205. package/admin/src/pages/Providers/utils/createProvidersArray.js +0 -21
  206. package/admin/src/pages/Roles/CreatePage.js +0 -185
  207. package/admin/src/pages/Roles/EditPage.js +0 -197
  208. package/admin/src/pages/Roles/ListPage/components/TableBody.js +0 -93
  209. package/admin/src/pages/Roles/ListPage/utils/api.js +0 -30
  210. package/admin/src/pages/Roles/ProtectedCreatePage.js +0 -15
  211. package/admin/src/pages/Roles/ProtectedEditPage.js +0 -15
  212. package/admin/src/pages/Roles/ProtectedListPage.js +0 -17
  213. package/admin/src/pages/Roles/index.js +0 -30
  214. package/server/bootstrap/grant-config.js +0 -131
  215. package/strapi-admin.js +0 -3
  216. package/strapi-server.js +0 -3
  217. /package/admin/src/components/Permissions/PermissionRow/{index.js → index.jsx} +0 -0
  218. /package/admin/src/contexts/UsersPermissionsContext/{index.js → index.jsx} +0 -0
@@ -0,0 +1,617 @@
1
+ import { jsxs, jsx } from "react/jsx-runtime";
2
+ import * as React from "react";
3
+ import { u as useTracking, L as Layouts } from "./index-DUWx-KG9-C1E7ADbJ.mjs";
4
+ import { Field, Toggle, TextInput, Modal, Breadcrumbs, Crumb, Flex, Grid, Button, useCollator, Table, Thead, Tr, Th, Typography, VisuallyHidden, Tbody, Td, IconButton } from "@strapi/design-system";
5
+ import { Pencil } from "@strapi/icons";
6
+ import { translatedErrors, useNotification, useFetchClient, useAPIErrorHandler, useRBAC, Page } from "@strapi/strapi/admin";
7
+ import upperFirst from "lodash/upperFirst";
8
+ import { useIntl } from "react-intl";
9
+ import { useQueryClient, useQuery, useMutation } from "react-query";
10
+ import { Formik, Form } from "formik";
11
+ import PropTypes from "prop-types";
12
+ import { g as getTrad, P as PERMISSIONS } from "./index-R4R0dEOv.mjs";
13
+ import "lodash/isEmpty";
14
+ import * as yup from "yup";
15
+ const Input = ({
16
+ description,
17
+ disabled,
18
+ intlLabel,
19
+ error,
20
+ name,
21
+ onChange,
22
+ placeholder,
23
+ providerToEditName,
24
+ type,
25
+ value
26
+ }) => {
27
+ const { formatMessage } = useIntl();
28
+ const inputValue = name === "noName" ? `${window.strapi.backendURL}/api/connect/${providerToEditName}/callback` : value;
29
+ const label = formatMessage(
30
+ { id: intlLabel.id, defaultMessage: intlLabel.defaultMessage },
31
+ { provider: providerToEditName, ...intlLabel.values }
32
+ );
33
+ const hint = description ? formatMessage(
34
+ { id: description.id, defaultMessage: description.defaultMessage },
35
+ { provider: providerToEditName, ...description.values }
36
+ ) : "";
37
+ if (type === "bool") {
38
+ return /* @__PURE__ */ jsxs(Field.Root, { hint, name, children: [
39
+ /* @__PURE__ */ jsx(Field.Label, { children: label }),
40
+ /* @__PURE__ */ jsx(
41
+ Toggle,
42
+ {
43
+ "aria-label": name,
44
+ checked: value,
45
+ disabled,
46
+ offLabel: formatMessage({
47
+ id: "app.components.ToggleCheckbox.off-label",
48
+ defaultMessage: "Off"
49
+ }),
50
+ onLabel: formatMessage({
51
+ id: "app.components.ToggleCheckbox.on-label",
52
+ defaultMessage: "On"
53
+ }),
54
+ onChange: (e) => {
55
+ onChange({ target: { name, value: e.target.checked } });
56
+ }
57
+ }
58
+ ),
59
+ /* @__PURE__ */ jsx(Field.Hint, {})
60
+ ] });
61
+ }
62
+ const formattedPlaceholder = placeholder ? formatMessage(
63
+ { id: placeholder.id, defaultMessage: placeholder.defaultMessage },
64
+ { ...placeholder.values }
65
+ ) : "";
66
+ const errorMessage = error ? formatMessage({ id: error, defaultMessage: error }) : "";
67
+ return /* @__PURE__ */ jsxs(Field.Root, { error: errorMessage, name, children: [
68
+ /* @__PURE__ */ jsx(Field.Label, { children: label }),
69
+ /* @__PURE__ */ jsx(
70
+ TextInput,
71
+ {
72
+ disabled,
73
+ onChange,
74
+ placeholder: formattedPlaceholder,
75
+ type,
76
+ value: inputValue
77
+ }
78
+ ),
79
+ /* @__PURE__ */ jsx(Field.Error, {})
80
+ ] });
81
+ };
82
+ Input.defaultProps = {
83
+ description: null,
84
+ disabled: false,
85
+ error: "",
86
+ placeholder: null,
87
+ value: ""
88
+ };
89
+ Input.propTypes = {
90
+ description: PropTypes.shape({
91
+ id: PropTypes.string.isRequired,
92
+ defaultMessage: PropTypes.string.isRequired,
93
+ values: PropTypes.object
94
+ }),
95
+ disabled: PropTypes.bool,
96
+ error: PropTypes.string,
97
+ intlLabel: PropTypes.shape({
98
+ id: PropTypes.string.isRequired,
99
+ defaultMessage: PropTypes.string.isRequired,
100
+ values: PropTypes.object
101
+ }).isRequired,
102
+ name: PropTypes.string.isRequired,
103
+ onChange: PropTypes.func.isRequired,
104
+ placeholder: PropTypes.shape({
105
+ id: PropTypes.string.isRequired,
106
+ defaultMessage: PropTypes.string.isRequired,
107
+ values: PropTypes.object
108
+ }),
109
+ providerToEditName: PropTypes.string.isRequired,
110
+ type: PropTypes.string.isRequired,
111
+ value: PropTypes.oneOfType([PropTypes.bool, PropTypes.string])
112
+ };
113
+ const FormModal = ({
114
+ headerBreadcrumbs,
115
+ initialData,
116
+ isSubmiting,
117
+ layout,
118
+ isOpen,
119
+ onSubmit,
120
+ onToggle,
121
+ providerToEditName
122
+ }) => {
123
+ const { formatMessage } = useIntl();
124
+ return /* @__PURE__ */ jsx(Modal.Root, { open: isOpen, onOpenChange: onToggle, children: /* @__PURE__ */ jsxs(Modal.Content, { children: [
125
+ /* @__PURE__ */ jsx(Modal.Header, { children: /* @__PURE__ */ jsx(Breadcrumbs, { label: headerBreadcrumbs.join(", "), children: headerBreadcrumbs.map((crumb, index, arr) => /* @__PURE__ */ jsx(Crumb, { isCurrent: index === arr.length - 1, children: crumb }, crumb)) }) }),
126
+ /* @__PURE__ */ jsx(
127
+ Formik,
128
+ {
129
+ onSubmit: (values) => onSubmit(values),
130
+ initialValues: initialData,
131
+ validationSchema: layout.schema,
132
+ validateOnChange: false,
133
+ children: ({ errors, handleChange, values }) => {
134
+ return /* @__PURE__ */ jsxs(Form, { children: [
135
+ /* @__PURE__ */ jsx(Modal.Body, { children: /* @__PURE__ */ jsx(Flex, { direction: "column", alignItems: "stretch", gap: 1, children: /* @__PURE__ */ jsx(Grid.Root, { gap: 5, children: layout.form.map((row) => {
136
+ return row.map((input) => {
137
+ return /* @__PURE__ */ jsx(
138
+ Grid.Item,
139
+ {
140
+ col: input.size,
141
+ xs: 12,
142
+ direction: "column",
143
+ alignItems: "stretch",
144
+ children: /* @__PURE__ */ jsx(
145
+ Input,
146
+ {
147
+ ...input,
148
+ error: errors[input.name],
149
+ onChange: handleChange,
150
+ value: values[input.name],
151
+ providerToEditName
152
+ }
153
+ )
154
+ },
155
+ input.name
156
+ );
157
+ });
158
+ }) }) }) }),
159
+ /* @__PURE__ */ jsxs(Modal.Footer, { children: [
160
+ /* @__PURE__ */ jsx(Button, { variant: "tertiary", onClick: onToggle, type: "button", children: formatMessage({
161
+ id: "app.components.Button.cancel",
162
+ defaultMessage: "Cancel"
163
+ }) }),
164
+ /* @__PURE__ */ jsx(Button, { type: "submit", loading: isSubmiting, children: formatMessage({ id: "global.save", defaultMessage: "Save" }) })
165
+ ] })
166
+ ] });
167
+ }
168
+ }
169
+ )
170
+ ] }) });
171
+ };
172
+ FormModal.defaultProps = {
173
+ initialData: null,
174
+ providerToEditName: null
175
+ };
176
+ FormModal.propTypes = {
177
+ headerBreadcrumbs: PropTypes.arrayOf(PropTypes.string).isRequired,
178
+ initialData: PropTypes.object,
179
+ layout: PropTypes.shape({
180
+ form: PropTypes.arrayOf(PropTypes.array),
181
+ schema: PropTypes.object
182
+ }).isRequired,
183
+ isOpen: PropTypes.bool.isRequired,
184
+ isSubmiting: PropTypes.bool.isRequired,
185
+ onSubmit: PropTypes.func.isRequired,
186
+ onToggle: PropTypes.func.isRequired,
187
+ providerToEditName: PropTypes.string
188
+ };
189
+ const callbackLabel = {
190
+ id: getTrad("PopUpForm.Providers.redirectURL.front-end.label"),
191
+ defaultMessage: "The redirect URL to your front-end app"
192
+ };
193
+ const callbackPlaceholder = {
194
+ id: "http://www.client-app.com",
195
+ defaultMessage: "http://www.client-app.com"
196
+ };
197
+ const enabledDescription = {
198
+ id: getTrad("PopUpForm.Providers.enabled.description"),
199
+ defaultMessage: "If disabled, users won't be able to use this provider."
200
+ };
201
+ const enabledLabel = {
202
+ id: getTrad("PopUpForm.Providers.enabled.label"),
203
+ defaultMessage: "Enable"
204
+ };
205
+ const keyLabel = { id: getTrad("PopUpForm.Providers.key.label"), defaultMessage: "Client ID" };
206
+ const hintLabel = {
207
+ id: getTrad("PopUpForm.Providers.redirectURL.label"),
208
+ defaultMessage: "The redirect URL to add in your {provider} application configurations"
209
+ };
210
+ const textPlaceholder = {
211
+ id: getTrad("PopUpForm.Providers.key.placeholder"),
212
+ defaultMessage: "TEXT"
213
+ };
214
+ const secretLabel = {
215
+ id: getTrad("PopUpForm.Providers.secret.label"),
216
+ defaultMessage: "Client Secret"
217
+ };
218
+ const CALLBACK_REGEX = /^$|^[a-z][a-z0-9+.-]*:\/\/[^\s/$.?#](?:[^\s]*[^\s/$.?#])?$/i;
219
+ const SUBDOMAIN_REGEX = /^(([a-zA-Z0-9-]+\.)*[a-zA-Z0-9-]+)(:\d+)?(\/\S*)?$/i;
220
+ const forms = {
221
+ email: {
222
+ form: [
223
+ [
224
+ {
225
+ intlLabel: enabledLabel,
226
+ name: "enabled",
227
+ type: "bool",
228
+ description: enabledDescription,
229
+ size: 6
230
+ // TODO check if still needed
231
+ // validations: {
232
+ // required: true,
233
+ // },
234
+ }
235
+ ]
236
+ ],
237
+ schema: yup.object().shape({
238
+ enabled: yup.bool().required(translatedErrors.required.id)
239
+ })
240
+ },
241
+ providers: {
242
+ form: [
243
+ [
244
+ {
245
+ intlLabel: enabledLabel,
246
+ name: "enabled",
247
+ type: "bool",
248
+ description: enabledDescription,
249
+ size: 6,
250
+ validations: {
251
+ required: true
252
+ }
253
+ }
254
+ ],
255
+ [
256
+ {
257
+ intlLabel: keyLabel,
258
+ name: "key",
259
+ type: "text",
260
+ placeholder: textPlaceholder,
261
+ size: 12,
262
+ validations: {
263
+ required: true
264
+ }
265
+ }
266
+ ],
267
+ [
268
+ {
269
+ intlLabel: secretLabel,
270
+ name: "secret",
271
+ type: "text",
272
+ placeholder: textPlaceholder,
273
+ size: 12,
274
+ validations: {
275
+ required: true
276
+ }
277
+ }
278
+ ],
279
+ [
280
+ {
281
+ intlLabel: callbackLabel,
282
+ placeholder: callbackPlaceholder,
283
+ name: "callback",
284
+ type: "text",
285
+ size: 12,
286
+ validations: {
287
+ required: true
288
+ }
289
+ }
290
+ ],
291
+ [
292
+ {
293
+ intlLabel: hintLabel,
294
+ name: "noName",
295
+ type: "text",
296
+ validations: {},
297
+ size: 12,
298
+ disabled: true
299
+ }
300
+ ]
301
+ ],
302
+ schema: yup.object().shape({
303
+ enabled: yup.bool().required(translatedErrors.required.id),
304
+ key: yup.string().when("enabled", {
305
+ is: true,
306
+ then: yup.string().required(translatedErrors.required.id),
307
+ otherwise: yup.string()
308
+ }),
309
+ secret: yup.string().when("enabled", {
310
+ is: true,
311
+ then: yup.string().required(translatedErrors.required.id),
312
+ otherwise: yup.string()
313
+ }),
314
+ callback: yup.string().when("enabled", {
315
+ is: true,
316
+ then: yup.string().matches(CALLBACK_REGEX, translatedErrors.regex.id).required(translatedErrors.required.id),
317
+ otherwise: yup.string()
318
+ })
319
+ })
320
+ },
321
+ providersWithSubdomain: {
322
+ form: [
323
+ [
324
+ {
325
+ intlLabel: enabledLabel,
326
+ name: "enabled",
327
+ type: "bool",
328
+ description: enabledDescription,
329
+ size: 6,
330
+ validations: {
331
+ required: true
332
+ }
333
+ }
334
+ ],
335
+ [
336
+ {
337
+ intlLabel: keyLabel,
338
+ name: "key",
339
+ type: "text",
340
+ placeholder: textPlaceholder,
341
+ size: 12,
342
+ validations: {
343
+ required: true
344
+ }
345
+ }
346
+ ],
347
+ [
348
+ {
349
+ intlLabel: secretLabel,
350
+ name: "secret",
351
+ type: "text",
352
+ placeholder: textPlaceholder,
353
+ size: 12,
354
+ validations: {
355
+ required: true
356
+ }
357
+ }
358
+ ],
359
+ [
360
+ {
361
+ intlLabel: {
362
+ id: getTrad({ id: "PopUpForm.Providers.jwksurl.label" }),
363
+ defaultMessage: "JWKS URL"
364
+ },
365
+ name: "jwksurl",
366
+ type: "text",
367
+ placeholder: textPlaceholder,
368
+ size: 12,
369
+ validations: {
370
+ required: false
371
+ }
372
+ }
373
+ ],
374
+ [
375
+ {
376
+ intlLabel: {
377
+ id: getTrad("PopUpForm.Providers.subdomain.label"),
378
+ defaultMessage: "Host URI (Subdomain)"
379
+ },
380
+ name: "subdomain",
381
+ type: "text",
382
+ placeholder: {
383
+ id: getTrad("PopUpForm.Providers.subdomain.placeholder"),
384
+ defaultMessage: "my.subdomain.com"
385
+ },
386
+ size: 12,
387
+ validations: {
388
+ required: true
389
+ }
390
+ }
391
+ ],
392
+ [
393
+ {
394
+ intlLabel: callbackLabel,
395
+ placeholder: callbackPlaceholder,
396
+ name: "callback",
397
+ type: "text",
398
+ size: 12,
399
+ validations: {
400
+ required: true
401
+ }
402
+ }
403
+ ],
404
+ [
405
+ {
406
+ intlLabel: hintLabel,
407
+ name: "noName",
408
+ type: "text",
409
+ validations: {},
410
+ size: 12,
411
+ disabled: true
412
+ }
413
+ ]
414
+ ],
415
+ schema: yup.object().shape({
416
+ enabled: yup.bool().required(translatedErrors.required.id),
417
+ key: yup.string().when("enabled", {
418
+ is: true,
419
+ then: yup.string().required(translatedErrors.required.id),
420
+ otherwise: yup.string()
421
+ }),
422
+ secret: yup.string().when("enabled", {
423
+ is: true,
424
+ then: yup.string().required(translatedErrors.required.id),
425
+ otherwise: yup.string()
426
+ }),
427
+ subdomain: yup.string().when("enabled", {
428
+ is: true,
429
+ then: yup.string().matches(SUBDOMAIN_REGEX, translatedErrors.regex.id).required(translatedErrors.required.id),
430
+ otherwise: yup.string()
431
+ }),
432
+ callback: yup.string().when("enabled", {
433
+ is: true,
434
+ then: yup.string().matches(CALLBACK_REGEX, translatedErrors.regex.id).required(translatedErrors.required.id),
435
+ otherwise: yup.string()
436
+ })
437
+ })
438
+ }
439
+ };
440
+ const ProvidersPage = () => {
441
+ const { formatMessage, locale } = useIntl();
442
+ const queryClient = useQueryClient();
443
+ const { trackUsage } = useTracking();
444
+ const [isOpen, setIsOpen] = React.useState(false);
445
+ const [providerToEditName, setProviderToEditName] = React.useState(null);
446
+ const { toggleNotification } = useNotification();
447
+ const { get, put } = useFetchClient();
448
+ const { formatAPIError } = useAPIErrorHandler();
449
+ const formatter = useCollator(locale, {
450
+ sensitivity: "base"
451
+ });
452
+ const {
453
+ isLoading: isLoadingPermissions,
454
+ allowedActions: { canUpdate }
455
+ } = useRBAC({ update: PERMISSIONS.updateProviders });
456
+ const { isLoading: isLoadingData, data } = useQuery(
457
+ ["users-permissions", "get-providers"],
458
+ async () => {
459
+ const { data: data2 } = await get("/users-permissions/providers");
460
+ return data2;
461
+ },
462
+ {
463
+ initialData: {}
464
+ }
465
+ );
466
+ const submitMutation = useMutation((body) => put("/users-permissions/providers", body), {
467
+ async onSuccess() {
468
+ await queryClient.invalidateQueries(["users-permissions", "get-providers"]);
469
+ toggleNotification({
470
+ type: "success",
471
+ message: formatMessage({ id: getTrad("notification.success.submit") })
472
+ });
473
+ trackUsage("didEditAuthenticationProvider");
474
+ handleToggleModal();
475
+ },
476
+ onError(error) {
477
+ toggleNotification({
478
+ type: "danger",
479
+ message: formatAPIError(error)
480
+ });
481
+ },
482
+ refetchActive: false
483
+ });
484
+ const providers = Object.entries(data).reduce((acc, [name, provider]) => {
485
+ const { icon, enabled, subdomain } = provider;
486
+ acc.push({
487
+ name,
488
+ icon: icon === "envelope" ? ["fas", "envelope"] : ["fab", icon],
489
+ enabled,
490
+ subdomain
491
+ });
492
+ return acc;
493
+ }, []).sort((a, b) => formatter.compare(a.name, b.name));
494
+ const isLoading = isLoadingData || isLoadingPermissions;
495
+ const isProviderWithSubdomain = React.useMemo(() => {
496
+ if (!providerToEditName) {
497
+ return false;
498
+ }
499
+ const providerToEdit = providers.find((obj) => obj.name === providerToEditName);
500
+ return !!providerToEdit?.subdomain;
501
+ }, [providers, providerToEditName]);
502
+ const layoutToRender = React.useMemo(() => {
503
+ if (providerToEditName === "email") {
504
+ return forms.email;
505
+ }
506
+ if (isProviderWithSubdomain) {
507
+ return forms.providersWithSubdomain;
508
+ }
509
+ return forms.providers;
510
+ }, [providerToEditName, isProviderWithSubdomain]);
511
+ const handleToggleModal = () => {
512
+ setIsOpen((prev) => !prev);
513
+ };
514
+ const handleClickEdit = (provider) => {
515
+ if (canUpdate) {
516
+ setProviderToEditName(provider.name);
517
+ handleToggleModal();
518
+ }
519
+ };
520
+ const handleSubmit = async (values) => {
521
+ trackUsage("willEditAuthenticationProvider");
522
+ submitMutation.mutate({ providers: { ...data, [providerToEditName]: values } });
523
+ };
524
+ if (isLoading) {
525
+ return /* @__PURE__ */ jsx(Page.Loading, {});
526
+ }
527
+ return /* @__PURE__ */ jsxs(Layouts.Root, { children: [
528
+ /* @__PURE__ */ jsx(Page.Title, { children: formatMessage(
529
+ { id: "Settings.PageTitle", defaultMessage: "Settings - {name}" },
530
+ {
531
+ name: formatMessage({
532
+ id: getTrad("HeaderNav.link.providers"),
533
+ defaultMessage: "Providers"
534
+ })
535
+ }
536
+ ) }),
537
+ /* @__PURE__ */ jsxs(Page.Main, { children: [
538
+ /* @__PURE__ */ jsx(
539
+ Layouts.Header,
540
+ {
541
+ title: formatMessage({
542
+ id: getTrad("HeaderNav.link.providers"),
543
+ defaultMessage: "Providers"
544
+ })
545
+ }
546
+ ),
547
+ /* @__PURE__ */ jsx(Layouts.Content, { children: /* @__PURE__ */ jsxs(Table, { colCount: 3, rowCount: providers.length + 1, children: [
548
+ /* @__PURE__ */ jsx(Thead, { children: /* @__PURE__ */ jsxs(Tr, { children: [
549
+ /* @__PURE__ */ jsx(Th, { children: /* @__PURE__ */ jsx(Typography, { variant: "sigma", textColor: "neutral600", children: formatMessage({ id: "global.name", defaultMessage: "Name" }) }) }),
550
+ /* @__PURE__ */ jsx(Th, { children: /* @__PURE__ */ jsx(Typography, { variant: "sigma", textColor: "neutral600", children: formatMessage({ id: getTrad("Providers.status"), defaultMessage: "Status" }) }) }),
551
+ /* @__PURE__ */ jsx(Th, { children: /* @__PURE__ */ jsx(Typography, { variant: "sigma", children: /* @__PURE__ */ jsx(VisuallyHidden, { children: formatMessage({
552
+ id: "global.settings",
553
+ defaultMessage: "Settings"
554
+ }) }) }) })
555
+ ] }) }),
556
+ /* @__PURE__ */ jsx(Tbody, { children: providers.map((provider) => /* @__PURE__ */ jsxs(
557
+ Tr,
558
+ {
559
+ onClick: () => canUpdate ? handleClickEdit(provider) : void 0,
560
+ children: [
561
+ /* @__PURE__ */ jsx(Td, { width: "45%", children: /* @__PURE__ */ jsx(Typography, { fontWeight: "semiBold", textColor: "neutral800", children: provider.name }) }),
562
+ /* @__PURE__ */ jsx(Td, { width: "65%", children: /* @__PURE__ */ jsx(
563
+ Typography,
564
+ {
565
+ textColor: provider.enabled ? "success600" : "danger600",
566
+ "data-testid": `enable-${provider.name}`,
567
+ children: provider.enabled ? formatMessage({
568
+ id: "global.enabled",
569
+ defaultMessage: "Enabled"
570
+ }) : formatMessage({
571
+ id: "global.disabled",
572
+ defaultMessage: "Disabled"
573
+ })
574
+ }
575
+ ) }),
576
+ /* @__PURE__ */ jsx(Td, { onClick: (e) => e.stopPropagation(), children: canUpdate && /* @__PURE__ */ jsx(
577
+ IconButton,
578
+ {
579
+ onClick: () => handleClickEdit(provider),
580
+ variant: "ghost",
581
+ label: "Edit",
582
+ children: /* @__PURE__ */ jsx(Pencil, {})
583
+ }
584
+ ) })
585
+ ]
586
+ },
587
+ provider.name
588
+ )) })
589
+ ] }) })
590
+ ] }),
591
+ /* @__PURE__ */ jsx(
592
+ FormModal,
593
+ {
594
+ initialData: data[providerToEditName],
595
+ isOpen,
596
+ isSubmiting: submitMutation.isLoading,
597
+ layout: layoutToRender,
598
+ headerBreadcrumbs: [
599
+ formatMessage({
600
+ id: getTrad("PopUpForm.header.edit.providers"),
601
+ defaultMessage: "Edit Provider"
602
+ }),
603
+ upperFirst(providerToEditName)
604
+ ],
605
+ onToggle: handleToggleModal,
606
+ onSubmit: handleSubmit,
607
+ providerToEditName
608
+ }
609
+ )
610
+ ] });
611
+ };
612
+ const ProtectedProvidersPage = () => /* @__PURE__ */ jsx(Page.Protect, { permissions: PERMISSIONS.readProviders, children: /* @__PURE__ */ jsx(ProvidersPage, {}) });
613
+ export {
614
+ ProvidersPage,
615
+ ProtectedProvidersPage as default
616
+ };
617
+ //# sourceMappingURL=index-XOcQhJMB.mjs.map