@kuadrant/kuadrant-backstage-plugin-frontend 0.0.2-dev-83763ff → 0.0.2-dev-b696169

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 (91) hide show
  1. package/dist/components/ApiKeyDetailPage/ApiKeyDetailPage.esm.js +319 -0
  2. package/dist/components/ApiKeyDetailPage/ApiKeyDetailPage.esm.js.map +1 -0
  3. package/dist/components/ApiKeyDetailPage/index.esm.js +2 -0
  4. package/dist/components/ApiKeyDetailPage/index.esm.js.map +1 -0
  5. package/dist/components/ApiKeyManagementTab/ApiKeyManagementTab.esm.js +340 -162
  6. package/dist/components/ApiKeyManagementTab/ApiKeyManagementTab.esm.js.map +1 -1
  7. package/dist/components/ApiKeysPage/ApiKeysPage.esm.js +43 -0
  8. package/dist/components/ApiKeysPage/ApiKeysPage.esm.js.map +1 -0
  9. package/dist/components/ApiKeysPage/index.esm.js +2 -0
  10. package/dist/components/ApiKeysPage/index.esm.js.map +1 -0
  11. package/dist/components/ApprovalQueueCard/ApprovalQueueCard.esm.js +312 -118
  12. package/dist/components/ApprovalQueueCard/ApprovalQueueCard.esm.js.map +1 -1
  13. package/dist/components/ApprovalQueueTable/ApprovalQueueTable.esm.js +645 -0
  14. package/dist/components/ApprovalQueueTable/ApprovalQueueTable.esm.js.map +1 -0
  15. package/dist/components/EditAPIKeyDialog/EditAPIKeyDialog.esm.js +14 -2
  16. package/dist/components/EditAPIKeyDialog/EditAPIKeyDialog.esm.js.map +1 -1
  17. package/dist/components/EntityApiApprovalTab/EntityApiApprovalTab.esm.js +276 -0
  18. package/dist/components/EntityApiApprovalTab/EntityApiApprovalTab.esm.js.map +1 -0
  19. package/dist/components/EntityApiApprovalTab/index.esm.js +2 -0
  20. package/dist/components/EntityApiApprovalTab/index.esm.js.map +1 -0
  21. package/dist/components/FilterPanel/FilterPanel.esm.js +127 -0
  22. package/dist/components/FilterPanel/FilterPanel.esm.js.map +1 -0
  23. package/dist/components/KuadrantPage/KuadrantPage.esm.js +123 -40
  24. package/dist/components/KuadrantPage/KuadrantPage.esm.js.map +1 -1
  25. package/dist/components/KuadrantPage/index.esm.js +1 -1
  26. package/dist/components/{MyApiKeysCard/MyApiKeysCard.esm.js → MyApiKeysTable/MyApiKeysTable.esm.js} +294 -176
  27. package/dist/components/MyApiKeysTable/MyApiKeysTable.esm.js.map +1 -0
  28. package/dist/components/PlanPolicyDetailsCard/PlanPolicyDetails.esm.js +5 -4
  29. package/dist/components/PlanPolicyDetailsCard/PlanPolicyDetails.esm.js.map +1 -1
  30. package/dist/index.d.ts +8 -4
  31. package/dist/index.esm.js +1 -1
  32. package/dist/plugin.esm.js +30 -1
  33. package/dist/plugin.esm.js.map +1 -1
  34. package/dist/utils/styles.esm.js +14 -0
  35. package/dist/utils/styles.esm.js.map +1 -0
  36. package/dist-scalprum/{internal.plugin-kuadrant.6dd50c0e0265251d3d4f.js → internal.plugin-kuadrant.e37d8046ec4d7ed991e0.js} +2 -2
  37. package/dist-scalprum/internal.plugin-kuadrant.e37d8046ec4d7ed991e0.js.map +1 -0
  38. package/dist-scalprum/plugin-manifest.json +2 -2
  39. package/dist-scalprum/static/2967.c684efaf.chunk.js +2 -0
  40. package/dist-scalprum/static/2967.c684efaf.chunk.js.map +1 -0
  41. package/dist-scalprum/static/3097.4bd6b35f.chunk.js +2 -0
  42. package/dist-scalprum/static/3097.4bd6b35f.chunk.js.map +1 -0
  43. package/dist-scalprum/static/4306.3a218ff1.chunk.js +2 -0
  44. package/dist-scalprum/static/4306.3a218ff1.chunk.js.map +1 -0
  45. package/dist-scalprum/static/5010.acf9a415.chunk.js +3 -0
  46. package/dist-scalprum/static/5010.acf9a415.chunk.js.map +1 -0
  47. package/dist-scalprum/static/5453.ae292ab1.chunk.js +2 -0
  48. package/dist-scalprum/static/5453.ae292ab1.chunk.js.map +1 -0
  49. package/dist-scalprum/static/6800.736d5da3.chunk.js +2 -0
  50. package/dist-scalprum/static/6800.736d5da3.chunk.js.map +1 -0
  51. package/dist-scalprum/static/6813.036a322f.chunk.js +2 -0
  52. package/dist-scalprum/static/6813.036a322f.chunk.js.map +1 -0
  53. package/dist-scalprum/static/6840.4728fab9.chunk.js +2 -0
  54. package/dist-scalprum/static/6840.4728fab9.chunk.js.map +1 -0
  55. package/dist-scalprum/static/6979.9699b350.chunk.js +2 -0
  56. package/dist-scalprum/static/6979.9699b350.chunk.js.map +1 -0
  57. package/dist-scalprum/static/7684.3d1fc1a1.chunk.js +2 -0
  58. package/dist-scalprum/static/7684.3d1fc1a1.chunk.js.map +1 -0
  59. package/dist-scalprum/static/8365.d3360f18.chunk.js +2 -0
  60. package/dist-scalprum/static/8365.d3360f18.chunk.js.map +1 -0
  61. package/dist-scalprum/static/8416.3604a311.chunk.js +2 -0
  62. package/dist-scalprum/static/8416.3604a311.chunk.js.map +1 -0
  63. package/dist-scalprum/static/8563.7e068fb0.chunk.js +3 -0
  64. package/dist-scalprum/static/8563.7e068fb0.chunk.js.map +1 -0
  65. package/dist-scalprum/static/exposed-PluginRoot.0717f1ce.chunk.js +2 -0
  66. package/dist-scalprum/static/exposed-PluginRoot.0717f1ce.chunk.js.map +1 -0
  67. package/package.json +1 -1
  68. package/dist/components/MyApiKeysCard/MyApiKeysCard.esm.js.map +0 -1
  69. package/dist-scalprum/internal.plugin-kuadrant.6dd50c0e0265251d3d4f.js.map +0 -1
  70. package/dist-scalprum/static/3483.2f14a8ca.chunk.js +0 -3
  71. package/dist-scalprum/static/3483.2f14a8ca.chunk.js.map +0 -1
  72. package/dist-scalprum/static/4306.b68910c9.chunk.js +0 -2
  73. package/dist-scalprum/static/4306.b68910c9.chunk.js.map +0 -1
  74. package/dist-scalprum/static/4556.c6bedfc4.chunk.js +0 -3
  75. package/dist-scalprum/static/4556.c6bedfc4.chunk.js.map +0 -1
  76. package/dist-scalprum/static/5222.796292ca.chunk.js +0 -2
  77. package/dist-scalprum/static/5222.796292ca.chunk.js.map +0 -1
  78. package/dist-scalprum/static/532.e406b85b.chunk.js +0 -2
  79. package/dist-scalprum/static/532.e406b85b.chunk.js.map +0 -1
  80. package/dist-scalprum/static/5453.29118045.chunk.js +0 -2
  81. package/dist-scalprum/static/5453.29118045.chunk.js.map +0 -1
  82. package/dist-scalprum/static/6281.eb6e16a2.chunk.js +0 -2
  83. package/dist-scalprum/static/6281.eb6e16a2.chunk.js.map +0 -1
  84. package/dist-scalprum/static/8365.75ea3581.chunk.js +0 -2
  85. package/dist-scalprum/static/8365.75ea3581.chunk.js.map +0 -1
  86. package/dist-scalprum/static/8441.62394cfd.chunk.js +0 -2
  87. package/dist-scalprum/static/8441.62394cfd.chunk.js.map +0 -1
  88. package/dist-scalprum/static/exposed-PluginRoot.7023ce03.chunk.js +0 -2
  89. package/dist-scalprum/static/exposed-PluginRoot.7023ce03.chunk.js.map +0 -1
  90. /package/dist-scalprum/static/{4556.c6bedfc4.chunk.js.LICENSE.txt → 5010.acf9a415.chunk.js.LICENSE.txt} +0 -0
  91. /package/dist-scalprum/static/{3483.2f14a8ca.chunk.js.LICENSE.txt → 8563.7e068fb0.chunk.js.LICENSE.txt} +0 -0
