@kuadrant/kuadrant-backstage-plugin-frontend 0.0.2-dev-19b0e56 → 0.0.2-dev-8ec2009
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.
- package/README.md +4 -4
- package/dist/components/ApiKeyManagementTab/ApiKeyManagementTab.esm.js +3 -3
- package/dist/components/ApiKeyManagementTab/ApiKeyManagementTab.esm.js.map +1 -1
- package/dist/components/ApprovalQueueCard/ApprovalQueueCard.esm.js +44 -28
- package/dist/components/ApprovalQueueCard/ApprovalQueueCard.esm.js.map +1 -1
- package/dist/components/EditAPIKeyRequestDialog/EditAPIKeyRequestDialog.esm.js +2 -2
- package/dist/components/EditAPIKeyRequestDialog/EditAPIKeyRequestDialog.esm.js.map +1 -1
- package/dist/components/KuadrantPage/KuadrantPage.esm.js +31 -20
- package/dist/components/KuadrantPage/KuadrantPage.esm.js.map +1 -1
- package/dist/components/MyApiKeysCard/MyApiKeysCard.esm.js +1 -1
- package/dist/components/MyApiKeysCard/MyApiKeysCard.esm.js.map +1 -1
- package/dist-scalprum/{internal.plugin-kuadrant.41278d0bcfa31baa5fae.js → internal.plugin-kuadrant.810174392d0016a5a388.js} +2 -2
- package/dist-scalprum/{internal.plugin-kuadrant.41278d0bcfa31baa5fae.js.map → internal.plugin-kuadrant.810174392d0016a5a388.js.map} +1 -1
- package/dist-scalprum/plugin-manifest.json +2 -2
- package/dist-scalprum/static/4306.4587e025.chunk.js +2 -0
- package/dist-scalprum/static/{4306.7eedfb3a.chunk.js.map → 4306.4587e025.chunk.js.map} +1 -1
- package/dist-scalprum/static/6281.b000c79f.chunk.js +2 -0
- package/dist-scalprum/static/6281.b000c79f.chunk.js.map +1 -0
- package/dist-scalprum/static/7632.daef27e4.chunk.js +2 -0
- package/dist-scalprum/static/7632.daef27e4.chunk.js.map +1 -0
- package/package.json +1 -1
- package/dist-scalprum/static/4306.7eedfb3a.chunk.js +0 -2
- package/dist-scalprum/static/6281.5f4d856e.chunk.js +0 -2
- package/dist-scalprum/static/6281.5f4d856e.chunk.js.map +0 -1
- package/dist-scalprum/static/7632.af7b3369.chunk.js +0 -2
- package/dist-scalprum/static/7632.af7b3369.chunk.js.map +0 -1
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"static/4306.7eedfb3a.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/Visibility.js","webpack://internal.plugin-kuadrant/./src/permissions.ts","webpack://internal.plugin-kuadrant/./src/utils/permissions.ts","webpack://internal.plugin-kuadrant/./src/components/ApiKeyManagementTab/ApiKeyManagementTab.tsx","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/icons/VisibilityOff.js","webpack://internal.plugin-kuadrant/./src/components/ConfirmDeleteDialog/ConfirmDeleteDialog.tsx","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/icons/MoreVert.js","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/icons/Warning.js","webpack://internal.plugin-kuadrant/./src/components/EditAPIKeyRequestDialog/EditAPIKeyRequestDialog.tsx","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 4.5C7 4.5 2.73 7.61 1 12c1.73 4.39 6 7.5 11 7.5s9.27-3.11 11-7.5c-1.73-4.39-6-7.5-11-7.5zM12 17c-2.76 0-5-2.24-5-5s2.24-5 5-5 5 2.24 5 5-2.24 5-5 5zm0-8c-1.66 0-3 1.34-3 3s1.34 3 3 3 3-1.34 3-3-1.34-3-3-3z\"\n}), 'Visibility');\n\nexports.default = _default;","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// apikeyrequest permissions\n\n/**\n * permission to create API key requests\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(kuadrantApiKeyRequestCreatePermission)\n * use in backend with resource: { permission, resourceRef: 'apiproduct:namespace/name' }\n */\nexport const kuadrantApiKeyRequestCreatePermission = createPermission({\n name: 'kuadrant.apikeyrequest.create',\n attributes: { action: 'create' },\n resourceType: 'apiproduct',\n});\n\n/**\n * permission to read API key requests created by the current user\n * use this for allowing users to see their own request history\n */\nexport const kuadrantApiKeyRequestReadOwnPermission = createPermission({\n name: 'kuadrant.apikeyrequest.read.own',\n attributes: { action: 'read' },\n});\n\n/**\n * permission to read all API key requests regardless of who created them\n * use this for platform engineers/admins who need to view the approval queue\n */\nexport const kuadrantApiKeyRequestReadAllPermission = createPermission({\n name: 'kuadrant.apikeyrequest.read.all',\n attributes: { action: 'read' },\n});\n\n/**\n * permission to update API key requests owned by the current user\n * allows users to edit their own pending requests (change plan tier, use case)\n */\nexport const kuadrantApiKeyRequestUpdateOwnPermission = createPermission({\n name: 'kuadrant.apikeyrequest.update.own',\n attributes: { action: 'update' },\n});\n\n/**\n * permission to update any API key request regardless of ownership\n * typically granted to API owners and platform engineers for approving/rejecting requests\n */\nexport const kuadrantApiKeyRequestUpdateAllPermission = createPermission({\n name: 'kuadrant.apikeyrequest.update.all',\n attributes: { action: 'update' },\n});\n\n/**\n * permission to delete API key requests created by the current user\n * allows users to cancel their own pending requests\n */\nexport const kuadrantApiKeyRequestDeleteOwnPermission = createPermission({\n name: 'kuadrant.apikeyrequest.delete.own',\n attributes: { action: 'delete' },\n});\n\n/**\n * permission to delete any API key request regardless of ownership\n * for platform engineers/admins\n */\nexport const kuadrantApiKeyRequestDeleteAllPermission = createPermission({\n name: 'kuadrant.apikeyrequest.delete.all',\n attributes: { action: 'delete' },\n});\n\nexport const kuadrantApiKeyRequestListPermission = createPermission({\n name: 'kuadrant.apikeyrequest.list',\n attributes: { action: 'read' },\n});\n\n// api key permissions\n\n/**\n * permission to read API keys owned by the current user\n * allows users to view their own active API keys\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 audit keys\n */\nexport const kuadrantApiKeyReadAllPermission = createPermission({\n name: 'kuadrant.apikey.read.all',\n attributes: { action: 'read' },\n});\n\n/**\n * permission to delete API keys owned by the current user\n * allows users to 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\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 kuadrantApiKeyRequestCreatePermission,\n kuadrantApiKeyRequestReadOwnPermission,\n kuadrantApiKeyRequestReadAllPermission,\n kuadrantApiKeyRequestUpdateOwnPermission,\n kuadrantApiKeyRequestUpdateAllPermission,\n kuadrantApiKeyRequestDeleteOwnPermission,\n kuadrantApiKeyRequestDeleteAllPermission,\n kuadrantApiKeyRequestListPermission,\n kuadrantApiKeyReadOwnPermission,\n kuadrantApiKeyReadAllPermission,\n kuadrantApiKeyDeleteOwnPermission,\n kuadrantApiKeyDeleteAllPermission,\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 * kuadrantApiKeyRequestCreatePermission,\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","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 { useApi, configApiRef, identityApiRef, fetchApiRef, alertApiRef } 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 { APIKeyRequest } from '../../types/api-management';\nimport {\n kuadrantApiKeyRequestCreatePermission,\n kuadrantApiKeyDeleteOwnPermission,\n kuadrantApiKeyDeleteAllPermission,\n kuadrantApiKeyRequestUpdateOwnPermission,\n} from '../../permissions';\nimport { useKuadrantPermission, canDeleteResource } from '../../utils/permissions';\nimport { EditAPIKeyRequestDialog } from '../EditAPIKeyRequestDialog';\nimport { ConfirmDeleteDialog } from '../ConfirmDeleteDialog';\n\ninterface APIProduct {\n metadata: {\n name: string;\n namespace: string;\n };\n spec: {\n plans?: Array<{\n tier: string;\n description?: string;\n limits?: any;\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 = ({ namespace: propNamespace }: 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<APIKeyRequest | null>(null);\n const [menuAnchor, setMenuAnchor] = useState<{ top: number; left: number } | null>(null);\n const [menuRequest, setMenuRequest] = useState<APIKeyRequest | null>(null);\n const [deleting, setDeleting] = useState<string | null>(null);\n const [optimisticallyDeleted, setOptimisticallyDeleted] = useState<Set<string>>(new Set());\n const [deleteDialogState, setDeleteDialogState] = useState<{\n open: boolean;\n request: APIKeyRequest | null;\n }>({ open: false, request: null });\n\n // get apiproduct name from entity annotation (set by entity provider)\n const apiProductName = entity.metadata.annotations?.['kuadrant.io/apiproduct'] || entity.metadata.name;\n const namespace = entity.metadata.annotations?.['kuadrant.io/namespace'] || propNamespace || '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 { value: requests, loading: requestsLoading, error: requestsError } = 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: APIKeyRequest) => r.spec.apiName === apiProductName && r.spec.apiNamespace === namespace\n );\n }, [apiProductName, namespace, refresh, fetchApi, backendUrl]);\n\n const { value: apiProduct, loading: plansLoading, error: plansError } = useAsync(async () => {\n const response = await fetchApi.fetch(`${backendUrl}/api/kuadrant/apiproducts`);\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((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 ? `apiproduct:${apiProduct.metadata.namespace}/${apiProduct.metadata.name}` : undefined;\n\n const {\n allowed: canCreateRequest,\n loading: createRequestPermissionLoading,\n error: createRequestPermissionError,\n } = useKuadrantPermission(kuadrantApiKeyRequestCreatePermission, 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(kuadrantApiKeyRequestUpdateOwnPermission);\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 request deleted successfully',\n severity: 'success',\n display: 'transient',\n });\n setRefresh(r => r + 1);\n } catch (err) {\n const errorMessage = 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 request: ${errorMessage}`,\n severity: 'error',\n display: 'transient',\n });\n } finally {\n setDeleting(null);\n }\n };\n\n const handleEditRequest = (request: APIKeyRequest) => {\n setRequestToEdit(request);\n setEditDialogOpen(true);\n };\n\n const handleEditSuccess = () => {\n setRefresh(r => r + 1);\n setEditDialogOpen(false);\n alertApi.post({ message: 'Request updated', severity: 'success', display: 'transient' });\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(`${backendUrl}/api/kuadrant/requests`, {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n },\n body: JSON.stringify({\n apiName: apiProductName,\n apiNamespace: namespace,\n userId,\n userEmail,\n planTier: selectedPlan,\n useCase: useCase.trim() || '',\n namespace,\n }),\n });\n\n if (!response.ok) {\n const errorData = await response.json().catch(() => ({}));\n throw new Error(errorData.error || `failed to create request: ${response.status}`);\n }\n\n alertApi.post({\n message: 'API access request submitted 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 = err instanceof Error ? err.message : 'unknown error occurred';\n alertApi.post({\n message: `Failed to create API access request: ${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 render: (data: any) => {\n // backstage Table wraps the data in { rowData: actualData }\n const request = data.rowData as APIKeyRequest;\n if (!request?.metadata?.name) {\n return <Box />;\n }\n\n return <DetailPanelContent request={request} apiName={apiProductName} />;\n },\n },\n ], [apiProductName]);\n\n // separate component to isolate state\n const DetailPanelContent = ({ request, apiName: api }: { request: APIKeyRequest; apiName: string }) => {\n const [selectedLanguage, setSelectedLanguage] = useState(0);\n const hostname = request.status?.apiHostname || `${api}.apps.example.com`;\n\n return (\n <Box p={3} bgcolor=\"background.default\" onClick={(e) => e.stopPropagation()}>\n {request.spec.useCase && (\n <Box mb={3}>\n <Typography variant=\"h6\" gutterBottom>\n Use Case\n </Typography>\n <Box p={2} bgcolor=\"background.paper\" borderRadius={1} border=\"1px solid rgba(0, 0, 0, 0.12)\">\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 {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 ${request.status?.apiKey}\"`}\n language=\"bash\"\n showCopyCodeButton\n />\n )}\n {selectedLanguage === 1 && (\n <CodeSnippet\n text={`const fetch = require('node-fetch');\n\nconst apiKey = '${request.status?.apiKey}';\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 = '${request.status?.apiKey}'\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 := \"${request.status?.apiKey}\"\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 = requestsLoading || plansLoading || createRequestPermissionLoading || deleteOwnPermissionLoading || deleteAllPermissionLoading || updateRequestPermissionLoading;\n const error = requestsError || plansError;\n const permissionError = createRequestPermissionError || deleteOwnPermissionError || deleteAllPermissionError || 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 ? 'kuadrant.apikeyrequest.create' :\n deleteOwnPermissionError ? 'kuadrant.apikey.delete.own' :\n deleteAllPermissionError ? 'kuadrant.apikey.delete.all' :\n updateRequestPermissionError ? 'kuadrant.apikeyrequest.update.own' : '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 APIKeyRequest[]).filter(\n r => !optimisticallyDeleted.has(r.metadata.name)\n );\n const plans = (apiProduct?.spec?.plans || []) as Plan[];\n\n const pendingRequests = myRequests.filter(r => !r.status?.phase || r.status.phase === 'Pending');\n const approvedRequests = myRequests.filter(r => r.status?.phase === 'Approved');\n const rejectedRequests = myRequests.filter(r => r.status?.phase === 'Rejected');\n\n const approvedColumns: TableColumn<APIKeyRequest>[] = [\n {\n title: 'Plan Tier',\n field: 'spec.planTier',\n render: (row: APIKeyRequest) => (\n <Chip label={row.spec.planTier} color=\"primary\" size=\"small\" />\n ),\n },\n {\n title: 'Approved',\n field: 'status.reviewedAt',\n render: (row: APIKeyRequest) => (\n <Typography variant=\"body2\">\n {row.status?.reviewedAt ? new Date(row.status.reviewedAt).toLocaleDateString() : '-'}\n </Typography>\n ),\n },\n {\n title: 'API Key',\n field: 'status.apiKey',\n searchable: false,\n filtering: false,\n render: (row: APIKeyRequest) => {\n const isVisible = visibleKeys.has(row.metadata.name);\n const apiKey = row.status?.apiKey || 'N/A';\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 {isVisible ? apiKey : '••••••••••••••••'}\n </Typography>\n <IconButton\n size=\"small\"\n onClick={() => toggleVisibility(row.metadata.name)}\n >\n {isVisible ? <VisibilityOffIcon /> : <VisibilityIcon />}\n </IconButton>\n </Box>\n );\n },\n },\n {\n title: '',\n field: 'actions',\n searchable: false,\n filtering: false,\n render: (row: APIKeyRequest) => {\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(ownerId, userId, canDeleteOwnKey, canDeleteAllKeys);\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<APIKeyRequest>[] = [\n {\n title: 'Status',\n field: 'status.phase',\n render: (row: APIKeyRequest) => {\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: 'Plan Tier',\n field: 'spec.planTier',\n render: (row: APIKeyRequest) => (\n <Chip label={row.spec.planTier} color=\"primary\" size=\"small\" />\n ),\n },\n {\n title: 'Use Case',\n field: 'spec.useCase',\n render: (row: APIKeyRequest) => {\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: 'spec.requestedAt',\n render: (row: APIKeyRequest) => (\n <Typography variant=\"body2\">\n {row.spec.requestedAt ? new Date(row.spec.requestedAt).toLocaleDateString() : '-'}\n </Typography>\n ),\n },\n {\n title: 'Reviewed',\n field: 'status.reviewedAt',\n render: (row: APIKeyRequest) => {\n if (!row.status?.reviewedAt) 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: 'Reason',\n field: 'status.reason',\n render: (row: APIKeyRequest) => {\n if (!row.status?.reason) {\n return <Typography variant=\"body2\">-</Typography>;\n }\n return (\n <Tooltip title={row.status.reason} 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.status.reason}\n </Typography>\n </Tooltip>\n );\n },\n },\n {\n title: '',\n field: 'actions',\n searchable: false,\n filtering: false,\n render: (row: APIKeyRequest) => {\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(ownerId, userId, canDeleteOwnKey, canDeleteAllKeys);\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 display=\"flex\" flexDirection=\"column\" alignItems=\"flex-end\" mb={2}>\n <Button\n variant=\"contained\"\n color=\"primary\"\n startIcon={<AddIcon />}\n onClick={() => setOpen(true)}\n disabled={plans.length === 0}\n >\n Request API Access\n </Button>\n {plans.length === 0 && (\n <Typography variant=\"caption\" color=\"textSecondary\" style={{ marginTop: 4 }}>\n {!apiProduct ? 'API product not found' : 'No plans available'}\n </Typography>\n )}\n </Box>\n </Grid>\n )}\n {pendingRequests.length === 0 && rejectedRequests.length === 0 && 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 open={open} onClose={() => setOpen(false)} maxWidth=\"sm\" fullWidth>\n <DialogTitle>Request API Access</DialogTitle>\n <DialogContent>\n {createError && (\n <Box mb={2} p={2} bgcolor=\"error.main\" color=\"error.contrastText\" borderRadius={1}>\n <Typography variant=\"body2\">{createError}</Typography>\n </Box>\n )}\n <FormControl fullWidth margin=\"normal\" disabled={creating}>\n <InputLabel>Select Plan Tier</InputLabel>\n <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 key={plan.tier} value={plan.tier}>\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}>Cancel</Button>\n <Button\n onClick={handleRequestAccess}\n color=\"primary\"\n variant=\"contained\"\n disabled={!selectedPlan || creating}\n startIcon={creating ? <CircularProgress size={16} color=\"inherit\" /> : undefined}\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 const isPending = !menuRequest.status?.phase || 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(<MenuItem key=\"edit\" onClick={handleMenuEdit}>Edit</MenuItem>);\n }\n items.push(<MenuItem key=\"delete\" onClick={handleMenuDeleteClick}>Delete</MenuItem>);\n return items;\n })()}\n </Menu>\n\n {requestToEdit && (\n <EditAPIKeyRequestDialog\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 </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 7c2.76 0 5 2.24 5 5 0 .65-.13 1.26-.36 1.83l2.92 2.92c1.51-1.26 2.7-2.89 3.43-4.75-1.73-4.39-6-7.5-11-7.5-1.4 0-2.74.25-3.98.7l2.16 2.16C10.74 7.13 11.35 7 12 7zM2 4.27l2.28 2.28.46.46C3.08 8.3 1.78 10.02 1 12c1.73 4.39 6 7.5 11 7.5 1.55 0 3.03-.3 4.38-.84l.42.42L19.73 22 21 20.73 3.27 3 2 4.27zM7.53 9.8l1.55 1.55c-.05.21-.08.43-.08.65 0 1.66 1.34 3 3 3 .22 0 .44-.03.65-.08l1.55 1.55c-.67.33-1.41.53-2.2.53-2.76 0-5-2.24-5-5 0-.79.2-1.53.53-2.2zm4.31-.78l3.15 3.15.02-.16c0-1.66-1.34-3-3-3l-.17.01z\"\n}), 'VisibilityOff');\n\nexports.default = _default;","import React, { useState, useEffect } from 'react';\nimport {\n Dialog,\n DialogTitle,\n DialogContent,\n DialogContentText,\n DialogActions,\n Button,\n TextField,\n Typography,\n Box,\n CircularProgress,\n} from '@material-ui/core';\nimport WarningIcon from '@material-ui/icons/Warning';\n\nexport interface ConfirmDeleteDialogProps {\n open: boolean;\n title: string;\n description: string;\n // for dangerous deletes, require typing this text to confirm\n confirmText?: string;\n // severity affects styling - 'high' shows warning icon and requires text confirmation\n severity?: 'normal' | 'high';\n deleting?: boolean;\n onConfirm: () => void;\n onCancel: () => void;\n}\n\nexport const ConfirmDeleteDialog = ({\n open,\n title,\n description,\n confirmText,\n severity = 'normal',\n deleting = false,\n onConfirm,\n onCancel,\n}: ConfirmDeleteDialogProps) => {\n const [inputValue, setInputValue] = useState('');\n\n // reset input when dialog opens/closes\n useEffect(() => {\n if (!open) {\n setInputValue('');\n }\n }, [open]);\n\n const requiresTextConfirmation = severity === 'high' && confirmText;\n const canConfirm = requiresTextConfirmation ? inputValue === confirmText : true;\n\n const handleConfirm = () => {\n if (canConfirm) {\n onConfirm();\n }\n };\n\n return (\n <Dialog\n open={open}\n onClose={deleting ? undefined : onCancel}\n maxWidth=\"sm\"\n fullWidth\n >\n <DialogTitle>\n {severity === 'high' && (\n <Box display=\"flex\" alignItems=\"center\" style={{ gap: 8 }}>\n <WarningIcon color=\"error\" />\n <span>{title}</span>\n </Box>\n )}\n {severity !== 'high' && title}\n </DialogTitle>\n <DialogContent>\n <DialogContentText>{description}</DialogContentText>\n {requiresTextConfirmation && (\n <Box mt={2}>\n <Typography variant=\"body2\" color=\"textSecondary\" gutterBottom>\n Type <strong>{confirmText}</strong> to confirm:\n </Typography>\n <TextField\n fullWidth\n variant=\"outlined\"\n size=\"small\"\n value={inputValue}\n onChange={e => setInputValue(e.target.value)}\n disabled={deleting}\n autoFocus\n placeholder={confirmText}\n />\n </Box>\n )}\n </DialogContent>\n <DialogActions>\n <Button onClick={onCancel} disabled={deleting}>\n Cancel\n </Button>\n <Button\n onClick={handleConfirm}\n color=\"secondary\"\n variant=\"contained\"\n disabled={deleting || !canConfirm}\n startIcon={deleting ? <CircularProgress size={16} color=\"inherit\" /> : undefined}\n >\n {deleting ? 'Deleting...' : 'Delete'}\n </Button>\n </DialogActions>\n </Dialog>\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: \"M1 21h22L12 2 1 21zm12-3h-2v-2h2v2zm0-4h-2v-4h2v4z\"\n}), 'Warning');\n\nexports.default = _default;","import React, { useState, useEffect } from 'react';\nimport {\n Dialog,\n DialogTitle,\n DialogContent,\n DialogActions,\n Button,\n TextField,\n Box,\n Typography,\n FormControl,\n InputLabel,\n Select,\n MenuItem,\n CircularProgress,\n} from '@material-ui/core';\nimport { useApi, configApiRef, fetchApiRef } from '@backstage/core-plugin-api';\nimport { APIKeyRequest } from '../../types/api-management';\n\ninterface EditAPIKeyRequestDialogProps {\n open: boolean;\n onClose: () => void;\n onSuccess: () => void;\n request: APIKeyRequest;\n availablePlans: Array<{\n tier: string;\n description?: string;\n limits?: any;\n }>;\n}\n\nexport const EditAPIKeyRequestDialog = ({\n open,\n onClose,\n onSuccess,\n request,\n availablePlans,\n}: EditAPIKeyRequestDialogProps) => {\n const config = useApi(configApiRef);\n const fetchApi = useApi(fetchApiRef);\n const backendUrl = config.getString('backend.baseUrl');\n\n const [planTier, setPlanTier] = useState('');\n const [useCase, setUseCase] = useState('');\n const [saving, setSaving] = useState(false);\n const [error, setError] = useState('');\n\n useEffect(() => {\n if (open && request) {\n setPlanTier(request.spec.planTier || '');\n setUseCase(request.spec.useCase || '');\n setError('');\n }\n }, [open, request]);\n\n const handleSave = async () => {\n if (!planTier) {\n setError('Please select a plan tier');\n return;\n }\n\n setError('');\n setSaving(true);\n\n try {\n const patch = {\n spec: {\n planTier,\n useCase: useCase.trim(),\n },\n };\n\n const response = await fetchApi.fetch(\n `${backendUrl}/api/kuadrant/requests/${request.metadata.namespace}/${request.metadata.name}`,\n {\n method: 'PATCH',\n headers: {\n 'Content-Type': 'application/json',\n },\n body: JSON.stringify(patch),\n }\n );\n\n if (!response.ok) {\n const errorData = await response.json().catch(() => ({}));\n throw new Error(errorData.error || `Failed to update request: ${response.status}`);\n }\n\n onSuccess();\n onClose();\n } catch (err) {\n console.error('Error updating API key request:', err);\n setError(err instanceof Error ? err.message : 'Unknown error occurred');\n } finally {\n setSaving(false);\n }\n };\n\n const handleClose = () => {\n if (!saving) {\n setError('');\n onClose();\n }\n };\n\n return (\n <Dialog open={open} onClose={handleClose} maxWidth=\"sm\" fullWidth>\n <DialogTitle>Edit API Access Request</DialogTitle>\n <DialogContent>\n {error && (\n <Box mb={2} p={2} bgcolor=\"error.main\" color=\"error.contrastText\" borderRadius={1}>\n <Typography variant=\"body2\">{error}</Typography>\n </Box>\n )}\n\n <FormControl fullWidth margin=\"normal\">\n <InputLabel>Plan Tier</InputLabel>\n <Select\n value={planTier}\n onChange={(e) => setPlanTier(e.target.value as string)}\n disabled={saving}\n >\n {availablePlans.map((plan) => {\n const limitDesc = Object.entries(plan.limits || {})\n .map(([key, val]) => `${val} per ${key}`)\n .join(', ');\n return (\n <MenuItem key={plan.tier} value={plan.tier}>\n {plan.tier} {limitDesc ? `(${limitDesc})` : ''}\n </MenuItem>\n );\n })}\n </Select>\n </FormControl>\n\n <TextField\n label=\"Use Case\"\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 disabled={saving}\n helperText=\"Explain your intended use of this API for admin review\"\n />\n </DialogContent>\n <DialogActions>\n <Button onClick={handleClose} disabled={saving}>\n Cancel\n </Button>\n <Button\n onClick={handleSave}\n color=\"primary\"\n variant=\"contained\"\n disabled={!planTier || saving}\n startIcon={saving ? <CircularProgress size={16} color=\"inherit\" /> : undefined}\n >\n {saving ? 'Saving...' : 'Save Changes'}\n </Button>\n </DialogActions>\n </Dialog>\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: \"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","createPermission","name","attributes","action","kuadrantPlanPolicyListPermission","kuadrantApiProductCreatePermission","kuadrantApiProductReadAllPermission","kuadrantApiProductUpdateOwnPermission","kuadrantApiProductUpdateAllPermission","kuadrantApiProductDeleteOwnPermission","kuadrantApiProductDeleteAllPermission","kuadrantApiProductListPermission","kuadrantApiKeyRequestCreatePermission","resourceType","kuadrantApiKeyRequestReadOwnPermission","kuadrantApiKeyRequestReadAllPermission","kuadrantApiKeyRequestUpdateOwnPermission","kuadrantApiKeyRequestUpdateAllPermission","kuadrantApiKeyDeleteOwnPermission","kuadrantApiKeyDeleteAllPermission","useKuadrantPermission","permission","resourceRef","permissionRequest","result","usePermission","allowed","loading","error","canDeleteResource","ownerId","currentUserId","canDeleteOwn","canDeleteAll","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","apiProductName","metadata","annotations","useAsync","async","identity","getBackstageIdentity","profile","getProfileInfo","userEntityRef","email","requests","requestsLoading","requestsError","response","fetch","ok","json","items","filter","r","spec","apiName","apiNamespace","plansLoading","plansError","data","find","p","undefined","canCreateRequest","createRequestPermissionLoading","createRequestPermissionError","canDeleteOwnKey","deleteOwnPermissionLoading","deleteOwnPermissionError","canDeleteAllKeys","deleteAllPermissionLoading","deleteAllPermissionError","canUpdateRequest","updateRequestPermissionLoading","updateRequestPermissionError","handleMenuClose","handleMenuEdit","handleMenuDeleteClick","detailPanelConfig","useMemo","render","rowData","DetailPanelContent","Box","api","selectedLanguage","setSelectedLanguage","hostname","status","apiHostname","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","apiKey","language","showCopyCodeButton","permissionError","Progress","ResponseErrorPanel","failedPermission","color","message","myRequests","has","plans","pendingRequests","phase","approvedRequests","rejectedRequests","approvedColumns","title","field","row","Chip","size","reviewedAt","Date","toLocaleDateString","searchable","filtering","isVisible","display","alignItems","fontFamily","marginRight","IconButton","toggleVisibility","keyName","prev","newSet","delete","add","VisibilityOffIcon","VisibilityIcon","CircularProgress","requestedBy","rect","currentTarget","getBoundingClientRect","top","bottom","left","aria-controls","aria-haspopup","MoreVertIcon","requestColumns","isPending","icon","HourglassEmptyIcon","CancelIcon","Tooltip","placement","maxWidth","overflow","textOverflow","requestedAt","reason","canDelete","pendingRequestColumns","col","Grid","container","spacing","direction","item","flexDirection","Button","startIcon","AddIcon","disabled","length","marginTop","textAlign","Table","options","paging","pageSize","search","debounceInterval","toolbar","emptyRowsWhenPaging","columns","detailPanel","Dialog","onClose","fullWidth","DialogTitle","DialogContent","FormControl","margin","InputLabel","Select","target","map","plan","limitDesc","Object","entries","limits","key","val","join","MenuItem","tier","TextField","placeholder","multiline","rows","helperText","DialogActions","method","headers","body","JSON","stringify","trim","errorData","catch","post","severity","err","errorMessage","Menu","id","Boolean","anchorReference","anchorPosition","push","EditAPIKeyRequestDialog","onSuccess","availablePlans","ConfirmDeleteDialog","description","onConfirm","next","handleDeleteRequest","onCancel","confirmText","inputValue","setInputValue","useEffect","requiresTextConfirmation","canConfirm","gap","WarningIcon","span","DialogContentText","strong","autoFocus","setPlanTier","saving","setSaving","setError","handleClose","patch","console"],"sourceRoot":""}
|
|
1
|
+
{"version":3,"file":"static/4306.4587e025.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/Visibility.js","webpack://internal.plugin-kuadrant/./src/permissions.ts","webpack://internal.plugin-kuadrant/./src/utils/permissions.ts","webpack://internal.plugin-kuadrant/./src/components/ApiKeyManagementTab/ApiKeyManagementTab.tsx","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/icons/VisibilityOff.js","webpack://internal.plugin-kuadrant/./src/components/ConfirmDeleteDialog/ConfirmDeleteDialog.tsx","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/icons/MoreVert.js","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/icons/Warning.js","webpack://internal.plugin-kuadrant/./src/components/EditAPIKeyRequestDialog/EditAPIKeyRequestDialog.tsx","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 4.5C7 4.5 2.73 7.61 1 12c1.73 4.39 6 7.5 11 7.5s9.27-3.11 11-7.5c-1.73-4.39-6-7.5-11-7.5zM12 17c-2.76 0-5-2.24-5-5s2.24-5 5-5 5 2.24 5 5-2.24 5-5 5zm0-8c-1.66 0-3 1.34-3 3s1.34 3 3 3 3-1.34 3-3-1.34-3-3-3z\"\n}), 'Visibility');\n\nexports.default = _default;","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// apikeyrequest permissions\n\n/**\n * permission to create API key requests\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(kuadrantApiKeyRequestCreatePermission)\n * use in backend with resource: { permission, resourceRef: 'apiproduct:namespace/name' }\n */\nexport const kuadrantApiKeyRequestCreatePermission = createPermission({\n name: 'kuadrant.apikeyrequest.create',\n attributes: { action: 'create' },\n resourceType: 'apiproduct',\n});\n\n/**\n * permission to read API key requests created by the current user\n * use this for allowing users to see their own request history\n */\nexport const kuadrantApiKeyRequestReadOwnPermission = createPermission({\n name: 'kuadrant.apikeyrequest.read.own',\n attributes: { action: 'read' },\n});\n\n/**\n * permission to read all API key requests regardless of who created them\n * use this for platform engineers/admins who need to view the approval queue\n */\nexport const kuadrantApiKeyRequestReadAllPermission = createPermission({\n name: 'kuadrant.apikeyrequest.read.all',\n attributes: { action: 'read' },\n});\n\n/**\n * permission to update API key requests owned by the current user\n * allows users to edit their own pending requests (change plan tier, use case)\n */\nexport const kuadrantApiKeyRequestUpdateOwnPermission = createPermission({\n name: 'kuadrant.apikeyrequest.update.own',\n attributes: { action: 'update' },\n});\n\n/**\n * permission to update any API key request regardless of ownership\n * typically granted to API owners and platform engineers for approving/rejecting requests\n */\nexport const kuadrantApiKeyRequestUpdateAllPermission = createPermission({\n name: 'kuadrant.apikeyrequest.update.all',\n attributes: { action: 'update' },\n});\n\n/**\n * permission to delete API key requests created by the current user\n * allows users to cancel their own pending requests\n */\nexport const kuadrantApiKeyRequestDeleteOwnPermission = createPermission({\n name: 'kuadrant.apikeyrequest.delete.own',\n attributes: { action: 'delete' },\n});\n\n/**\n * permission to delete any API key request regardless of ownership\n * for platform engineers/admins\n */\nexport const kuadrantApiKeyRequestDeleteAllPermission = createPermission({\n name: 'kuadrant.apikeyrequest.delete.all',\n attributes: { action: 'delete' },\n});\n\nexport const kuadrantApiKeyRequestListPermission = createPermission({\n name: 'kuadrant.apikeyrequest.list',\n attributes: { action: 'read' },\n});\n\n// api key permissions\n\n/**\n * permission to read API keys owned by the current user\n * allows users to view their own active API keys\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 audit keys\n */\nexport const kuadrantApiKeyReadAllPermission = createPermission({\n name: 'kuadrant.apikey.read.all',\n attributes: { action: 'read' },\n});\n\n/**\n * permission to delete API keys owned by the current user\n * allows users to 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\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 kuadrantApiKeyRequestCreatePermission,\n kuadrantApiKeyRequestReadOwnPermission,\n kuadrantApiKeyRequestReadAllPermission,\n kuadrantApiKeyRequestUpdateOwnPermission,\n kuadrantApiKeyRequestUpdateAllPermission,\n kuadrantApiKeyRequestDeleteOwnPermission,\n kuadrantApiKeyRequestDeleteAllPermission,\n kuadrantApiKeyRequestListPermission,\n kuadrantApiKeyReadOwnPermission,\n kuadrantApiKeyReadAllPermission,\n kuadrantApiKeyDeleteOwnPermission,\n kuadrantApiKeyDeleteAllPermission,\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 * kuadrantApiKeyRequestCreatePermission,\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","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 { useApi, configApiRef, identityApiRef, fetchApiRef, alertApiRef } 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 { APIKeyRequest } from '../../types/api-management';\nimport {\n kuadrantApiKeyRequestCreatePermission,\n kuadrantApiKeyDeleteOwnPermission,\n kuadrantApiKeyDeleteAllPermission,\n kuadrantApiKeyRequestUpdateOwnPermission,\n} from '../../permissions';\nimport { useKuadrantPermission, canDeleteResource } from '../../utils/permissions';\nimport { EditAPIKeyRequestDialog } from '../EditAPIKeyRequestDialog';\nimport { ConfirmDeleteDialog } from '../ConfirmDeleteDialog';\n\ninterface APIProduct {\n metadata: {\n name: string;\n namespace: string;\n };\n spec: {\n plans?: Array<{\n tier: string;\n description?: string;\n limits?: any;\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 = ({ namespace: propNamespace }: 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<APIKeyRequest | null>(null);\n const [menuAnchor, setMenuAnchor] = useState<{ top: number; left: number } | null>(null);\n const [menuRequest, setMenuRequest] = useState<APIKeyRequest | null>(null);\n const [deleting, setDeleting] = useState<string | null>(null);\n const [optimisticallyDeleted, setOptimisticallyDeleted] = useState<Set<string>>(new Set());\n const [deleteDialogState, setDeleteDialogState] = useState<{\n open: boolean;\n request: APIKeyRequest | null;\n }>({ open: false, request: null });\n\n // get apiproduct name from entity annotation (set by entity provider)\n const apiProductName = entity.metadata.annotations?.['kuadrant.io/apiproduct'] || entity.metadata.name;\n const namespace = entity.metadata.annotations?.['kuadrant.io/namespace'] || propNamespace || '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 { value: requests, loading: requestsLoading, error: requestsError } = 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: APIKeyRequest) => r.spec.apiName === apiProductName && r.spec.apiNamespace === namespace\n );\n }, [apiProductName, namespace, refresh, fetchApi, backendUrl]);\n\n const { value: apiProduct, loading: plansLoading, error: plansError } = useAsync(async () => {\n const response = await fetchApi.fetch(`${backendUrl}/api/kuadrant/apiproducts`);\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((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 ? `apiproduct:${apiProduct.metadata.namespace}/${apiProduct.metadata.name}` : undefined;\n\n const {\n allowed: canCreateRequest,\n loading: createRequestPermissionLoading,\n error: createRequestPermissionError,\n } = useKuadrantPermission(kuadrantApiKeyRequestCreatePermission, 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(kuadrantApiKeyRequestUpdateOwnPermission);\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 request deleted successfully',\n severity: 'success',\n display: 'transient',\n });\n setRefresh(r => r + 1);\n } catch (err) {\n const errorMessage = 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 request: ${errorMessage}`,\n severity: 'error',\n display: 'transient',\n });\n } finally {\n setDeleting(null);\n }\n };\n\n const handleEditRequest = (request: APIKeyRequest) => {\n setRequestToEdit(request);\n setEditDialogOpen(true);\n };\n\n const handleEditSuccess = () => {\n setRefresh(r => r + 1);\n setEditDialogOpen(false);\n alertApi.post({ message: 'Request updated', severity: 'success', display: 'transient' });\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(`${backendUrl}/api/kuadrant/requests`, {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n },\n body: JSON.stringify({\n apiName: apiProductName,\n apiNamespace: namespace,\n userId,\n userEmail,\n planTier: selectedPlan,\n useCase: useCase.trim() || '',\n namespace,\n }),\n });\n\n if (!response.ok) {\n const errorData = await response.json().catch(() => ({}));\n throw new Error(errorData.error || `failed to create request: ${response.status}`);\n }\n\n alertApi.post({\n message: 'API access request submitted 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 = err instanceof Error ? err.message : 'unknown error occurred';\n alertApi.post({\n message: `Failed to create API access request: ${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 render: (data: any) => {\n // backstage Table wraps the data in { rowData: actualData }\n const request = data.rowData as APIKeyRequest;\n if (!request?.metadata?.name) {\n return <Box />;\n }\n\n return <DetailPanelContent request={request} apiName={apiProductName} />;\n },\n },\n ], [apiProductName]);\n\n // separate component to isolate state\n const DetailPanelContent = ({ request, apiName: api }: { request: APIKeyRequest; apiName: string }) => {\n const [selectedLanguage, setSelectedLanguage] = useState(0);\n const hostname = request.status?.apiHostname || `${api}.apps.example.com`;\n\n return (\n <Box p={3} bgcolor=\"background.default\" onClick={(e) => e.stopPropagation()}>\n {request.spec.useCase && (\n <Box mb={3}>\n <Typography variant=\"h6\" gutterBottom>\n Use Case\n </Typography>\n <Box p={2} bgcolor=\"background.paper\" borderRadius={1} border=\"1px solid rgba(0, 0, 0, 0.12)\">\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 {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 ${request.status?.apiKey}\"`}\n language=\"bash\"\n showCopyCodeButton\n />\n )}\n {selectedLanguage === 1 && (\n <CodeSnippet\n text={`const fetch = require('node-fetch');\n\nconst apiKey = '${request.status?.apiKey}';\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 = '${request.status?.apiKey}'\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 := \"${request.status?.apiKey}\"\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 = requestsLoading || plansLoading || createRequestPermissionLoading || deleteOwnPermissionLoading || deleteAllPermissionLoading || updateRequestPermissionLoading;\n const error = requestsError || plansError;\n const permissionError = createRequestPermissionError || deleteOwnPermissionError || deleteAllPermissionError || 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 ? 'kuadrant.apikeyrequest.create' :\n deleteOwnPermissionError ? 'kuadrant.apikey.delete.own' :\n deleteAllPermissionError ? 'kuadrant.apikey.delete.all' :\n updateRequestPermissionError ? 'kuadrant.apikeyrequest.update.own' : '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 APIKeyRequest[]).filter(\n r => !optimisticallyDeleted.has(r.metadata.name)\n );\n const plans = (apiProduct?.spec?.plans || []) as Plan[];\n\n const pendingRequests = myRequests.filter(r => !r.status?.phase || r.status.phase === 'Pending');\n const approvedRequests = myRequests.filter(r => r.status?.phase === 'Approved');\n const rejectedRequests = myRequests.filter(r => r.status?.phase === 'Rejected');\n\n const approvedColumns: TableColumn<APIKeyRequest>[] = [\n {\n title: 'Tier',\n field: 'spec.planTier',\n render: (row: APIKeyRequest) => (\n <Chip label={row.spec.planTier} color=\"primary\" size=\"small\" />\n ),\n },\n {\n title: 'Approved',\n field: 'status.reviewedAt',\n render: (row: APIKeyRequest) => (\n <Typography variant=\"body2\">\n {row.status?.reviewedAt ? new Date(row.status.reviewedAt).toLocaleDateString() : '-'}\n </Typography>\n ),\n },\n {\n title: 'API Key',\n field: 'status.apiKey',\n searchable: false,\n filtering: false,\n render: (row: APIKeyRequest) => {\n const isVisible = visibleKeys.has(row.metadata.name);\n const apiKey = row.status?.apiKey || 'N/A';\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 {isVisible ? apiKey : '••••••••••••••••'}\n </Typography>\n <IconButton\n size=\"small\"\n onClick={() => toggleVisibility(row.metadata.name)}\n >\n {isVisible ? <VisibilityOffIcon /> : <VisibilityIcon />}\n </IconButton>\n </Box>\n );\n },\n },\n {\n title: '',\n field: 'actions',\n searchable: false,\n filtering: false,\n render: (row: APIKeyRequest) => {\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(ownerId, userId, canDeleteOwnKey, canDeleteAllKeys);\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<APIKeyRequest>[] = [\n {\n title: 'Status',\n field: 'status.phase',\n render: (row: APIKeyRequest) => {\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: APIKeyRequest) => (\n <Chip label={row.spec.planTier} color=\"primary\" size=\"small\" />\n ),\n },\n {\n title: 'Use Case',\n field: 'spec.useCase',\n render: (row: APIKeyRequest) => {\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: 'spec.requestedAt',\n render: (row: APIKeyRequest) => (\n <Typography variant=\"body2\">\n {row.spec.requestedAt ? new Date(row.spec.requestedAt).toLocaleDateString() : '-'}\n </Typography>\n ),\n },\n {\n title: 'Reviewed',\n field: 'status.reviewedAt',\n render: (row: APIKeyRequest) => {\n if (!row.status?.reviewedAt) 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: 'Reason',\n field: 'status.reason',\n render: (row: APIKeyRequest) => {\n if (!row.status?.reason) {\n return <Typography variant=\"body2\">-</Typography>;\n }\n return (\n <Tooltip title={row.status.reason} 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.status.reason}\n </Typography>\n </Tooltip>\n );\n },\n },\n {\n title: '',\n field: 'actions',\n searchable: false,\n filtering: false,\n render: (row: APIKeyRequest) => {\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(ownerId, userId, canDeleteOwnKey, canDeleteAllKeys);\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 display=\"flex\" flexDirection=\"column\" alignItems=\"flex-end\" mb={2}>\n <Button\n variant=\"contained\"\n color=\"primary\"\n startIcon={<AddIcon />}\n onClick={() => setOpen(true)}\n disabled={plans.length === 0}\n >\n Request API Access\n </Button>\n {plans.length === 0 && (\n <Typography variant=\"caption\" color=\"textSecondary\" style={{ marginTop: 4 }}>\n {!apiProduct ? 'API product not found' : 'No plans available'}\n </Typography>\n )}\n </Box>\n </Grid>\n )}\n {pendingRequests.length === 0 && rejectedRequests.length === 0 && 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 open={open} onClose={() => setOpen(false)} maxWidth=\"sm\" fullWidth>\n <DialogTitle>Request API Access</DialogTitle>\n <DialogContent>\n {createError && (\n <Box mb={2} p={2} bgcolor=\"error.main\" color=\"error.contrastText\" borderRadius={1}>\n <Typography variant=\"body2\">{createError}</Typography>\n </Box>\n )}\n <FormControl fullWidth margin=\"normal\" disabled={creating}>\n <InputLabel>Select Tier</InputLabel>\n <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 key={plan.tier} value={plan.tier}>\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}>Cancel</Button>\n <Button\n onClick={handleRequestAccess}\n color=\"primary\"\n variant=\"contained\"\n disabled={!selectedPlan || creating}\n startIcon={creating ? <CircularProgress size={16} color=\"inherit\" /> : undefined}\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 const isPending = !menuRequest.status?.phase || 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(<MenuItem key=\"edit\" onClick={handleMenuEdit}>Edit</MenuItem>);\n }\n items.push(<MenuItem key=\"delete\" onClick={handleMenuDeleteClick}>Delete</MenuItem>);\n return items;\n })()}\n </Menu>\n\n {requestToEdit && (\n <EditAPIKeyRequestDialog\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 </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 7c2.76 0 5 2.24 5 5 0 .65-.13 1.26-.36 1.83l2.92 2.92c1.51-1.26 2.7-2.89 3.43-4.75-1.73-4.39-6-7.5-11-7.5-1.4 0-2.74.25-3.98.7l2.16 2.16C10.74 7.13 11.35 7 12 7zM2 4.27l2.28 2.28.46.46C3.08 8.3 1.78 10.02 1 12c1.73 4.39 6 7.5 11 7.5 1.55 0 3.03-.3 4.38-.84l.42.42L19.73 22 21 20.73 3.27 3 2 4.27zM7.53 9.8l1.55 1.55c-.05.21-.08.43-.08.65 0 1.66 1.34 3 3 3 .22 0 .44-.03.65-.08l1.55 1.55c-.67.33-1.41.53-2.2.53-2.76 0-5-2.24-5-5 0-.79.2-1.53.53-2.2zm4.31-.78l3.15 3.15.02-.16c0-1.66-1.34-3-3-3l-.17.01z\"\n}), 'VisibilityOff');\n\nexports.default = _default;","import React, { useState, useEffect } from 'react';\nimport {\n Dialog,\n DialogTitle,\n DialogContent,\n DialogContentText,\n DialogActions,\n Button,\n TextField,\n Typography,\n Box,\n CircularProgress,\n} from '@material-ui/core';\nimport WarningIcon from '@material-ui/icons/Warning';\n\nexport interface ConfirmDeleteDialogProps {\n open: boolean;\n title: string;\n description: string;\n // for dangerous deletes, require typing this text to confirm\n confirmText?: string;\n // severity affects styling - 'high' shows warning icon and requires text confirmation\n severity?: 'normal' | 'high';\n deleting?: boolean;\n onConfirm: () => void;\n onCancel: () => void;\n}\n\nexport const ConfirmDeleteDialog = ({\n open,\n title,\n description,\n confirmText,\n severity = 'normal',\n deleting = false,\n onConfirm,\n onCancel,\n}: ConfirmDeleteDialogProps) => {\n const [inputValue, setInputValue] = useState('');\n\n // reset input when dialog opens/closes\n useEffect(() => {\n if (!open) {\n setInputValue('');\n }\n }, [open]);\n\n const requiresTextConfirmation = severity === 'high' && confirmText;\n const canConfirm = requiresTextConfirmation ? inputValue === confirmText : true;\n\n const handleConfirm = () => {\n if (canConfirm) {\n onConfirm();\n }\n };\n\n return (\n <Dialog\n open={open}\n onClose={deleting ? undefined : onCancel}\n maxWidth=\"sm\"\n fullWidth\n >\n <DialogTitle>\n {severity === 'high' && (\n <Box display=\"flex\" alignItems=\"center\" style={{ gap: 8 }}>\n <WarningIcon color=\"error\" />\n <span>{title}</span>\n </Box>\n )}\n {severity !== 'high' && title}\n </DialogTitle>\n <DialogContent>\n <DialogContentText>{description}</DialogContentText>\n {requiresTextConfirmation && (\n <Box mt={2}>\n <Typography variant=\"body2\" color=\"textSecondary\" gutterBottom>\n Type <strong>{confirmText}</strong> to confirm:\n </Typography>\n <TextField\n fullWidth\n variant=\"outlined\"\n size=\"small\"\n value={inputValue}\n onChange={e => setInputValue(e.target.value)}\n disabled={deleting}\n autoFocus\n placeholder={confirmText}\n />\n </Box>\n )}\n </DialogContent>\n <DialogActions>\n <Button onClick={onCancel} disabled={deleting}>\n Cancel\n </Button>\n <Button\n onClick={handleConfirm}\n color=\"secondary\"\n variant=\"contained\"\n disabled={deleting || !canConfirm}\n startIcon={deleting ? <CircularProgress size={16} color=\"inherit\" /> : undefined}\n >\n {deleting ? 'Deleting...' : 'Delete'}\n </Button>\n </DialogActions>\n </Dialog>\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: \"M1 21h22L12 2 1 21zm12-3h-2v-2h2v2zm0-4h-2v-4h2v4z\"\n}), 'Warning');\n\nexports.default = _default;","import React, { useState, useEffect } from 'react';\nimport {\n Dialog,\n DialogTitle,\n DialogContent,\n DialogActions,\n Button,\n TextField,\n Box,\n Typography,\n FormControl,\n InputLabel,\n Select,\n MenuItem,\n CircularProgress,\n} from '@material-ui/core';\nimport { useApi, configApiRef, fetchApiRef } from '@backstage/core-plugin-api';\nimport { APIKeyRequest } from '../../types/api-management';\n\ninterface EditAPIKeyRequestDialogProps {\n open: boolean;\n onClose: () => void;\n onSuccess: () => void;\n request: APIKeyRequest;\n availablePlans: Array<{\n tier: string;\n description?: string;\n limits?: any;\n }>;\n}\n\nexport const EditAPIKeyRequestDialog = ({\n open,\n onClose,\n onSuccess,\n request,\n availablePlans,\n}: EditAPIKeyRequestDialogProps) => {\n const config = useApi(configApiRef);\n const fetchApi = useApi(fetchApiRef);\n const backendUrl = config.getString('backend.baseUrl');\n\n const [planTier, setPlanTier] = useState('');\n const [useCase, setUseCase] = useState('');\n const [saving, setSaving] = useState(false);\n const [error, setError] = useState('');\n\n useEffect(() => {\n if (open && request) {\n setPlanTier(request.spec.planTier || '');\n setUseCase(request.spec.useCase || '');\n setError('');\n }\n }, [open, request]);\n\n const handleSave = async () => {\n if (!planTier) {\n setError('Please select a tier');\n return;\n }\n\n setError('');\n setSaving(true);\n\n try {\n const patch = {\n spec: {\n planTier,\n useCase: useCase.trim(),\n },\n };\n\n const response = await fetchApi.fetch(\n `${backendUrl}/api/kuadrant/requests/${request.metadata.namespace}/${request.metadata.name}`,\n {\n method: 'PATCH',\n headers: {\n 'Content-Type': 'application/json',\n },\n body: JSON.stringify(patch),\n }\n );\n\n if (!response.ok) {\n const errorData = await response.json().catch(() => ({}));\n throw new Error(errorData.error || `Failed to update request: ${response.status}`);\n }\n\n onSuccess();\n onClose();\n } catch (err) {\n console.error('Error updating API key request:', err);\n setError(err instanceof Error ? err.message : 'Unknown error occurred');\n } finally {\n setSaving(false);\n }\n };\n\n const handleClose = () => {\n if (!saving) {\n setError('');\n onClose();\n }\n };\n\n return (\n <Dialog open={open} onClose={handleClose} maxWidth=\"sm\" fullWidth>\n <DialogTitle>Edit API Access Request</DialogTitle>\n <DialogContent>\n {error && (\n <Box mb={2} p={2} bgcolor=\"error.main\" color=\"error.contrastText\" borderRadius={1}>\n <Typography variant=\"body2\">{error}</Typography>\n </Box>\n )}\n\n <FormControl fullWidth margin=\"normal\">\n <InputLabel>Tier</InputLabel>\n <Select\n value={planTier}\n onChange={(e) => setPlanTier(e.target.value as string)}\n disabled={saving}\n >\n {availablePlans.map((plan) => {\n const limitDesc = Object.entries(plan.limits || {})\n .map(([key, val]) => `${val} per ${key}`)\n .join(', ');\n return (\n <MenuItem key={plan.tier} value={plan.tier}>\n {plan.tier} {limitDesc ? `(${limitDesc})` : ''}\n </MenuItem>\n );\n })}\n </Select>\n </FormControl>\n\n <TextField\n label=\"Use Case\"\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 disabled={saving}\n helperText=\"Explain your intended use of this API for admin review\"\n />\n </DialogContent>\n <DialogActions>\n <Button onClick={handleClose} disabled={saving}>\n Cancel\n </Button>\n <Button\n onClick={handleSave}\n color=\"primary\"\n variant=\"contained\"\n disabled={!planTier || saving}\n startIcon={saving ? <CircularProgress size={16} color=\"inherit\" /> : undefined}\n >\n {saving ? 'Saving...' : 'Save Changes'}\n </Button>\n </DialogActions>\n </Dialog>\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: \"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","createPermission","name","attributes","action","kuadrantPlanPolicyListPermission","kuadrantApiProductCreatePermission","kuadrantApiProductReadAllPermission","kuadrantApiProductUpdateOwnPermission","kuadrantApiProductUpdateAllPermission","kuadrantApiProductDeleteOwnPermission","kuadrantApiProductDeleteAllPermission","kuadrantApiProductListPermission","kuadrantApiKeyRequestCreatePermission","resourceType","kuadrantApiKeyRequestReadOwnPermission","kuadrantApiKeyRequestReadAllPermission","kuadrantApiKeyRequestUpdateOwnPermission","kuadrantApiKeyRequestUpdateAllPermission","kuadrantApiKeyDeleteOwnPermission","kuadrantApiKeyDeleteAllPermission","useKuadrantPermission","permission","resourceRef","permissionRequest","result","usePermission","allowed","loading","error","canDeleteResource","ownerId","currentUserId","canDeleteOwn","canDeleteAll","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","apiProductName","metadata","annotations","useAsync","async","identity","getBackstageIdentity","profile","getProfileInfo","userEntityRef","email","requests","requestsLoading","requestsError","response","fetch","ok","json","items","filter","r","spec","apiName","apiNamespace","plansLoading","plansError","data","find","p","undefined","canCreateRequest","createRequestPermissionLoading","createRequestPermissionError","canDeleteOwnKey","deleteOwnPermissionLoading","deleteOwnPermissionError","canDeleteAllKeys","deleteAllPermissionLoading","deleteAllPermissionError","canUpdateRequest","updateRequestPermissionLoading","updateRequestPermissionError","handleMenuClose","handleMenuEdit","handleMenuDeleteClick","detailPanelConfig","useMemo","render","rowData","DetailPanelContent","Box","api","selectedLanguage","setSelectedLanguage","hostname","status","apiHostname","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","apiKey","language","showCopyCodeButton","permissionError","Progress","ResponseErrorPanel","failedPermission","color","message","myRequests","has","plans","pendingRequests","phase","approvedRequests","rejectedRequests","approvedColumns","title","field","row","Chip","size","reviewedAt","Date","toLocaleDateString","searchable","filtering","isVisible","display","alignItems","fontFamily","marginRight","IconButton","toggleVisibility","keyName","prev","newSet","delete","add","VisibilityOffIcon","VisibilityIcon","CircularProgress","requestedBy","rect","currentTarget","getBoundingClientRect","top","bottom","left","aria-controls","aria-haspopup","MoreVertIcon","requestColumns","isPending","icon","HourglassEmptyIcon","CancelIcon","Tooltip","placement","maxWidth","overflow","textOverflow","requestedAt","reason","canDelete","pendingRequestColumns","col","Grid","container","spacing","direction","item","flexDirection","Button","startIcon","AddIcon","disabled","length","marginTop","textAlign","Table","options","paging","pageSize","search","debounceInterval","toolbar","emptyRowsWhenPaging","columns","detailPanel","Dialog","onClose","fullWidth","DialogTitle","DialogContent","FormControl","margin","InputLabel","Select","target","map","plan","limitDesc","Object","entries","limits","key","val","join","MenuItem","tier","TextField","placeholder","multiline","rows","helperText","DialogActions","method","headers","body","JSON","stringify","trim","errorData","catch","post","severity","err","errorMessage","Menu","id","Boolean","anchorReference","anchorPosition","push","EditAPIKeyRequestDialog","onSuccess","availablePlans","ConfirmDeleteDialog","description","onConfirm","next","handleDeleteRequest","onCancel","confirmText","inputValue","setInputValue","useEffect","requiresTextConfirmation","canConfirm","gap","WarningIcon","span","DialogContentText","strong","autoFocus","setPlanTier","saving","setSaving","setError","handleClose","patch","console"],"sourceRoot":""}
|
|
@@ -0,0 +1,2 @@
|
|
|
1
|
+
"use strict";(self.webpackChunkinternal_plugin_kuadrant=self.webpackChunkinternal_plugin_kuadrant||[]).push([[6281],{16281:(e,t,a)=>{a.d(t,{d:()=>P});var r=a(31085),s=a(95478),n=a(22097),i=a(35015),o=a(42367),d=a(86687),l=a(25010),c=a(96040),p=a(34955),u=a(46205),m=a(76891),v=a(61477),h=a(46805),A=a(10394),y=a(72501),j=a(16249),g=a(93453),x=a(64947),b=a(78467),f=a(67720),q=a(71677),w=a(31653),k=a(38605),C=a(8973),$=a(29655),N=a(91567),R=a(55429),S=a(92399),I=a(6924);const B=({open:e,request:t,action:a,processing:n,onClose:i,onConfirm:o})=>{const[d,l]=(0,s.useState)("");(0,s.useEffect)(()=>{e||l("")},[e]);const c="approve"===a?"Approve":"Reject",p="approve"===a?"Approving...":"Rejecting...";return(0,r.jsxs)(m.A,{open:e,onClose:n?void 0:i,maxWidth:"sm",fullWidth:!0,children:[(0,r.jsxs)(v.A,{children:[c," API Key Request"]}),(0,r.jsx)(h.A,{children:t&&(0,r.jsxs)(r.Fragment,{children:[(0,r.jsxs)("p",{children:[(0,r.jsx)("strong",{children:"User:"})," ",t.spec.requestedBy.userId]}),(0,r.jsxs)("p",{children:[(0,r.jsx)("strong",{children:"API:"})," ",t.spec.apiName]}),(0,r.jsxs)("p",{children:[(0,r.jsx)("strong",{children:"Tier:"})," ",t.spec.planTier]}),(0,r.jsxs)(A.A,{mb:2,children:[(0,r.jsx)(y.A,{variant:"body2",component:"span",style:{fontWeight:"bold"},children:"Use Case:"})," ",(0,r.jsx)(y.A,{variant:"body2",component:"span",style:{whiteSpace:"pre-wrap"},children:t.spec.useCase||"-"})]}),(0,r.jsx)(j.A,{label:"Comment (optional)",multiline:!0,rows:3,fullWidth:!0,margin:"normal",value:d,onChange:e=>l(e.target.value),disabled:n})]})}),(0,r.jsxs)(g.A,{children:[(0,r.jsx)(x.A,{onClick:i,disabled:n,children:"Cancel"}),(0,r.jsx)(x.A,{onClick:()=>{o(d)},color:"approve"===a?"primary":"secondary",variant:"contained",disabled:n,startIcon:n?(0,r.jsx)(b.A,{size:16,color:"inherit"}):void 0,children:n?p:c})]})]})},T=({open:e,requests:t,action:a,processing:n,onClose:i,onConfirm:o})=>{const[d,l]=(0,s.useState)("");(0,s.useEffect)(()=>{e||l("")},[e]);const c="approve"===a,p=c?"Approve All":"Reject All",u=c?"Approving...":"Rejecting...";return(0,r.jsxs)(m.A,{open:e,onClose:n?void 0:i,maxWidth:"md",fullWidth:!0,children:[(0,r.jsxs)(v.A,{children:[c?"Approve":"Reject"," ",t.length," API Key Requests"]}),(0,r.jsxs)(h.A,{children:[(0,r.jsxs)(y.A,{variant:"body2",paragraph:!0,children:["You are about to ",c?"approve":"reject"," the following requests:"]}),(0,r.jsx)(A.A,{mb:2,maxHeight:200,overflow:"auto",children:t.map(e=>(0,r.jsx)(A.A,{mb:1,p:1,bgcolor:"background.default",children:(0,r.jsxs)(y.A,{variant:"body2",children:[(0,r.jsx)("strong",{children:e.spec.requestedBy.userId})," - ",e.spec.apiName," (",e.spec.planTier,")"]})},`${e.metadata.namespace}/${e.metadata.name}`))}),(0,r.jsx)(j.A,{label:"Comment (optional)",multiline:!0,fullWidth:!0,margin:"normal",value:d,onChange:e=>l(e.target.value),helperText:`This comment will be applied to all ${c?"approved":"rejected"} requests`,disabled:n})]}),(0,r.jsxs)(g.A,{children:[(0,r.jsx)(x.A,{onClick:i,disabled:n,children:"Cancel"}),(0,r.jsx)(x.A,{onClick:()=>{o(d)},color:c?"primary":"secondary",variant:"contained",disabled:n,startIcon:n?(0,r.jsx)(b.A,{size:16,color:"inherit"}):void 0,children:n?u:p})]})]})},P=()=>{const e=(0,n.useApi)(n.configApiRef),t=(0,n.useApi)(n.fetchApiRef),a=(0,n.useApi)(n.identityApiRef),m=(0,n.useApi)(n.alertApiRef),v=e.getString("backend.baseUrl"),[h,j]=(0,s.useState)(0),[g,b]=(0,s.useState)(0),[P,z]=(0,s.useState)([]),[W,U]=(0,s.useState)({open:!1,request:null,action:"approve",processing:!1}),[E,O]=(0,s.useState)({open:!1,requests:[],action:"approve",processing:!1}),{allowed:_,loading:Q,error:F}=(0,u.l)(p.ai),{allowed:J,loading:D,error:K}=(0,u.l)(p.vs),L=Q||D,M=F||K,{value:H,loading:G,error:X}=(0,i.A)(async()=>{const e=(await a.getBackstageIdentity()).userEntityRef;console.log("ApprovalQueueCard: fetching all requests from",`${v}/api/kuadrant/requests`);const[r,s]=await Promise.all([t.fetch(`${v}/api/kuadrant/requests`),t.fetch(`${v}/api/kuadrant/apiproducts`)]);if(!r.ok)return console.log("ApprovalQueueCard: failed to fetch requests, status:",r.status),{pending:[],approved:[],rejected:[],reviewedBy:e,ownedApiProducts:new Set};const n=r.headers.get("content-type");if(!n||!n.includes("application/json"))return console.log("ApprovalQueueCard: received non-json response"),{pending:[],approved:[],rejected:[],reviewedBy:e,ownedApiProducts:new Set};const i=(await r.json()).items||[],o=new Set;if(s.ok){const t=await s.json();for(const a of t.items||[]){var d,l;(null===(l=a.metadata)||void 0===l||null===(d=l.annotations)||void 0===d?void 0:d["backstage.io/owner"])===e&&o.add(`${a.metadata.namespace}/${a.metadata.name}`)}}console.log("ApprovalQueueCard: received",i.length,"total requests"),console.log("ApprovalQueueCard: user owns",o.size,"api products");const c=i.filter(e=>{var t;return"Pending"===((null===(t=e.status)||void 0===t?void 0:t.phase)||"Pending")}),p=i.filter(e=>{var t;return"Approved"===(null===(t=e.status)||void 0===t?void 0:t.phase)}),u=i.filter(e=>{var t;return"Rejected"===(null===(t=e.status)||void 0===t?void 0:t.phase)});return console.log("ApprovalQueueCard: grouped -",{pending:c.length,approved:p.length,rejected:u.length}),{pending:c,approved:p,rejected:u,reviewedBy:e,ownedApiProducts:o}},[v,t,a,h]);if(G||L)return(0,r.jsx)(d.k,{});if(X)return(0,r.jsx)(o._,{error:X});if(M)return(0,r.jsxs)(A.A,{p:2,children:[(0,r.jsxs)(y.A,{color:"error",children:["Unable to check permissions: ",M.message]}),(0,r.jsx)(y.A,{variant:"body2",color:"textSecondary",children:"Permission: kuadrant.apikeyrequest.update.all"}),(0,r.jsx)(y.A,{variant:"body2",color:"textSecondary",children:"Please try again or contact your administrator"})]});const Y=(null==H?void 0:H.pending)||[],V=(null==H?void 0:H.approved)||[],Z=(null==H?void 0:H.rejected)||[],ee=e=>new Date(e).toLocaleDateString("en-GB",{year:"numeric",month:"short",day:"numeric",hour:"2-digit",minute:"2-digit"}),te=[{title:"Request Name",field:"metadata.name",render:e=>(0,r.jsx)(y.A,{variant:"body2",children:e.metadata.name})},{title:"User",field:"spec.requestedBy.userId",render:e=>(0,r.jsx)(y.A,{variant:"body2",children:e.spec.requestedBy.userId})},{title:"API",field:"spec.apiName",render:e=>(0,r.jsx)(y.A,{variant:"body2",children:(0,r.jsx)("strong",{children:e.spec.apiName})})},{title:"Namespace",field:"spec.apiNamespace",render:e=>(0,r.jsx)(y.A,{variant:"body2",children:e.spec.apiNamespace})},{title:"Tier",field:"spec.planTier",render:e=>(0,r.jsx)(f.A,{label:e.spec.planTier,size:"small"})},{title:"Use Case",field:"spec.useCase",render:e=>e.spec.useCase?(0,r.jsx)(q.Ay,{title:e.spec.useCase,placement:"top",children:(0,r.jsx)(y.A,{variant:"body2",style:{maxWidth:"200px",overflow:"hidden",textOverflow:"ellipsis",whiteSpace:"nowrap"},children:e.spec.useCase})}):(0,r.jsx)(y.A,{variant:"body2",children:"-"})},{title:"Requested",field:"spec.requestedAt",render:e=>(0,r.jsx)(y.A,{variant:"body2",children:e.spec.requestedAt?ee(e.spec.requestedAt):"-"})},{title:"Actions",filtering:!1,render:e=>{var t;const a=`${e.spec.apiNamespace}/${e.spec.apiName}`;var s;const n=null!==(s=null==H||null===(t=H.ownedApiProducts)||void 0===t?void 0:t.has(a))&&void 0!==s&&s;return _||J&&n?(0,r.jsxs)(A.A,{display:"flex",style:{gap:8},children:[(0,r.jsx)(x.A,{size:"small",startIcon:(0,r.jsx)(R.A,{}),onClick:()=>{U({open:!0,request:e,action:"approve",processing:!1})},color:"primary",variant:"outlined",children:"Approve"}),(0,r.jsx)(x.A,{size:"small",startIcon:(0,r.jsx)(S.A,{}),onClick:()=>{U({open:!0,request:e,action:"reject",processing:!1})},color:"secondary",variant:"outlined",children:"Reject"})]}):null}}],ae=[{title:"Request Name",field:"metadata.name",render:e=>(0,r.jsx)(y.A,{variant:"body2",children:e.metadata.name})},{title:"User",field:"spec.requestedBy.userId",render:e=>(0,r.jsx)(y.A,{variant:"body2",children:e.spec.requestedBy.userId})},{title:"API",field:"spec.apiName",render:e=>(0,r.jsx)(y.A,{variant:"body2",children:(0,r.jsx)("strong",{children:e.spec.apiName})})},{title:"Namespace",field:"spec.apiNamespace",render:e=>(0,r.jsx)(y.A,{variant:"body2",children:e.spec.apiNamespace})},{title:"Tier",field:"spec.planTier",render:e=>(0,r.jsx)(f.A,{label:e.spec.planTier,size:"small"})},{title:"Requested",field:"spec.requestedAt",render:e=>(0,r.jsx)(y.A,{variant:"body2",children:e.spec.requestedAt?ee(e.spec.requestedAt):"-"})},{title:"Approved",field:"status.reviewedAt",render:e=>{var t;return(0,r.jsx)(y.A,{variant:"body2",children:(null===(t=e.status)||void 0===t?void 0:t.reviewedAt)?ee(e.status.reviewedAt):"-"})}},{title:"Reviewed By",field:"status.reviewedBy",render:e=>{var t;return(0,r.jsx)(y.A,{variant:"body2",children:(null===(t=e.status)||void 0===t?void 0:t.reviewedBy)||"-"})}},{title:"Approval Type",field:"status.reviewedBy",render:e=>{var t;const a="system"===(null===(t=e.status)||void 0===t?void 0:t.reviewedBy);return(0,r.jsx)(f.A,{label:a?"Automatic":"Manual",size:"small",color:a?"default":"primary"})}}],re=[{title:"Request Name",field:"metadata.name",render:e=>(0,r.jsx)(y.A,{variant:"body2",children:e.metadata.name})},{title:"User",field:"spec.requestedBy.userId",render:e=>(0,r.jsx)(y.A,{variant:"body2",children:e.spec.requestedBy.userId})},{title:"API",field:"spec.apiName",render:e=>(0,r.jsx)(y.A,{variant:"body2",children:(0,r.jsx)("strong",{children:e.spec.apiName})})},{title:"Namespace",field:"spec.apiNamespace",render:e=>(0,r.jsx)(y.A,{variant:"body2",children:e.spec.apiNamespace})},{title:"Tier",field:"spec.planTier",render:e=>(0,r.jsx)(f.A,{label:e.spec.planTier,size:"small"})},{title:"Requested",field:"spec.requestedAt",render:e=>(0,r.jsx)(y.A,{variant:"body2",children:e.spec.requestedAt?ee(e.spec.requestedAt):"-"})},{title:"Rejected",field:"status.reviewedAt",render:e=>{var t;return(0,r.jsx)(y.A,{variant:"body2",children:(null===(t=e.status)||void 0===t?void 0:t.reviewedAt)?ee(e.status.reviewedAt):"-"})}},{title:"Reviewed By",field:"status.reviewedBy",render:e=>{var t;return(0,r.jsx)(y.A,{variant:"body2",children:(null===(t=e.status)||void 0===t?void 0:t.reviewedBy)||"-"})}},{title:"Reason",field:"status.reason",render:e=>{var t;return(null===(t=e.status)||void 0===t?void 0:t.reason)?(0,r.jsx)(q.Ay,{title:e.status.reason,placement:"top",children:(0,r.jsx)(y.A,{variant:"body2",style:{maxWidth:"200px",overflow:"hidden",textOverflow:"ellipsis",whiteSpace:"nowrap"},children:e.status.reason})}):(0,r.jsx)(y.A,{variant:"body2",children:"-"})}}],se=(()=>{const e=e=>e.map(e=>({...e,id:e.metadata.name}));switch(g){case 0:default:return{data:e(V),columns:ae,showSelection:!1};case 1:return{data:Y.map(e=>{const t=P.some(t=>t.metadata.name===e.metadata.name&&t.metadata.namespace===e.metadata.namespace);return{...e,tableData:{checked:t}}}),columns:te,showSelection:!0};case 2:return{data:e(Z),columns:re,showSelection:!1}}})(),ne=(e=>{const t=new Map;return e.forEach(e=>{const a=`${e.spec.apiNamespace}/${e.spec.apiName}`;t.has(a)||t.set(a,[]),t.get(a).push(e)}),t})(se.data),ie=Array.from(ne.keys()).sort();return(0,r.jsxs)(r.Fragment,{children:[(0,r.jsxs)(c.n,{title:"API Access Requests",subheader:`${Y.length} pending, ${V.length} approved, ${Z.length} rejected`,children:[(0,r.jsx)(A.A,{mb:2,children:(0,r.jsxs)(w.A,{value:g,onChange:(e,t)=>{b(t),z([])},indicatorColor:"primary",textColor:"primary",children:[(0,r.jsx)(k.A,{label:`Approved (${V.length})`}),(0,r.jsx)(k.A,{label:`Pending (${Y.length})`}),(0,r.jsx)(k.A,{label:`Rejected (${Z.length})`})]})}),1===g&&P.length>0&&(0,r.jsxs)(A.A,{mb:2,display:"flex",alignItems:"center",justifyContent:"space-between",p:2,bgcolor:"background.default",children:[(0,r.jsxs)(y.A,{variant:"body2",children:[P.length," request",1!==P.length?"s":""," selected"]}),(0,r.jsxs)(A.A,{display:"flex",style:{gap:8},children:[(0,r.jsx)(x.A,{size:"small",variant:"contained",color:"primary",startIcon:(0,r.jsx)(R.A,{}),onClick:()=>{0!==P.length&&O({open:!0,requests:P,action:"approve",processing:!1})},children:"Approve Selected"}),(0,r.jsx)(x.A,{size:"small",variant:"contained",color:"secondary",startIcon:(0,r.jsx)(S.A,{}),onClick:()=>{0!==P.length&&O({open:!0,requests:P,action:"reject",processing:!1})},children:"Reject Selected"})]})]}),0===se.data.length?(0,r.jsx)(A.A,{p:3,textAlign:"center",children:(0,r.jsxs)(y.A,{variant:"body1",color:"textSecondary",children:[0===g&&"No approved requests.",1===g&&"No pending requests.",2===g&&"No rejected requests."]})}):(0,r.jsx)(A.A,{children:ie.map(e=>{var t,a;const s=ne.get(e)||[],n=(null===(t=s[0])||void 0===t?void 0:t.spec.apiName)||e;var i;const o=null!==(i=null==H||null===(a=H.ownedApiProducts)||void 0===a?void 0:a.has(e))&&void 0!==i&&i,d=_||J&&o;return(0,r.jsxs)(C.A,{defaultExpanded:1===ie.length,children:[(0,r.jsx)($.A,{expandIcon:(0,r.jsx)(I.A,{}),children:(0,r.jsxs)(A.A,{display:"flex",alignItems:"center",justifyContent:"space-between",width:"100%",children:[(0,r.jsx)(y.A,{variant:"h6",children:n}),(0,r.jsx)(f.A,{label:`${s.length} request${1!==s.length?"s":""}`,size:"small",color:"primary",style:{marginRight:16}})]})}),(0,r.jsx)(N.A,{children:(0,r.jsx)(A.A,{width:"100%",children:(0,r.jsx)(l.X,{options:{selection:d&&se.showSelection,paging:s.length>5,pageSize:20,search:!0,filtering:!0,debounceInterval:300,showTextRowsSelected:!1,toolbar:!0,emptyRowsWhenPaging:!1},data:s,columns:se.columns,onSelectionChange:t=>{const a=P.filter(t=>`${t.spec.apiNamespace}/${t.spec.apiName}`!==e);z([...a,...t])}})})})]},e)})})]}),(0,r.jsx)(B,{open:W.open,request:W.request,action:W.action,processing:W.processing,onClose:()=>U({open:!1,request:null,action:"approve",processing:!1}),onConfirm:async e=>{if(!W.request||!H)return;U(e=>({...e,processing:!0}));const a="approve"===W.action?`${v}/api/kuadrant/requests/${W.request.metadata.namespace}/${W.request.metadata.name}/approve`:`${v}/api/kuadrant/requests/${W.request.metadata.namespace}/${W.request.metadata.name}/reject`;try{if(!(await t.fetch(a,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({comment:e,reviewedBy:H.reviewedBy})})).ok)throw new Error(`failed to ${W.action} request`);U({open:!1,request:null,action:"approve",processing:!1}),j(e=>e+1);const r="approve"===W.action?"approved":"rejected";m.post({message:`Request ${r}`,severity:"success",display:"transient"})}catch(e){console.error(`error ${W.action}ing request:`,e),U(e=>({...e,processing:!1})),m.post({message:`Failed to ${W.action} request`,severity:"error",display:"transient"})}}}),(0,r.jsx)(T,{open:E.open,requests:E.requests,action:E.action,processing:E.processing,onClose:()=>O({open:!1,requests:[],action:"approve",processing:!1}),onConfirm:async e=>{if(!H||0===E.requests.length)return;O(e=>({...e,processing:!0}));const a="approve"===E.action,r=a?`${v}/api/kuadrant/requests/bulk-approve`:`${v}/api/kuadrant/requests/bulk-reject`;try{if(!(await t.fetch(r,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({requests:E.requests.map(e=>({namespace:e.metadata.namespace,name:e.metadata.name})),comment:e,reviewedBy:H.reviewedBy})})).ok)throw new Error(`failed to bulk ${E.action} requests`);const s=E.requests.length,n=a?"approved":"rejected";O({open:!1,requests:[],action:"approve",processing:!1}),z([]),j(e=>e+1),m.post({message:`${s} requests ${n}`,severity:"success",display:"transient"})}catch(e){console.error(`error bulk ${E.action}ing requests:`,e),O(e=>({...e,processing:!1})),m.post({message:`Failed to bulk ${E.action} requests`,severity:"error",display:"transient"})}}})]})}},34955:(e,t,a)=>{a.d(t,{Al:()=>i,EM:()=>l,FL:()=>n,J:()=>s,KN:()=>u,OP:()=>p,R_:()=>c,SU:()=>m,TE:()=>v,U3:()=>o,ai:()=>A,jH:()=>j,uL:()=>y,v_:()=>d,vs:()=>h});var r=a(83572);(0,r.i)({name:"kuadrant.planpolicy.create",attributes:{action:"create"}}),(0,r.i)({name:"kuadrant.planpolicy.read",attributes:{action:"read"}}),(0,r.i)({name:"kuadrant.planpolicy.update",attributes:{action:"update"}}),(0,r.i)({name:"kuadrant.planpolicy.delete",attributes:{action:"delete"}});const s=(0,r.i)({name:"kuadrant.planpolicy.list",attributes:{action:"read"}}),n=(0,r.i)({name:"kuadrant.apiproduct.create",attributes:{action:"create"}}),i=((0,r.i)({name:"kuadrant.apiproduct.read.own",attributes:{action:"read"}}),(0,r.i)({name:"kuadrant.apiproduct.read.all",attributes:{action:"read"}})),o=(0,r.i)({name:"kuadrant.apiproduct.update.own",attributes:{action:"update"}}),d=(0,r.i)({name:"kuadrant.apiproduct.update.all",attributes:{action:"update"}}),l=(0,r.i)({name:"kuadrant.apiproduct.delete.own",attributes:{action:"delete"}}),c=(0,r.i)({name:"kuadrant.apiproduct.delete.all",attributes:{action:"delete"}}),p=(0,r.i)({name:"kuadrant.apiproduct.list",attributes:{action:"read"}}),u=(0,r.i)({name:"kuadrant.apikeyrequest.create",attributes:{action:"create"},resourceType:"apiproduct"}),m=(0,r.i)({name:"kuadrant.apikeyrequest.read.own",attributes:{action:"read"}}),v=(0,r.i)({name:"kuadrant.apikeyrequest.read.all",attributes:{action:"read"}}),h=(0,r.i)({name:"kuadrant.apikeyrequest.update.own",attributes:{action:"update"}}),A=(0,r.i)({name:"kuadrant.apikeyrequest.update.all",attributes:{action:"update"}}),y=((0,r.i)({name:"kuadrant.apikeyrequest.delete.own",attributes:{action:"delete"}}),(0,r.i)({name:"kuadrant.apikeyrequest.delete.all",attributes:{action:"delete"}}),(0,r.i)({name:"kuadrant.apikeyrequest.list",attributes:{action:"read"}}),(0,r.i)({name:"kuadrant.apikey.read.own",attributes:{action:"read"}}),(0,r.i)({name:"kuadrant.apikey.read.all",attributes:{action:"read"}}),(0,r.i)({name:"kuadrant.apikey.delete.own",attributes:{action:"delete"}})),j=(0,r.i)({name:"kuadrant.apikey.delete.all",attributes:{action:"delete"}})},46205:(e,t,a)=>{a.d(t,{W:()=>n,l:()=>s});var r=a(87421);function s(e,t){const a="resourceType"in e?{permission:e,resourceRef:t}:{permission:e},s=(0,r.J)(a);return{allowed:s.allowed,loading:s.loading,error:s.error}}function n(e,t,a,r){return!!r||!(!a||e!==t)}}}]);
|
|
2
|
+
//# sourceMappingURL=6281.b000c79f.chunk.js.map
|