@kuadrant/kuadrant-backstage-plugin-frontend 0.0.2-dev-25e5ccb → 0.0.2-dev-087fcd3
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/dist/components/ApiKeyManagementTab/ApiKeyManagementTab.esm.js +89 -36
- package/dist/components/ApiKeyManagementTab/ApiKeyManagementTab.esm.js.map +1 -1
- package/dist/components/ApiProductInfoCard/ApiProductInfoCard.esm.js +4 -3
- package/dist/components/ApiProductInfoCard/ApiProductInfoCard.esm.js.map +1 -1
- package/dist/components/ApprovalQueueCard/ApprovalQueueCard.esm.js +6 -6
- package/dist/components/ApprovalQueueCard/ApprovalQueueCard.esm.js.map +1 -1
- package/dist/components/MyApiKeysCard/MyApiKeysCard.esm.js +83 -10
- package/dist/components/MyApiKeysCard/MyApiKeysCard.esm.js.map +1 -1
- package/dist/components/PlanPolicyDetailsCard/PlanPolicyDetails.esm.js +1 -1
- package/dist/components/PlanPolicyDetailsCard/PlanPolicyDetails.esm.js.map +1 -1
- package/dist-scalprum/{internal.plugin-kuadrant.4475e90c8a94cfeda29d.js → internal.plugin-kuadrant.c93747b8ad44bbc1d661.js} +2 -2
- package/dist-scalprum/{internal.plugin-kuadrant.4475e90c8a94cfeda29d.js.map → internal.plugin-kuadrant.c93747b8ad44bbc1d661.js.map} +1 -1
- package/dist-scalprum/plugin-manifest.json +2 -2
- package/dist-scalprum/static/4306.a3a2e598.chunk.js +2 -0
- package/dist-scalprum/static/4306.a3a2e598.chunk.js.map +1 -0
- package/dist-scalprum/static/532.0e09e787.chunk.js +2 -0
- package/dist-scalprum/static/532.0e09e787.chunk.js.map +1 -0
- package/dist-scalprum/static/{5453.bd86702b.chunk.js → 5453.bad88dd0.chunk.js} +2 -2
- package/dist-scalprum/static/5453.bad88dd0.chunk.js.map +1 -0
- package/dist-scalprum/static/{6281.66e6f0ff.chunk.js → 6281.650b1d6c.chunk.js} +2 -2
- package/dist-scalprum/static/6281.650b1d6c.chunk.js.map +1 -0
- package/package.json +1 -1
- package/dist-scalprum/static/4306.a6a669e4.chunk.js +0 -2
- package/dist-scalprum/static/4306.a6a669e4.chunk.js.map +0 -1
- package/dist-scalprum/static/532.bf600078.chunk.js +0 -2
- package/dist-scalprum/static/532.bf600078.chunk.js.map +0 -1
- package/dist-scalprum/static/5453.bd86702b.chunk.js.map +0 -1
- package/dist-scalprum/static/6281.66e6f0ff.chunk.js.map +0 -1
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"static/6281.66e6f0ff.chunk.js","mappings":"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","sources":["webpack://internal.plugin-kuadrant/./src/components/ApprovalQueueCard/ApprovalQueueCard.tsx","webpack://internal.plugin-kuadrant/./src/permissions.ts","webpack://internal.plugin-kuadrant/./src/utils/permissions.ts"],"sourcesContent":["import React, { useState } from 'react';\nimport { useApi, fetchApiRef, identityApiRef, configApiRef, alertApiRef } from '@backstage/core-plugin-api';\nimport { useAsync } from 'react-use';\nimport {\n Table,\n TableColumn,\n Progress,\n ResponseErrorPanel,\n InfoCard,\n} from '@backstage/core-components';\nimport { kuadrantApiKeyRequestUpdateAllPermission, kuadrantApiKeyRequestUpdateOwnPermission } from '../../permissions';\nimport { useKuadrantPermission } from '../../utils/permissions';\nimport {\n Button,\n Dialog,\n DialogTitle,\n DialogContent,\n DialogActions,\n Chip,\n Typography,\n Box,\n Tabs,\n Tab,\n Tooltip,\n Accordion,\n AccordionSummary,\n AccordionDetails,\n CircularProgress,\n} from '@material-ui/core';\nimport CheckCircleIcon from '@material-ui/icons/CheckCircle';\nimport CancelIcon from '@material-ui/icons/Cancel';\nimport ExpandMoreIcon from '@material-ui/icons/ExpandMore';\nimport { APIKey } from '../../types/api-management';\n\ninterface ApprovalDialogProps {\n open: boolean;\n request: APIKey | null;\n action: 'approve' | 'reject';\n processing: boolean;\n onClose: () => void;\n onConfirm: () => void;\n}\n\nconst ApprovalDialog = ({ open, request, action, processing, onClose, onConfirm }: ApprovalDialogProps) => {\n const actionLabel = action === 'approve' ? 'Approve' : 'Reject';\n const processingLabel = action === 'approve' ? 'Approving...' : 'Rejecting...';\n\n return (\n <Dialog open={open} onClose={processing ? undefined : onClose} maxWidth=\"sm\" fullWidth>\n <DialogTitle>\n {actionLabel} API Key Request\n </DialogTitle>\n <DialogContent>\n {request && (\n <>\n <p><strong>User:</strong> {request.spec.requestedBy.userId}</p>\n <p><strong>API:</strong> {request.spec.apiProductRef?.name || 'unknown'}</p>\n <p><strong>Tier:</strong> {request.spec.planTier}</p>\n <Box mb={2}>\n <Typography variant=\"body2\" component=\"span\" style={{ fontWeight: 'bold' }}>\n Use Case:\n </Typography>\n {' '}\n <Typography variant=\"body2\" component=\"span\" style={{ whiteSpace: 'pre-wrap' }}>\n {request.spec.useCase || '-'}\n </Typography>\n </Box>\n </>\n )}\n </DialogContent>\n <DialogActions>\n <Button onClick={onClose} disabled={processing}>Cancel</Button>\n <Button\n onClick={onConfirm}\n color={action === 'approve' ? 'primary' : 'secondary'}\n variant=\"contained\"\n disabled={processing}\n startIcon={processing ? <CircularProgress size={16} color=\"inherit\" /> : undefined}\n >\n {processing ? processingLabel : actionLabel}\n </Button>\n </DialogActions>\n </Dialog>\n );\n};\n\ninterface BulkActionDialogProps {\n open: boolean;\n requests: APIKey[];\n action: 'approve' | 'reject';\n processing: boolean;\n onClose: () => void;\n onConfirm: () => void;\n}\n\nconst BulkActionDialog = ({ open, requests, action, processing, onClose, onConfirm }: BulkActionDialogProps) => {\n const isApprove = action === 'approve';\n const actionLabel = isApprove ? 'Approve All' : 'Reject All';\n const processingLabel = isApprove ? 'Approving...' : 'Rejecting...';\n\n return (\n <Dialog open={open} onClose={processing ? undefined : onClose} maxWidth=\"md\" fullWidth>\n <DialogTitle>\n {isApprove ? 'Approve' : 'Reject'} {requests.length} API Key Requests\n </DialogTitle>\n <DialogContent>\n <Typography variant=\"body2\" paragraph>\n You are about to {isApprove ? 'approve' : 'reject'} the following requests:\n </Typography>\n <Box mb={2} maxHeight={200} overflow=\"auto\">\n {requests.map(request => (\n <Box key={`${request.metadata.namespace}/${request.metadata.name}`} mb={1} p={1} bgcolor=\"background.default\">\n <Typography variant=\"body2\">\n <strong>{request.spec.requestedBy.userId}</strong> - {request.spec.apiProductRef?.name || 'unknown'} ({request.spec.planTier})\n </Typography>\n </Box>\n ))}\n </Box>\n </DialogContent>\n <DialogActions>\n <Button onClick={onClose} disabled={processing}>Cancel</Button>\n <Button\n onClick={onConfirm}\n color={isApprove ? 'primary' : 'secondary'}\n variant=\"contained\"\n disabled={processing}\n startIcon={processing ? <CircularProgress size={16} color=\"inherit\" /> : undefined}\n >\n {processing ? processingLabel : actionLabel}\n </Button>\n </DialogActions>\n </Dialog>\n );\n};\n\nexport const ApprovalQueueCard = () => {\n const config = useApi(configApiRef);\n const fetchApi = useApi(fetchApiRef);\n const identityApi = useApi(identityApiRef);\n const alertApi = useApi(alertApiRef);\n const backendUrl = config.getString('backend.baseUrl');\n const [refresh, setRefresh] = useState(0);\n const [selectedTab, setSelectedTab] = useState(0);\n const [selectedRequests, setSelectedRequests] = useState<APIKey[]>([]);\n const [dialogState, setDialogState] = useState<{\n open: boolean;\n request: APIKey | null;\n action: 'approve' | 'reject';\n processing: boolean;\n }>({\n open: false,\n request: null,\n action: 'approve',\n processing: false,\n });\n const [bulkDialogState, setBulkDialogState] = useState<{\n open: boolean;\n requests: APIKey[];\n action: 'approve' | 'reject';\n processing: boolean;\n }>({\n open: false,\n requests: [],\n action: 'approve',\n processing: false,\n });\n\n const {\n allowed: canUpdateAllRequests,\n loading: updateAllPermissionLoading,\n error: updateAllPermissionError,\n } = useKuadrantPermission(kuadrantApiKeyRequestUpdateAllPermission);\n\n const {\n allowed: canUpdateOwnRequests,\n loading: updateOwnPermissionLoading,\n error: updateOwnPermissionError,\n } = useKuadrantPermission(kuadrantApiKeyRequestUpdateOwnPermission);\n\n const updatePermissionLoading = updateAllPermissionLoading || updateOwnPermissionLoading;\n const updatePermissionError = updateAllPermissionError || updateOwnPermissionError;\n\n const { value, loading, error } = useAsync(async () => {\n const identity = await identityApi.getBackstageIdentity();\n const reviewedBy = identity.userEntityRef;\n\n console.log('ApprovalQueueCard: fetching all requests from', `${backendUrl}/api/kuadrant/requests`);\n\n // fetch requests and api products in parallel\n const [requestsResponse, apiProductsResponse] = await Promise.all([\n fetchApi.fetch(`${backendUrl}/api/kuadrant/requests`),\n fetchApi.fetch(`${backendUrl}/api/kuadrant/apiproducts`),\n ]);\n\n if (!requestsResponse.ok) {\n console.log('ApprovalQueueCard: failed to fetch requests, status:', requestsResponse.status);\n return { pending: [] as APIKey[], approved: [] as APIKey[], rejected: [] as APIKey[], reviewedBy, ownedApiProducts: new Set<string>() };\n }\n\n // check content-type before parsing json\n const contentType = requestsResponse.headers.get('content-type');\n if (!contentType?.includes('application/json')) {\n console.error('ApprovalQueueCard: received non-json response');\n alertApi.post({\n message: 'Unexpected content-type from the server response. Please contact support.',\n display: 'transient',\n severity: 'warning'\n });\n return { pending: [] as APIKey[], approved: [] as APIKey[], rejected: [] as APIKey[], reviewedBy, ownedApiProducts: new Set<string>() };\n\n }\n\n const data = await requestsResponse.json();\n const allRequests = data.items || [];\n\n // build set of api products owned by current user\n const ownedApiProducts = new Set<string>();\n if (apiProductsResponse.ok) {\n const apiProductsData = await apiProductsResponse.json();\n for (const product of apiProductsData.items || []) {\n const owner = product.metadata?.annotations?.['backstage.io/owner'];\n if (owner === reviewedBy) {\n // key is namespace/name to match against request's namespace/apiProductRef.name\n ownedApiProducts.add(`${product.metadata.namespace}/${product.metadata.name}`);\n }\n }\n }\n\n console.log('ApprovalQueueCard: received', allRequests.length, 'total requests');\n console.log('ApprovalQueueCard: user owns', ownedApiProducts.size, 'api products');\n\n // group by status (field is 'phase' not 'status')\n const pending = allRequests.filter((r: APIKey) => {\n const phase = (r.status as any)?.phase || 'Pending';\n return phase === 'Pending';\n });\n const approved = allRequests.filter((r: APIKey) => {\n const phase = (r.status as any)?.phase;\n return phase === 'Approved';\n });\n const rejected = allRequests.filter((r: APIKey) => {\n const phase = (r.status as any)?.phase;\n return phase === 'Rejected';\n });\n\n console.log('ApprovalQueueCard: grouped -', {\n pending: pending.length,\n approved: approved.length,\n rejected: rejected.length,\n });\n\n return { pending, approved, rejected, reviewedBy, ownedApiProducts };\n }, [backendUrl, fetchApi, identityApi, refresh]);\n\n const handleApprove = (request: APIKey) => {\n setDialogState({ open: true, request, action: 'approve', processing: false });\n };\n\n const handleReject = (request: APIKey) => {\n setDialogState({ open: true, request, action: 'reject', processing: false });\n };\n\n const handleConfirm = async () => {\n if (!dialogState.request || !value) return;\n\n setDialogState(prev => ({ ...prev, processing: true }));\n\n const endpoint = dialogState.action === 'approve'\n ? `${backendUrl}/api/kuadrant/requests/${dialogState.request.metadata.namespace}/${dialogState.request.metadata.name}/approve`\n : `${backendUrl}/api/kuadrant/requests/${dialogState.request.metadata.namespace}/${dialogState.request.metadata.name}/reject`;\n\n try {\n const response = await fetchApi.fetch(endpoint, {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body: JSON.stringify({\n reviewedBy: value.reviewedBy,\n }),\n });\n\n if (!response.ok) {\n throw new Error(`failed to ${dialogState.action} request`);\n }\n\n setDialogState({ open: false, request: null, action: 'approve', processing: false });\n setRefresh(r => r + 1);\n const action = dialogState.action === 'approve' ? 'approved' : 'rejected';\n alertApi.post({ message: `Request ${action}`, severity: 'success', display: 'transient' });\n } catch (err) {\n console.error(`error ${dialogState.action}ing request:`, err);\n setDialogState(prev => ({ ...prev, processing: false }));\n alertApi.post({ message: `Failed to ${dialogState.action} request`, severity: 'error', display: 'transient' });\n }\n };\n\n const handleBulkApprove = () => {\n if (selectedRequests.length === 0) return;\n setBulkDialogState({ open: true, requests: selectedRequests, action: 'approve', processing: false });\n };\n\n const handleBulkReject = () => {\n if (selectedRequests.length === 0) return;\n setBulkDialogState({ open: true, requests: selectedRequests, action: 'reject', processing: false });\n };\n\n const handleBulkConfirm = async () => {\n if (!value || bulkDialogState.requests.length === 0) return;\n\n setBulkDialogState(prev => ({ ...prev, processing: true }));\n\n const isApprove = bulkDialogState.action === 'approve';\n const endpoint = isApprove\n ? `${backendUrl}/api/kuadrant/requests/bulk-approve`\n : `${backendUrl}/api/kuadrant/requests/bulk-reject`;\n\n try {\n const response = await fetchApi.fetch(endpoint, {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body: JSON.stringify({\n requests: bulkDialogState.requests.map(r => ({\n namespace: r.metadata.namespace,\n name: r.metadata.name,\n })),\n reviewedBy: value.reviewedBy,\n }),\n });\n\n if (!response.ok) {\n throw new Error(`failed to bulk ${bulkDialogState.action} requests`);\n }\n\n const count = bulkDialogState.requests.length;\n const action = isApprove ? 'approved' : 'rejected';\n setBulkDialogState({ open: false, requests: [], action: 'approve', processing: false });\n setSelectedRequests([]);\n setRefresh(r => r + 1);\n alertApi.post({ message: `${count} requests ${action}`, severity: 'success', display: 'transient' });\n } catch (err) {\n console.error(`error bulk ${bulkDialogState.action}ing requests:`, err);\n setBulkDialogState(prev => ({ ...prev, processing: false }));\n alertApi.post({ message: `Failed to bulk ${bulkDialogState.action} requests`, severity: 'error', display: 'transient' });\n }\n };\n\n if (loading || updatePermissionLoading) {\n return <Progress />;\n }\n\n if (error) {\n return <ResponseErrorPanel error={error} />;\n }\n\n if (updatePermissionError) {\n return (\n <Box p={2}>\n <Typography color=\"error\">\n Unable to check permissions: {updatePermissionError.message}\n </Typography>\n <Typography variant=\"body2\" color=\"textSecondary\">\n Permission: kuadrant.apikeyrequest.update.all\n </Typography>\n <Typography variant=\"body2\" color=\"textSecondary\">\n Please try again or contact your administrator\n </Typography>\n </Box>\n );\n }\n\n const pending = value?.pending || [];\n const approved = value?.approved || [];\n const rejected = value?.rejected || [];\n\n const formatDate = (dateString: string) => {\n const date = new Date(dateString);\n return date.toLocaleDateString('en-GB', {\n year: 'numeric',\n month: 'short',\n day: 'numeric',\n hour: '2-digit',\n minute: '2-digit',\n });\n };\n\n const pendingColumns: TableColumn<APIKey>[] = [\n {\n title: 'Request Name',\n field: 'metadata.name',\n render: (row) => <Typography variant=\"body2\">{row.metadata.name}</Typography>,\n },\n {\n title: 'User',\n field: 'spec.requestedBy.userId',\n render: (row) => <Typography variant=\"body2\">{row.spec.requestedBy.userId}</Typography>,\n },\n {\n title: 'API',\n field: 'spec.apiProductRef.name',\n render: (row) => <Typography variant=\"body2\"><strong>{row.spec.apiProductRef?.name || 'unknown'}</strong></Typography>,\n },\n {\n title: 'Namespace',\n field: 'metadata.namespace',\n render: (row) => <Typography variant=\"body2\">{row.metadata.namespace}</Typography>,\n },\n {\n title: 'Tier',\n field: 'spec.planTier',\n render: (row) => (\n <Chip\n label={row.spec.planTier}\n size=\"small\"\n />\n ),\n },\n {\n title: 'Use Case',\n field: 'spec.useCase',\n render: (row) => {\n if (!row.spec.useCase) {\n return <Typography variant=\"body2\">-</Typography>;\n }\n return (\n <Tooltip title={row.spec.useCase} placement=\"top\">\n <Typography\n variant=\"body2\"\n style={{\n maxWidth: '200px',\n overflow: 'hidden',\n textOverflow: 'ellipsis',\n whiteSpace: 'nowrap',\n }}\n >\n {row.spec.useCase}\n </Typography>\n </Tooltip>\n );\n },\n },\n {\n title: 'Requested',\n field: 'metadata.creationTimestamp',\n render: (row) => (\n <Typography variant=\"body2\">\n {row.metadata.creationTimestamp ? formatDate(row.metadata.creationTimestamp) : '-'}\n </Typography>\n ),\n },\n {\n title: 'Actions',\n filtering: false,\n render: (row) => {\n const apiProductKey = `${row.metadata.namespace}/${row.spec.apiProductRef?.name || 'unknown'}`;\n const ownsApiProduct = value?.ownedApiProducts?.has(apiProductKey) ?? false;\n const canUpdate = canUpdateAllRequests || (canUpdateOwnRequests && ownsApiProduct);\n if (!canUpdate) return null;\n return (\n <Box display=\"flex\" style={{ gap: 8 }}>\n <Button\n size=\"small\"\n startIcon={<CheckCircleIcon />}\n onClick={() => handleApprove(row)}\n color=\"primary\"\n variant=\"outlined\"\n >\n Approve\n </Button>\n <Button\n size=\"small\"\n startIcon={<CancelIcon />}\n onClick={() => handleReject(row)}\n color=\"secondary\"\n variant=\"outlined\"\n >\n Reject\n </Button>\n </Box>\n );\n },\n },\n ];\n\n const approvedColumns: TableColumn<APIKey>[] = [\n {\n title: 'Request Name',\n field: 'metadata.name',\n render: (row) => <Typography variant=\"body2\">{row.metadata.name}</Typography>,\n },\n {\n title: 'User',\n field: 'spec.requestedBy.userId',\n render: (row) => <Typography variant=\"body2\">{row.spec.requestedBy.userId}</Typography>,\n },\n {\n title: 'API',\n field: 'spec.apiProductRef.name',\n render: (row) => <Typography variant=\"body2\"><strong>{row.spec.apiProductRef?.name || 'unknown'}</strong></Typography>,\n },\n {\n title: 'Namespace',\n field: 'metadata.namespace',\n render: (row) => <Typography variant=\"body2\">{row.metadata.namespace}</Typography>,\n },\n {\n title: 'Tier',\n field: 'spec.planTier',\n render: (row) => (\n <Chip\n label={row.spec.planTier}\n size=\"small\"\n />\n ),\n },\n {\n title: 'Requested',\n field: 'metadata.creationTimestamp',\n render: (row) => (\n <Typography variant=\"body2\">\n {row.metadata.creationTimestamp ? formatDate(row.metadata.creationTimestamp) : '-'}\n </Typography>\n ),\n },\n {\n title: 'Approved',\n field: 'status.reviewedAt',\n render: (row) => (\n <Typography variant=\"body2\">\n {row.status?.reviewedAt ? formatDate(row.status.reviewedAt) : '-'}\n </Typography>\n ),\n },\n {\n title: 'Reviewed By',\n field: 'status.reviewedBy',\n render: (row) => (\n <Typography variant=\"body2\">\n {row.status?.reviewedBy || '-'}\n </Typography>\n ),\n },\n {\n title: 'Approval Type',\n field: 'status.reviewedBy',\n render: (row) => {\n const isAutomatic = row.status?.reviewedBy === 'system';\n return (\n <Chip\n label={isAutomatic ? 'Automatic' : 'Manual'}\n size=\"small\"\n color={isAutomatic ? 'default' : 'primary'}\n />\n );\n },\n },\n ];\n\n const rejectedColumns: TableColumn<APIKey>[] = [\n {\n title: 'Request Name',\n field: 'metadata.name',\n render: (row) => <Typography variant=\"body2\">{row.metadata.name}</Typography>,\n },\n {\n title: 'User',\n field: 'spec.requestedBy.userId',\n render: (row) => <Typography variant=\"body2\">{row.spec.requestedBy.userId}</Typography>,\n },\n {\n title: 'API',\n field: 'spec.apiProductRef.name',\n render: (row) => <Typography variant=\"body2\"><strong>{row.spec.apiProductRef?.name || 'unknown'}</strong></Typography>,\n },\n {\n title: 'Namespace',\n field: 'metadata.namespace',\n render: (row) => <Typography variant=\"body2\">{row.metadata.namespace}</Typography>,\n },\n {\n title: 'Tier',\n field: 'spec.planTier',\n render: (row) => (\n <Chip\n label={row.spec.planTier}\n size=\"small\"\n />\n ),\n },\n {\n title: 'Requested',\n field: 'metadata.creationTimestamp',\n render: (row) => (\n <Typography variant=\"body2\">\n {row.metadata.creationTimestamp ? formatDate(row.metadata.creationTimestamp) : '-'}\n </Typography>\n ),\n },\n {\n title: 'Rejected',\n field: 'status.reviewedAt',\n render: (row) => (\n <Typography variant=\"body2\">\n {row.status?.reviewedAt ? formatDate(row.status.reviewedAt) : '-'}\n </Typography>\n ),\n },\n {\n title: 'Reviewed By',\n field: 'status.reviewedBy',\n render: (row) => (\n <Typography variant=\"body2\">\n {row.status?.reviewedBy || '-'}\n </Typography>\n ),\n },\n ];\n\n const getTabData = () => {\n const addIds = (data: APIKey[]) =>\n data.map(item => ({ ...item, id: item.metadata.name }));\n\n switch (selectedTab) {\n case 0:\n return { data: addIds(approved), columns: approvedColumns, showSelection: false };\n case 1:\n // Add tableData.checked to control checkbox state\n const pendingWithSelection = pending.map((row: APIKey) => {\n const isSelected = selectedRequests.some(\n selected => selected.metadata.name === row.metadata.name &&\n selected.metadata.namespace === row.metadata.namespace\n );\n return {\n ...row,\n tableData: { checked: isSelected },\n };\n });\n return { data: pendingWithSelection, columns: pendingColumns, showSelection: true };\n case 2:\n return { data: addIds(rejected), columns: rejectedColumns, showSelection: false };\n default:\n return { data: addIds(approved), columns: approvedColumns, showSelection: false };\n }\n };\n\n const tabData = getTabData();\n\n // group requests by api product (namespace/name)\n const groupByApiProduct = (requests: APIKey[]) => {\n const grouped = new Map<string, APIKey[]>();\n requests.forEach(request => {\n const key = `${request.metadata.namespace}/${request.spec.apiProductRef?.name || 'unknown'}`;\n if (!grouped.has(key)) {\n grouped.set(key, []);\n }\n grouped.get(key)!.push(request);\n });\n return grouped;\n };\n\n const groupedData = groupByApiProduct(tabData.data);\n const apiProductKeys = Array.from(groupedData.keys()).sort();\n\n return (\n <>\n <InfoCard\n title=\"API Access Requests\"\n subheader={`${pending.length} pending, ${approved.length} approved, ${rejected.length} rejected`}\n >\n <Box mb={2}>\n <Tabs\n value={selectedTab}\n onChange={(_, newValue) => {\n setSelectedTab(newValue);\n setSelectedRequests([]);\n }}\n indicatorColor=\"primary\"\n textColor=\"primary\"\n >\n <Tab label={`Approved (${approved.length})`} />\n <Tab label={`Pending (${pending.length})`} />\n <Tab label={`Rejected (${rejected.length})`} />\n </Tabs>\n </Box>\n\n {selectedTab === 1 && selectedRequests.length > 0 && (\n <Box mb={2} display=\"flex\" alignItems=\"center\" justifyContent=\"space-between\" p={2} bgcolor=\"background.default\">\n <Typography variant=\"body2\">\n {selectedRequests.length} request{selectedRequests.length !== 1 ? 's' : ''} selected\n </Typography>\n <Box display=\"flex\" style={{ gap: 8 }}>\n <Button\n size=\"small\"\n variant=\"contained\"\n color=\"primary\"\n startIcon={<CheckCircleIcon />}\n onClick={handleBulkApprove}\n >\n Approve Selected\n </Button>\n <Button\n size=\"small\"\n variant=\"contained\"\n color=\"secondary\"\n startIcon={<CancelIcon />}\n onClick={handleBulkReject}\n >\n Reject Selected\n </Button>\n </Box>\n </Box>\n )}\n\n {tabData.data.length === 0 ? (\n <Box p={3} textAlign=\"center\">\n <Typography variant=\"body1\" color=\"textSecondary\">\n {selectedTab === 0 && 'No approved requests.'}\n {selectedTab === 1 && 'No pending requests.'}\n {selectedTab === 2 && 'No rejected requests.'}\n </Typography>\n </Box>\n ) : (\n <Box>\n {apiProductKeys.map(apiProductKey => {\n const requests = groupedData.get(apiProductKey) || [];\n const displayName = requests[0]?.spec.apiProductRef?.name || apiProductKey;\n const ownsThisApiProduct = value?.ownedApiProducts?.has(apiProductKey) ?? false;\n const canSelectRows = canUpdateAllRequests || (canUpdateOwnRequests && ownsThisApiProduct);\n return (\n <Accordion key={apiProductKey} defaultExpanded={apiProductKeys.length === 1}>\n <AccordionSummary expandIcon={<ExpandMoreIcon />}>\n <Box display=\"flex\" alignItems=\"center\" justifyContent=\"space-between\" width=\"100%\">\n <Typography variant=\"h6\">{displayName}</Typography>\n <Chip\n label={`${requests.length} request${requests.length !== 1 ? 's' : ''}`}\n size=\"small\"\n color=\"primary\"\n style={{ marginRight: 16 }}\n />\n </Box>\n </AccordionSummary>\n <AccordionDetails>\n <Box width=\"100%\">\n <Table\n options={{\n selection: canSelectRows && tabData.showSelection,\n paging: requests.length > 5,\n pageSize: 20,\n search: true,\n filtering: true,\n debounceInterval: 300,\n showTextRowsSelected: false,\n toolbar: true,\n emptyRowsWhenPaging: false,\n }}\n data={requests}\n columns={tabData.columns}\n onSelectionChange={(rows) => {\n // merge selections from this api product with selections from other products\n const otherSelections = selectedRequests.filter(\n r => `${r.metadata.namespace}/${r.spec.apiProductRef?.name || 'unknown'}` !== apiProductKey\n );\n setSelectedRequests([...otherSelections, ...(rows as APIKey[])]);\n }}\n />\n </Box>\n </AccordionDetails>\n </Accordion>\n );\n })}\n </Box>\n )}\n </InfoCard>\n <ApprovalDialog\n open={dialogState.open}\n request={dialogState.request}\n action={dialogState.action}\n processing={dialogState.processing}\n onClose={() => setDialogState({ open: false, request: null, action: 'approve', processing: false })}\n onConfirm={handleConfirm}\n />\n <BulkActionDialog\n open={bulkDialogState.open}\n requests={bulkDialogState.requests}\n action={bulkDialogState.action}\n processing={bulkDialogState.processing}\n onClose={() => setBulkDialogState({ open: false, requests: [], action: 'approve', processing: false })}\n onConfirm={handleBulkConfirm}\n />\n </>\n );\n};\n","import { createPermission } from '@backstage/plugin-permission-common';\n\n/**\n * permission definitions for the kuadrant plugin\n *\n * these permissions control access to kuadrant resources and operations.\n * they must match the permissions defined in the backend plugin.\n *\n * permission types:\n * - BasicPermission: standard permission that applies globally\n * - ResourcePermission: permission scoped to specific resource types (e.g., apiproduct)\n *\n * permission patterns:\n * - `.create` - create new resources\n * - `.read` - read resource details\n * - `.read.own` - read only resources owned by the user\n * - `.read.all` - read all resources regardless of ownership\n * - `.update` - modify existing resources\n * - `.delete` - delete resources\n * - `.delete.own` - delete only resources owned by the user\n * - `.delete.all` - delete any resource regardless of ownership\n * - `.list` - list/view collections of resources\n */\n\n// planpolicy permissions\nexport const kuadrantPlanPolicyCreatePermission = createPermission({\n name: 'kuadrant.planpolicy.create',\n attributes: { action: 'create' },\n});\n\nexport const kuadrantPlanPolicyReadPermission = createPermission({\n name: 'kuadrant.planpolicy.read',\n attributes: { action: 'read' },\n});\n\nexport const kuadrantPlanPolicyUpdatePermission = createPermission({\n name: 'kuadrant.planpolicy.update',\n attributes: { action: 'update' },\n});\n\nexport const kuadrantPlanPolicyDeletePermission = createPermission({\n name: 'kuadrant.planpolicy.delete',\n attributes: { action: 'delete' },\n});\n\nexport const kuadrantPlanPolicyListPermission = createPermission({\n name: 'kuadrant.planpolicy.list',\n attributes: { action: 'read' },\n});\n\n// apiproduct permissions\n\n/**\n * permission to create new API products\n * granted to api owners and admins\n */\nexport const kuadrantApiProductCreatePermission = createPermission({\n name: 'kuadrant.apiproduct.create',\n attributes: { action: 'create' },\n});\n\n/**\n * permission to read API products owned by the current user\n * for api owners to view their own products\n */\nexport const kuadrantApiProductReadOwnPermission = createPermission({\n name: 'kuadrant.apiproduct.read.own',\n attributes: { action: 'read' },\n});\n\n/**\n * permission to read all API products regardless of ownership\n * for platform engineers/admins who need to view all products\n */\nexport const kuadrantApiProductReadAllPermission = createPermission({\n name: 'kuadrant.apiproduct.read.all',\n attributes: { action: 'read' },\n});\n\n/**\n * permission to update API products owned by the current user\n * for api owners to modify their own products\n */\nexport const kuadrantApiProductUpdateOwnPermission = createPermission({\n name: 'kuadrant.apiproduct.update.own',\n attributes: { action: 'update' },\n});\n\n/**\n * permission to update any API product regardless of ownership\n * for platform engineers/admins\n */\nexport const kuadrantApiProductUpdateAllPermission = createPermission({\n name: 'kuadrant.apiproduct.update.all',\n attributes: { action: 'update' },\n});\n\n/**\n * permission to delete API products owned by the current user\n * for api owners to remove their own products\n */\nexport const kuadrantApiProductDeleteOwnPermission = createPermission({\n name: 'kuadrant.apiproduct.delete.own',\n attributes: { action: 'delete' },\n});\n\n/**\n * permission to delete any API product regardless of ownership\n * for platform engineers/admins\n */\nexport const kuadrantApiProductDeleteAllPermission = createPermission({\n name: 'kuadrant.apiproduct.delete.all',\n attributes: { action: 'delete' },\n});\n\n/**\n * permission to list API products\n * backend filters results based on .own vs .all read permissions\n */\nexport const kuadrantApiProductListPermission = createPermission({\n name: 'kuadrant.apiproduct.list',\n attributes: { action: 'read' },\n});\n\n// apikey request 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(kuadrantAPIKeyCreatePermission)\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"],"names":["ApprovalDialog","open","request","action","processing","onClose","onConfirm","actionLabel","processingLabel","Dialog","undefined","maxWidth","fullWidth","DialogTitle","DialogContent","p","strong","spec","requestedBy","userId","apiProductRef","name","planTier","Box","mb","Typography","variant","component","style","fontWeight","whiteSpace","useCase","DialogActions","Button","onClick","disabled","color","startIcon","CircularProgress","size","BulkActionDialog","requests","isApprove","length","paragraph","maxHeight","overflow","map","bgcolor","metadata","namespace","ApprovalQueueCard","config","useApi","configApiRef","fetchApi","fetchApiRef","identityApi","identityApiRef","alertApi","alertApiRef","backendUrl","getString","refresh","setRefresh","useState","selectedTab","setSelectedTab","selectedRequests","setSelectedRequests","dialogState","setDialogState","bulkDialogState","setBulkDialogState","allowed","canUpdateAllRequests","loading","updateAllPermissionLoading","error","updateAllPermissionError","useKuadrantPermission","kuadrantApiKeyRequestUpdateAllPermission","canUpdateOwnRequests","updateOwnPermissionLoading","updateOwnPermissionError","kuadrantApiKeyRequestUpdateOwnPermission","updatePermissionLoading","updatePermissionError","value","useAsync","async","reviewedBy","getBackstageIdentity","userEntityRef","console","log","requestsResponse","apiProductsResponse","Promise","all","fetch","ok","status","pending","approved","rejected","ownedApiProducts","Set","contentType","headers","get","includes","post","message","display","severity","allRequests","json","items","apiProductsData","product","annotations","add","filter","r","phase","Progress","ResponseErrorPanel","formatDate","dateString","Date","toLocaleDateString","year","month","day","hour","minute","pendingColumns","title","field","render","row","Chip","label","Tooltip","placement","textOverflow","creationTimestamp","filtering","apiProductKey","ownsApiProduct","has","gap","CheckCircleIcon","CancelIcon","approvedColumns","reviewedAt","isAutomatic","rejectedColumns","tabData","addIds","data","item","id","columns","showSelection","isSelected","some","selected","tableData","checked","getTabData","groupedData","grouped","Map","forEach","key","set","push","groupByApiProduct","apiProductKeys","Array","from","keys","sort","InfoCard","subheader","Tabs","onChange","_","newValue","indicatorColor","textColor","Tab","alignItems","justifyContent","textAlign","displayName","ownsThisApiProduct","canSelectRows","Accordion","defaultExpanded","AccordionSummary","expandIcon","ExpandMoreIcon","width","marginRight","AccordionDetails","Table","options","selection","paging","pageSize","search","debounceInterval","showTextRowsSelected","toolbar","emptyRowsWhenPaging","onSelectionChange","rows","otherSelections","prev","endpoint","method","body","JSON","stringify","Error","err","count","createPermission","attributes","kuadrantPlanPolicyListPermission","kuadrantApiProductCreatePermission","kuadrantApiProductReadAllPermission","kuadrantApiProductUpdateOwnPermission","kuadrantApiProductUpdateAllPermission","kuadrantApiProductDeleteOwnPermission","kuadrantApiProductDeleteAllPermission","kuadrantApiProductListPermission","kuadrantApiKeyRequestCreatePermission","resourceType","kuadrantApiKeyRequestReadOwnPermission","kuadrantApiKeyRequestReadAllPermission","kuadrantApiKeyDeleteOwnPermission","kuadrantApiKeyDeleteAllPermission","permission","resourceRef","permissionRequest","result","usePermission","canDeleteResource","ownerId","currentUserId","canDeleteOwn","canDeleteAll"],"sourceRoot":""}
|