@@ -0,0 +1 @@
1
+ {"version":3,"file":"static/3097.4bd6b35f.chunk.js","mappings":"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","sources":["webpack://internal.plugin-kuadrant/./src/permissions.ts","webpack://internal.plugin-kuadrant/../../node_modules/react-use/esm/useAsync.js","webpack://internal.plugin-kuadrant/../../node_modules/react-use/esm/useAsyncFn.js","webpack://internal.plugin-kuadrant/./node_modules/@backstage/core-components/dist/components/Link/Link.esm.js","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/icons/OpenInNew.js","webpack://internal.plugin-kuadrant/./src/components/ApprovalQueueCard/ApprovalQueueCard.tsx","webpack://internal.plugin-kuadrant/./src/utils/permissions.ts","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/icons/ArrowForward.js","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/icons/CheckCircle.js","webpack://internal.plugin-kuadrant/./node_modules/@backstage/core-components/dist/layout/BottomLink/BottomLink.esm.js","webpack://internal.plugin-kuadrant/./node_modules/@backstage/core-components/dist/components/LinkButton/LinkButton.esm.js","webpack://internal.plugin-kuadrant/./node_modules/@backstage/core-components/dist/layout/ErrorBoundary/ErrorBoundary.esm.js","webpack://internal.plugin-kuadrant/./node_modules/@backstage/core-components/dist/layout/InfoCard/InfoCard.esm.js"],"sourcesContent":["import { createPermission } from '@backstage/plugin-permission-common';\n\n/**\n * permission definitions for the kuadrant plugin\n *\n * these permissions control access to kuadrant resources and operations.\n * they must match the permissions defined in the backend plugin.\n *\n * permission types:\n * - BasicPermission: standard permission that applies globally\n * - ResourcePermission: permission scoped to specific resource types (e.g., apiproduct)\n *\n * permission patterns:\n * - `.create` - create new resources\n * - `.read` - read resource details\n * - `.read.own` - read only resources owned by the user\n * - `.read.all` - read all resources regardless of ownership\n * - `.update` - modify existing resources\n * - `.delete` - delete resources\n * - `.delete.own` - delete only resources owned by the user\n * - `.delete.all` - delete any resource regardless of ownership\n * - `.list` - list/view collections of resources\n */\n\n// planpolicy permissions\nexport const kuadrantPlanPolicyCreatePermission = createPermission({\n name: 'kuadrant.planpolicy.create',\n attributes: { action: 'create' },\n});\n\nexport const kuadrantPlanPolicyReadPermission = createPermission({\n name: 'kuadrant.planpolicy.read',\n attributes: { action: 'read' },\n});\n\nexport const kuadrantPlanPolicyUpdatePermission = createPermission({\n name: 'kuadrant.planpolicy.update',\n attributes: { action: 'update' },\n});\n\nexport const kuadrantPlanPolicyDeletePermission = createPermission({\n name: 'kuadrant.planpolicy.delete',\n attributes: { action: 'delete' },\n});\n\nexport const kuadrantPlanPolicyListPermission = createPermission({\n name: 'kuadrant.planpolicy.list',\n attributes: { action: 'read' },\n});\n\n// apiproduct permissions\n\n/**\n * permission to create new API products\n * granted to api owners and admins\n */\nexport const kuadrantApiProductCreatePermission = createPermission({\n name: 'kuadrant.apiproduct.create',\n attributes: { action: 'create' },\n});\n\n/**\n * permission to read API products owned by the current user\n * for api owners to view their own products\n */\nexport const kuadrantApiProductReadOwnPermission = createPermission({\n name: 'kuadrant.apiproduct.read.own',\n attributes: { action: 'read' },\n});\n\n/**\n * permission to read all API products regardless of ownership\n * for platform engineers/admins who need to view all products\n */\nexport const kuadrantApiProductReadAllPermission = createPermission({\n name: 'kuadrant.apiproduct.read.all',\n attributes: { action: 'read' },\n});\n\n/**\n * permission to update API products owned by the current user\n * for api owners to modify their own products\n */\nexport const kuadrantApiProductUpdateOwnPermission = createPermission({\n name: 'kuadrant.apiproduct.update.own',\n attributes: { action: 'update' },\n});\n\n/**\n * permission to update any API product regardless of ownership\n * for platform engineers/admins\n */\nexport const kuadrantApiProductUpdateAllPermission = createPermission({\n name: 'kuadrant.apiproduct.update.all',\n attributes: { action: 'update' },\n});\n\n/**\n * permission to delete API products owned by the current user\n * for api owners to remove their own products\n */\nexport const kuadrantApiProductDeleteOwnPermission = createPermission({\n name: 'kuadrant.apiproduct.delete.own',\n attributes: { action: 'delete' },\n});\n\n/**\n * permission to delete any API product regardless of ownership\n * for platform engineers/admins\n */\nexport const kuadrantApiProductDeleteAllPermission = createPermission({\n name: 'kuadrant.apiproduct.delete.all',\n attributes: { action: 'delete' },\n});\n\n/**\n * permission to list API products\n * backend filters results based on .own vs .all read permissions\n */\nexport const kuadrantApiProductListPermission = createPermission({\n name: 'kuadrant.apiproduct.list',\n attributes: { action: 'read' },\n});\n\n// apikey permissions\n\n/**\n * permission to create API keys (request API access)\n *\n * this is a ResourcePermission scoped to 'apiproduct', allowing\n * fine-grained control over which API products users can request access to.\n *\n * use in frontend: useKuadrantPermission(kuadrantApiKeyCreatePermission)\n * use in backend with resource: { permission, resourceRef: 'apiproduct:namespace/name' }\n */\nexport const kuadrantApiKeyCreatePermission = createPermission({\n name: 'kuadrant.apikey.create',\n attributes: { action: 'create' },\n resourceType: 'apiproduct',\n});\n\n/**\n * permission to read API keys owned by the current user\n * allows users to view their own API keys and request history\n */\nexport const kuadrantApiKeyReadOwnPermission = createPermission({\n name: 'kuadrant.apikey.read.own',\n attributes: { action: 'read' },\n});\n\n/**\n * permission to read all API keys regardless of ownership\n * for platform engineers/admins who need to view the approval queue and audit keys\n */\nexport const kuadrantApiKeyReadAllPermission = createPermission({\n name: 'kuadrant.apikey.read.all',\n attributes: { action: 'read' },\n});\n\n/**\n * permission to update API keys owned by the current user\n * allows users to edit their own pending requests (change plan tier, use case)\n */\nexport const kuadrantApiKeyUpdateOwnPermission = createPermission({\n name: 'kuadrant.apikey.update.own',\n attributes: { action: 'update' },\n});\n\n/**\n * permission to update any API key regardless of ownership\n * typically granted to API owners and platform engineers for approving/rejecting requests\n */\nexport const kuadrantApiKeyUpdateAllPermission = createPermission({\n name: 'kuadrant.apikey.update.all',\n attributes: { action: 'update' },\n});\n\n/**\n * permission to delete API keys owned by the current user\n * allows users to cancel their own requests or revoke their own access\n */\nexport const kuadrantApiKeyDeleteOwnPermission = createPermission({\n name: 'kuadrant.apikey.delete.own',\n attributes: { action: 'delete' },\n});\n\n/**\n * permission to delete any API key regardless of ownership\n * for platform engineers/admins who need to revoke access\n */\nexport const kuadrantApiKeyDeleteAllPermission = createPermission({\n name: 'kuadrant.apikey.delete.all',\n attributes: { action: 'delete' },\n});\n\n/**\n * permission to approve/reject API key requests\n * grants access to the approval queue - for API owners and admins only\n * separate from update.own which consumers use to edit their pending requests\n */\nexport const kuadrantApiKeyApprovePermission = createPermission({\n name: 'kuadrant.apikey.approve',\n attributes: { action: 'update' },\n});\n\nexport const kuadrantPermissions = [\n kuadrantPlanPolicyCreatePermission,\n kuadrantPlanPolicyReadPermission,\n kuadrantPlanPolicyUpdatePermission,\n kuadrantPlanPolicyDeletePermission,\n kuadrantPlanPolicyListPermission,\n kuadrantApiProductCreatePermission,\n kuadrantApiProductReadOwnPermission,\n kuadrantApiProductReadAllPermission,\n kuadrantApiProductUpdateOwnPermission,\n kuadrantApiProductUpdateAllPermission,\n kuadrantApiProductDeleteOwnPermission,\n kuadrantApiProductDeleteAllPermission,\n kuadrantApiProductListPermission,\n kuadrantApiKeyCreatePermission,\n kuadrantApiKeyReadOwnPermission,\n kuadrantApiKeyReadAllPermission,\n kuadrantApiKeyUpdateOwnPermission,\n kuadrantApiKeyUpdateAllPermission,\n kuadrantApiKeyDeleteOwnPermission,\n kuadrantApiKeyDeleteAllPermission,\n kuadrantApiKeyApprovePermission,\n];\n","import { useEffect } from 'react';\nimport useAsyncFn from './useAsyncFn';\nexport default function useAsync(fn, deps) {\n if (deps === void 0) { deps = []; }\n var _a = useAsyncFn(fn, deps, {\n loading: true,\n }), state = _a[0], callback = _a[1];\n useEffect(function () {\n callback();\n }, [callback]);\n return state;\n}\n","import { __assign } from \"tslib\";\nimport { useCallback, useRef, useState } from 'react';\nimport useMountedState from './useMountedState';\nexport default function useAsyncFn(fn, deps, initialState) {\n if (deps === void 0) { deps = []; }\n if (initialState === void 0) { initialState = { loading: false }; }\n var lastCallId = useRef(0);\n var isMounted = useMountedState();\n var _a = useState(initialState), state = _a[0], set = _a[1];\n var callback = useCallback(function () {\n var args = [];\n for (var _i = 0; _i < arguments.length; _i++) {\n args[_i] = arguments[_i];\n }\n var callId = ++lastCallId.current;\n if (!state.loading) {\n set(function (prevState) { return (__assign(__assign({}, prevState), { loading: true })); });\n }\n return fn.apply(void 0, args).then(function (value) {\n isMounted() && callId === lastCallId.current && set({ value: value, loading: false });\n return value;\n }, function (error) {\n isMounted() && callId === lastCallId.current && set({ error: error, loading: false });\n return error;\n });\n }, deps);\n return [state, callback];\n}\n","import { jsxs, jsx } from 'react/jsx-runtime';\nimport { useAnalytics, useApp, useApi, configApiRef } from '@backstage/core-plugin-api';\nimport MaterialLink from '@material-ui/core/Link';\nimport { makeStyles } from '@material-ui/core/styles';\nimport Typography from '@material-ui/core/Typography';\nimport classNames from 'classnames';\nimport { trimEnd } from 'lodash';\nimport { forwardRef } from 'react';\nimport { Link as Link$1, createRoutesFromChildren, Route } from 'react-router-dom';\nimport OpenInNew from '@material-ui/icons/OpenInNew';\n\nfunction isReactRouterBeta() {\n const [obj] = createRoutesFromChildren(/* @__PURE__ */ jsx(Route, { index: true, element: /* @__PURE__ */ jsx(\"div\", {}) }));\n return !obj.index;\n}\nconst useStyles = makeStyles(\n (theme) => ({\n visuallyHidden: {\n clip: \"rect(0 0 0 0)\",\n clipPath: \"inset(50%)\",\n overflow: \"hidden\",\n position: \"absolute\",\n userSelect: \"none\",\n whiteSpace: \"nowrap\",\n height: 1,\n width: 1\n },\n externalLink: {\n position: \"relative\"\n },\n externalLinkIcon: {\n verticalAlign: \"bottom\",\n marginLeft: theme.spacing(0.5)\n }\n }),\n { name: \"Link\" }\n);\nconst ExternalLinkIcon = () => {\n const app = useApp();\n const Icon = app.getSystemIcon(\"externalLink\") || OpenInNew;\n const classes = useStyles();\n return /* @__PURE__ */ jsx(Icon, { className: classes.externalLinkIcon });\n};\nconst isExternalUri = (uri) => /^([a-z+.-]+):/.test(uri);\nconst scriptProtocolPattern = (\n // eslint-disable-next-line no-control-regex\n /^[\\u0000-\\u001F ]*j[\\r\\n\\t]*a[\\r\\n\\t]*v[\\r\\n\\t]*a[\\r\\n\\t]*s[\\r\\n\\t]*c[\\r\\n\\t]*r[\\r\\n\\t]*i[\\r\\n\\t]*p[\\r\\n\\t]*t[\\r\\n\\t]*\\:/i\n);\nconst originalWindowOpen = window.open;\nif (originalWindowOpen && !originalWindowOpen.__backstage) {\n const newOpen = function open(...args) {\n const url = String(args[0]);\n if (scriptProtocolPattern.test(url)) {\n throw new Error(\n \"Rejected window.open() with a javascript: URL as a security precaution\"\n );\n }\n return originalWindowOpen.apply(this, args);\n };\n newOpen.__backstage = true;\n window.open = newOpen;\n}\nconst useBaseUrl = () => {\n try {\n const config = useApi(configApiRef);\n return config.getOptionalString(\"app.baseUrl\");\n } catch {\n return void 0;\n }\n};\nconst useBasePath = () => {\n const base = \"http://sample.dev\";\n const url = useBaseUrl() ?? \"/\";\n const { pathname } = new URL(url, base);\n return trimEnd(pathname, \"/\");\n};\nconst useResolvedPath = (uri) => {\n let resolvedPath = String(uri);\n const basePath = useBasePath();\n const external = isExternalUri(resolvedPath);\n const startsWithBasePath = resolvedPath.startsWith(basePath);\n if (!external && !startsWithBasePath) {\n resolvedPath = basePath.concat(resolvedPath);\n }\n return resolvedPath;\n};\nconst getNodeText = (node) => {\n if (node instanceof Array) {\n return node.map(getNodeText).join(\" \").trim();\n }\n if (typeof node === \"object\" && node) {\n return getNodeText(node?.props?.children);\n }\n if ([\"string\", \"number\"].includes(typeof node)) {\n return String(node);\n }\n return \"\";\n};\nconst Link = forwardRef(\n ({ onClick, noTrack, externalLinkIcon, ...props }, ref) => {\n const classes = useStyles();\n const analytics = useAnalytics();\n const to = isReactRouterBeta() ? useResolvedPath(props.to) : props.to;\n const linkText = getNodeText(props.children) || to;\n const external = isExternalUri(to);\n const newWindow = external && !!/^https?:/.exec(to);\n if (scriptProtocolPattern.test(to)) {\n throw new Error(\n \"Link component rejected javascript: URL as a security precaution\"\n );\n }\n const handleClick = (event) => {\n onClick?.(event);\n if (!noTrack) {\n analytics.captureEvent(\"click\", linkText, { attributes: { to } });\n }\n };\n return external ? (\n // External links\n /* @__PURE__ */ jsxs(\n MaterialLink,\n {\n ...newWindow ? { target: \"_blank\", rel: \"noopener\" } : {},\n ...props,\n ...props[\"aria-label\"] ? { \"aria-label\": `${props[\"aria-label\"]}, Opens in a new window` } : {},\n ref,\n href: to,\n onClick: handleClick,\n className: classNames(classes.externalLink, props.className),\n children: [\n props.children,\n externalLinkIcon && /* @__PURE__ */ jsx(ExternalLinkIcon, {}),\n /* @__PURE__ */ jsx(Typography, { component: \"span\", className: classes.visuallyHidden, children: \", Opens in a new window\" })\n ]\n }\n )\n ) : (\n // Interact with React Router for internal links\n /* @__PURE__ */ jsx(\n MaterialLink,\n {\n ...props,\n ref,\n component: Link$1,\n to,\n onClick: handleClick\n }\n )\n );\n }\n);\n\nexport { Link, isExternalUri, isReactRouterBeta, useResolvedPath };\n//# sourceMappingURL=Link.esm.js.map\n","\"use strict\";\n\nvar _interopRequireDefault = require(\"@babel/runtime/helpers/interopRequireDefault\");\n\nvar _interopRequireWildcard = require(\"@babel/runtime/helpers/interopRequireWildcard\");\n\nObject.defineProperty(exports, \"__esModule\", {\n value: true\n});\nexports.default = void 0;\n\nvar React = _interopRequireWildcard(require(\"react\"));\n\nvar _createSvgIcon = _interopRequireDefault(require(\"./utils/createSvgIcon\"));\n\nvar _default = (0, _createSvgIcon.default)( /*#__PURE__*/React.createElement(\"path\", {\n d: \"M19 19H5V5h7V3H5c-1.11 0-2 .9-2 2v14c0 1.1.89 2 2 2h14c1.1 0 2-.9 2-2v-7h-2v7zM14 3v2h3.59l-9.83 9.83 1.41 1.41L19 6.41V10h2V3h-7z\"\n}), 'OpenInNew');\n\nexports.default = _default;","import React, { useState } from \"react\";\nimport {\n useApi,\n fetchApiRef,\n identityApiRef,\n configApiRef,\n alertApiRef,\n} from \"@backstage/core-plugin-api\";\nimport { useAsync } from \"react-use\";\nimport {\n Table,\n TableColumn,\n Progress,\n ResponseErrorPanel,\n InfoCard,\n} from \"@backstage/core-components\";\nimport {\n kuadrantApiKeyUpdateAllPermission,\n kuadrantApiKeyUpdateOwnPermission,\n} from \"../../permissions\";\nimport { useKuadrantPermission } from \"../../utils/permissions\";\nimport {\n Button,\n Dialog,\n DialogTitle,\n DialogContent,\n DialogActions,\n Chip,\n Typography,\n Box,\n Tabs,\n Tab,\n Tooltip,\n Accordion,\n AccordionSummary,\n AccordionDetails,\n CircularProgress,\n} from \"@material-ui/core\";\nimport CheckCircleIcon from \"@material-ui/icons/CheckCircle\";\nimport CancelIcon from \"@material-ui/icons/Cancel\";\nimport ExpandMoreIcon from \"@material-ui/icons/ExpandMore\";\nimport { APIKey } from \"../../types/api-management\";\n\ninterface ApprovalDialogProps {\n open: boolean;\n request: APIKey | null;\n action: \"approve\" | \"reject\";\n processing: boolean;\n onClose: () => void;\n onConfirm: () => void;\n}\n\nconst ApprovalDialog = ({\n open,\n request,\n action,\n processing,\n onClose,\n onConfirm,\n}: ApprovalDialogProps) => {\n const actionLabel = action === \"approve\" ? \"Approve\" : \"Reject\";\n const processingLabel =\n action === \"approve\" ? \"Approving...\" : \"Rejecting...\";\n\n return (\n <Dialog\n open={open}\n onClose={processing ? undefined : onClose}\n maxWidth=\"sm\"\n fullWidth\n >\n <DialogTitle>{actionLabel} API Key</DialogTitle>\n <DialogContent>\n {request && (\n <>\n <p>\n <strong>User:</strong> {request.spec.requestedBy.userId}\n </p>\n <p>\n <strong>API:</strong>{\" \"}\n {request.spec.apiProductRef?.name || \"unknown\"}\n </p>\n <p>\n <strong>Tier:</strong> {request.spec.planTier}\n </p>\n <Box mb={2}>\n <Typography\n variant=\"body2\"\n component=\"span\"\n style={{ fontWeight: \"bold\" }}\n >\n Use Case:\n </Typography>{\" \"}\n <Typography\n variant=\"body2\"\n component=\"span\"\n style={{ whiteSpace: \"pre-wrap\" }}\n >\n {request.spec.useCase || \"-\"}\n </Typography>\n </Box>\n </>\n )}\n </DialogContent>\n <DialogActions>\n <Button onClick={onClose} disabled={processing}>\n Cancel\n </Button>\n <Button\n onClick={onConfirm}\n color={action === \"approve\" ? \"primary\" : \"secondary\"}\n variant=\"contained\"\n disabled={processing}\n startIcon={\n processing ? (\n <CircularProgress size={16} color=\"inherit\" />\n ) : undefined\n }\n >\n {processing ? processingLabel : actionLabel}\n </Button>\n </DialogActions>\n </Dialog>\n );\n};\n\ninterface BulkActionDialogProps {\n open: boolean;\n requests: APIKey[];\n action: \"approve\" | \"reject\";\n processing: boolean;\n onClose: () => void;\n onConfirm: () => void;\n}\n\nconst BulkActionDialog = ({\n open,\n requests,\n action,\n processing,\n onClose,\n onConfirm,\n}: BulkActionDialogProps) => {\n const isApprove = action === \"approve\";\n const actionLabel = isApprove ? \"Approve All\" : \"Reject All\";\n const processingLabel = isApprove ? \"Approving...\" : \"Rejecting...\";\n\n return (\n <Dialog\n open={open}\n onClose={processing ? undefined : onClose}\n maxWidth=\"md\"\n fullWidth\n >\n <DialogTitle>\n {isApprove ? \"Approve\" : \"Reject\"} {requests.length} API Keys\n </DialogTitle>\n <DialogContent>\n <Typography variant=\"body2\" paragraph>\n You are about to {isApprove ? \"approve\" : \"reject\"} the following API\n keys:\n </Typography>\n <Box mb={2} maxHeight={200} overflow=\"auto\">\n {requests.map((request) => (\n <Box\n key={`${request.metadata.namespace}/${request.metadata.name}`}\n mb={1}\n p={1}\n bgcolor=\"background.default\"\n >\n <Typography variant=\"body2\">\n <strong>{request.spec.requestedBy.userId}</strong> -{\" \"}\n {request.spec.apiProductRef?.name || \"unknown\"} (\n {request.spec.planTier})\n </Typography>\n </Box>\n ))}\n </Box>\n </DialogContent>\n <DialogActions>\n <Button onClick={onClose} disabled={processing}>\n Cancel\n </Button>\n <Button\n onClick={onConfirm}\n color={isApprove ? \"primary\" : \"secondary\"}\n variant=\"contained\"\n disabled={processing}\n startIcon={\n processing ? (\n <CircularProgress size={16} color=\"inherit\" />\n ) : undefined\n }\n >\n {processing ? processingLabel : actionLabel}\n </Button>\n </DialogActions>\n </Dialog>\n );\n};\n\nexport const ApprovalQueueCard = () => {\n const config = useApi(configApiRef);\n const fetchApi = useApi(fetchApiRef);\n const identityApi = useApi(identityApiRef);\n const alertApi = useApi(alertApiRef);\n const backendUrl = config.getString(\"backend.baseUrl\");\n const [refresh, setRefresh] = useState(0);\n const [selectedTab, setSelectedTab] = useState(0);\n const [selectedRequests, setSelectedRequests] = useState<APIKey[]>([]);\n const [dialogState, setDialogState] = useState<{\n open: boolean;\n request: APIKey | null;\n action: \"approve\" | \"reject\";\n processing: boolean;\n }>({\n open: false,\n request: null,\n action: \"approve\",\n processing: false,\n });\n const [bulkDialogState, setBulkDialogState] = useState<{\n open: boolean;\n requests: APIKey[];\n action: \"approve\" | \"reject\";\n processing: boolean;\n }>({\n open: false,\n requests: [],\n action: \"approve\",\n processing: false,\n });\n\n const {\n allowed: canUpdateAllRequests,\n loading: updateAllPermissionLoading,\n error: updateAllPermissionError,\n } = useKuadrantPermission(kuadrantApiKeyUpdateAllPermission);\n\n const {\n allowed: canUpdateOwnRequests,\n loading: updateOwnPermissionLoading,\n error: updateOwnPermissionError,\n } = useKuadrantPermission(kuadrantApiKeyUpdateOwnPermission);\n\n const updatePermissionLoading =\n updateAllPermissionLoading || updateOwnPermissionLoading;\n const updatePermissionError =\n updateAllPermissionError || updateOwnPermissionError;\n\n const { value, loading, error } = useAsync(async () => {\n const identity = await identityApi.getBackstageIdentity();\n const reviewedBy = identity.userEntityRef;\n\n console.log(\n \"ApprovalQueueCard: fetching all requests from\",\n `${backendUrl}/api/kuadrant/requests`,\n );\n\n // fetch requests and api products in parallel\n const [requestsResponse, apiProductsResponse] = await Promise.all([\n fetchApi.fetch(`${backendUrl}/api/kuadrant/requests`),\n fetchApi.fetch(`${backendUrl}/api/kuadrant/apiproducts`),\n ]);\n\n if (!requestsResponse.ok) {\n console.log(\n \"ApprovalQueueCard: failed to fetch requests, status:\",\n requestsResponse.status,\n );\n return {\n pending: [] as APIKey[],\n approved: [] as APIKey[],\n rejected: [] as APIKey[],\n reviewedBy,\n ownedApiProducts: new Set<string>(),\n };\n }\n\n // check content-type before parsing json\n const contentType = requestsResponse.headers.get(\"content-type\");\n if (!contentType?.includes(\"application/json\")) {\n console.error(\"ApprovalQueueCard: received non-json response\");\n alertApi.post({\n message:\n \"Unexpected content-type from the server response. Please contact support.\",\n display: \"transient\",\n severity: \"warning\",\n });\n return {\n pending: [] as APIKey[],\n approved: [] as APIKey[],\n rejected: [] as APIKey[],\n reviewedBy,\n ownedApiProducts: new Set<string>(),\n };\n }\n\n const data = await requestsResponse.json();\n const allRequests = data.items || [];\n\n // build set of api products owned by current user\n const ownedApiProducts = new Set<string>();\n if (apiProductsResponse.ok) {\n const apiProductsData = await apiProductsResponse.json();\n for (const product of apiProductsData.items || []) {\n const owner = product.metadata?.annotations?.[\"backstage.io/owner\"];\n if (owner === reviewedBy) {\n // key is namespace/name to match against request's namespace/apiProductRef.name\n ownedApiProducts.add(\n `${product.metadata.namespace}/${product.metadata.name}`,\n );\n }\n }\n }\n\n console.log(\n \"ApprovalQueueCard: received\",\n allRequests.length,\n \"total requests\",\n );\n console.log(\n \"ApprovalQueueCard: user owns\",\n ownedApiProducts.size,\n \"api products\",\n );\n\n // group by status (field is 'phase' not 'status')\n const pending = allRequests.filter((r: APIKey) => {\n const phase = (r.status as any)?.phase || \"Pending\";\n return phase === \"Pending\";\n });\n const approved = allRequests.filter((r: APIKey) => {\n const phase = (r.status as any)?.phase;\n return phase === \"Approved\";\n });\n const rejected = allRequests.filter((r: APIKey) => {\n const phase = (r.status as any)?.phase;\n return phase === \"Rejected\";\n });\n\n console.log(\"ApprovalQueueCard: grouped -\", {\n pending: pending.length,\n approved: approved.length,\n rejected: rejected.length,\n });\n\n return { pending, approved, rejected, reviewedBy, ownedApiProducts };\n }, [backendUrl, fetchApi, identityApi, refresh]);\n\n const handleApprove = (request: APIKey) => {\n setDialogState({\n open: true,\n request,\n action: \"approve\",\n processing: false,\n });\n };\n\n const handleReject = (request: APIKey) => {\n setDialogState({\n open: true,\n request,\n action: \"reject\",\n processing: false,\n });\n };\n\n const handleConfirm = async () => {\n if (!dialogState.request || !value) return;\n\n setDialogState((prev) => ({ ...prev, processing: true }));\n\n const endpoint =\n dialogState.action === \"approve\"\n ? `${backendUrl}/api/kuadrant/requests/${dialogState.request.metadata.namespace}/${dialogState.request.metadata.name}/approve`\n : `${backendUrl}/api/kuadrant/requests/${dialogState.request.metadata.namespace}/${dialogState.request.metadata.name}/reject`;\n\n try {\n const response = await fetchApi.fetch(endpoint, {\n method: \"POST\",\n headers: { \"Content-Type\": \"application/json\" },\n body: JSON.stringify({\n reviewedBy: value.reviewedBy,\n }),\n });\n\n if (!response.ok) {\n throw new Error(`failed to ${dialogState.action} request`);\n }\n\n setDialogState({\n open: false,\n request: null,\n action: \"approve\",\n processing: false,\n });\n setRefresh((r) => r + 1);\n const action = dialogState.action === \"approve\" ? \"approved\" : \"rejected\";\n alertApi.post({\n message: `API key ${action}`,\n severity: \"success\",\n display: \"transient\",\n });\n } catch (err) {\n console.error(`error ${dialogState.action}ing request:`, err);\n setDialogState((prev) => ({ ...prev, processing: false }));\n alertApi.post({\n message: `Failed to ${dialogState.action} API key`,\n severity: \"error\",\n display: \"transient\",\n });\n }\n };\n\n const handleBulkApprove = () => {\n if (selectedRequests.length === 0) return;\n setBulkDialogState({\n open: true,\n requests: selectedRequests,\n action: \"approve\",\n processing: false,\n });\n };\n\n const handleBulkReject = () => {\n if (selectedRequests.length === 0) return;\n setBulkDialogState({\n open: true,\n requests: selectedRequests,\n action: \"reject\",\n processing: false,\n });\n };\n\n const handleBulkConfirm = async () => {\n if (!value || bulkDialogState.requests.length === 0) return;\n\n setBulkDialogState((prev) => ({ ...prev, processing: true }));\n\n const isApprove = bulkDialogState.action === \"approve\";\n const endpoint = isApprove\n ? `${backendUrl}/api/kuadrant/requests/bulk-approve`\n : `${backendUrl}/api/kuadrant/requests/bulk-reject`;\n\n try {\n const response = await fetchApi.fetch(endpoint, {\n method: \"POST\",\n headers: { \"Content-Type\": \"application/json\" },\n body: JSON.stringify({\n requests: bulkDialogState.requests.map((r) => ({\n namespace: r.metadata.namespace,\n name: r.metadata.name,\n })),\n reviewedBy: value.reviewedBy,\n }),\n });\n\n if (!response.ok) {\n throw new Error(`failed to bulk ${bulkDialogState.action} requests`);\n }\n\n const count = bulkDialogState.requests.length;\n const action = isApprove ? \"approved\" : \"rejected\";\n setBulkDialogState({\n open: false,\n requests: [],\n action: \"approve\",\n processing: false,\n });\n setSelectedRequests([]);\n setRefresh((r) => r + 1);\n alertApi.post({\n message: `${count} API keys ${action}`,\n severity: \"success\",\n display: \"transient\",\n });\n } catch (err) {\n console.error(`error bulk ${bulkDialogState.action}ing requests:`, err);\n setBulkDialogState((prev) => ({ ...prev, processing: false }));\n alertApi.post({\n message: `Failed to bulk ${bulkDialogState.action} API keys`,\n severity: \"error\",\n display: \"transient\",\n });\n }\n };\n\n if (loading || updatePermissionLoading) {\n return <Progress />;\n }\n\n if (error) {\n return <ResponseErrorPanel error={error} />;\n }\n\n if (updatePermissionError) {\n return (\n <Box p={2}>\n <Typography color=\"error\">\n Unable to check permissions: {updatePermissionError.message}\n </Typography>\n <Typography variant=\"body2\" color=\"textSecondary\">\n Permission: kuadrant.apikey.update.all\n </Typography>\n <Typography variant=\"body2\" color=\"textSecondary\">\n Please try again or contact your administrator\n </Typography>\n </Box>\n );\n }\n\n const pending = value?.pending || [];\n const approved = value?.approved || [];\n const rejected = value?.rejected || [];\n\n const formatDate = (dateString: string) => {\n const date = new Date(dateString);\n return date.toLocaleDateString(\"en-GB\", {\n year: \"numeric\",\n month: \"short\",\n day: \"numeric\",\n hour: \"2-digit\",\n minute: \"2-digit\",\n });\n };\n\n const pendingColumns: TableColumn<APIKey>[] = [\n {\n title: \"Name\",\n field: \"metadata.name\",\n render: (row) => (\n <Typography variant=\"body2\">{row.metadata.name}</Typography>\n ),\n },\n {\n title: \"User\",\n field: \"spec.requestedBy.userId\",\n render: (row) => (\n <Typography variant=\"body2\">{row.spec.requestedBy.userId}</Typography>\n ),\n },\n {\n title: \"API\",\n field: \"spec.apiProductRef.name\",\n render: (row) => (\n <Typography variant=\"body2\">\n <strong>{row.spec.apiProductRef?.name || \"unknown\"}</strong>\n </Typography>\n ),\n },\n {\n title: \"Namespace\",\n field: \"metadata.namespace\",\n render: (row) => (\n <Typography variant=\"body2\">{row.metadata.namespace}</Typography>\n ),\n },\n {\n title: \"Tier\",\n field: \"spec.planTier\",\n render: (row) => <Chip label={row.spec.planTier} size=\"small\" />,\n },\n {\n title: \"Use Case\",\n field: \"spec.useCase\",\n render: (row) => {\n if (!row.spec.useCase) {\n return <Typography variant=\"body2\">-</Typography>;\n }\n return (\n <Tooltip title={row.spec.useCase} placement=\"top\">\n <Typography\n variant=\"body2\"\n style={{\n maxWidth: \"200px\",\n overflow: \"hidden\",\n textOverflow: \"ellipsis\",\n whiteSpace: \"nowrap\",\n }}\n >\n {row.spec.useCase}\n </Typography>\n </Tooltip>\n );\n },\n },\n {\n title: \"Requested\",\n field: \"metadata.creationTimestamp\",\n render: (row) => (\n <Typography variant=\"body2\">\n {row.metadata.creationTimestamp\n ? formatDate(row.metadata.creationTimestamp)\n : \"-\"}\n </Typography>\n ),\n },\n {\n title: \"Actions\",\n filtering: false,\n render: (row) => {\n const apiProductKey = `${row.metadata.namespace}/${row.spec.apiProductRef?.name || \"unknown\"}`;\n const ownsApiProduct =\n value?.ownedApiProducts?.has(apiProductKey) ?? false;\n const canUpdate =\n canUpdateAllRequests || (canUpdateOwnRequests && ownsApiProduct);\n if (!canUpdate) return null;\n return (\n <Box display=\"flex\" style={{ gap: 8 }}>\n <Button\n size=\"small\"\n startIcon={<CheckCircleIcon />}\n onClick={() => handleApprove(row)}\n color=\"primary\"\n variant=\"outlined\"\n >\n Approve\n </Button>\n <Button\n size=\"small\"\n startIcon={<CancelIcon />}\n onClick={() => handleReject(row)}\n color=\"secondary\"\n variant=\"outlined\"\n >\n Reject\n </Button>\n </Box>\n );\n },\n },\n ];\n\n const approvedColumns: TableColumn<APIKey>[] = [\n {\n title: \"Name\",\n field: \"metadata.name\",\n render: (row) => (\n <Typography variant=\"body2\">{row.metadata.name}</Typography>\n ),\n },\n {\n title: \"User\",\n field: \"spec.requestedBy.userId\",\n render: (row) => (\n <Typography variant=\"body2\">{row.spec.requestedBy.userId}</Typography>\n ),\n },\n {\n title: \"API\",\n field: \"spec.apiProductRef.name\",\n render: (row) => (\n <Typography variant=\"body2\">\n <strong>{row.spec.apiProductRef?.name || \"unknown\"}</strong>\n </Typography>\n ),\n },\n {\n title: \"Namespace\",\n field: \"metadata.namespace\",\n render: (row) => (\n <Typography variant=\"body2\">{row.metadata.namespace}</Typography>\n ),\n },\n {\n title: \"Tier\",\n field: \"spec.planTier\",\n render: (row) => <Chip label={row.spec.planTier} size=\"small\" />,\n },\n {\n title: \"Requested\",\n field: \"metadata.creationTimestamp\",\n render: (row) => (\n <Typography variant=\"body2\">\n {row.metadata.creationTimestamp\n ? formatDate(row.metadata.creationTimestamp)\n : \"-\"}\n </Typography>\n ),\n },\n {\n title: \"Approved\",\n field: \"status.reviewedAt\",\n render: (row) => (\n <Typography variant=\"body2\">\n {row.status?.reviewedAt ? formatDate(row.status.reviewedAt) : \"-\"}\n </Typography>\n ),\n },\n {\n title: \"Reviewed By\",\n field: \"status.reviewedBy\",\n render: (row) => (\n <Typography variant=\"body2\">{row.status?.reviewedBy || \"-\"}</Typography>\n ),\n },\n {\n title: \"Approval Type\",\n field: \"status.reviewedBy\",\n render: (row) => {\n const isAutomatic = row.status?.reviewedBy === \"system\";\n return (\n <Chip\n label={isAutomatic ? \"Automatic\" : \"Manual\"}\n size=\"small\"\n color={isAutomatic ? \"default\" : \"primary\"}\n />\n );\n },\n },\n ];\n\n const rejectedColumns: TableColumn<APIKey>[] = [\n {\n title: \"Name\",\n field: \"metadata.name\",\n render: (row) => (\n <Typography variant=\"body2\">{row.metadata.name}</Typography>\n ),\n },\n {\n title: \"User\",\n field: \"spec.requestedBy.userId\",\n render: (row) => (\n <Typography variant=\"body2\">{row.spec.requestedBy.userId}</Typography>\n ),\n },\n {\n title: \"API\",\n field: \"spec.apiProductRef.name\",\n render: (row) => (\n <Typography variant=\"body2\">\n <strong>{row.spec.apiProductRef?.name || \"unknown\"}</strong>\n </Typography>\n ),\n },\n {\n title: \"Namespace\",\n field: \"metadata.namespace\",\n render: (row) => (\n <Typography variant=\"body2\">{row.metadata.namespace}</Typography>\n ),\n },\n {\n title: \"Tier\",\n field: \"spec.planTier\",\n render: (row) => <Chip label={row.spec.planTier} size=\"small\" />,\n },\n {\n title: \"Requested\",\n field: \"metadata.creationTimestamp\",\n render: (row) => (\n <Typography variant=\"body2\">\n {row.metadata.creationTimestamp\n ? formatDate(row.metadata.creationTimestamp)\n : \"-\"}\n </Typography>\n ),\n },\n {\n title: \"Rejected\",\n field: \"status.reviewedAt\",\n render: (row) => (\n <Typography variant=\"body2\">\n {row.status?.reviewedAt ? formatDate(row.status.reviewedAt) : \"-\"}\n </Typography>\n ),\n },\n {\n title: \"Reviewed By\",\n field: \"status.reviewedBy\",\n render: (row) => (\n <Typography variant=\"body2\">{row.status?.reviewedBy || \"-\"}</Typography>\n ),\n },\n ];\n\n const getTabData = () => {\n const addIds = (data: APIKey[]) =>\n data.map((item) => ({ ...item, id: item.metadata.name }));\n\n switch (selectedTab) {\n case 0:\n // pending tab - add tableData.checked to control checkbox state\n const pendingWithSelection = pending.map((row: APIKey) => {\n const isSelected = selectedRequests.some(\n (selected) =>\n selected.metadata.name === row.metadata.name &&\n selected.metadata.namespace === row.metadata.namespace,\n );\n return {\n ...row,\n tableData: { checked: isSelected },\n };\n });\n return {\n data: pendingWithSelection,\n columns: pendingColumns,\n showSelection: true,\n };\n case 1:\n return {\n data: addIds(approved),\n columns: approvedColumns,\n showSelection: false,\n };\n case 2:\n return {\n data: addIds(rejected),\n columns: rejectedColumns,\n showSelection: false,\n };\n default:\n return {\n data: addIds(pending),\n columns: pendingColumns,\n showSelection: true,\n };\n }\n };\n\n const tabData = getTabData();\n\n // group requests by api product (namespace/name)\n const groupByApiProduct = (requests: APIKey[]) => {\n const grouped = new Map<string, APIKey[]>();\n requests.forEach((request) => {\n const key = `${request.metadata.namespace}/${request.spec.apiProductRef?.name || \"unknown\"}`;\n if (!grouped.has(key)) {\n grouped.set(key, []);\n }\n grouped.get(key)!.push(request);\n });\n return grouped;\n };\n\n const groupedData = groupByApiProduct(tabData.data);\n const apiProductKeys = Array.from(groupedData.keys()).sort();\n\n return (\n <>\n <InfoCard\n title=\"API Access Requests\"\n subheader={`${pending.length} pending, ${approved.length} approved, ${rejected.length} rejected`}\n >\n <Box mb={2} data-testid=\"approval-queue-card\">\n <Tabs\n value={selectedTab}\n onChange={(_, newValue) => {\n setSelectedTab(newValue);\n setSelectedRequests([]);\n }}\n indicatorColor=\"primary\"\n textColor=\"primary\"\n data-testid=\"approval-queue-tabs\"\n >\n <Tab\n label={`Pending (${pending.length})`}\n data-testid=\"approval-queue-pending-tab\"\n />\n <Tab\n label={`Approved (${approved.length})`}\n data-testid=\"approval-queue-approved-tab\"\n />\n <Tab\n label={`Rejected (${rejected.length})`}\n data-testid=\"approval-queue-rejected-tab\"\n />\n </Tabs>\n </Box>\n\n {selectedTab === 0 && selectedRequests.length > 0 && (\n <Box\n mb={2}\n display=\"flex\"\n alignItems=\"center\"\n justifyContent=\"space-between\"\n p={2}\n bgcolor=\"background.default\"\n >\n <Typography variant=\"body2\">\n {selectedRequests.length} request\n {selectedRequests.length !== 1 ? \"s\" : \"\"} selected\n </Typography>\n <Box display=\"flex\" style={{ gap: 8 }}>\n <Button\n size=\"small\"\n variant=\"contained\"\n color=\"primary\"\n startIcon={<CheckCircleIcon />}\n onClick={handleBulkApprove}\n >\n Approve Selected\n </Button>\n <Button\n size=\"small\"\n variant=\"contained\"\n color=\"secondary\"\n startIcon={<CancelIcon />}\n onClick={handleBulkReject}\n >\n Reject Selected\n </Button>\n </Box>\n </Box>\n )}\n\n {tabData.data.length === 0 ? (\n <Box p={3} textAlign=\"center\">\n <Typography variant=\"body1\" color=\"textSecondary\">\n {selectedTab === 0 && \"No pending requests.\"}\n {selectedTab === 1 && \"No approved requests.\"}\n {selectedTab === 2 && \"No rejected requests.\"}\n </Typography>\n </Box>\n ) : (\n <Box>\n {apiProductKeys.map((apiProductKey) => {\n const requests = groupedData.get(apiProductKey) || [];\n const displayName =\n requests[0]?.spec.apiProductRef?.name || apiProductKey;\n const ownsThisApiProduct =\n value?.ownedApiProducts?.has(apiProductKey) ?? false;\n const canSelectRows =\n canUpdateAllRequests ||\n (canUpdateOwnRequests && ownsThisApiProduct);\n return (\n <Accordion\n key={apiProductKey}\n defaultExpanded={apiProductKeys.length === 1}\n >\n <AccordionSummary expandIcon={<ExpandMoreIcon />}>\n <Box\n display=\"flex\"\n alignItems=\"center\"\n justifyContent=\"space-between\"\n width=\"100%\"\n >\n <Typography variant=\"h6\">{displayName}</Typography>\n <Chip\n label={`${requests.length} request${requests.length !== 1 ? \"s\" : \"\"}`}\n size=\"small\"\n color=\"primary\"\n style={{ marginRight: 16 }}\n />\n </Box>\n </AccordionSummary>\n <AccordionDetails>\n <Box width=\"100%\">\n <Table\n options={{\n selection: canSelectRows && tabData.showSelection,\n paging: requests.length > 5,\n pageSize: 20,\n search: true,\n filtering: true,\n debounceInterval: 300,\n showTextRowsSelected: false,\n toolbar: true,\n emptyRowsWhenPaging: false,\n }}\n data={requests}\n columns={tabData.columns}\n onSelectionChange={(rows) => {\n // merge selections from this api product with selections from other products\n const otherSelections = selectedRequests.filter(\n (r) =>\n `${r.metadata.namespace}/${r.spec.apiProductRef?.name || \"unknown\"}` !==\n apiProductKey,\n );\n setSelectedRequests([\n ...otherSelections,\n ...(rows as APIKey[]),\n ]);\n }}\n />\n </Box>\n </AccordionDetails>\n </Accordion>\n );\n })}\n </Box>\n )}\n </InfoCard>\n <ApprovalDialog\n open={dialogState.open}\n request={dialogState.request}\n action={dialogState.action}\n processing={dialogState.processing}\n onClose={() =>\n setDialogState({\n open: false,\n request: null,\n action: \"approve\",\n processing: false,\n })\n }\n onConfirm={handleConfirm}\n />\n <BulkActionDialog\n open={bulkDialogState.open}\n requests={bulkDialogState.requests}\n action={bulkDialogState.action}\n processing={bulkDialogState.processing}\n onClose={() =>\n setBulkDialogState({\n open: false,\n requests: [],\n action: \"approve\",\n processing: false,\n })\n }\n onConfirm={handleBulkConfirm}\n />\n </>\n );\n};\n","import { usePermission } from '@backstage/plugin-permission-react';\nimport { Permission, ResourcePermission } from '@backstage/plugin-permission-common';\n\n/**\n * result of a permission check including error state\n */\nexport interface PermissionCheckResult {\n allowed: boolean;\n loading: boolean;\n error?: Error;\n}\n\n/**\n * custom hook for checking kuadrant permissions that handles both\n * BasicPermission and ResourcePermission types without type bypasses\n *\n * @param permission - the permission to check\n * @param resourceRef - optional resource reference for ResourcePermissions\n * @returns permission check result with error handling\n *\n * @example\n * // basic permission\n * const { allowed, loading, error } = useKuadrantPermission(\n * kuadrantApiProductListPermission\n * );\n *\n * @example\n * // resource permission\n * const { allowed, loading, error } = useKuadrantPermission(\n * kuadrantApiKeyCreatePermission,\n * 'apiproduct:namespace/name'\n * );\n */\nexport function useKuadrantPermission(\n permission: Permission,\n resourceRef?: string,\n): PermissionCheckResult {\n // construct the permission request based on whether it's a ResourcePermission\n const permissionRequest = 'resourceType' in permission\n ? { permission: permission as ResourcePermission, resourceRef }\n : { permission };\n\n const result = usePermission(permissionRequest as any);\n\n return {\n allowed: result.allowed,\n loading: result.loading,\n error: result.error,\n };\n}\n\n/**\n * helper to determine if a user can delete a specific API key or request\n *\n * @param ownerId - the user id who owns the key/request\n * @param currentUserId - the current user's id\n * @param canDeleteOwn - whether user has permission to delete their own keys\n * @param canDeleteAll - whether user has permission to delete all keys\n * @returns true if user can delete this specific key/request\n */\nexport function canDeleteResource(\n ownerId: string,\n currentUserId: string,\n canDeleteOwn: boolean,\n canDeleteAll: boolean,\n): boolean {\n if (canDeleteAll) return true;\n if (canDeleteOwn && ownerId === currentUserId) return true;\n return false;\n}\n","\"use strict\";\n\nvar _interopRequireDefault = require(\"@babel/runtime/helpers/interopRequireDefault\");\n\nvar _interopRequireWildcard = require(\"@babel/runtime/helpers/interopRequireWildcard\");\n\nObject.defineProperty(exports, \"__esModule\", {\n value: true\n});\nexports.default = void 0;\n\nvar React = _interopRequireWildcard(require(\"react\"));\n\nvar _createSvgIcon = _interopRequireDefault(require(\"./utils/createSvgIcon\"));\n\nvar _default = (0, _createSvgIcon.default)( /*#__PURE__*/React.createElement(\"path\", {\n d: \"M12 4l-1.41 1.41L16.17 11H4v2h12.17l-5.58 5.59L12 20l8-8z\"\n}), 'ArrowForward');\n\nexports.default = _default;","\"use strict\";\n\nvar _interopRequireDefault = require(\"@babel/runtime/helpers/interopRequireDefault\");\n\nvar _interopRequireWildcard = require(\"@babel/runtime/helpers/interopRequireWildcard\");\n\nObject.defineProperty(exports, \"__esModule\", {\n value: true\n});\nexports.default = void 0;\n\nvar React = _interopRequireWildcard(require(\"react\"));\n\nvar _createSvgIcon = _interopRequireDefault(require(\"./utils/createSvgIcon\"));\n\nvar _default = (0, _createSvgIcon.default)( /*#__PURE__*/React.createElement(\"path\", {\n d: \"M12 2C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10S17.52 2 12 2zm-2 15l-5-5 1.41-1.41L10 14.17l7.59-7.59L19 8l-9 9z\"\n}), 'CheckCircle');\n\nexports.default = _default;","import { jsxs, jsx } from 'react/jsx-runtime';\nimport Box from '@material-ui/core/Box';\nimport Divider from '@material-ui/core/Divider';\nimport { makeStyles } from '@material-ui/core/styles';\nimport Typography from '@material-ui/core/Typography';\nimport ArrowIcon from '@material-ui/icons/ArrowForward';\nimport { Link } from '../../components/Link/Link.esm.js';\n\nconst useStyles = makeStyles(\n (theme) => ({\n root: {\n maxWidth: \"fit-content\",\n padding: theme.spacing(2, 2, 2, 2.5)\n },\n boxTitle: {\n margin: 0,\n color: theme.palette.textSubtle\n },\n arrow: {\n color: theme.palette.textSubtle\n }\n }),\n { name: \"BackstageBottomLink\" }\n);\nfunction BottomLink(props) {\n const { link, title, onClick } = props;\n const classes = useStyles();\n return /* @__PURE__ */ jsxs(Box, { children: [\n /* @__PURE__ */ jsx(Divider, {}),\n /* @__PURE__ */ jsx(Link, { to: link, onClick, underline: \"none\", children: /* @__PURE__ */ jsxs(Box, { display: \"flex\", alignItems: \"center\", className: classes.root, children: [\n /* @__PURE__ */ jsx(Box, { className: classes.boxTitle, fontWeight: \"fontWeightBold\", m: 1, children: /* @__PURE__ */ jsx(Typography, { children: /* @__PURE__ */ jsx(\"strong\", { children: title }) }) }),\n /* @__PURE__ */ jsx(ArrowIcon, { className: classes.arrow })\n ] }) })\n ] });\n}\n\nexport { BottomLink };\n//# sourceMappingURL=BottomLink.esm.js.map\n","import { jsx } from 'react/jsx-runtime';\nimport Button$1 from '@material-ui/core/Button';\nimport { forwardRef } from 'react';\nimport { Link } from '../Link/Link.esm.js';\n\nconst LinkWrapper = forwardRef((props, ref) => /* @__PURE__ */ jsx(Link, { ref, ...props, color: \"initial\" }));\nconst LinkButton = forwardRef((props, ref) => /* @__PURE__ */ jsx(Button$1, { ref, component: LinkWrapper, ...props }));\nconst Button = LinkButton;\n\nexport { Button, LinkButton };\n//# sourceMappingURL=LinkButton.esm.js.map\n","import { jsx } from 'react/jsx-runtime';\nimport Typography from '@material-ui/core/Typography';\nimport { Component } from 'react';\nimport { LinkButton } from '../../components/LinkButton/LinkButton.esm.js';\nimport { ErrorPanel } from '../../components/ErrorPanel/ErrorPanel.esm.js';\nimport { coreComponentsTranslationRef } from '../../translation.esm.js';\nimport { useTranslationRef } from '@backstage/core-plugin-api/alpha';\n\nconst SlackLink = (props) => {\n const { slackChannel } = props;\n const { t } = useTranslationRef(coreComponentsTranslationRef);\n if (!slackChannel) {\n return null;\n } else if (typeof slackChannel === \"string\") {\n return /* @__PURE__ */ jsx(Typography, { children: t(\"errorBoundary.title\", { slackChannel }) });\n } else if (!slackChannel.href) {\n return /* @__PURE__ */ jsx(Typography, { children: t(\"errorBoundary.title\", {\n slackChannel: slackChannel.name\n }) });\n }\n return /* @__PURE__ */ jsx(LinkButton, { to: slackChannel.href, variant: \"contained\", children: slackChannel.name });\n};\nconst ErrorBoundary = class ErrorBoundary2 extends Component {\n constructor(props) {\n super(props);\n this.state = {\n error: void 0,\n errorInfo: void 0\n };\n }\n componentDidCatch(error, errorInfo) {\n console.error(`ErrorBoundary, error: ${error}`, { error, errorInfo });\n this.setState({ error, errorInfo });\n }\n render() {\n const { slackChannel, children } = this.props;\n const { error } = this.state;\n if (!error) {\n return children;\n }\n return /* @__PURE__ */ jsx(ErrorPanel, { title: \"Something Went Wrong\", error, children: /* @__PURE__ */ jsx(SlackLink, { slackChannel }) });\n }\n};\n\nexport { ErrorBoundary };\n//# sourceMappingURL=ErrorBoundary.esm.js.map\n","import { jsx, jsxs } from 'react/jsx-runtime';\nimport Card from '@material-ui/core/Card';\nimport CardActions from '@material-ui/core/CardActions';\nimport CardContent from '@material-ui/core/CardContent';\nimport CardHeader from '@material-ui/core/CardHeader';\nimport Divider from '@material-ui/core/Divider';\nimport { makeStyles, withStyles } from '@material-ui/core/styles';\nimport classNames from 'classnames';\nimport { BottomLink } from '../BottomLink/BottomLink.esm.js';\nimport { ErrorBoundary } from '../ErrorBoundary/ErrorBoundary.esm.js';\n\nconst useStyles = makeStyles(\n (theme) => ({\n noPadding: {\n padding: 0,\n \"&:last-child\": {\n paddingBottom: 0\n }\n },\n contentAlignBottom: {\n display: \"flex\",\n alignItems: \"self-end\"\n },\n header: {\n padding: theme.spacing(2, 2, 2, 2.5)\n },\n headerTitle: {\n fontWeight: theme.typography.fontWeightBold\n },\n headerSubheader: {\n paddingTop: theme.spacing(1)\n },\n headerAvatar: {},\n headerAction: {},\n headerContent: {},\n subheader: {\n display: \"flex\"\n }\n }),\n { name: \"BackstageInfoCard\" }\n);\nconst CardActionsTopRight = withStyles(\n (theme) => ({\n root: {\n display: \"inline-block\",\n padding: theme.spacing(8, 8, 0, 0),\n float: \"right\"\n }\n }),\n { name: \"BackstageInfoCardCardActionsTopRight\" }\n)(CardActions);\nconst VARIANT_STYLES = {\n card: {\n flex: {\n display: \"flex\",\n flexDirection: \"column\"\n },\n fullHeight: {\n display: \"flex\",\n flexDirection: \"column\",\n height: \"100%\"\n },\n gridItem: {\n display: \"flex\",\n flexDirection: \"column\",\n height: \"calc(100% - 10px)\",\n // for pages without content header\n marginBottom: \"10px\",\n breakInside: \"avoid-page\",\n \"@media print\": {\n height: \"auto\"\n }\n }\n },\n cardContent: {\n fullHeight: {\n flex: 1\n },\n gridItem: {\n flex: 1\n }\n }\n};\nfunction InfoCard(props) {\n const {\n title,\n subheader,\n divider = true,\n deepLink,\n slackChannel,\n errorBoundaryProps,\n variant,\n alignContent = \"normal\",\n children,\n headerStyle,\n headerProps,\n icon,\n action,\n actionsClassName,\n actions,\n cardClassName,\n actionsTopRight,\n className,\n noPadding,\n titleTypographyProps,\n subheaderTypographyProps\n } = props;\n const classes = useStyles();\n let calculatedStyle = {};\n let calculatedCardStyle = {};\n if (variant) {\n const variants = variant.split(/[\\s]+/g);\n variants.forEach((name) => {\n calculatedStyle = {\n ...calculatedStyle,\n ...VARIANT_STYLES.card[name]\n };\n calculatedCardStyle = {\n ...calculatedCardStyle,\n ...VARIANT_STYLES.cardContent[name]\n };\n });\n }\n const cardSubTitle = () => {\n if (!subheader && !icon) {\n return null;\n }\n return /* @__PURE__ */ jsxs(\"div\", { \"data-testid\": \"info-card-subheader\", children: [\n subheader && /* @__PURE__ */ jsx(\"div\", { className: classes.subheader, children: subheader }),\n icon\n ] });\n };\n const errProps = errorBoundaryProps || (slackChannel ? { slackChannel } : {});\n return /* @__PURE__ */ jsx(Card, { style: calculatedStyle, className, children: /* @__PURE__ */ jsxs(ErrorBoundary, { ...errProps, children: [\n title && /* @__PURE__ */ jsx(\n CardHeader,\n {\n classes: {\n root: classNames(classes.header),\n title: classes.headerTitle,\n subheader: classes.headerSubheader,\n avatar: classes.headerAvatar,\n action: classes.headerAction,\n content: classes.headerContent\n },\n title,\n subheader: cardSubTitle(),\n action,\n style: { ...headerStyle },\n titleTypographyProps,\n subheaderTypographyProps,\n ...headerProps\n }\n ),\n actionsTopRight && /* @__PURE__ */ jsx(CardActionsTopRight, { children: actionsTopRight }),\n divider && /* @__PURE__ */ jsx(Divider, {}),\n /* @__PURE__ */ jsx(\n CardContent,\n {\n className: classNames(cardClassName, {\n [classes.noPadding]: noPadding,\n [classes.contentAlignBottom]: alignContent === \"bottom\"\n }),\n style: calculatedCardStyle,\n children\n }\n ),\n actions && /* @__PURE__ */ jsx(CardActions, { className: actionsClassName, children: actions }),\n deepLink && /* @__PURE__ */ jsx(BottomLink, { ...deepLink })\n ] }) });\n}\n\nexport { InfoCard };\n//# sourceMappingURL=InfoCard.esm.js.map\n"],"names":["createPermission","name","attributes","action","kuadrantPlanPolicyListPermission","kuadrantApiProductCreatePermission","kuadrantApiProductReadAllPermission","kuadrantApiProductUpdateOwnPermission","kuadrantApiProductUpdateAllPermission","kuadrantApiProductDeleteOwnPermission","kuadrantApiProductDeleteAllPermission","kuadrantApiProductListPermission","kuadrantApiKeyCreatePermission","resourceType","kuadrantApiKeyReadOwnPermission","kuadrantApiKeyUpdateOwnPermission","kuadrantApiKeyUpdateAllPermission","kuadrantApiKeyDeleteOwnPermission","kuadrantApiKeyDeleteAllPermission","kuadrantApiKeyApprovePermission","useAsync","fn","deps","_a","initialState","loading","lastCallId","useRef","isMounted","useMountedState","useState","state","set","callback","useCallback","args","_i","arguments","length","callId","current","prevState","__assign","apply","then","value","error","useAsyncFn","useEffect","useStyles","makeStyles","theme","visuallyHidden","clip","clipPath","overflow","position","userSelect","whiteSpace","height","width","externalLink","externalLinkIcon","verticalAlign","marginLeft","spacing","ExternalLinkIcon","Icon","useApp","getSystemIcon","classes","jsx","className","isExternalUri","uri","test","scriptProtocolPattern","originalWindowOpen","window","open","__backstage","newOpen","url","String","Error","this","getNodeText","node","Array","map","join","trim","props","children","includes","Link","forwardRef","onClick","noTrack","ref","analytics","useAnalytics","to","obj","createRoutesFromChildren","Route","index","element","isReactRouterBeta","resolvedPath","basePath","useApi","configApiRef","getOptionalString","useBaseUrl","pathname","URL","trimEnd","useBasePath","external","startsWithBasePath","startsWith","concat","useResolvedPath","linkText","newWindow","exec","handleClick","event","captureEvent","jsxs","target","rel","href","component","_interopRequireDefault","_interopRequireWildcard","exports","React","_default","default","createElement","d","ApprovalDialog","request","processing","onClose","onConfirm","actionLabel","processingLabel","Dialog","undefined","maxWidth","fullWidth","DialogTitle","DialogContent","p","strong","spec","requestedBy","userId","apiProductRef","planTier","Box","mb","Typography","variant","style","fontWeight","useCase","DialogActions","Button","disabled","color","startIcon","CircularProgress","size","BulkActionDialog","requests","isApprove","paragraph","maxHeight","bgcolor","metadata","namespace","ApprovalQueueCard","config","fetchApi","fetchApiRef","identityApi","identityApiRef","alertApi","alertApiRef","backendUrl","getString","refresh","setRefresh","selectedTab","setSelectedTab","selectedRequests","setSelectedRequests","dialogState","setDialogState","bulkDialogState","setBulkDialogState","allowed","canUpdateAllRequests","updateAllPermissionLoading","updateAllPermissionError","useKuadrantPermission","canUpdateOwnRequests","updateOwnPermissionLoading","updateOwnPermissionError","updatePermissionLoading","updatePermissionError","async","reviewedBy","getBackstageIdentity","userEntityRef","console","log","requestsResponse","apiProductsResponse","Promise","all","fetch","ok","status","pending","approved","rejected","ownedApiProducts","Set","contentType","headers","get","post","message","display","severity","allRequests","json","items","apiProductsData","product","annotations","add","filter","r","phase","Progress","ResponseErrorPanel","formatDate","dateString","Date","toLocaleDateString","year","month","day","hour","minute","pendingColumns","title","field","render","row","Chip","label","Tooltip","placement","textOverflow","creationTimestamp","filtering","apiProductKey","ownsApiProduct","has","gap","CheckCircleIcon","CancelIcon","approvedColumns","reviewedAt","isAutomatic","rejectedColumns","tabData","addIds","data","item","id","isSelected","some","selected","tableData","checked","columns","showSelection","getTabData","groupedData","grouped","Map","forEach","key","push","groupByApiProduct","apiProductKeys","from","keys","sort","InfoCard","subheader","data-testid","Tabs","onChange","_","newValue","indicatorColor","textColor","Tab","alignItems","justifyContent","textAlign","displayName","ownsThisApiProduct","canSelectRows","Accordion","defaultExpanded","AccordionSummary","expandIcon","ExpandMoreIcon","marginRight","AccordionDetails","Table","options","selection","paging","pageSize","search","debounceInterval","showTextRowsSelected","toolbar","emptyRowsWhenPaging","onSelectionChange","rows","otherSelections","prev","endpoint","method","body","JSON","stringify","err","count","permission","resourceRef","permissionRequest","result","usePermission","canDeleteResource","ownerId","currentUserId","canDeleteOwn","canDeleteAll","root","padding","boxTitle","margin","palette","textSubtle","arrow","BottomLink","link","Divider","underline","m","A","LinkWrapper","LinkButton","SlackLink","slackChannel","t","ErrorBoundary","Component","constructor","super","errorInfo","componentDidCatch","setState","noPadding","paddingBottom","contentAlignBottom","header","headerTitle","typography","fontWeightBold","headerSubheader","paddingTop","headerAvatar","headerAction","headerContent","CardActionsTopRight","withStyles","float","CardActions","VARIANT_STYLES","card","flex","flexDirection","fullHeight","gridItem","marginBottom","breakInside","cardContent","divider","deepLink","errorBoundaryProps","alignContent","headerStyle","headerProps","icon","actionsClassName","actions","cardClassName","actionsTopRight","titleTypographyProps","subheaderTypographyProps","calculatedStyle","calculatedCardStyle","split","errProps","Card","CardHeader","avatar","content","CardContent"],"sourceRoot":""}
@@ -0,0 +1,2 @@
1
+ "use strict";(self.webpackChunkinternal_plugin_kuadrant=self.webpackChunkinternal_plugin_kuadrant||[]).push([[4306],{16400:(e,t,a)=>{a.d(t,{tN:()=>s}),a(31085),a(26659);var n=a(64218);function s(){const e=(0,n.useVersionedContext)("entity-context");if(!e)throw new Error("Entity context is not available");const t=e.atVersion(1);if(!t)throw new Error("EntityContext v1 not available");if(!t.entity)throw new Error("useEntity hook is being called outside of an EntityLayout where the entity has not been loaded. If this is intentional, please use useAsyncEntity instead.");return{entity:t.entity}}(0,n.createVersionedContext)("entity-context")},18466:(e,t,a)=>{var n=a(4293),s=a(78920);t.A=void 0;var r=s(a(95478)),i=(0,n(a(74044)).default)(r.createElement("path",{d:"M19 13h-6v6h-2v-6H5v-2h6V5h2v6h6v2z"}),"Add");t.A=i},39053:(e,t,a)=>{var n=a(4293),s=a(78920);t.A=void 0;var r=s(a(95478)),i=(0,n(a(74044)).default)(r.createElement("path",{d:"M12 2C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10S17.52 2 12 2zm1 15h-2v-6h2v6zm0-8h-2V7h2v2z"}),"Info");t.A=i},54306:(e,t,a)=>{a.r(t),a.d(t,{ApiKeyManagementTab:()=>U});var n=a(31085),s=a(95478),r=a(35015),i=a(26648),o=a(86687),l=a(42367),d=a(25010),c=a(10394),u=a(72501),p=a(31653),h=a(38605),m=a(67720),y=a(71677),v=a(29365),A=a(78467),g=a(42899),x=a(64947),f=a(76891),j=a(61477),w=a(46805),b=a(95061),k=a(48543),P=a(81215),S=a(26343),C=a(16249),I=a(93453),R=a(37757),q=a(22097),$=a(16400),z=a(32269),T=a(61524),E=a(93246),B=a(92399),D=a(18466),K=a(71407),H=a(99594),W=a(77225),M=a(39053),N=a(34955),V=a(46205),L=a(16397),_=a(63221);const U=({namespace:e})=>{var t,a,U,F,G;const{entity:O}=(0,$.tN)(),X=(0,q.useApi)(q.configApiRef),J=(0,q.useApi)(q.identityApiRef),Y=(0,q.useApi)(q.fetchApiRef),Q=(0,q.useApi)(q.alertApiRef),Z=X.getString("backend.baseUrl"),[ee,te]=(0,s.useState)(new Set),[ae,ne]=(0,s.useState)(0),[se,re]=(0,s.useState)(""),[ie,oe]=(0,s.useState)(""),[le,de]=(0,s.useState)(!1),[ce,ue]=(0,s.useState)(""),[pe,he]=(0,s.useState)(""),[me,ye]=(0,s.useState)(!1),[ve,Ae]=(0,s.useState)(null),[ge,xe]=(0,s.useState)(!1),[fe,je]=(0,s.useState)(null),[we,be]=(0,s.useState)(null),[ke,Pe]=(0,s.useState)(null),[Se,Ce]=(0,s.useState)(null),[Ie,Re]=(0,s.useState)(new Set),[qe,$e]=(0,s.useState)({open:!1,request:null}),[ze,Te]=(0,s.useState)(new Map),[Ee,Be]=(0,s.useState)(new Set),[De,Ke]=(0,s.useState)(new Set),[He,We]=(0,s.useState)(!1),[Me,Ne]=(0,s.useState)(null),Ve=(null===(t=O.metadata.annotations)||void 0===t?void 0:t["kuadrant.io/apiproduct"])||O.metadata.name,Le=(null===(a=O.metadata.annotations)||void 0===a?void 0:a["kuadrant.io/namespace"])||e||"default";(0,r.A)(async()=>{const e=await J.getBackstageIdentity(),t=await J.getProfileInfo();re(e.userEntityRef),oe(t.email||"")},[J]);const{value:_e,loading:Ue,error:Fe}=(0,r.A)(async()=>{const e=await Y.fetch(`${Z}/api/kuadrant/requests/my?namespace=${Le}`);if(!e.ok)throw new Error("failed to fetch requests");return((await e.json()).items||[]).filter(e=>e.spec.apiProductRef.name===Ve&&e.metadata.namespace===Le)},[Ve,Le,ae,Y,Z]),{value:Ge,loading:Oe,error:Xe}=(0,r.A)(async()=>{var e;const t=await Y.fetch(`${Z}/api/kuadrant/apiproducts`);if(!t.ok)throw new Error("failed to fetch api products");return null===(e=(await t.json()).items)||void 0===e?void 0:e.find(e=>e.metadata.namespace===Le&&e.metadata.name===Ve)},[Le,Ve,Y]),Je=Ge?`apiproduct:${Ge.metadata.namespace}/${Ge.metadata.name}`:void 0,{allowed:Ye,loading:Qe,error:Ze}=(0,V.l)(N.dp,Je),{allowed:et,loading:tt,error:at}=(0,V.l)(N.uL),{allowed:nt,loading:st,error:rt}=(0,V.l)(N.jH),{allowed:it,loading:ot,error:lt}=(0,V.l)(N.q0),dt=()=>{be(null),Pe(null)},ct=()=>{ke&&(je(ke),xe(!0),dt())},ut=()=>{if(!ke)return;const e=ke;dt(),$e({open:!0,request:e})},pt=e=>{te(t=>{const a=new Set(t);return a.has(e)?a.delete(e):a.add(e),a})},ht=(0,s.useMemo)(()=>[{render:e=>{var t;const a=e.rowData;if(!(null==a||null===(t=a.metadata)||void 0===t?void 0:t.name))return(0,n.jsx)(c.A,{});const s=`${a.metadata.namespace}/${a.metadata.name}`,r=ze.get(s);return(0,n.jsx)(mt,{request:a,apiName:Ve,revealedApiKey:r})}}],[Ve,ze]),mt=({request:e,apiName:t,revealedApiKey:a})=>{var r;const[o,l]=(0,s.useState)(0),d=(null===(r=e.status)||void 0===r?void 0:r.apiHostname)||`${t}.apps.example.com`,m=a||"<your-api-key>";return(0,n.jsxs)(c.A,{p:3,bgcolor:"background.default",onClick:e=>e.stopPropagation(),children:[e.spec.useCase&&(0,n.jsxs)(c.A,{mb:3,children:[(0,n.jsx)(u.A,{variant:"h6",gutterBottom:!0,children:"Use Case"}),(0,n.jsx)(c.A,{p:2,bgcolor:"background.paper",borderRadius:1,border:"1px solid rgba(0, 0, 0, 0.12)",children:(0,n.jsx)(u.A,{variant:"body2",style:{whiteSpace:"pre-wrap",wordBreak:"break-word",overflowWrap:"break-word"},children:e.spec.useCase})})]}),(0,n.jsx)(u.A,{variant:"h6",gutterBottom:!0,children:"Usage Examples"}),(0,n.jsxs)(u.A,{variant:"body2",paragraph:!0,children:["Use these code examples to test the API with your"," ",e.spec.planTier," tier key."]}),(0,n.jsx)(c.A,{onClick:e=>e.stopPropagation(),children:(0,n.jsxs)(p.A,{value:o,onChange:(e,t)=>{e.stopPropagation(),l(t)},indicatorColor:"primary",children:[(0,n.jsx)(h.A,{label:"cURL",onClick:e=>e.stopPropagation()}),(0,n.jsx)(h.A,{label:"Node.js",onClick:e=>e.stopPropagation()}),(0,n.jsx)(h.A,{label:"Python",onClick:e=>e.stopPropagation()}),(0,n.jsx)(h.A,{label:"Go",onClick:e=>e.stopPropagation()})]})}),(0,n.jsxs)(c.A,{mt:2,children:[0===o&&(0,n.jsx)(i.z,{text:`curl -X GET https://${d}/api/v1/endpoint \\\n -H "Authorization: Bearer ${m}"`,language:"bash",showCopyCodeButton:!0}),1===o&&(0,n.jsx)(i.z,{text:`const fetch = require('node-fetch');\n\nconst apiKey = '${m}';\nconst endpoint = 'https://${d}/api/v1/endpoint';\n\nfetch(endpoint, {\n method: 'GET',\n headers: {\n 'Authorization': \`Bearer \${apiKey}\`\n }\n})\n .then(response => response.json())\n .then(data => console.log(data))\n .catch(error => console.error('Error:', error));`,language:"javascript",showCopyCodeButton:!0}),2===o&&(0,n.jsx)(i.z,{text:`import requests\n\napi_key = '${m}'\nendpoint = 'https://${d}/api/v1/endpoint'\n\nheaders = {\n 'Authorization': f'Bearer {api_key}'\n}\n\nresponse = requests.get(endpoint, headers=headers)\nprint(response.json())`,language:"python",showCopyCodeButton:!0}),3===o&&(0,n.jsx)(i.z,{text:`package main\n\nimport (\n "fmt"\n "net/http"\n "io"\n)\n\nfunc main() {\n apiKey := "${m}"\n endpoint := "https://${d}/api/v1/endpoint"\n\n client := &http.Client{}\n req, _ := http.NewRequest("GET", endpoint, nil)\n req.Header.Add("Authorization", "Bearer " + apiKey)\n\n resp, err := client.Do(req)\n if err != nil {\n fmt.Println("Error:", err)\n return\n }\n defer resp.Body.Close()\n\n body, _ := io.ReadAll(resp.Body)\n fmt.Println(string(body))\n}`,language:"go",showCopyCodeButton:!0})]})]})},yt=Fe||Xe,vt=Ze||at||rt||lt;if(Ue||Oe||Qe||tt||st||ot)return(0,n.jsx)(o.k,{});if(yt)return(0,n.jsx)(l._,{error:yt});if(vt){const e=Ze?"kuadrant.apikey.create":at?"kuadrant.apikey.delete.own":rt?"kuadrant.apikey.delete.all":lt?"kuadrant.apikey.update.own":"unknown";return(0,n.jsxs)(c.A,{p:2,children:[(0,n.jsxs)(u.A,{color:"error",children:["Unable to check permissions: ",vt.message]}),(0,n.jsxs)(u.A,{variant:"body2",color:"textSecondary",children:["Permission: ",e]}),(0,n.jsx)(u.A,{variant:"body2",color:"textSecondary",children:"Please try again or contact your administrator"})]})}const At=(_e||[]).filter(e=>!Ie.has(e.metadata.name)),gt=(null==Ge||null===(U=Ge.status)||void 0===U?void 0:U.discoveredPlans)||[],xt=At.filter(e=>{var t;return!(null===(t=e.status)||void 0===t?void 0:t.phase)||"Pending"===e.status.phase}),ft=At.filter(e=>{var t;return"Approved"===(null===(t=e.status)||void 0===t?void 0:t.phase)}),jt=At.filter(e=>{var t;return"Rejected"===(null===(t=e.status)||void 0===t?void 0:t.phase)}),wt=[{title:"Tier",field:"spec.planTier",render:e=>(0,n.jsx)(m.A,{label:e.spec.planTier,color:"primary",size:"small"})},{title:"Approved",field:"status.reviewedAt",render:e=>{var t;return(0,n.jsx)(u.A,{variant:"body2",children:(null===(t=e.status)||void 0===t?void 0:t.reviewedAt)?new Date(e.status.reviewedAt).toLocaleDateString():"-"})}},{title:"API Key",field:"status.secretRef",searchable:!1,filtering:!1,render:e=>{var t,a,s;const r=`${e.metadata.namespace}/${e.metadata.name}`,i=ee.has(e.metadata.name),o=Ee.has(r),l=ze.get(r),d=null===(a=e.status)||void 0===a||null===(t=a.secretRef)||void 0===t?void 0:t.name,p=!1!==(null===(s=e.status)||void 0===s?void 0:s.canReadSecret),h=De.has(r)||!p;return d?h&&!l?(0,n.jsx)(y.Ay,{title:"This API key has already been viewed and cannot be retrieved again",children:(0,n.jsxs)(c.A,{display:"flex",alignItems:"center",children:[(0,n.jsx)(u.A,{variant:"body2",color:"textSecondary",style:{fontFamily:"monospace",marginRight:8},children:"Already viewed"}),(0,n.jsx)(T.A,{fontSize:"small",color:"disabled"})]})}):(0,n.jsxs)(c.A,{display:"flex",alignItems:"center",children:[(0,n.jsx)(u.A,{variant:"body2",style:{fontFamily:"monospace",marginRight:8},children:o?"Loading...":i&&l?l:"••••••••••••••••"}),i&&l&&(0,n.jsx)(y.Ay,{title:"Copy to clipboard",children:(0,n.jsx)(v.A,{size:"small",onClick:async()=>{l&&(await navigator.clipboard.writeText(l),Q.post({message:"API key copied to clipboard",severity:"success",display:"transient"}))},children:(0,n.jsx)(H.A,{fontSize:"small"})})}),(0,n.jsx)(y.Ay,{title:i?"Hide API key":"Reveal API key (one-time only)",children:(0,n.jsx)("span",{children:(0,n.jsx)(v.A,{size:"small",onClick:()=>{i?(((e,t)=>{const a=`${e}/${t}`;Te(e=>{const t=new Map(e);return t.delete(a),t})})(e.metadata.namespace,e.metadata.name),pt(e.metadata.name)):h||(Ne({namespace:e.metadata.namespace,name:e.metadata.name}),We(!0))},disabled:o||h&&!l,children:i?(0,n.jsx)(T.A,{}):(0,n.jsx)(z.A,{})})})})]}):(0,n.jsx)(u.A,{variant:"body2",color:"textSecondary",children:"Awaiting secret..."})}},{title:"",field:"actions",searchable:!1,filtering:!1,render:e=>{if(Se===e.metadata.name)return(0,n.jsx)(A.A,{size:20});const t=e.spec.requestedBy.userId;return(0,V.W)(t,se,et,nt)?(0,n.jsx)(v.A,{size:"small",onClick:t=>{t.stopPropagation();const a=t.currentTarget.getBoundingClientRect();be({top:a.bottom,left:a.left}),Pe(e)},title:"Actions","aria-controls":we?"actions-menu":void 0,"aria-haspopup":"true",children:(0,n.jsx)(K.A,{})}):null}}],bt=[{title:"Status",field:"status.phase",render:e=>{var t;const a=(null===(t=e.status)||void 0===t?void 0:t.phase)||"Pending",s="Pending"===a;return(0,n.jsx)(m.A,{label:a,size:"small",icon:s?(0,n.jsx)(E.A,{}):(0,n.jsx)(B.A,{}),color:s?"default":"secondary"})}},{title:"Tier",field:"spec.planTier",render:e=>(0,n.jsx)(m.A,{label:e.spec.planTier,color:"primary",size:"small"})},{title:"Use Case",field:"spec.useCase",render:e=>e.spec.useCase?(0,n.jsx)(y.Ay,{title:e.spec.useCase,placement:"top",children:(0,n.jsx)(u.A,{variant:"body2",style:{maxWidth:"200px",overflow:"hidden",textOverflow:"ellipsis",whiteSpace:"nowrap"},children:e.spec.useCase})}):(0,n.jsx)(u.A,{variant:"body2",children:"-"})},{title:"Requested",field:"metadata.creationTimestamp",render:e=>(0,n.jsx)(u.A,{variant:"body2",children:e.metadata.creationTimestamp?new Date(e.metadata.creationTimestamp).toLocaleDateString():"-"})},{title:"Reviewed",field:"status.reviewedAt",render:e=>{var t;return(null===(t=e.status)||void 0===t?void 0:t.reviewedAt)?(0,n.jsx)(u.A,{variant:"body2",children:new Date(e.status.reviewedAt).toLocaleDateString()}):(0,n.jsx)(u.A,{variant:"body2",children:"-"})}},{title:"",field:"actions",searchable:!1,filtering:!1,render:e=>{var t;if(Se===e.metadata.name)return(0,n.jsx)(A.A,{size:20});const a=!(null===(t=e.status)||void 0===t?void 0:t.phase)||"Pending"===e.status.phase,s=e.spec.requestedBy.userId,r=(0,V.W)(s,se,et,nt);return a&&(it&&s===se||r)?(0,n.jsx)(v.A,{size:"small",onClick:t=>{t.stopPropagation();const a=t.currentTarget.getBoundingClientRect();be({top:a.bottom,left:a.left}),Pe(e)},title:"Actions","aria-controls":we?"actions-menu":void 0,"aria-haspopup":"true",children:(0,n.jsx)(K.A,{})}):null}}],kt=bt.filter(e=>"Reviewed"!==e.title&&"Reason"!==e.title);return(0,n.jsxs)(c.A,{p:2,children:[(0,n.jsxs)(g.A,{container:!0,spacing:3,direction:"column",children:[Ye&&(0,n.jsx)(g.A,{item:!0,children:(0,n.jsxs)(c.A,{display:"flex",flexDirection:"column",alignItems:"flex-end",mb:2,children:[(0,n.jsx)(x.A,{variant:"contained",color:"primary",startIcon:(0,n.jsx)(D.A,{}),onClick:()=>de(!0),disabled:0===gt.length,"data-testid":"request-api-access-button","data-plans-count":gt.length,children:"Request API Access"}),0===gt.length&&(0,n.jsx)(u.A,{variant:"caption",color:"textSecondary",style:{marginTop:4},"data-testid":"no-plans-message",children:Ge?(()=>{var e,t,a,n;const s=null===(t=Ge.status)||void 0===t||null===(e=t.conditions)||void 0===e?void 0:e.find(e=>"Ready"===e.type),r=null===(n=Ge.status)||void 0===n||null===(a=n.conditions)||void 0===a?void 0:a.find(e=>"PlanPolicyDiscovered"===e.type);return"True"!==(null==s?void 0:s.status)?`HTTPRoute not ready: ${(null==s?void 0:s.message)||"unknown"}`:"True"!==(null==r?void 0:r.status)?`No plans discovered: ${(null==r?void 0:r.message)||"no PlanPolicy found"}`:"No plans available"})():"API product not found"})]})}),0===xt.length&&0===jt.length&&0===ft.length&&(0,n.jsx)(g.A,{item:!0,children:(0,n.jsx)(c.A,{p:3,textAlign:"center",children:(0,n.jsx)(u.A,{variant:"body1",color:"textSecondary",children:"No API keys yet. Request access to get started."})})}),xt.length>0&&(0,n.jsx)(g.A,{item:!0,children:(0,n.jsx)(d.X,{title:"Pending Requests",options:{paging:xt.length>5,pageSize:20,search:!0,filtering:!0,debounceInterval:300,toolbar:!0,emptyRowsWhenPaging:!1},columns:kt,data:xt})}),jt.length>0&&(0,n.jsx)(g.A,{item:!0,children:(0,n.jsx)(d.X,{title:"Rejected Requests",options:{paging:jt.length>5,pageSize:20,search:!0,filtering:!0,debounceInterval:300,toolbar:!0,emptyRowsWhenPaging:!1},columns:bt,data:jt})}),ft.length>0&&(0,n.jsx)(g.A,{item:!0,children:(0,n.jsx)(d.X,{title:"API Keys",options:{paging:ft.length>5,pageSize:20,search:!0,filtering:!0,debounceInterval:300,toolbar:!0,emptyRowsWhenPaging:!1},columns:wt,data:ft,detailPanel:ht},"api-keys-table")})]}),(0,n.jsxs)(f.A,{open:le,onClose:()=>de(!1),maxWidth:"sm",fullWidth:!0,children:[(0,n.jsx)(j.A,{children:"Request API Access"}),(0,n.jsxs)(w.A,{children:[(0,n.jsxs)(c.A,{mb:2,p:1.5,bgcolor:"info.light",borderRadius:1,display:"flex",alignItems:"flex-start",style:{gap:8},children:[(0,n.jsx)(M.A,{color:"primary",fontSize:"small",style:{marginTop:2}}),(0,n.jsx)(u.A,{variant:"body2",children:"Your request will be reviewed by an API owner before access is granted."})]}),ve&&(0,n.jsx)(c.A,{mb:2,p:2,bgcolor:"error.main",color:"error.contrastText",borderRadius:1,children:(0,n.jsx)(u.A,{variant:"body2",children:ve})}),(0,n.jsxs)(b.A,{fullWidth:!0,margin:"normal",disabled:me,"data-testid":"tier-select-form",children:[(0,n.jsx)(k.A,{id:"tier-select-label",children:"Select Tier"}),(0,n.jsx)(P.A,{labelId:"tier-select-label","data-testid":"tier-select",value:ce,onChange:e=>ue(e.target.value),disabled:me,children:gt.map(e=>{const t=Object.entries(e.limits||{}).map(([e,t])=>`${t} per ${e}`).join(", ");return(0,n.jsxs)(S.A,{value:e.tier,"data-testid":`tier-option-${e.tier}`,children:[e.tier," ",t?`(${t})`:""]},e.tier)})})]}),(0,n.jsx)(C.A,{label:"Use Case (optional)",placeholder:"Describe how you plan to use this API",multiline:!0,rows:3,fullWidth:!0,margin:"normal",value:pe,onChange:e=>he(e.target.value),helperText:"Explain your intended use of this API for admin review",disabled:me})]}),(0,n.jsxs)(I.A,{children:[(0,n.jsx)(x.A,{onClick:()=>de(!1),disabled:me,children:"Cancel"}),(0,n.jsx)(x.A,{onClick:async()=>{if(ce){ye(!0),Ae(null);try{const e=await Y.fetch(`${Z}/api/kuadrant/requests`,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({apiProductName:Ve,namespace:Le,planTier:ce,useCase:pe.trim()||"",userEmail:ie})});if(!e.ok){const t=await e.json().catch(()=>({}));throw new Error(t.error||`failed to create request: ${e.status}`)}Q.post({message:"API key requested successfully",severity:"success",display:"transient"}),de(!1),ue(""),he(""),ne(e=>e+1)}catch(e){const t=e instanceof Error?e.message:"unknown error occurred";Q.post({message:`Failed to request API key: ${t}`,severity:"error",display:"transient"}),Ae(t)}finally{ye(!1)}}},color:"primary",variant:"contained",disabled:!ce||me,startIcon:me?(0,n.jsx)(A.A,{size:16,color:"inherit"}):void 0,children:me?"Submitting...":"Submit Request"})]})]}),(0,n.jsx)(R.A,{id:"actions-menu",open:Boolean(we),onClose:dt,anchorReference:"anchorPosition",anchorPosition:we||{top:0,left:0},children:ke&&(()=>{var e;const t=!(null===(e=ke.status)||void 0===e?void 0:e.phase)||"Pending"===ke.status.phase,a=ke.spec.requestedBy.userId,s=[];return it&&a===se&&t&&s.push((0,n.jsx)(S.A,{onClick:ct,children:"Edit"},"edit")),s.push((0,n.jsx)(S.A,{onClick:ut,children:"Delete"},"delete")),s})()}),fe&&(0,n.jsx)(L.e,{open:ge,onClose:()=>{xe(!1),je(null)},onSuccess:()=>{ne(e=>e+1),xe(!1),Q.post({message:"API key updated",severity:"success",display:"transient"}),je(null)},request:fe,availablePlans:gt}),(0,n.jsx)(_.K,{open:qe.open,title:"Delete Request",description:`Are you sure you want to delete this ${"Approved"===(null===(G=qe.request)||void 0===G||null===(F=G.status)||void 0===F?void 0:F.phase)?"API key":"request"}?`,deleting:null!==Se,onConfirm:async()=>{qe.request&&(await(async e=>{Re(t=>new Set(t).add(e)),Ce(e);try{if(!(await Y.fetch(`${Z}/api/kuadrant/requests/${Le}/${e}`,{method:"DELETE"})).ok)throw new Error("failed to delete request");Q.post({message:"API key deleted successfully",severity:"success",display:"transient"}),ne(e=>e+1)}catch(t){const a=t instanceof Error?t.message:"unknown error occurred";Re(t=>{const a=new Set(t);return a.delete(e),a}),Q.post({message:`Failed to delete API key: ${a}`,severity:"error",display:"transient"})}finally{Ce(null)}})(qe.request.metadata.name),$e({open:!1,request:null}))},onCancel:()=>{$e({open:!1,request:null})}}),(0,n.jsxs)(f.A,{open:He,onClose:()=>{We(!1),Ne(null)},maxWidth:"sm",children:[(0,n.jsx)(j.A,{children:(0,n.jsxs)(c.A,{display:"flex",alignItems:"center",children:[(0,n.jsx)(W.A,{color:"primary",style:{marginRight:8}}),"View API Key"]})}),(0,n.jsxs)(w.A,{children:[(0,n.jsxs)(u.A,{variant:"body1",paragraph:!0,children:["This API key can only be viewed ",(0,n.jsx)("strong",{children:"once"}),". After you reveal it, you will not be able to retrieve it again."]}),(0,n.jsx)(u.A,{variant:"body2",color:"textSecondary",children:"Make sure to copy and store it securely before closing this view."})]}),(0,n.jsxs)(I.A,{children:[(0,n.jsx)(x.A,{onClick:()=>{We(!1),Ne(null)},children:"Cancel"}),(0,n.jsx)(x.A,{variant:"contained",color:"primary",onClick:()=>{Me&&((async(e,t)=>{const a=`${e}/${t}`;if(!Ee.has(a)){Be(e=>new Set(e).add(a));try{const n=await Y.fetch(`${Z}/api/kuadrant/apikeys/${e}/${t}/secret`);if(n.ok){const e=await n.json();Te(t=>new Map(t).set(a,e.apiKey)),Ke(e=>new Set(e).add(a))}else 403===n.status&&(Ke(e=>new Set(e).add(a)),Q.post({message:"This API key has already been viewed and cannot be retrieved again.",severity:"warning",display:"transient"}))}catch(e){console.error("failed to fetch api key:",e)}finally{Be(e=>{const t=new Set(e);return t.delete(a),t})}}})(Me.namespace,Me.name),pt(Me.name)),We(!1),Ne(null)},children:"Reveal API Key"})]})]})]})}},71407:(e,t,a)=>{var n=a(4293),s=a(78920);t.A=void 0;var r=s(a(95478)),i=(0,n(a(74044)).default)(r.createElement("path",{d:"M12 8c1.1 0 2-.9 2-2s-.9-2-2-2-2 .9-2 2 .9 2 2 2zm0 2c-1.1 0-2 .9-2 2s.9 2 2 2 2-.9 2-2-.9-2-2-2zm0 6c-1.1 0-2 .9-2 2s.9 2 2 2 2-.9 2-2-.9-2-2-2z"}),"MoreVert");t.A=i},93246:(e,t,a)=>{var n=a(4293),s=a(78920);t.A=void 0;var r=s(a(95478)),i=(0,n(a(74044)).default)(r.createElement("path",{d:"M6 2v6h.01L6 8.01 10 12l-4 4 .01.01H6V22h12v-5.99h-.01L18 16l-4-4 4-3.99-.01-.01H18V2H6zm10 14.5V20H8v-3.5l4-4 4 4zm-4-5l-4-4V4h8v3.5l-4 4z"}),"HourglassEmpty");t.A=i}}]);
2
+ //# sourceMappingURL=4306.3a218ff1.chunk.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"static/4306.3a218ff1.chunk.js","mappings":"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","sources":["webpack://internal.plugin-kuadrant/../../node_modules/@backstage/plugin-catalog-react/dist/hooks/useEntity.esm.js","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/icons/Add.js","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/icons/Info.js","webpack://internal.plugin-kuadrant/./src/components/ApiKeyManagementTab/ApiKeyManagementTab.tsx","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/icons/MoreVert.js","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/icons/HourglassEmpty.js"],"sourcesContent":["import { jsx } from 'react/jsx-runtime';\nimport { stringifyEntityRef } from '@backstage/catalog-model';\nimport { AnalyticsContext } from '@backstage/core-plugin-api';\nimport { createVersionedContext, useVersionedContext, createVersionedValueMap } from '@backstage/version-bridge';\n\nconst NewEntityContext = createVersionedContext(\n \"entity-context\"\n);\nconst AsyncEntityProvider = (props) => {\n const { children, entity, loading, error, refresh } = props;\n const value = { entity, loading, error, refresh };\n return /* @__PURE__ */ jsx(NewEntityContext.Provider, { value: createVersionedValueMap({ 1: value }), children: /* @__PURE__ */ jsx(\n AnalyticsContext,\n {\n attributes: {\n ...entity ? { entityRef: stringifyEntityRef(entity) } : void 0\n },\n children\n }\n ) });\n};\nconst EntityProvider = (props) => /* @__PURE__ */ jsx(\n AsyncEntityProvider,\n {\n entity: props.entity,\n loading: !Boolean(props.entity),\n error: void 0,\n refresh: void 0,\n children: props.children\n }\n);\nfunction useEntity() {\n const versionedHolder = useVersionedContext(\n \"entity-context\"\n );\n if (!versionedHolder) {\n throw new Error(\"Entity context is not available\");\n }\n const value = versionedHolder.atVersion(1);\n if (!value) {\n throw new Error(\"EntityContext v1 not available\");\n }\n if (!value.entity) {\n throw new Error(\n \"useEntity hook is being called outside of an EntityLayout where the entity has not been loaded. If this is intentional, please use useAsyncEntity instead.\"\n );\n }\n return { entity: value.entity };\n}\nfunction useAsyncEntity() {\n const versionedHolder = useVersionedContext(\n \"entity-context\"\n );\n if (!versionedHolder) {\n throw new Error(\"Entity context is not available\");\n }\n const value = versionedHolder.atVersion(1);\n if (!value) {\n throw new Error(\"EntityContext v1 not available\");\n }\n const { entity, loading, error, refresh } = value;\n return { entity, loading, error, refresh };\n}\n\nexport { AsyncEntityProvider, EntityProvider, useAsyncEntity, useEntity };\n//# sourceMappingURL=useEntity.esm.js.map\n","\"use strict\";\n\nvar _interopRequireDefault = require(\"@babel/runtime/helpers/interopRequireDefault\");\n\nvar _interopRequireWildcard = require(\"@babel/runtime/helpers/interopRequireWildcard\");\n\nObject.defineProperty(exports, \"__esModule\", {\n value: true\n});\nexports.default = void 0;\n\nvar React = _interopRequireWildcard(require(\"react\"));\n\nvar _createSvgIcon = _interopRequireDefault(require(\"./utils/createSvgIcon\"));\n\nvar _default = (0, _createSvgIcon.default)( /*#__PURE__*/React.createElement(\"path\", {\n d: \"M19 13h-6v6h-2v-6H5v-2h6V5h2v6h6v2z\"\n}), 'Add');\n\nexports.default = _default;","\"use strict\";\n\nvar _interopRequireDefault = require(\"@babel/runtime/helpers/interopRequireDefault\");\n\nvar _interopRequireWildcard = require(\"@babel/runtime/helpers/interopRequireWildcard\");\n\nObject.defineProperty(exports, \"__esModule\", {\n value: true\n});\nexports.default = void 0;\n\nvar React = _interopRequireWildcard(require(\"react\"));\n\nvar _createSvgIcon = _interopRequireDefault(require(\"./utils/createSvgIcon\"));\n\nvar _default = (0, _createSvgIcon.default)( /*#__PURE__*/React.createElement(\"path\", {\n d: \"M12 2C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10S17.52 2 12 2zm1 15h-2v-6h2v6zm0-8h-2V7h2v2z\"\n}), 'Info');\n\nexports.default = _default;","import React, { useState, useMemo } from \"react\";\nimport { useAsync } from \"react-use\";\nimport {\n Table,\n TableColumn,\n Progress,\n ResponseErrorPanel,\n CodeSnippet,\n} from \"@backstage/core-components\";\nimport {\n IconButton,\n Typography,\n Box,\n Chip,\n Grid,\n Button,\n Dialog,\n DialogTitle,\n DialogContent,\n DialogActions,\n TextField,\n Select,\n MenuItem,\n FormControl,\n InputLabel,\n Tabs,\n Tab,\n Menu,\n Tooltip,\n CircularProgress,\n} from \"@material-ui/core\";\nimport {\n useApi,\n configApiRef,\n identityApiRef,\n fetchApiRef,\n alertApiRef,\n} from \"@backstage/core-plugin-api\";\nimport { useEntity } from \"@backstage/plugin-catalog-react\";\nimport VisibilityIcon from \"@material-ui/icons/Visibility\";\nimport VisibilityOffIcon from \"@material-ui/icons/VisibilityOff\";\nimport HourglassEmptyIcon from \"@material-ui/icons/HourglassEmpty\";\nimport CancelIcon from \"@material-ui/icons/Cancel\";\nimport AddIcon from \"@material-ui/icons/Add\";\nimport MoreVertIcon from \"@material-ui/icons/MoreVert\";\nimport FileCopyIcon from \"@material-ui/icons/FileCopy\";\nimport WarningIcon from \"@material-ui/icons/Warning\";\nimport InfoIcon from \"@material-ui/icons/Info\";\nimport { APIKey } from \"../../types/api-management\";\nimport {\n kuadrantApiKeyCreatePermission,\n kuadrantApiKeyDeleteOwnPermission,\n kuadrantApiKeyDeleteAllPermission,\n kuadrantApiKeyUpdateOwnPermission,\n} from \"../../permissions\";\nimport {\n useKuadrantPermission,\n canDeleteResource,\n} from \"../../utils/permissions\";\nimport { EditAPIKeyDialog } from \"../EditAPIKeyDialog\";\nimport { ConfirmDeleteDialog } from \"../ConfirmDeleteDialog\";\n\ninterface APIProduct {\n metadata: {\n name: string;\n namespace: string;\n };\n spec: {\n displayName?: string;\n };\n status?: {\n discoveredPlans?: Array<{\n tier: string;\n description?: string;\n limits?: any;\n }>;\n conditions?: Array<{\n type: string;\n status: \"True\" | \"False\" | \"Unknown\";\n reason?: string;\n message?: string;\n lastTransitionTime?: string;\n }>;\n };\n}\n\ninterface Plan {\n tier: string;\n limits: any;\n}\n\nexport interface ApiKeyManagementTabProps {\n namespace?: string;\n}\n\nexport const ApiKeyManagementTab = ({\n namespace: propNamespace,\n}: ApiKeyManagementTabProps) => {\n const { entity } = useEntity();\n const config = useApi(configApiRef);\n const identityApi = useApi(identityApiRef);\n const fetchApi = useApi(fetchApiRef);\n const alertApi = useApi(alertApiRef);\n const backendUrl = config.getString(\"backend.baseUrl\");\n const [visibleKeys, setVisibleKeys] = useState<Set<string>>(new Set());\n const [refresh, setRefresh] = useState(0);\n const [userId, setUserId] = useState<string>(\"\");\n const [userEmail, setUserEmail] = useState<string>(\"\");\n const [open, setOpen] = useState(false);\n const [selectedPlan, setSelectedPlan] = useState(\"\");\n const [useCase, setUseCase] = useState(\"\");\n const [creating, setCreating] = useState(false);\n const [createError, setCreateError] = useState<string | null>(null);\n const [editDialogOpen, setEditDialogOpen] = useState(false);\n const [requestToEdit, setRequestToEdit] = useState<APIKey | null>(null);\n const [menuAnchor, setMenuAnchor] = useState<{\n top: number;\n left: number;\n } | null>(null);\n const [menuRequest, setMenuRequest] = useState<APIKey | null>(null);\n const [deleting, setDeleting] = useState<string | null>(null);\n const [optimisticallyDeleted, setOptimisticallyDeleted] = useState<\n Set<string>\n >(new Set());\n const [deleteDialogState, setDeleteDialogState] = useState<{\n open: boolean;\n request: APIKey | null;\n }>({ open: false, request: null });\n const [apiKeyValues, setApiKeyValues] = useState<Map<string, string>>(\n new Map(),\n );\n const [apiKeyLoading, setApiKeyLoading] = useState<Set<string>>(new Set());\n const [alreadyReadKeys, setAlreadyReadKeys] = useState<Set<string>>(\n new Set(),\n );\n const [showOnceWarningOpen, setShowOnceWarningOpen] = useState(false);\n const [pendingKeyReveal, setPendingKeyReveal] = useState<{\n namespace: string;\n name: string;\n } | null>(null);\n\n // get apiproduct name from entity annotation (set by entity provider)\n const apiProductName =\n entity.metadata.annotations?.[\"kuadrant.io/apiproduct\"] ||\n entity.metadata.name;\n const namespace =\n entity.metadata.annotations?.[\"kuadrant.io/namespace\"] ||\n propNamespace ||\n \"default\";\n\n useAsync(async () => {\n const identity = await identityApi.getBackstageIdentity();\n const profile = await identityApi.getProfileInfo();\n setUserId(identity.userEntityRef);\n setUserEmail(profile.email || \"\");\n }, [identityApi]);\n\n const {\n value: requests,\n loading: requestsLoading,\n error: requestsError,\n } = useAsync(async () => {\n const response = await fetchApi.fetch(\n `${backendUrl}/api/kuadrant/requests/my?namespace=${namespace}`,\n );\n if (!response.ok) {\n throw new Error(\"failed to fetch requests\");\n }\n const data = await response.json();\n // filter by apiproduct name, not httproute name\n return (data.items || []).filter(\n (r: APIKey) =>\n r.spec.apiProductRef.name === apiProductName &&\n r.metadata.namespace === namespace, // APIProducts and APIKeys (and its Secret) will be in the same NS\n );\n }, [apiProductName, namespace, refresh, fetchApi, backendUrl]);\n\n const {\n value: apiProduct,\n loading: plansLoading,\n error: plansError,\n } = useAsync(async () => {\n const response = await fetchApi.fetch(\n `${backendUrl}/api/kuadrant/apiproducts`,\n );\n if (!response.ok) {\n throw new Error(\"failed to fetch api products\");\n }\n const data = await response.json();\n\n const product = data.items?.find(\n (p: APIProduct) =>\n p.metadata.namespace === namespace &&\n p.metadata.name === apiProductName,\n );\n\n return product;\n }, [namespace, apiProductName, fetchApi]);\n\n // check permissions with resource reference once we have the apiproduct\n const resourceRef = apiProduct\n ? `apiproduct:${apiProduct.metadata.namespace}/${apiProduct.metadata.name}`\n : undefined;\n\n const {\n allowed: canCreateRequest,\n loading: createRequestPermissionLoading,\n error: createRequestPermissionError,\n } = useKuadrantPermission(kuadrantApiKeyCreatePermission, resourceRef);\n\n const {\n allowed: canDeleteOwnKey,\n loading: deleteOwnPermissionLoading,\n error: deleteOwnPermissionError,\n } = useKuadrantPermission(kuadrantApiKeyDeleteOwnPermission);\n\n const {\n allowed: canDeleteAllKeys,\n loading: deleteAllPermissionLoading,\n error: deleteAllPermissionError,\n } = useKuadrantPermission(kuadrantApiKeyDeleteAllPermission);\n\n const {\n allowed: canUpdateRequest,\n loading: updateRequestPermissionLoading,\n error: updateRequestPermissionError,\n } = useKuadrantPermission(kuadrantApiKeyUpdateOwnPermission);\n\n const handleDeleteRequest = async (name: string) => {\n // optimistic update - remove from UI immediately\n setOptimisticallyDeleted((prev) => new Set(prev).add(name));\n setDeleting(name);\n try {\n const response = await fetchApi.fetch(\n `${backendUrl}/api/kuadrant/requests/${namespace}/${name}`,\n { method: \"DELETE\" },\n );\n if (!response.ok) {\n throw new Error(\"failed to delete request\");\n }\n alertApi.post({\n message: \"API key deleted successfully\",\n severity: \"success\",\n display: \"transient\",\n });\n setRefresh((r) => r + 1);\n } catch (err) {\n const errorMessage =\n err instanceof Error ? err.message : \"unknown error occurred\";\n // rollback optimistic update on error\n setOptimisticallyDeleted((prev) => {\n const next = new Set(prev);\n next.delete(name);\n return next;\n });\n alertApi.post({\n message: `Failed to delete API key: ${errorMessage}`,\n severity: \"error\",\n display: \"transient\",\n });\n } finally {\n setDeleting(null);\n }\n };\n\n const fetchApiKeyFromSecret = async (\n requestNamespace: string,\n requestName: string,\n ) => {\n const key = `${requestNamespace}/${requestName}`;\n if (apiKeyLoading.has(key)) {\n return;\n }\n\n setApiKeyLoading((prev) => new Set(prev).add(key));\n try {\n const response = await fetchApi.fetch(\n `${backendUrl}/api/kuadrant/apikeys/${requestNamespace}/${requestName}/secret`,\n );\n if (response.ok) {\n const data = await response.json();\n setApiKeyValues((prev) => new Map(prev).set(key, data.apiKey));\n // after successful read, mark as already read (show-once behaviour)\n setAlreadyReadKeys((prev) => new Set(prev).add(key));\n } else if (response.status === 403) {\n // secret has already been read\n setAlreadyReadKeys((prev) => new Set(prev).add(key));\n alertApi.post({\n message:\n \"This API key has already been viewed and cannot be retrieved again.\",\n severity: \"warning\",\n display: \"transient\",\n });\n }\n } catch (err) {\n console.error(\"failed to fetch api key:\", err);\n } finally {\n setApiKeyLoading((prev) => {\n const next = new Set(prev);\n next.delete(key);\n return next;\n });\n }\n };\n\n const clearApiKeyValue = (requestNamespace: string, requestName: string) => {\n const key = `${requestNamespace}/${requestName}`;\n setApiKeyValues((prev) => {\n const next = new Map(prev);\n next.delete(key);\n return next;\n });\n };\n\n const handleEditRequest = (request: APIKey) => {\n setRequestToEdit(request);\n setEditDialogOpen(true);\n };\n\n const handleEditSuccess = () => {\n setRefresh((r) => r + 1);\n setEditDialogOpen(false);\n alertApi.post({\n message: \"API key updated\",\n severity: \"success\",\n display: \"transient\",\n });\n setRequestToEdit(null);\n };\n\n const handleMenuClose = () => {\n setMenuAnchor(null);\n setMenuRequest(null);\n };\n\n const handleMenuEdit = () => {\n if (!menuRequest) return;\n handleEditRequest(menuRequest);\n handleMenuClose();\n };\n\n const handleMenuDeleteClick = () => {\n if (!menuRequest) return;\n const request = menuRequest;\n handleMenuClose();\n setDeleteDialogState({ open: true, request });\n };\n\n const handleDeleteConfirm = async () => {\n if (!deleteDialogState.request) return;\n await handleDeleteRequest(deleteDialogState.request.metadata.name);\n setDeleteDialogState({ open: false, request: null });\n };\n\n const handleDeleteCancel = () => {\n setDeleteDialogState({ open: false, request: null });\n };\n\n const toggleVisibility = (keyName: string) => {\n setVisibleKeys((prev) => {\n const newSet = new Set(prev);\n if (newSet.has(keyName)) {\n newSet.delete(keyName);\n } else {\n newSet.add(keyName);\n }\n return newSet;\n });\n };\n\n const handleRequestAccess = async () => {\n if (!selectedPlan) return;\n\n setCreating(true);\n setCreateError(null);\n try {\n const response = await fetchApi.fetch(\n `${backendUrl}/api/kuadrant/requests`,\n {\n method: \"POST\",\n headers: {\n \"Content-Type\": \"application/json\",\n },\n body: JSON.stringify({\n apiProductName,\n namespace,\n planTier: selectedPlan,\n useCase: useCase.trim() || \"\",\n userEmail,\n }),\n },\n );\n\n if (!response.ok) {\n const errorData = await response.json().catch(() => ({}));\n throw new Error(\n errorData.error || `failed to create request: ${response.status}`,\n );\n }\n\n alertApi.post({\n message: \"API key requested successfully\",\n severity: \"success\",\n display: \"transient\",\n });\n\n setOpen(false);\n setSelectedPlan(\"\");\n setUseCase(\"\");\n setRefresh((r) => r + 1);\n } catch (err) {\n const errorMessage =\n err instanceof Error ? err.message : \"unknown error occurred\";\n alertApi.post({\n message: `Failed to request API key: ${errorMessage}`,\n severity: \"error\",\n display: \"transient\",\n });\n setCreateError(errorMessage);\n } finally {\n setCreating(false);\n }\n };\n\n const detailPanelConfig = useMemo(\n () => [\n {\n render: (data: any) => {\n // backstage Table wraps the data in { rowData: actualData }\n const request = data.rowData as APIKey;\n if (!request?.metadata?.name) {\n return <Box />;\n }\n\n // pass already-revealed key from parent state (don't auto-fetch - that consumes show-once)\n const key = `${request.metadata.namespace}/${request.metadata.name}`;\n const revealedKey = apiKeyValues.get(key);\n return (\n <DetailPanelContent\n request={request}\n apiName={apiProductName}\n revealedApiKey={revealedKey}\n />\n );\n },\n },\n ],\n [apiProductName, apiKeyValues],\n );\n\n // separate component to isolate state\n const DetailPanelContent = ({\n request,\n apiName: api,\n revealedApiKey,\n }: {\n request: APIKey;\n apiName: string;\n revealedApiKey?: string;\n }) => {\n const [selectedLanguage, setSelectedLanguage] = useState(0);\n const hostname = request.status?.apiHostname || `${api}.apps.example.com`;\n\n // use revealed key if available, otherwise show placeholder\n const displayApiKey = revealedApiKey || \"<your-api-key>\";\n\n return (\n <Box\n p={3}\n bgcolor=\"background.default\"\n onClick={(e) => e.stopPropagation()}\n >\n {request.spec.useCase && (\n <Box mb={3}>\n <Typography variant=\"h6\" gutterBottom>\n Use Case\n </Typography>\n <Box\n p={2}\n bgcolor=\"background.paper\"\n borderRadius={1}\n border=\"1px solid rgba(0, 0, 0, 0.12)\"\n >\n <Typography\n variant=\"body2\"\n style={{\n whiteSpace: \"pre-wrap\",\n wordBreak: \"break-word\",\n overflowWrap: \"break-word\",\n }}\n >\n {request.spec.useCase}\n </Typography>\n </Box>\n </Box>\n )}\n <Typography variant=\"h6\" gutterBottom>\n Usage Examples\n </Typography>\n <Typography variant=\"body2\" paragraph>\n Use these code examples to test the API with your{\" \"}\n {request.spec.planTier} tier key.\n </Typography>\n <Box onClick={(e) => e.stopPropagation()}>\n <Tabs\n value={selectedLanguage}\n onChange={(e, newValue) => {\n e.stopPropagation();\n setSelectedLanguage(newValue);\n }}\n indicatorColor=\"primary\"\n >\n <Tab label=\"cURL\" onClick={(e) => e.stopPropagation()} />\n <Tab label=\"Node.js\" onClick={(e) => e.stopPropagation()} />\n <Tab label=\"Python\" onClick={(e) => e.stopPropagation()} />\n <Tab label=\"Go\" onClick={(e) => e.stopPropagation()} />\n </Tabs>\n </Box>\n <Box mt={2}>\n {selectedLanguage === 0 && (\n <CodeSnippet\n text={`curl -X GET https://${hostname}/api/v1/endpoint \\\\\n -H \"Authorization: Bearer ${displayApiKey}\"`} // notsecret - template for user's own api key\n language=\"bash\"\n showCopyCodeButton\n />\n )}\n {selectedLanguage === 1 && (\n <CodeSnippet\n text={`const fetch = require('node-fetch');\n\nconst apiKey = '${displayApiKey}';\nconst endpoint = 'https://${hostname}/api/v1/endpoint';\n\nfetch(endpoint, {\n method: 'GET',\n headers: {\n 'Authorization': \\`Bearer \\${apiKey}\\`\n }\n})\n .then(response => response.json())\n .then(data => console.log(data))\n .catch(error => console.error('Error:', error));`}\n language=\"javascript\"\n showCopyCodeButton\n />\n )}\n {selectedLanguage === 2 && (\n <CodeSnippet\n text={`import requests\n\napi_key = '${displayApiKey}'\nendpoint = 'https://${hostname}/api/v1/endpoint'\n\nheaders = {\n 'Authorization': f'Bearer {api_key}'\n}\n\nresponse = requests.get(endpoint, headers=headers)\nprint(response.json())`}\n language=\"python\"\n showCopyCodeButton\n />\n )}\n {selectedLanguage === 3 && (\n <CodeSnippet\n text={`package main\n\nimport (\n \"fmt\"\n \"net/http\"\n \"io\"\n)\n\nfunc main() {\n apiKey := \"${displayApiKey}\"\n endpoint := \"https://${hostname}/api/v1/endpoint\"\n\n client := &http.Client{}\n req, _ := http.NewRequest(\"GET\", endpoint, nil)\n req.Header.Add(\"Authorization\", \"Bearer \" + apiKey)\n\n resp, err := client.Do(req)\n if err != nil {\n fmt.Println(\"Error:\", err)\n return\n }\n defer resp.Body.Close()\n\n body, _ := io.ReadAll(resp.Body)\n fmt.Println(string(body))\n}`}\n language=\"go\"\n showCopyCodeButton\n />\n )}\n </Box>\n </Box>\n );\n };\n\n const loading =\n requestsLoading ||\n plansLoading ||\n createRequestPermissionLoading ||\n deleteOwnPermissionLoading ||\n deleteAllPermissionLoading ||\n updateRequestPermissionLoading;\n const error = requestsError || plansError;\n const permissionError =\n createRequestPermissionError ||\n deleteOwnPermissionError ||\n deleteAllPermissionError ||\n updateRequestPermissionError;\n\n if (loading) {\n return <Progress />;\n }\n\n if (error) {\n return <ResponseErrorPanel error={error} />;\n }\n\n if (permissionError) {\n const failedPermission = createRequestPermissionError\n ? \"kuadrant.apikey.create\"\n : deleteOwnPermissionError\n ? \"kuadrant.apikey.delete.own\"\n : deleteAllPermissionError\n ? \"kuadrant.apikey.delete.all\"\n : updateRequestPermissionError\n ? \"kuadrant.apikey.update.own\"\n : \"unknown\";\n return (\n <Box p={2}>\n <Typography color=\"error\">\n Unable to check permissions: {permissionError.message}\n </Typography>\n <Typography variant=\"body2\" color=\"textSecondary\">\n Permission: {failedPermission}\n </Typography>\n <Typography variant=\"body2\" color=\"textSecondary\">\n Please try again or contact your administrator\n </Typography>\n </Box>\n );\n }\n\n const myRequests = ((requests || []) as APIKey[]).filter(\n (r) => !optimisticallyDeleted.has(r.metadata.name),\n );\n const plans = (apiProduct?.status?.discoveredPlans || []) as Plan[];\n\n const pendingRequests = myRequests.filter(\n (r) => !r.status?.phase || r.status.phase === \"Pending\",\n );\n const approvedRequests = myRequests.filter(\n (r) => r.status?.phase === \"Approved\",\n );\n const rejectedRequests = myRequests.filter(\n (r) => r.status?.phase === \"Rejected\",\n );\n\n const approvedColumns: TableColumn<APIKey>[] = [\n {\n title: \"Tier\",\n field: \"spec.planTier\",\n render: (row: APIKey) => (\n <Chip label={row.spec.planTier} color=\"primary\" size=\"small\" />\n ),\n },\n {\n title: \"Approved\",\n field: \"status.reviewedAt\",\n render: (row: APIKey) => (\n <Typography variant=\"body2\">\n {row.status?.reviewedAt\n ? new Date(row.status.reviewedAt).toLocaleDateString()\n : \"-\"}\n </Typography>\n ),\n },\n {\n title: \"API Key\",\n field: \"status.secretRef\",\n searchable: false,\n filtering: false,\n render: (row: APIKey) => {\n const key = `${row.metadata.namespace}/${row.metadata.name}`;\n const isVisible = visibleKeys.has(row.metadata.name);\n const isLoading = apiKeyLoading.has(key);\n const apiKeyValue = apiKeyValues.get(key);\n const hasSecretRef = row.status?.secretRef?.name;\n const canReadSecret = row.status?.canReadSecret !== false;\n const isAlreadyRead = alreadyReadKeys.has(key) || !canReadSecret;\n\n if (!hasSecretRef) {\n return (\n <Typography variant=\"body2\" color=\"textSecondary\">\n Awaiting secret...\n </Typography>\n );\n }\n\n // key has already been viewed and cannot be retrieved again\n if (isAlreadyRead && !apiKeyValue) {\n return (\n <Tooltip title=\"This API key has already been viewed and cannot be retrieved again\">\n <Box display=\"flex\" alignItems=\"center\">\n <Typography\n variant=\"body2\"\n color=\"textSecondary\"\n style={{ fontFamily: \"monospace\", marginRight: 8 }}\n >\n Already viewed\n </Typography>\n <VisibilityOffIcon fontSize=\"small\" color=\"disabled\" />\n </Box>\n </Tooltip>\n );\n }\n\n const handleRevealClick = () => {\n if (isVisible) {\n // hiding - clear the value from memory\n clearApiKeyValue(row.metadata.namespace, row.metadata.name);\n toggleVisibility(row.metadata.name);\n } else if (!isAlreadyRead) {\n // show warning dialog before first reveal\n setPendingKeyReveal({\n namespace: row.metadata.namespace,\n name: row.metadata.name,\n });\n setShowOnceWarningOpen(true);\n }\n };\n\n const handleCopy = async () => {\n if (apiKeyValue) {\n await navigator.clipboard.writeText(apiKeyValue);\n alertApi.post({\n message: \"API key copied to clipboard\",\n severity: \"success\",\n display: \"transient\",\n });\n }\n };\n\n return (\n <Box display=\"flex\" alignItems=\"center\">\n <Typography\n variant=\"body2\"\n style={{\n fontFamily: \"monospace\",\n marginRight: 8,\n }}\n >\n {isLoading\n ? \"Loading...\"\n : isVisible && apiKeyValue\n ? apiKeyValue\n : \"••••••••••••••••\"}\n </Typography>\n {isVisible && apiKeyValue && (\n <Tooltip title=\"Copy to clipboard\">\n <IconButton size=\"small\" onClick={handleCopy}>\n <FileCopyIcon fontSize=\"small\" />\n </IconButton>\n </Tooltip>\n )}\n <Tooltip\n title={\n isVisible ? \"Hide API key\" : \"Reveal API key (one-time only)\"\n }\n >\n <span>\n <IconButton\n size=\"small\"\n onClick={handleRevealClick}\n disabled={isLoading || (isAlreadyRead && !apiKeyValue)}\n >\n {isVisible ? <VisibilityOffIcon /> : <VisibilityIcon />}\n </IconButton>\n </span>\n </Tooltip>\n </Box>\n );\n },\n },\n {\n title: \"\",\n field: \"actions\",\n searchable: false,\n filtering: false,\n render: (row: APIKey) => {\n const isDeleting = deleting === row.metadata.name;\n if (isDeleting) {\n return <CircularProgress size={20} />;\n }\n const ownerId = row.spec.requestedBy.userId;\n const canDelete = canDeleteResource(\n ownerId,\n userId,\n canDeleteOwnKey,\n canDeleteAllKeys,\n );\n if (!canDelete) return null;\n return (\n <IconButton\n size=\"small\"\n onClick={(e) => {\n e.stopPropagation();\n const rect = e.currentTarget.getBoundingClientRect();\n setMenuAnchor({ top: rect.bottom, left: rect.left });\n setMenuRequest(row);\n }}\n title=\"Actions\"\n aria-controls={menuAnchor ? \"actions-menu\" : undefined}\n aria-haspopup=\"true\"\n >\n <MoreVertIcon />\n </IconButton>\n );\n },\n },\n ];\n\n const requestColumns: TableColumn<APIKey>[] = [\n {\n title: \"Status\",\n field: \"status.phase\",\n render: (row: APIKey) => {\n const phase = row.status?.phase || \"Pending\";\n const isPending = phase === \"Pending\";\n return (\n <Chip\n label={phase}\n size=\"small\"\n icon={isPending ? <HourglassEmptyIcon /> : <CancelIcon />}\n color={isPending ? \"default\" : \"secondary\"}\n />\n );\n },\n },\n {\n title: \"Tier\",\n field: \"spec.planTier\",\n render: (row: APIKey) => (\n <Chip label={row.spec.planTier} color=\"primary\" size=\"small\" />\n ),\n },\n {\n title: \"Use Case\",\n field: \"spec.useCase\",\n render: (row: APIKey) => {\n if (!row.spec.useCase) {\n return <Typography variant=\"body2\">-</Typography>;\n }\n return (\n <Tooltip title={row.spec.useCase} placement=\"top\">\n <Typography\n variant=\"body2\"\n style={{\n maxWidth: \"200px\",\n overflow: \"hidden\",\n textOverflow: \"ellipsis\",\n whiteSpace: \"nowrap\",\n }}\n >\n {row.spec.useCase}\n </Typography>\n </Tooltip>\n );\n },\n },\n {\n title: \"Requested\",\n field: \"metadata.creationTimestamp\",\n render: (row: APIKey) => (\n <Typography variant=\"body2\">\n {row.metadata.creationTimestamp\n ? new Date(row.metadata.creationTimestamp).toLocaleDateString()\n : \"-\"}\n </Typography>\n ),\n },\n {\n title: \"Reviewed\",\n field: \"status.reviewedAt\",\n render: (row: APIKey) => {\n if (!row.status?.reviewedAt)\n return <Typography variant=\"body2\">-</Typography>;\n return (\n <Typography variant=\"body2\">\n {new Date(row.status.reviewedAt).toLocaleDateString()}\n </Typography>\n );\n },\n },\n {\n title: \"\",\n field: \"actions\",\n searchable: false,\n filtering: false,\n render: (row: APIKey) => {\n const isDeleting = deleting === row.metadata.name;\n if (isDeleting) {\n return <CircularProgress size={20} />;\n }\n const isPending = !row.status?.phase || row.status.phase === \"Pending\";\n const ownerId = row.spec.requestedBy.userId;\n const canDelete = canDeleteResource(\n ownerId,\n userId,\n canDeleteOwnKey,\n canDeleteAllKeys,\n );\n const canEdit = canUpdateRequest && ownerId === userId;\n if (!isPending || (!canEdit && !canDelete)) return null;\n return (\n <IconButton\n size=\"small\"\n onClick={(e) => {\n e.stopPropagation();\n const rect = e.currentTarget.getBoundingClientRect();\n setMenuAnchor({ top: rect.bottom, left: rect.left });\n setMenuRequest(row);\n }}\n title=\"Actions\"\n aria-controls={menuAnchor ? \"actions-menu\" : undefined}\n aria-haspopup=\"true\"\n >\n <MoreVertIcon />\n </IconButton>\n );\n },\n },\n ];\n\n // Filter columns for pending requests (no Reviewed or Reason)\n const pendingRequestColumns = requestColumns.filter(\n (col) => col.title !== \"Reviewed\" && col.title !== \"Reason\",\n );\n\n return (\n <Box p={2}>\n <Grid container spacing={3} direction=\"column\">\n {canCreateRequest && (\n <Grid item>\n <Box\n display=\"flex\"\n flexDirection=\"column\"\n alignItems=\"flex-end\"\n mb={2}\n >\n <Button\n variant=\"contained\"\n color=\"primary\"\n startIcon={<AddIcon />}\n onClick={() => setOpen(true)}\n disabled={plans.length === 0}\n data-testid=\"request-api-access-button\"\n data-plans-count={plans.length}\n >\n Request API Access\n </Button>\n {plans.length === 0 && (\n <Typography\n variant=\"caption\"\n color=\"textSecondary\"\n style={{ marginTop: 4 }}\n data-testid=\"no-plans-message\"\n >\n {!apiProduct\n ? \"API product not found\"\n : (() => {\n const readyCondition =\n apiProduct.status?.conditions?.find(\n (c: any) => c.type === \"Ready\",\n );\n const planCondition =\n apiProduct.status?.conditions?.find(\n (c: any) => c.type === \"PlanPolicyDiscovered\",\n );\n\n if (readyCondition?.status !== \"True\") {\n return `HTTPRoute not ready: ${readyCondition?.message || \"unknown\"}`;\n }\n if (planCondition?.status !== \"True\") {\n return `No plans discovered: ${planCondition?.message || \"no PlanPolicy found\"}`;\n }\n return \"No plans available\";\n })()}\n </Typography>\n )}\n </Box>\n </Grid>\n )}\n {pendingRequests.length === 0 &&\n rejectedRequests.length === 0 &&\n approvedRequests.length === 0 && (\n <Grid item>\n <Box p={3} textAlign=\"center\">\n <Typography variant=\"body1\" color=\"textSecondary\">\n No API keys yet. Request access to get started.\n </Typography>\n </Box>\n </Grid>\n )}\n {pendingRequests.length > 0 && (\n <Grid item>\n <Table\n title=\"Pending Requests\"\n options={{\n paging: pendingRequests.length > 5,\n pageSize: 20,\n search: true,\n filtering: true,\n debounceInterval: 300,\n toolbar: true,\n emptyRowsWhenPaging: false,\n }}\n columns={pendingRequestColumns}\n data={pendingRequests}\n />\n </Grid>\n )}\n {rejectedRequests.length > 0 && (\n <Grid item>\n <Table\n title=\"Rejected Requests\"\n options={{\n paging: rejectedRequests.length > 5,\n pageSize: 20,\n search: true,\n filtering: true,\n debounceInterval: 300,\n toolbar: true,\n emptyRowsWhenPaging: false,\n }}\n columns={requestColumns}\n data={rejectedRequests}\n />\n </Grid>\n )}\n {approvedRequests.length > 0 && (\n <Grid item>\n <Table\n key=\"api-keys-table\"\n title=\"API Keys\"\n options={{\n paging: approvedRequests.length > 5,\n pageSize: 20,\n search: true,\n filtering: true,\n debounceInterval: 300,\n toolbar: true,\n emptyRowsWhenPaging: false,\n }}\n columns={approvedColumns}\n data={approvedRequests}\n detailPanel={detailPanelConfig}\n />\n </Grid>\n )}\n </Grid>\n\n <Dialog\n open={open}\n onClose={() => setOpen(false)}\n maxWidth=\"sm\"\n fullWidth\n >\n <DialogTitle>Request API Access</DialogTitle>\n <DialogContent>\n <Box\n mb={2}\n p={1.5}\n bgcolor=\"info.light\"\n borderRadius={1}\n display=\"flex\"\n alignItems=\"flex-start\"\n style={{ gap: 8 }}\n >\n <InfoIcon\n color=\"primary\"\n fontSize=\"small\"\n style={{ marginTop: 2 }}\n />\n <Typography variant=\"body2\">\n Your request will be reviewed by an API owner before access is\n granted.\n </Typography>\n </Box>\n {createError && (\n <Box\n mb={2}\n p={2}\n bgcolor=\"error.main\"\n color=\"error.contrastText\"\n borderRadius={1}\n >\n <Typography variant=\"body2\">{createError}</Typography>\n </Box>\n )}\n <FormControl\n fullWidth\n margin=\"normal\"\n disabled={creating}\n data-testid=\"tier-select-form\"\n >\n <InputLabel id=\"tier-select-label\">Select Tier</InputLabel>\n <Select\n labelId=\"tier-select-label\"\n data-testid=\"tier-select\"\n value={selectedPlan}\n onChange={(e) => setSelectedPlan(e.target.value as string)}\n disabled={creating}\n >\n {plans.map((plan: Plan) => {\n const limitDesc = Object.entries(plan.limits || {})\n .map(([key, val]) => `${val} per ${key}`)\n .join(\", \");\n return (\n <MenuItem\n key={plan.tier}\n value={plan.tier}\n data-testid={`tier-option-${plan.tier}`}\n >\n {plan.tier} {limitDesc ? `(${limitDesc})` : \"\"}\n </MenuItem>\n );\n })}\n </Select>\n </FormControl>\n <TextField\n label=\"Use Case (optional)\"\n placeholder=\"Describe how you plan to use this API\"\n multiline\n rows={3}\n fullWidth\n margin=\"normal\"\n value={useCase}\n onChange={(e) => setUseCase(e.target.value)}\n helperText=\"Explain your intended use of this API for admin review\"\n disabled={creating}\n />\n </DialogContent>\n <DialogActions>\n <Button onClick={() => setOpen(false)} disabled={creating}>\n Cancel\n </Button>\n <Button\n onClick={handleRequestAccess}\n color=\"primary\"\n variant=\"contained\"\n disabled={!selectedPlan || creating}\n startIcon={\n creating ? (\n <CircularProgress size={16} color=\"inherit\" />\n ) : undefined\n }\n >\n {creating ? \"Submitting...\" : \"Submit Request\"}\n </Button>\n </DialogActions>\n </Dialog>\n\n <Menu\n id=\"actions-menu\"\n open={Boolean(menuAnchor)}\n onClose={handleMenuClose}\n anchorReference=\"anchorPosition\"\n anchorPosition={menuAnchor || { top: 0, left: 0 }}\n >\n {menuRequest &&\n (() => {\n const isPending =\n !menuRequest.status?.phase ||\n menuRequest.status.phase === \"Pending\";\n const ownerId = menuRequest.spec.requestedBy.userId;\n const canEdit = canUpdateRequest && ownerId === userId && isPending;\n\n const items = [];\n if (canEdit) {\n items.push(\n <MenuItem key=\"edit\" onClick={handleMenuEdit}>\n Edit\n </MenuItem>,\n );\n }\n items.push(\n <MenuItem key=\"delete\" onClick={handleMenuDeleteClick}>\n Delete\n </MenuItem>,\n );\n return items;\n })()}\n </Menu>\n\n {requestToEdit && (\n <EditAPIKeyDialog\n open={editDialogOpen}\n onClose={() => {\n setEditDialogOpen(false);\n setRequestToEdit(null);\n }}\n onSuccess={handleEditSuccess}\n request={requestToEdit}\n availablePlans={plans}\n />\n )}\n\n <ConfirmDeleteDialog\n open={deleteDialogState.open}\n title=\"Delete Request\"\n description={`Are you sure you want to delete this ${deleteDialogState.request?.status?.phase === \"Approved\" ? \"API key\" : \"request\"}?`}\n deleting={deleting !== null}\n onConfirm={handleDeleteConfirm}\n onCancel={handleDeleteCancel}\n />\n\n <Dialog\n open={showOnceWarningOpen}\n onClose={() => {\n setShowOnceWarningOpen(false);\n setPendingKeyReveal(null);\n }}\n maxWidth=\"sm\"\n >\n <DialogTitle>\n <Box display=\"flex\" alignItems=\"center\">\n <WarningIcon color=\"primary\" style={{ marginRight: 8 }} />\n View API Key\n </Box>\n </DialogTitle>\n <DialogContent>\n <Typography variant=\"body1\" paragraph>\n This API key can only be viewed <strong>once</strong>. After you\n reveal it, you will not be able to retrieve it again.\n </Typography>\n <Typography variant=\"body2\" color=\"textSecondary\">\n Make sure to copy and store it securely before closing this view.\n </Typography>\n </DialogContent>\n <DialogActions>\n <Button\n onClick={() => {\n setShowOnceWarningOpen(false);\n setPendingKeyReveal(null);\n }}\n >\n Cancel\n </Button>\n <Button\n variant=\"contained\"\n color=\"primary\"\n onClick={() => {\n if (pendingKeyReveal) {\n fetchApiKeyFromSecret(\n pendingKeyReveal.namespace,\n pendingKeyReveal.name,\n );\n toggleVisibility(pendingKeyReveal.name);\n }\n setShowOnceWarningOpen(false);\n setPendingKeyReveal(null);\n }}\n >\n Reveal API Key\n </Button>\n </DialogActions>\n </Dialog>\n </Box>\n );\n};\n","\"use strict\";\n\nvar _interopRequireDefault = require(\"@babel/runtime/helpers/interopRequireDefault\");\n\nvar _interopRequireWildcard = require(\"@babel/runtime/helpers/interopRequireWildcard\");\n\nObject.defineProperty(exports, \"__esModule\", {\n value: true\n});\nexports.default = void 0;\n\nvar React = _interopRequireWildcard(require(\"react\"));\n\nvar _createSvgIcon = _interopRequireDefault(require(\"./utils/createSvgIcon\"));\n\nvar _default = (0, _createSvgIcon.default)( /*#__PURE__*/React.createElement(\"path\", {\n d: \"M12 8c1.1 0 2-.9 2-2s-.9-2-2-2-2 .9-2 2 .9 2 2 2zm0 2c-1.1 0-2 .9-2 2s.9 2 2 2 2-.9 2-2-.9-2-2-2zm0 6c-1.1 0-2 .9-2 2s.9 2 2 2 2-.9 2-2-.9-2-2-2z\"\n}), 'MoreVert');\n\nexports.default = _default;","\"use strict\";\n\nvar _interopRequireDefault = require(\"@babel/runtime/helpers/interopRequireDefault\");\n\nvar _interopRequireWildcard = require(\"@babel/runtime/helpers/interopRequireWildcard\");\n\nObject.defineProperty(exports, \"__esModule\", {\n value: true\n});\nexports.default = void 0;\n\nvar React = _interopRequireWildcard(require(\"react\"));\n\nvar _createSvgIcon = _interopRequireDefault(require(\"./utils/createSvgIcon\"));\n\nvar _default = (0, _createSvgIcon.default)( /*#__PURE__*/React.createElement(\"path\", {\n d: \"M6 2v6h.01L6 8.01 10 12l-4 4 .01.01H6V22h12v-5.99h-.01L18 16l-4-4 4-3.99-.01-.01H18V2H6zm10 14.5V20H8v-3.5l4-4 4 4zm-4-5l-4-4V4h8v3.5l-4 4z\"\n}), 'HourglassEmpty');\n\nexports.default = _default;"],"names":["useEntity","versionedHolder","useVersionedContext","Error","value","atVersion","entity","createVersionedContext","_interopRequireDefault","_interopRequireWildcard","exports","React","_default","default","createElement","d","ApiKeyManagementTab","namespace","propNamespace","apiProduct","deleteDialogState","config","useApi","configApiRef","identityApi","identityApiRef","fetchApi","fetchApiRef","alertApi","alertApiRef","backendUrl","getString","visibleKeys","setVisibleKeys","useState","Set","refresh","setRefresh","userId","setUserId","userEmail","setUserEmail","open","setOpen","selectedPlan","setSelectedPlan","useCase","setUseCase","creating","setCreating","createError","setCreateError","editDialogOpen","setEditDialogOpen","requestToEdit","setRequestToEdit","menuAnchor","setMenuAnchor","menuRequest","setMenuRequest","deleting","setDeleting","optimisticallyDeleted","setOptimisticallyDeleted","setDeleteDialogState","request","apiKeyValues","setApiKeyValues","Map","apiKeyLoading","setApiKeyLoading","alreadyReadKeys","setAlreadyReadKeys","showOnceWarningOpen","setShowOnceWarningOpen","pendingKeyReveal","setPendingKeyReveal","apiProductName","metadata","annotations","name","useAsync","async","identity","getBackstageIdentity","profile","getProfileInfo","userEntityRef","email","requests","loading","requestsLoading","error","requestsError","response","fetch","ok","json","items","filter","r","spec","apiProductRef","plansLoading","plansError","data","find","p","resourceRef","undefined","allowed","canCreateRequest","createRequestPermissionLoading","createRequestPermissionError","useKuadrantPermission","kuadrantApiKeyCreatePermission","canDeleteOwnKey","deleteOwnPermissionLoading","deleteOwnPermissionError","kuadrantApiKeyDeleteOwnPermission","canDeleteAllKeys","deleteAllPermissionLoading","deleteAllPermissionError","kuadrantApiKeyDeleteAllPermission","canUpdateRequest","updateRequestPermissionLoading","updateRequestPermissionError","kuadrantApiKeyUpdateOwnPermission","handleMenuClose","handleMenuEdit","handleMenuDeleteClick","toggleVisibility","keyName","prev","newSet","has","delete","add","detailPanelConfig","useMemo","render","rowData","Box","key","revealedKey","get","DetailPanelContent","apiName","revealedApiKey","api","selectedLanguage","setSelectedLanguage","hostname","status","apiHostname","displayApiKey","bgcolor","onClick","e","stopPropagation","mb","Typography","variant","gutterBottom","borderRadius","border","style","whiteSpace","wordBreak","overflowWrap","paragraph","planTier","Tabs","onChange","newValue","indicatorColor","Tab","label","mt","CodeSnippet","text","language","showCopyCodeButton","permissionError","Progress","ResponseErrorPanel","failedPermission","color","message","myRequests","plans","discoveredPlans","pendingRequests","phase","approvedRequests","rejectedRequests","approvedColumns","title","field","row","Chip","size","reviewedAt","Date","toLocaleDateString","searchable","filtering","isVisible","isLoading","apiKeyValue","hasSecretRef","secretRef","canReadSecret","isAlreadyRead","Tooltip","display","alignItems","fontFamily","marginRight","VisibilityOffIcon","fontSize","IconButton","navigator","clipboard","writeText","post","severity","FileCopyIcon","span","requestNamespace","requestName","next","clearApiKeyValue","disabled","VisibilityIcon","CircularProgress","ownerId","requestedBy","canDeleteResource","rect","currentTarget","getBoundingClientRect","top","bottom","left","aria-controls","aria-haspopup","MoreVertIcon","requestColumns","isPending","icon","HourglassEmptyIcon","CancelIcon","placement","maxWidth","overflow","textOverflow","creationTimestamp","canDelete","pendingRequestColumns","col","Grid","container","spacing","direction","item","flexDirection","Button","startIcon","AddIcon","length","data-testid","data-plans-count","marginTop","readyCondition","conditions","c","type","planCondition","textAlign","Table","options","paging","pageSize","search","debounceInterval","toolbar","emptyRowsWhenPaging","columns","detailPanel","Dialog","onClose","fullWidth","DialogTitle","DialogContent","gap","InfoIcon","FormControl","margin","InputLabel","id","Select","labelId","target","map","plan","limitDesc","Object","entries","limits","val","join","MenuItem","tier","TextField","placeholder","multiline","rows","helperText","DialogActions","method","headers","body","JSON","stringify","trim","errorData","catch","err","errorMessage","Menu","Boolean","anchorReference","anchorPosition","push","EditAPIKeyDialog","onSuccess","availablePlans","ConfirmDeleteDialog","description","onConfirm","handleDeleteRequest","onCancel","WarningIcon","strong","set","apiKey","console","fetchApiKeyFromSecret"],"sourceRoot":""}