@kuadrant/kuadrant-backstage-plugin-frontend 0.0.2-dev-8f2e157 → 0.0.2-dev-844db4d

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (24) hide show
  1. package/dist/components/ApiKeyManagementTab/ApiKeyManagementTab.esm.js +45 -10
  2. package/dist/components/ApiKeyManagementTab/ApiKeyManagementTab.esm.js.map +1 -1
  3. package/dist/components/ApprovalQueueCard/ApprovalQueueCard.esm.js +8 -3
  4. package/dist/components/ApprovalQueueCard/ApprovalQueueCard.esm.js.map +1 -1
  5. package/dist/components/KuadrantPage/KuadrantPage.esm.js +24 -3
  6. package/dist/components/KuadrantPage/KuadrantPage.esm.js.map +1 -1
  7. package/dist/components/MyApiKeysCard/MyApiKeysCard.esm.js +12 -4
  8. package/dist/components/MyApiKeysCard/MyApiKeysCard.esm.js.map +1 -1
  9. package/dist-scalprum/{internal.plugin-kuadrant.86c50e4da6b961c623e9.js → internal.plugin-kuadrant.58f9452dd6d088d4816c.js} +2 -2
  10. package/dist-scalprum/{internal.plugin-kuadrant.86c50e4da6b961c623e9.js.map → internal.plugin-kuadrant.58f9452dd6d088d4816c.js.map} +1 -1
  11. package/dist-scalprum/plugin-manifest.json +2 -2
  12. package/dist-scalprum/static/4306.bc289e67.chunk.js +2 -0
  13. package/dist-scalprum/static/4306.bc289e67.chunk.js.map +1 -0
  14. package/dist-scalprum/static/6281.e7e36f6a.chunk.js +2 -0
  15. package/dist-scalprum/static/6281.e7e36f6a.chunk.js.map +1 -0
  16. package/dist-scalprum/static/7632.4bb9bd69.chunk.js +2 -0
  17. package/dist-scalprum/static/7632.4bb9bd69.chunk.js.map +1 -0
  18. package/package.json +1 -1
  19. package/dist-scalprum/static/4306.04dfa125.chunk.js +0 -2
  20. package/dist-scalprum/static/4306.04dfa125.chunk.js.map +0 -1
  21. package/dist-scalprum/static/6281.14f57b13.chunk.js +0 -2
  22. package/dist-scalprum/static/6281.14f57b13.chunk.js.map +0 -1
  23. package/dist-scalprum/static/7632.646001ce.chunk.js +0 -2
  24. package/dist-scalprum/static/7632.646001ce.chunk.js.map +0 -1
@@ -1 +0,0 @@
1
- {"version":3,"file":"static/6281.14f57b13.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, useEffect } from 'react';\nimport { useApi, fetchApiRef, identityApiRef, configApiRef } 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 TextField,\n Chip,\n Typography,\n Box,\n Tabs,\n Tab,\n Tooltip,\n Accordion,\n AccordionSummary,\n AccordionDetails,\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 { APIKeyRequest } from '../../types/api-management';\n\ninterface ApprovalDialogProps {\n open: boolean;\n request: APIKeyRequest | null;\n action: 'approve' | 'reject';\n onClose: () => void;\n onConfirm: (comment: string) => void;\n}\n\nconst ApprovalDialog = ({ open, request, action, onClose, onConfirm }: ApprovalDialogProps) => {\n const [comment, setComment] = useState('');\n\n // reset comment when dialog closes\n useEffect(() => {\n if (!open) {\n setComment('');\n }\n }, [open]);\n\n const handleConfirm = () => {\n onConfirm(comment);\n onClose();\n };\n\n return (\n <Dialog open={open} onClose={onClose} maxWidth=\"sm\" fullWidth>\n <DialogTitle>\n {action === 'approve' ? 'Approve' : 'Reject'} 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.apiName}</p>\n <p><strong>Plan:</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 <TextField\n label=\"Comment (optional)\"\n multiline\n rows={3}\n fullWidth\n margin=\"normal\"\n value={comment}\n onChange={(e) => setComment(e.target.value)}\n />\n </>\n )}\n </DialogContent>\n <DialogActions>\n <Button onClick={onClose}>Cancel</Button>\n <Button\n onClick={handleConfirm}\n color={action === 'approve' ? 'primary' : 'secondary'}\n variant=\"contained\"\n >\n {action === 'approve' ? 'Approve' : 'Reject'}\n </Button>\n </DialogActions>\n </Dialog>\n );\n};\n\ninterface BulkActionDialogProps {\n open: boolean;\n requests: APIKeyRequest[];\n action: 'approve' | 'reject';\n onClose: () => void;\n onConfirm: (comment: string) => void;\n}\n\nconst BulkActionDialog = ({ open, requests, action, onClose, onConfirm }: BulkActionDialogProps) => {\n const [comment, setComment] = useState('');\n\n // reset comment when dialog closes\n useEffect(() => {\n if (!open) {\n setComment('');\n }\n }, [open]);\n\n const handleConfirm = () => {\n onConfirm(comment);\n onClose();\n };\n\n const isApprove = action === 'approve';\n\n return (\n <Dialog open={open} onClose={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.apiName} ({request.spec.planTier})\n </Typography>\n </Box>\n ))}\n </Box>\n <TextField\n label=\"Comment (optional)\"\n multiline\n fullWidth\n margin=\"normal\"\n value={comment}\n onChange={(e) => setComment(e.target.value)}\n helperText={`This comment will be applied to all ${isApprove ? 'approved' : 'rejected'} requests`}\n />\n </DialogContent>\n <DialogActions>\n <Button onClick={onClose}>Cancel</Button>\n <Button\n onClick={handleConfirm}\n color={isApprove ? 'primary' : 'secondary'}\n variant=\"contained\"\n >\n {isApprove ? 'Approve All' : 'Reject All'}\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 backendUrl = config.getString('backend.baseUrl');\n const [refresh, setRefresh] = useState(0);\n const [selectedTab, setSelectedTab] = useState(0);\n const [selectedRequests, setSelectedRequests] = useState<APIKeyRequest[]>([]);\n const [dialogState, setDialogState] = useState<{\n open: boolean;\n request: APIKeyRequest | null;\n action: 'approve' | 'reject';\n }>({\n open: false,\n request: null,\n action: 'approve',\n });\n const [bulkDialogState, setBulkDialogState] = useState<{\n open: boolean;\n requests: APIKeyRequest[];\n action: 'approve' | 'reject';\n }>({\n open: false,\n requests: [],\n action: 'approve',\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 canUpdateRequests = canUpdateAllRequests || canUpdateOwnRequests;\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 const response = await fetchApi.fetch(\n `${backendUrl}/api/kuadrant/requests`\n );\n if (!response.ok) {\n console.log('ApprovalQueueCard: failed to fetch requests, status:', response.status);\n return { pending: [] as APIKeyRequest[], approved: [] as APIKeyRequest[], rejected: [] as APIKeyRequest[], reviewedBy };\n }\n\n // check content-type before parsing json\n const contentType = response.headers.get('content-type');\n if (!contentType || !contentType.includes('application/json')) {\n console.log('ApprovalQueueCard: received non-json response');\n return { pending: [] as APIKeyRequest[], approved: [] as APIKeyRequest[], rejected: [] as APIKeyRequest[], reviewedBy };\n }\n\n const data = await response.json();\n const allRequests = data.items || [];\n\n console.log('ApprovalQueueCard: received', allRequests.length, 'total requests');\n console.log('ApprovalQueueCard: raw requests:', allRequests);\n\n // group by status (field is 'phase' not 'status')\n const pending = allRequests.filter((r: APIKeyRequest) => {\n const phase = (r.status as any)?.phase || 'Pending';\n return phase === 'Pending';\n });\n const approved = allRequests.filter((r: APIKeyRequest) => {\n const phase = (r.status as any)?.phase;\n return phase === 'Approved';\n });\n const rejected = allRequests.filter((r: APIKeyRequest) => {\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 };\n }, [backendUrl, fetchApi, identityApi, refresh]);\n\n const handleApprove = (request: APIKeyRequest) => {\n setDialogState({ open: true, request, action: 'approve' });\n };\n\n const handleReject = (request: APIKeyRequest) => {\n setDialogState({ open: true, request, action: 'reject' });\n };\n\n const handleConfirm = async (comment: string) => {\n if (!dialogState.request || !value) return;\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 comment,\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' });\n setRefresh(r => r + 1);\n } catch (err) {\n console.error(`error ${dialogState.action}ing request:`, err);\n }\n };\n\n const handleBulkApprove = () => {\n if (selectedRequests.length === 0) return;\n setBulkDialogState({ open: true, requests: selectedRequests, action: 'approve' });\n };\n\n const handleBulkReject = () => {\n if (selectedRequests.length === 0) return;\n setBulkDialogState({ open: true, requests: selectedRequests, action: 'reject' });\n };\n\n const handleBulkConfirm = async (comment: string) => {\n if (!value || bulkDialogState.requests.length === 0) return;\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 comment,\n reviewedBy: value.reviewedBy,\n }),\n });\n\n if (!response.ok) {\n throw new Error(`failed to bulk ${bulkDialogState.action} requests`);\n }\n\n setBulkDialogState({ open: false, requests: [], action: 'approve' });\n setSelectedRequests([]);\n setRefresh(r => r + 1);\n } catch (err) {\n console.error(`error bulk ${bulkDialogState.action}ing requests:`, err);\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<APIKeyRequest>[] = [\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.apiName',\n render: (row) => <Typography variant=\"body2\"><strong>{row.spec.apiName}</strong></Typography>,\n },\n {\n title: 'Namespace',\n field: 'spec.apiNamespace',\n render: (row) => <Typography variant=\"body2\">{row.spec.apiNamespace}</Typography>,\n },\n {\n title: 'Plan',\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: 'spec.requestedAt',\n render: (row) => (\n <Typography variant=\"body2\">\n {row.spec.requestedAt ? formatDate(row.spec.requestedAt) : '-'}\n </Typography>\n ),\n },\n {\n title: 'Actions',\n render: (row) => {\n if (!canUpdateRequests) 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<APIKeyRequest>[] = [\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.apiName',\n render: (row) => <Typography variant=\"body2\"><strong>{row.spec.apiName}</strong></Typography>,\n },\n {\n title: 'Namespace',\n field: 'spec.apiNamespace',\n render: (row) => <Typography variant=\"body2\">{row.spec.apiNamespace}</Typography>,\n },\n {\n title: 'Plan',\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: 'spec.requestedAt',\n render: (row) => (\n <Typography variant=\"body2\">\n {row.spec.requestedAt ? formatDate(row.spec.requestedAt) : '-'}\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<APIKeyRequest>[] = [\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.apiName',\n render: (row) => <Typography variant=\"body2\"><strong>{row.spec.apiName}</strong></Typography>,\n },\n {\n title: 'Namespace',\n field: 'spec.apiNamespace',\n render: (row) => <Typography variant=\"body2\">{row.spec.apiNamespace}</Typography>,\n },\n {\n title: 'Plan',\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: 'spec.requestedAt',\n render: (row) => (\n <Typography variant=\"body2\">\n {row.spec.requestedAt ? formatDate(row.spec.requestedAt) : '-'}\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 title: 'Reason',\n field: 'status.reason',\n render: (row) => {\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\n const getTabData = () => {\n const addIds = (data: APIKeyRequest[]) =>\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: APIKeyRequest) => {\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\n const groupByApiProduct = (requests: APIKeyRequest[]) => {\n const grouped = new Map<string, APIKeyRequest[]>();\n requests.forEach(request => {\n const apiName = request.spec.apiName;\n if (!grouped.has(apiName)) {\n grouped.set(apiName, []);\n }\n grouped.get(apiName)!.push(request);\n });\n return grouped;\n };\n\n const groupedData = groupByApiProduct(tabData.data);\n const apiProducts = 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 {canUpdateRequests && 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 {apiProducts.map(apiName => {\n const requests = groupedData.get(apiName) || [];\n return (\n <Accordion key={apiName} defaultExpanded={apiProducts.length === 1}>\n <AccordionSummary expandIcon={<ExpandMoreIcon />}>\n <Box display=\"flex\" alignItems=\"center\" justifyContent=\"space-between\" width=\"100%\">\n <Typography variant=\"h6\">{apiName}</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: canUpdateRequests && tabData.showSelection,\n paging: false,\n search: false,\n showTextRowsSelected: false,\n toolbar: 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.spec.apiName !== apiName\n );\n setSelectedRequests([...otherSelections, ...(rows as APIKeyRequest[])]);\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 onClose={() => setDialogState({ open: false, request: null, action: 'approve' })}\n onConfirm={handleConfirm}\n />\n <BulkActionDialog\n open={bulkDialogState.open}\n requests={bulkDialogState.requests}\n action={bulkDialogState.action}\n onClose={() => setBulkDialogState({ open: false, requests: [], action: 'approve' })}\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// 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"],"names":["ApprovalDialog","open","request","action","onClose","onConfirm","comment","setComment","useState","useEffect","Dialog","maxWidth","fullWidth","DialogTitle","DialogContent","p","strong","spec","requestedBy","userId","apiName","planTier","Box","mb","Typography","variant","component","style","fontWeight","whiteSpace","useCase","TextField","label","multiline","rows","margin","value","onChange","e","target","DialogActions","Button","onClick","color","BulkActionDialog","requests","isApprove","length","paragraph","maxHeight","overflow","map","bgcolor","metadata","namespace","name","helperText","ApprovalQueueCard","config","useApi","configApiRef","fetchApi","fetchApiRef","identityApi","identityApiRef","backendUrl","getString","refresh","setRefresh","selectedTab","setSelectedTab","selectedRequests","setSelectedRequests","dialogState","setDialogState","bulkDialogState","setBulkDialogState","allowed","canUpdateAllRequests","loading","updateAllPermissionLoading","error","updateAllPermissionError","useKuadrantPermission","kuadrantApiKeyRequestUpdateAllPermission","canUpdateOwnRequests","updateOwnPermissionLoading","updateOwnPermissionError","kuadrantApiKeyRequestUpdateOwnPermission","canUpdateRequests","updatePermissionLoading","updatePermissionError","useAsync","async","reviewedBy","getBackstageIdentity","userEntityRef","console","log","response","fetch","ok","status","pending","approved","rejected","contentType","headers","get","includes","allRequests","json","items","filter","r","phase","Progress","ResponseErrorPanel","message","formatDate","dateString","Date","toLocaleDateString","year","month","day","hour","minute","pendingColumns","title","field","render","row","apiNamespace","Chip","size","Tooltip","placement","textOverflow","requestedAt","display","gap","startIcon","CheckCircleIcon","CancelIcon","approvedColumns","reviewedAt","isAutomatic","rejectedColumns","reason","tabData","addIds","data","item","id","columns","showSelection","isSelected","some","selected","tableData","checked","getTabData","groupedData","grouped","Map","forEach","has","set","push","groupByApiProduct","apiProducts","Array","from","keys","sort","InfoCard","subheader","Tabs","_","newValue","indicatorColor","textColor","Tab","alignItems","justifyContent","textAlign","Accordion","defaultExpanded","AccordionSummary","expandIcon","ExpandMoreIcon","width","marginRight","AccordionDetails","Table","options","selection","paging","search","showTextRowsSelected","toolbar","onSelectionChange","otherSelections","endpoint","method","body","JSON","stringify","Error","err","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":""}
@@ -1,2 +0,0 @@
1
- "use strict";(self.webpackChunkinternal_plugin_kuadrant=self.webpackChunkinternal_plugin_kuadrant||[]).push([[7632],{84189:(e,a,t)=>{t.d(a,{n:()=>g});var l=t(31085),r=t(95478),s=t(76891),n=t(61477),i=t(46805),o=t(10394),d=t(72501),c=t(95061),u=t(48543),p=t(81215),m=t(26343),h=t(16249),x=t(93453),A=t(64947),v=t(22097);const g=({open:e,onClose:a,onSuccess:t,request:g,availablePlans:j})=>{const f=(0,v.useApi)(v.configApiRef),y=(0,v.useApi)(v.fetchApiRef),b=f.getString("backend.baseUrl"),[P,S]=(0,r.useState)(""),[k,w]=(0,r.useState)(""),[C,T]=(0,r.useState)(!1),[R,I]=(0,r.useState)("");(0,r.useEffect)(()=>{e&&g&&(S(g.spec.planTier||""),w(g.spec.useCase||""),I(""))},[e,g]);const $=()=>{C||(I(""),a())};return(0,l.jsxs)(s.A,{open:e,onClose:$,maxWidth:"sm",fullWidth:!0,children:[(0,l.jsx)(n.A,{children:"Edit API Access Request"}),(0,l.jsxs)(i.A,{children:[R&&(0,l.jsx)(o.A,{mb:2,p:2,bgcolor:"error.main",color:"error.contrastText",borderRadius:1,children:(0,l.jsx)(d.A,{variant:"body2",children:R})}),(0,l.jsxs)(c.A,{fullWidth:!0,margin:"normal",children:[(0,l.jsx)(u.A,{children:"Plan Tier"}),(0,l.jsx)(p.A,{value:P,onChange:e=>S(e.target.value),disabled:C,children:j.map(e=>{const a=Object.entries(e.limits||{}).map(([e,a])=>`${a} per ${e}`).join(", ");return(0,l.jsxs)(m.A,{value:e.tier,children:[e.tier," ",a?`(${a})`:""]},e.tier)})})]}),(0,l.jsx)(h.A,{label:"Use Case",placeholder:"Describe how you plan to use this API",multiline:!0,rows:3,fullWidth:!0,margin:"normal",value:k,onChange:e=>w(e.target.value),disabled:C,helperText:"Explain your intended use of this API for admin review"})]}),(0,l.jsxs)(x.A,{children:[(0,l.jsx)(A.A,{onClick:$,disabled:C,children:"Cancel"}),(0,l.jsx)(A.A,{onClick:async()=>{if(P){I(""),T(!0);try{const e={spec:{planTier:P,useCase:k.trim()}},l=await y.fetch(`${b}/api/kuadrant/requests/${g.metadata.namespace}/${g.metadata.name}`,{method:"PATCH",headers:{"Content-Type":"application/json"},body:JSON.stringify(e)});if(!l.ok){const e=await l.json().catch(()=>({}));throw new Error(e.error||`Failed to update request: ${l.status}`)}t(),a()}catch(e){console.error("Error updating API key request:",e),I(e instanceof Error?e.message:"Unknown error occurred")}finally{T(!1)}}else I("Please select a plan tier")},color:"primary",disabled:!P||C,children:C?"Saving...":"Save Changes"})]})]})}},95251:(e,a,t)=>{t.r(a),t.d(a,{KuadrantPage:()=>ee});var l=t(31085),r=t(95478),s=t.n(r),n=t(67720),i=t(10394),o=t(29365),d=t(72501),c=t(42899),u=t(64947),p=t(76891),m=t(61477),h=t(46805),x=t(59461),A=t(93453),v=t(18466),g=t(39590),j=t(75625),f=t(37725),y=t(25010),b=t(289),P=t(55639),S=t(45210),k=t(46681),w=t(86687),C=t(42367),T=t(96040),R=t(91638),I=t(22097),$=t(16281),W=t(71677),E=t(31653),q=t(38605),N=t(37757),D=t(26343),M=t(32269),z=t(61524),L=t(71407),U=t(84189);const H=()=>{const e=(0,I.useApi)(I.configApiRef),a=(0,I.useApi)(I.fetchApiRef),t=(0,I.useApi)(I.identityApiRef),s=e.getString("backend.baseUrl"),[c,u]=(0,r.useState)(0),[,p]=(0,r.useState)(""),[m,h]=(0,r.useState)(new Set),[x,A]=(0,r.useState)(null),[v,g]=(0,r.useState)(null),[j,b]=(0,r.useState)({open:!1,request:null,plans:[]}),[P,S]=(0,r.useState)(0);(0,R.A)(async()=>{const e=await t.getBackstageIdentity(),a=e.userEntityRef.split("/")[1]||"guest";console.log(`MyApiKeysCard: setting userId from userEntityRef: ${e.userEntityRef} -> "${a}"`),p(a)},[t]);const{value:k,loading:w,error:C}=(0,R.A)(async()=>{const e=await a.fetch(`${s}/api/kuadrant/requests/my`);if(!e.ok)throw new Error("failed to fetch requests");return(await e.json()).items||[]},[s,a,P]);if(w)return(0,l.jsx)(T.n,{title:"My API Keys",children:(0,l.jsx)(d.A,{children:"Loading..."})});if(C)return(0,l.jsx)(T.n,{title:"My API Keys",children:(0,l.jsxs)(d.A,{color:"error",children:["Error loading API keys: ",C.message]})});const $=k||[],H=$.filter(e=>{var a;return"Approved"===(null===(a=e.status)||void 0===a?void 0:a.phase)}),K=$.filter(e=>{var a;return!(null===(a=e.status)||void 0===a?void 0:a.phase)||"Pending"===e.status.phase}),B=$.filter(e=>{var a;return"Rejected"===(null===(a=e.status)||void 0===a?void 0:a.phase)}),F=()=>{A(null),g(null)},O=async()=>{if(!v)return;const e=v;F();try{const l=await a.fetch(`${s}/api/kuadrant/apiproducts/${e.spec.apiNamespace}/${e.spec.apiName}`);if(l.ok){var t;const a=(null===(t=(await l.json()).spec)||void 0===t?void 0:t.plans)||[];b({open:!0,request:e,plans:a})}else console.error("Failed to fetch API product"),b({open:!0,request:e,plans:[]})}catch(a){console.error("Error fetching plans:",a),b({open:!0,request:e,plans:[]})}},_=async()=>{if(!v)return;const e=v;if(F(),window.confirm("Are you sure you want to delete this request?"))try{if(!(await a.fetch(`${s}/api/kuadrant/requests/${e.metadata.namespace}/${e.metadata.name}`,{method:"DELETE"})).ok)throw new Error("Failed to delete request");S(e=>e+1)}catch(e){console.error("Error deleting request:",e),alert("Failed to delete request")}},V=[{title:"API Product",field:"spec.apiName",render:e=>(0,l.jsx)(f.N_,{to:`/catalog/default/api/${e.spec.apiName}/api-keys`,children:(0,l.jsx)("strong",{children:e.spec.apiName})})},{title:"Plan",field:"spec.planTier",render:e=>{const a="gold"===e.spec.planTier?"primary":"silver"===e.spec.planTier?"default":"secondary";return(0,l.jsx)(n.A,{label:e.spec.planTier,color:a,size:"small"})}},{title:"Use Case",field:"spec.useCase",render:e=>e.spec.useCase?(0,l.jsx)(W.Ay,{title:e.spec.useCase,placement:"top",children:(0,l.jsx)(d.A,{variant:"body2",style:{maxWidth:"200px",overflow:"hidden",textOverflow:"ellipsis",whiteSpace:"nowrap"},children:e.spec.useCase})}):(0,l.jsx)(d.A,{variant:"body2",children:"-"})},{title:"Status",field:"status.phase",render:e=>{var a;const t=(null===(a=e.status)||void 0===a?void 0:a.phase)||"Pending",r="Approved"===t?"primary":"Rejected"===t?"secondary":"default";return(0,l.jsx)(n.A,{label:t,color:r,size:"small"})}},{title:"Reason",render:e=>{var a;if(null===(a=e.status)||void 0===a?void 0:a.reason){const a="Rejected"===e.status.phase?"error":"textPrimary";return(0,l.jsx)(W.Ay,{title:e.status.reason,placement:"top",children:(0,l.jsx)(d.A,{variant:"body2",color:a,style:{maxWidth:"200px",overflow:"hidden",textOverflow:"ellipsis",whiteSpace:"nowrap"},children:e.status.reason})})}return(0,l.jsx)(d.A,{variant:"body2",color:"textSecondary",children:"-"})}},{title:"Reviewed By",render:e=>{var a,t;if(("Approved"===(null===(a=e.status)||void 0===a?void 0:a.phase)||"Rejected"===(null===(t=e.status)||void 0===t?void 0:t.phase))&&e.status.reviewedBy){const a=e.status.reviewedAt?new Date(e.status.reviewedAt).toLocaleDateString():"";return(0,l.jsxs)(i.A,{children:[(0,l.jsx)(d.A,{variant:"body2",children:e.status.reviewedBy}),a&&(0,l.jsx)(d.A,{variant:"caption",color:"textSecondary",children:a})]})}return(0,l.jsx)(d.A,{variant:"body2",color:"textSecondary",children:"-"})}},{title:"API Key",render:e=>{var a;if("Approved"===(null===(a=e.status)||void 0===a?void 0:a.phase)&&e.status.apiKey){const a=m.has(e.metadata.name);return(0,l.jsxs)(i.A,{display:"flex",alignItems:"center",style:{gap:8},children:[(0,l.jsx)(i.A,{fontFamily:"monospace",fontSize:"0.875rem",children:a?e.status.apiKey:"•".repeat(20)+"..."}),(0,l.jsx)(W.Ay,{title:a?"hide key":"show key",children:(0,l.jsx)(o.A,{size:"small",onClick:()=>{return a=e.metadata.name,void h(e=>{const t=new Set(e);return t.has(a)?t.delete(a):t.add(a),t});var a},children:a?(0,l.jsx)(z.A,{fontSize:"small"}):(0,l.jsx)(M.A,{fontSize:"small"})})})]})}return(0,l.jsx)(d.A,{variant:"body2",color:"textSecondary",children:"-"})}},{title:"Requested",field:"metadata.creationTimestamp",render:e=>{if(!e.metadata.creationTimestamp)return(0,l.jsx)(d.A,{variant:"body2",children:"-"});const a=new Date(e.metadata.creationTimestamp);return(0,l.jsx)(d.A,{variant:"body2",children:a.toLocaleDateString()})}},{title:"",render:e=>(0,l.jsx)(o.A,{size:"small",onClick:a=>{a.stopPropagation();const t=a.currentTarget.getBoundingClientRect();A({top:t.bottom,left:t.left}),g(e)},"aria-controls":x?"myapikeys-menu":void 0,"aria-haspopup":"true",children:(0,l.jsx)(L.A,{})})}],J=(()=>{switch(c){case 0:return H;case 1:return K;case 2:return B;default:return $}})(),Y=(()=>{switch(c){case 0:return V.filter(e=>"Reason"!==e.title);case 1:return V.filter(e=>"Reason"!==e.title&&"Reviewed By"!==e.title&&"API Key"!==e.title);case 2:return V.filter(e=>"API Key"!==e.title);default:return V}})();return(0,l.jsxs)(l.Fragment,{children:[(0,l.jsxs)(T.n,{title:"My API Keys",subheader:`${H.length} active, ${K.length} pending`,children:[(0,l.jsx)(i.A,{mb:2,children:(0,l.jsxs)(E.A,{value:c,onChange:(e,a)=>u(a),indicatorColor:"primary",textColor:"primary",children:[(0,l.jsx)(q.A,{label:`Active (${H.length})`}),(0,l.jsx)(q.A,{label:`Pending (${K.length})`}),(0,l.jsx)(q.A,{label:`Rejected (${B.length})`})]})}),0===J.length?(0,l.jsx)(i.A,{p:3,textAlign:"center",children:(0,l.jsxs)(d.A,{variant:"body1",color:"textSecondary",children:[0===c&&"No active API keys. Request access to an API to get started.",1===c&&"No pending requests.",2===c&&"No rejected requests."]})}):(0,l.jsx)(y.X,{options:{paging:J.length>10,pageSize:10,search:!1,toolbar:!1},columns:Y,data:J.map(e=>({...e,id:e.metadata.name}))})]}),(0,l.jsx)(N.A,{id:"myapikeys-menu",open:Boolean(x),onClose:F,anchorReference:"anchorPosition",anchorPosition:x||{top:0,left:0},children:v&&(()=>{const e=[];var a;return(a=v).status&&"Pending"!==a.status.phase||e.push((0,l.jsx)(D.A,{onClick:O,children:"Edit"},"edit")),e.push((0,l.jsx)(D.A,{onClick:_,children:"Delete"},"delete")),e})()}),j.request&&(0,l.jsx)(U.n,{open:j.open,request:j.request,availablePlans:j.plans,onClose:()=>b({open:!1,request:null,plans:[]}),onSuccess:()=>{b({open:!1,request:null,plans:[]}),S(e=>e+1)}})]})};var K=t(46205);const B=({children:e,permission:a,fallback:t,errorMessage:r})=>{const{allowed:s,loading:n,error:o}=(0,K.l)(a);return n?(0,l.jsx)(w.k,{}):o?(0,l.jsxs)(i.A,{p:4,children:[(0,l.jsxs)(d.A,{color:"error",children:["Unable to check permissions: ",o.message]}),(0,l.jsx)(d.A,{variant:"body2",color:"textSecondary",children:"Please try again or contact your administrator"})]}):s?(0,l.jsx)(l.Fragment,{children:e}):t?(0,l.jsx)(l.Fragment,{children:t}):(0,l.jsxs)(i.A,{p:4,children:[(0,l.jsx)(d.A,{color:"textSecondary",children:r||"You don't have permission to view this page"}),(0,l.jsx)(i.A,{mt:1,children:(0,l.jsxs)(d.A,{variant:"caption",color:"textSecondary",children:["Required permission: ",a.name]})})]})};var F=t(58837),O=t(16249),_=t(84441);const V=({selectedPolicy:e,alertSeverity:a="warning",alertMessage:t="No PlanPolicy found for this HTTPRoute. API keys and rate limiting may not be available.",includeTopMargin:r=!0})=>(0,l.jsx)(i.A,{mt:r?1:0,p:2,bgcolor:"#f5f5f5",borderRadius:1,border:"1px solid #e0e0e0",children:e?(0,l.jsxs)(l.Fragment,{children:[(0,l.jsxs)(d.A,{variant:"subtitle2",gutterBottom:!0,style:{fontWeight:600},children:["Associated PlanPolicy: ",(0,l.jsx)("strong",{children:e.metadata.name})]}),e.plans&&e.plans.length>0?(0,l.jsxs)(l.Fragment,{children:[(0,l.jsx)(d.A,{variant:"caption",display:"block",gutterBottom:!0,color:"textSecondary",style:{marginTop:8},children:"Available Plans:"}),(0,l.jsx)(i.A,{display:"flex",flexWrap:"wrap",mt:1,style:{gap:8},children:e.plans.map((e,a)=>{var t,r,s;const i=(null===(t=e.limits)||void 0===t?void 0:t.daily)?`${e.limits.daily}/day`:(null===(r=e.limits)||void 0===r?void 0:r.monthly)?`${e.limits.monthly}/month`:(null===(s=e.limits)||void 0===s?void 0:s.yearly)?`${e.limits.yearly}/year`:"No limit";return(0,l.jsx)(n.A,{label:`${e.tier}: ${i}`,size:"small",variant:"outlined",color:"primary"},a)})}),e.plans.some(e=>e.description)&&(0,l.jsx)(i.A,{mt:1,children:e.plans.filter(e=>e.description).map((e,a)=>(0,l.jsxs)(d.A,{variant:"caption",display:"block",color:"textSecondary",children:["• ",(0,l.jsxs)("strong",{children:[e.tier,":"]})," ",e.description]},a))})]}):(0,l.jsx)(d.A,{variant:"caption",color:"textSecondary",children:"No plans defined in this PlanPolicy"})]}):(0,l.jsx)(_.A,{severity:a,children:t})}),J=(0,F.A)({asterisk:{color:"#f44336"}}),Y=({open:e,onClose:a,onSuccess:t})=>{const s=J(),o=(0,I.useApi)(I.configApiRef),x=(0,I.useApi)(I.fetchApiRef),v=o.getString("backend.baseUrl"),[g,j]=(0,r.useState)(""),[f,y]=(0,r.useState)(""),[b,P]=(0,r.useState)(""),[S,k]=(0,r.useState)("v1"),[w,C]=(0,r.useState)("manual"),[T,$]=(0,r.useState)("Published"),[W,E]=(0,r.useState)([]),[q,N]=(0,r.useState)(""),[M,z]=(0,r.useState)(""),[L,U]=(0,r.useState)(""),[H,K]=(0,r.useState)(""),[B,F]=(0,r.useState)(""),[Y,X]=(0,r.useState)(""),[G,Q]=(0,r.useState)(""),[Z,ee]=(0,r.useState)(!1),{value:ae,loading:te}=(0,R.A)(async()=>{const e=await x.fetch(`${v}/api/kuadrant/httproutes`);return((await e.json()).items||[]).filter(e=>{var a;return"true"===(null===(a=e.metadata.annotations)||void 0===a?void 0:a["backstage.io/expose"])})},[v,x,e]),{value:le}=(0,R.A)(async()=>{const e=await x.fetch(`${v}/api/kuadrant/planpolicies`);return await e.json()},[v,x,e]),re=M?M.split("/"):null,se=re?(ne=re[0],ie=re[1],(null==le?void 0:le.items)?le.items.find(e=>{const a=e.targetRef;return"HTTPRoute"===(null==a?void 0:a.kind)&&(null==a?void 0:a.name)===ie&&(!(null==a?void 0:a.namespace)||(null==a?void 0:a.namespace)===ne)}):null):null;var ne,ie;const oe=()=>{q.trim()&&!W.includes(q.trim())&&(E([...W,q.trim()]),N(""))},de=()=>{j(""),y(""),P(""),k("v1"),C("manual"),$("Published"),E([]),N(""),z(""),U(""),K(""),F(""),X(""),Q(""),a()};return(0,l.jsxs)(p.A,{open:e,onClose:de,maxWidth:"md",fullWidth:!0,children:[(0,l.jsx)(m.A,{children:"Create API Product"}),(0,l.jsxs)(h.A,{children:[G&&(0,l.jsx)(_.A,{severity:"error",style:{marginBottom:16},children:G}),(0,l.jsxs)(c.A,{container:!0,spacing:2,children:[(0,l.jsx)(c.A,{item:!0,xs:6,children:(0,l.jsx)(O.A,{fullWidth:!0,label:"Name",value:g,onChange:e=>j(e.target.value),placeholder:"my-api",helperText:"Kubernetes resource name (lowercase, hyphens)",margin:"normal",required:!0,InputLabelProps:{classes:{asterisk:s.asterisk}}})}),(0,l.jsx)(c.A,{item:!0,xs:6,children:(0,l.jsx)(O.A,{fullWidth:!0,label:"Display Name",value:f,onChange:e=>y(e.target.value),placeholder:"My API",margin:"normal",required:!0,InputLabelProps:{classes:{asterisk:s.asterisk}}})}),(0,l.jsx)(c.A,{item:!0,xs:6,children:(0,l.jsx)(O.A,{fullWidth:!0,label:"Version",value:S,onChange:e=>k(e.target.value),placeholder:"v1",margin:"normal"})}),(0,l.jsx)(c.A,{item:!0,xs:12,children:(0,l.jsxs)(O.A,{fullWidth:!0,select:!0,label:"Approval Mode",value:w,onChange:e=>C(e.target.value),margin:"normal",helperText:"Automatic: keys are created immediately. Manual: requires approval.",children:[(0,l.jsx)(D.A,{value:"manual",children:"Manual"}),(0,l.jsx)(D.A,{value:"automatic",children:"Automatic"})]})}),(0,l.jsx)(c.A,{item:!0,xs:12,children:(0,l.jsxs)(O.A,{fullWidth:!0,select:!0,label:"Publish Status",value:T,onChange:e=>$(e.target.value),margin:"normal",helperText:"Draft: hidden from catalog. Published: visible to consumers.",children:[(0,l.jsx)(D.A,{value:"Draft",children:"Draft"}),(0,l.jsx)(D.A,{value:"Published",children:"Published"})]})}),(0,l.jsx)(c.A,{item:!0,xs:12,children:(0,l.jsx)(O.A,{fullWidth:!0,label:"Description",value:b,onChange:e=>P(e.target.value),placeholder:"API description",margin:"normal",multiline:!0,rows:2,required:!0,InputLabelProps:{classes:{asterisk:s.asterisk}}})}),(0,l.jsxs)(c.A,{item:!0,xs:12,children:[(0,l.jsx)(d.A,{variant:"subtitle2",gutterBottom:!0,style:{marginTop:16},children:"Tags"}),(0,l.jsx)(i.A,{display:"flex",flexWrap:"wrap",marginBottom:1,style:{gap:8},children:W.map(e=>(0,l.jsx)(n.A,{label:e,onDelete:()=>{return a=e,void E(W.filter(e=>e!==a));var a},size:"small"},e))}),(0,l.jsxs)(i.A,{display:"flex",style:{gap:8},children:[(0,l.jsx)(O.A,{fullWidth:!0,size:"small",value:q,onChange:e=>N(e.target.value),onKeyPress:e=>"Enter"===e.key&&oe(),placeholder:"Add tag"}),(0,l.jsx)(u.A,{onClick:oe,variant:"outlined",size:"small",children:"Add"})]})]}),(0,l.jsx)(c.A,{item:!0,xs:12,children:(0,l.jsxs)(O.A,{fullWidth:!0,select:!0,label:"HTTPRoute",value:M,onChange:e=>z(e.target.value),margin:"normal",required:!0,helperText:"Select an HTTPRoute (backstage.io/expose: true). APIProduct will be created in the same namespace.",disabled:te,InputLabelProps:{classes:{asterisk:s.asterisk}},children:[te&&(0,l.jsx)(D.A,{value:"",children:"Loading..."}),!te&&ae&&0===ae.length&&(0,l.jsx)(D.A,{value:"",children:"No HTTPRoutes available"}),!te&&ae&&ae.map(e=>(0,l.jsxs)(D.A,{value:`${e.metadata.namespace}/${e.metadata.name}`,children:[e.metadata.name," (",e.metadata.namespace,")"]},`${e.metadata.namespace}/${e.metadata.name}`))]})}),M&&(0,l.jsx)(c.A,{item:!0,xs:12,children:(0,l.jsx)(V,{selectedPolicy:se,alertSeverity:"warning",alertMessage:"No PlanPolicy found for this HTTPRoute. API keys and rate limiting may not be available.",includeTopMargin:!0})}),(0,l.jsx)(c.A,{item:!0,xs:6,children:(0,l.jsx)(O.A,{fullWidth:!0,label:"Contact Email",value:L,onChange:e=>U(e.target.value),placeholder:"api-team@example.com",margin:"normal"})}),(0,l.jsx)(c.A,{item:!0,xs:6,children:(0,l.jsx)(O.A,{fullWidth:!0,label:"Contact Team",value:H,onChange:e=>K(e.target.value),placeholder:"platform-team",margin:"normal"})}),(0,l.jsx)(c.A,{item:!0,xs:6,children:(0,l.jsx)(O.A,{fullWidth:!0,label:"Docs URL",value:B,onChange:e=>F(e.target.value),placeholder:"https://api.example.com/docs",margin:"normal"})}),(0,l.jsx)(c.A,{item:!0,xs:6,children:(0,l.jsx)(O.A,{fullWidth:!0,label:"OpenAPI Spec URL",value:Y,onChange:e=>X(e.target.value),placeholder:"https://api.example.com/openapi.json",margin:"normal"})})]})]}),(0,l.jsxs)(A.A,{children:[(0,l.jsx)(u.A,{onClick:de,children:"Cancel"}),(0,l.jsx)(u.A,{onClick:async()=>{Q(""),ee(!0);try{if(!M)throw new Error("Please select an HTTPRoute");const[e,a]=M.split("/"),l={apiVersion:"extensions.kuadrant.io/v1alpha1",kind:"APIProduct",metadata:{name:g,namespace:e},spec:{displayName:f,description:b,version:S,approvalMode:w,publishStatus:T,tags:W,targetRef:{group:"gateway.networking.k8s.io",kind:"HTTPRoute",name:a,namespace:e},...L||H?{contact:{...L&&{email:L},...H&&{team:H}}}:{},...B||Y?{documentation:{...B&&{docsURL:B},...Y&&{openAPISpec:Y}}}:{}}},r=await x.fetch(`${v}/api/kuadrant/apiproducts`,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify(l)});if(!r.ok){const e=await r.json();throw new Error(e.error||"failed to create apiproduct")}t(),de()}catch(e){Q(e instanceof Error?e.message:String(e))}finally{ee(!1)}},color:"primary",variant:"contained",disabled:Z||!g||!f||!b||!M,children:Z?"Creating...":"Create"})]})]})};var X=t(34955);const G=(0,F.A)({asterisk:{color:"#f44336"}}),Q=({open:e,onClose:a,onSuccess:t,namespace:o,name:x})=>{const v=G(),g=(0,I.useApi)(I.configApiRef),j=(0,I.useApi)(I.fetchApiRef),f=g.getString("backend.baseUrl"),[y,b]=(0,r.useState)(!1),[P,S]=(0,r.useState)(""),[k,C]=(0,r.useState)(""),[T,$]=(0,r.useState)("v1"),[W,E]=(0,r.useState)("Draft"),[q,N]=(0,r.useState)("manual"),[M,z]=(0,r.useState)([]),[L,U]=(0,r.useState)(null),[H,K]=(0,r.useState)(""),[B,F]=(0,r.useState)(""),[J,Y]=(0,r.useState)(""),[X,Q]=(0,r.useState)(""),[Z,ee]=(0,r.useState)(""),[ae,te]=(0,r.useState)(""),[le,re]=(0,r.useState)(!1);(0,r.useEffect)(()=>{e&&o&&x&&(b(!0),te(""),j.fetch(`${f}/api/kuadrant/apiproducts/${o}/${x}`).then(async e=>{if(!e.ok){const a=await e.json();throw new Error(a.error||`Failed to fetch API product: ${e.status}`)}return e.json()}).then(e=>{var a,t,l,r;S(e.spec.displayName||""),C(e.spec.description||""),$(e.spec.version||"v1"),E(e.spec.publishStatus||"Draft"),N(e.spec.approvalMode||"manual"),z(e.spec.tags||[]),U(e.spec.targetRef||null),F((null===(a=e.spec.contact)||void 0===a?void 0:a.email)||""),Y((null===(t=e.spec.contact)||void 0===t?void 0:t.team)||""),Q((null===(l=e.spec.documentation)||void 0===l?void 0:l.docsURL)||""),ee((null===(r=e.spec.documentation)||void 0===r?void 0:r.openAPISpec)||""),b(!1)}).catch(e=>{te(e.message||"Failed to load API product"),b(!1)}))},[e,o,x,f,j]);const{value:se}=(0,R.A)(async()=>{if(!e)return null;const a=await j.fetch(`${f}/api/kuadrant/planpolicies`);return await a.json()},[f,j,e]),ne=s().useMemo(()=>(null==se?void 0:se.items)&&L?se.items.find(e=>{const a=e.targetRef;return"HTTPRoute"===(null==a?void 0:a.kind)&&(null==a?void 0:a.name)===L.name&&(!(null==a?void 0:a.namespace)||(null==a?void 0:a.namespace)===(L.namespace||o))}):null,[se,L,o]),ie=()=>{H.trim()&&!M.includes(H.trim())&&(z([...M,H.trim()]),K(""))};return(0,l.jsxs)(p.A,{open:e,onClose:a,maxWidth:"md",fullWidth:!0,children:[(0,l.jsx)(m.A,{children:"Edit API Product"}),(0,l.jsxs)(h.A,{children:[ae&&(0,l.jsx)(_.A,{severity:"error",style:{marginBottom:16},children:ae}),y?(0,l.jsx)(w.k,{}):(0,l.jsxs)(c.A,{container:!0,spacing:2,children:[(0,l.jsx)(c.A,{item:!0,xs:6,children:(0,l.jsx)(O.A,{fullWidth:!0,label:"Name",value:x,disabled:!0,helperText:"Kubernetes resource name (immutable)",margin:"normal"})}),(0,l.jsx)(c.A,{item:!0,xs:6,children:(0,l.jsx)(O.A,{fullWidth:!0,label:"Namespace",value:o,disabled:!0,helperText:"Derived from HTTPRoute (immutable)",margin:"normal"})}),(0,l.jsx)(c.A,{item:!0,xs:6,children:(0,l.jsx)(O.A,{fullWidth:!0,label:"Display Name",value:P,onChange:e=>S(e.target.value),placeholder:"My API",margin:"normal",required:!0,InputLabelProps:{classes:{asterisk:v.asterisk}}})}),(0,l.jsx)(c.A,{item:!0,xs:6,children:(0,l.jsx)(O.A,{fullWidth:!0,label:"Version",value:T,onChange:e=>$(e.target.value),placeholder:"v1",margin:"normal"})}),(0,l.jsx)(c.A,{item:!0,xs:6,children:(0,l.jsxs)(O.A,{fullWidth:!0,select:!0,label:"Publish Status",value:W,onChange:e=>E(e.target.value),margin:"normal",helperText:"Draft: Hidden from catalog. Published: Visible in catalog.",children:[(0,l.jsx)(D.A,{value:"Draft",children:"Draft (Hidden)"}),(0,l.jsx)(D.A,{value:"Published",children:"Published (Visible)"})]})}),(0,l.jsx)(c.A,{item:!0,xs:6,children:(0,l.jsxs)(O.A,{fullWidth:!0,select:!0,label:"Approval Mode",value:q,onChange:e=>N(e.target.value),margin:"normal",helperText:"Automatic: keys created immediately. Manual: requires approval.",children:[(0,l.jsx)(D.A,{value:"manual",children:"Manual"}),(0,l.jsx)(D.A,{value:"automatic",children:"Automatic"})]})}),(0,l.jsx)(c.A,{item:!0,xs:12,children:(0,l.jsx)(O.A,{fullWidth:!0,label:"Description",value:k,onChange:e=>C(e.target.value),placeholder:"API description",margin:"normal",multiline:!0,rows:2,required:!0,InputLabelProps:{classes:{asterisk:v.asterisk}}})}),(0,l.jsxs)(c.A,{item:!0,xs:12,children:[(0,l.jsx)(d.A,{variant:"subtitle2",gutterBottom:!0,style:{marginTop:16},children:"Tags"}),(0,l.jsx)(i.A,{display:"flex",flexWrap:"wrap",marginBottom:1,style:{gap:8},children:M.map(e=>(0,l.jsx)(n.A,{label:e,onDelete:()=>{return a=e,void z(M.filter(e=>e!==a));var a},size:"small"},e))}),(0,l.jsxs)(i.A,{display:"flex",style:{gap:8},children:[(0,l.jsx)(O.A,{fullWidth:!0,size:"small",value:H,onChange:e=>K(e.target.value),onKeyPress:e=>"Enter"===e.key&&ie(),placeholder:"Add tag"}),(0,l.jsx)(u.A,{onClick:ie,variant:"outlined",size:"small",children:"Add"})]})]}),L&&(0,l.jsxs)(l.Fragment,{children:[(0,l.jsx)(c.A,{item:!0,xs:12,children:(0,l.jsx)(O.A,{fullWidth:!0,label:"HTTPRoute",value:`${L.namespace||o}/${L.name}`,disabled:!0,helperText:"Target HTTPRoute (immutable)",margin:"normal"})}),(0,l.jsx)(c.A,{item:!0,xs:12,children:(0,l.jsx)(V,{selectedPolicy:ne,alertSeverity:"info",alertMessage:"No PlanPolicy found for this HTTPRoute.",includeTopMargin:!1})})]}),(0,l.jsx)(c.A,{item:!0,xs:6,children:(0,l.jsx)(O.A,{fullWidth:!0,label:"Contact Email",value:B,onChange:e=>F(e.target.value),placeholder:"api-team@example.com",margin:"normal"})}),(0,l.jsx)(c.A,{item:!0,xs:6,children:(0,l.jsx)(O.A,{fullWidth:!0,label:"Contact Team",value:J,onChange:e=>Y(e.target.value),placeholder:"platform-team",margin:"normal"})}),(0,l.jsx)(c.A,{item:!0,xs:6,children:(0,l.jsx)(O.A,{fullWidth:!0,label:"Docs URL",value:X,onChange:e=>Q(e.target.value),placeholder:"https://api.example.com/docs",margin:"normal"})}),(0,l.jsx)(c.A,{item:!0,xs:6,children:(0,l.jsx)(O.A,{fullWidth:!0,label:"OpenAPI Spec URL",value:Z,onChange:e=>ee(e.target.value),placeholder:"https://api.example.com/openapi.json",margin:"normal"})})]})]}),(0,l.jsxs)(A.A,{children:[(0,l.jsx)(u.A,{onClick:a,children:"Cancel"}),(0,l.jsx)(u.A,{onClick:async()=>{te(""),re(!0);try{const e={spec:{displayName:P,description:k,version:T,publishStatus:W,approvalMode:q,tags:M,targetRef:L,...B||J?{contact:{...B&&{email:B},...J&&{team:J}}}:{},...X||Z?{documentation:{...X&&{docsURL:X},...Z&&{openAPISpec:Z}}}:{}}},l=await j.fetch(`${f}/api/kuadrant/apiproducts/${o}/${x}`,{method:"PATCH",headers:{"Content-Type":"application/json"},body:JSON.stringify(e)});if(!l.ok){const e=await l.json();throw new Error(e.error||"failed to update apiproduct")}t(),a()}catch(e){te(e instanceof Error?e.message:String(e))}finally{re(!1)}},color:"primary",variant:"contained",disabled:le||y||!P||!k,children:le?"Saving...":"Save"})]})]})},Z=()=>{const e=(0,I.useApi)(I.configApiRef),a=(0,I.useApi)(I.fetchApiRef),t=e.getString("backend.baseUrl"),[s,W]=(0,r.useState)(!1),[E,q]=(0,r.useState)(!1),[N,D]=(0,r.useState)(0),[M,z]=(0,r.useState)(!1),[L,U]=(0,r.useState)(null),[B,F]=(0,r.useState)(null),[O,_]=(0,r.useState)(!1),{allowed:V,loading:J,error:G}=(0,K.l)(X.FL),{allowed:Z,loading:ee}=(0,K.l)(X.TE),{allowed:ae,loading:te,error:le}=(0,K.l)(X.SU),re=Z||ae,se=ee||te,{allowed:ne,loading:ie}=(0,K.l)(X.EM),{allowed:oe,loading:de,error:ce}=(0,K.l)(X.R_),{allowed:ue}=(0,K.l)(X.U3),{allowed:pe}=(0,K.l)(X.v_),me=ne||oe,he=ue||pe,xe=ie||de,{allowed:Ae,loading:ve,error:ge}=(0,K.l)(X.J),{value:je,loading:fe,error:ye}=(0,R.A)(async()=>{const e=await a.fetch(`${t}/api/kuadrant/apiproducts`);return await e.json()},[t,a,N]),{value:be,loading:Pe,error:Se}=(0,R.A)(async()=>{const e=await a.fetch(`${t}/api/kuadrant/planpolicies`);return await e.json()},[t,a,N]),ke=fe||Pe||J||se||xe||ve,we=ye||Se,Ce=G||le||ce||ge,Te=()=>{z(!1),U(null)},Re=[{title:"Name",field:"name",render:e=>{var a;return(0,l.jsx)(f.N_,{to:`/catalog/default/api/${e.metadata.name}/api-product`,children:(0,l.jsx)("strong",{children:(null===(a=e.spec)||void 0===a?void 0:a.displayName)||e.metadata.name})})}},{title:"Resource Name",field:"metadata.name"},{title:"Version",field:"spec.version",render:e=>{var a;return(null===(a=e.spec)||void 0===a?void 0:a.version)||"-"}},{title:"HTTPRoute",field:"spec.targetRef.name",render:e=>{var a,t;return(null===(t=e.spec)||void 0===t||null===(a=t.targetRef)||void 0===a?void 0:a.name)||"-"}},{title:"Publish Status",field:"spec.publishStatus",render:e=>{var a;const t=(null===(a=e.spec)||void 0===a?void 0:a.publishStatus)||"Draft";return(0,l.jsx)(n.A,{label:t,size:"small",color:"Published"===t?"primary":"default"})}},{title:"Approval Mode",field:"spec.approvalMode",render:e=>{var a;const t=(null===(a=e.spec)||void 0===a?void 0:a.approvalMode)||"manual";return(0,l.jsx)(n.A,{label:t,size:"small",color:"automatic"===t?"secondary":"default"})}},{title:"Namespace",field:"metadata.namespace"},{title:"Created",field:"metadata.creationTimestamp",render:e=>{return a=e.metadata.creationTimestamp,new Date(a).toLocaleDateString("en-GB",{year:"numeric",month:"short",day:"numeric"});var a}},{title:"Actions",field:"actions",render:e=>(0,l.jsxs)(i.A,{display:"flex",style:{gap:4},children:[he&&(0,l.jsx)(o.A,{size:"small",onClick:()=>{return a=e.metadata.namespace,t=e.metadata.name,F({namespace:a,name:t}),void q(!0);var a,t},title:"Edit API Product",children:(0,l.jsx)(j.A,{fontSize:"small"})}),me&&(0,l.jsx)(o.A,{size:"small",onClick:()=>{return a=e.metadata.namespace,t=e.metadata.name,U({namespace:a,name:t}),void z(!0);var a,t},title:"Delete API Product",children:(0,l.jsx)(g.A,{fontSize:"small"})})]})}],Ie=[{title:"Name",field:"metadata.name",render:e=>(0,l.jsx)(f.N_,{to:`/kuadrant/planpolicy/${e.metadata.namespace}/${e.metadata.name}`,children:(0,l.jsx)("strong",{children:e.metadata.name})})},{title:"Namespace",field:"metadata.namespace"}];return(0,l.jsxs)(b.Y,{themeId:"tool",children:[(0,l.jsx)(P.Y,{title:"Kuadrant",subtitle:"API management for Kubernetes",children:(0,l.jsx)(S.Y,{children:"Manage API products and access requests"})}),(0,l.jsxs)(k.U,{children:[ke&&(0,l.jsx)(w.k,{}),we&&(0,l.jsx)(C._,{error:we}),Ce&&(0,l.jsxs)(i.A,{p:2,children:[(0,l.jsxs)(d.A,{color:"error",children:["unable to check permissions: ",Ce.message]}),(0,l.jsxs)(d.A,{variant:"body2",color:"textSecondary",children:["permission: ",G?"kuadrant.apiproduct.create":ce?"kuadrant.apiproduct.delete":le?"kuadrant.apikeyrequest.read.all":ge?"kuadrant.planpolicy.list":"unknown"]}),(0,l.jsx)(d.A,{variant:"body2",color:"textSecondary",children:"please try again or contact your administrator"})]}),!ke&&!we&&!Ce&&(0,l.jsxs)(c.A,{container:!0,spacing:3,direction:"column",children:[(0,l.jsx)(c.A,{item:!0,children:(0,l.jsx)(H,{})}),(0,l.jsx)(c.A,{item:!0,children:(0,l.jsx)(T.n,{title:"API Products",action:V?(0,l.jsx)(i.A,{display:"flex",alignItems:"center",height:"100%",mt:1,children:(0,l.jsx)(u.A,{variant:"contained",color:"primary",size:"small",startIcon:(0,l.jsx)(v.A,{}),onClick:()=>W(!0),children:"Create API Product"})}):void 0,children:($e=null==je?void 0:je.items,$e&&0!==$e.length?(0,l.jsx)(y.X,{options:{paging:!1,search:!1,toolbar:!1},columns:Re,data:$e}):(0,l.jsx)(d.A,{variant:"body2",color:"textSecondary",children:"No API products found"}))})}),Ae&&(0,l.jsx)(c.A,{item:!0,children:(0,l.jsx)(T.n,{title:"Plan Policies",children:(e=>e&&0!==e.length?(0,l.jsx)(y.X,{options:{paging:!1,search:!1,toolbar:!1},columns:Ie,data:e}):(0,l.jsx)(d.A,{variant:"body2",color:"textSecondary",children:"No plan policies found"}))(null==be?void 0:be.items)})}),re&&(0,l.jsx)(c.A,{item:!0,children:(0,l.jsx)($.d,{})})]}),(0,l.jsx)(Y,{open:s,onClose:()=>W(!1),onSuccess:()=>{D(e=>e+1)}}),(0,l.jsx)(Q,{open:E,onClose:()=>q(!1),onSuccess:()=>{D(e=>e+1)},namespace:(null==B?void 0:B.namespace)||"",name:(null==B?void 0:B.name)||""}),(0,l.jsxs)(p.A,{open:M,onClose:Te,children:[(0,l.jsx)(m.A,{children:"Delete API Product"}),(0,l.jsx)(h.A,{children:(0,l.jsxs)(x.A,{children:["Are you sure you want to delete ",null==L?void 0:L.name," from namespace ",null==L?void 0:L.namespace,"? This will permanently remove the API Product from Kubernetes."]})}),(0,l.jsxs)(A.A,{children:[(0,l.jsx)(u.A,{onClick:Te,color:"primary",children:"Cancel"}),(0,l.jsx)(u.A,{onClick:async()=>{if(L){_(!0);try{if(!(await a.fetch(`${t}/api/kuadrant/apiproducts/${L.namespace}/${L.name}`,{method:"DELETE"})).ok)throw new Error("failed to delete apiproduct");D(e=>e+1)}catch(e){console.error("error deleting apiproduct:",e)}finally{_(!1),z(!1),U(null)}}},color:"secondary",disabled:O,children:O?"Deleting...":"Delete"})]})]})]})]});var $e},ee=()=>(0,l.jsx)(B,{permission:X.OP,errorMessage:"you don't have permission to view the Kuadrant page",children:(0,l.jsx)(Z,{})})}}]);
2
- //# sourceMappingURL=7632.646001ce.chunk.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"static/7632.646001ce.chunk.js","mappings":"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","sources":["webpack://internal.plugin-kuadrant/./src/components/EditAPIKeyRequestDialog/EditAPIKeyRequestDialog.tsx","webpack://internal.plugin-kuadrant/./src/components/MyApiKeysCard/MyApiKeysCard.tsx","webpack://internal.plugin-kuadrant/./src/components/PermissionGate/PermissionGate.tsx","webpack://internal.plugin-kuadrant/./src/components/PlanPolicyDetailsCard/PlanPolicyDetails.tsx","webpack://internal.plugin-kuadrant/./src/components/CreateAPIProductDialog/CreateAPIProductDialog.tsx","webpack://internal.plugin-kuadrant/./src/components/EditAPIProductDialog/EditAPIProductDialog.tsx","webpack://internal.plugin-kuadrant/./src/components/KuadrantPage/KuadrantPage.tsx"],"sourcesContent":["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} 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 disabled={!planTier || saving}\n >\n {saving ? 'Saving...' : 'Save Changes'}\n </Button>\n </DialogActions>\n </Dialog>\n );\n};\n","import React, { useState } from 'react';\nimport { InfoCard, Table, TableColumn, Link } from '@backstage/core-components';\nimport { useApi, configApiRef, fetchApiRef, identityApiRef } from '@backstage/core-plugin-api';\nimport useAsync from 'react-use/lib/useAsync';\nimport { Box, Chip, Typography, Tabs, Tab, IconButton, Tooltip, Menu, MenuItem } from '@material-ui/core';\nimport VisibilityIcon from '@material-ui/icons/Visibility';\nimport VisibilityOffIcon from '@material-ui/icons/VisibilityOff';\nimport MoreVertIcon from '@material-ui/icons/MoreVert';\nimport { EditAPIKeyRequestDialog } from '../EditAPIKeyRequestDialog';\nimport { APIKeyRequest } from '../../types/api-management';\n\nexport const MyApiKeysCard = () => {\n const config = useApi(configApiRef);\n const fetchApi = useApi(fetchApiRef);\n const identityApi = useApi(identityApiRef);\n const backendUrl = config.getString('backend.baseUrl');\n const [selectedTab, setSelectedTab] = useState(0);\n const [, setUserId] = useState<string>('');\n const [visibleKeys, setVisibleKeys] = useState<Set<string>>(new Set());\n const [menuAnchor, setMenuAnchor] = useState<{ top: number; left: number } | null>(null);\n const [menuRequest, setMenuRequest] = useState<APIKeyRequest | null>(null);\n const [editDialogState, setEditDialogState] = useState<{ open: boolean; request: APIKeyRequest | null; plans: any[] }>({\n open: false,\n request: null,\n plans: [],\n });\n const [refresh, setRefresh] = useState(0);\n\n useAsync(async () => {\n const identity = await identityApi.getBackstageIdentity();\n const extractedUserId = identity.userEntityRef.split('/')[1] || 'guest';\n console.log(`MyApiKeysCard: setting userId from userEntityRef: ${identity.userEntityRef} -> \"${extractedUserId}\"`);\n setUserId(extractedUserId);\n }, [identityApi]);\n\n const { value: requests, loading, error } = useAsync(async () => {\n const response = await fetchApi.fetch(\n `${backendUrl}/api/kuadrant/requests/my`\n );\n if (!response.ok) {\n throw new Error('failed to fetch requests');\n }\n const data = await response.json();\n return data.items || [];\n }, [backendUrl, fetchApi, refresh]);\n\n if (loading) {\n return (\n <InfoCard title=\"My API Keys\">\n <Typography>Loading...</Typography>\n </InfoCard>\n );\n }\n\n if (error) {\n return (\n <InfoCard title=\"My API Keys\">\n <Typography color=\"error\">Error loading API keys: {error.message}</Typography>\n </InfoCard>\n );\n }\n\n const allRequests = requests || [];\n const approvedRequests = allRequests.filter((r: APIKeyRequest) => r.status?.phase === 'Approved');\n const pendingRequests = allRequests.filter((r: APIKeyRequest) => !r.status?.phase || r.status.phase === 'Pending');\n const rejectedRequests = allRequests.filter((r: APIKeyRequest) => r.status?.phase === 'Rejected');\n\n const toggleKeyVisibility = (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 handleMenuClose = () => {\n setMenuAnchor(null);\n setMenuRequest(null);\n };\n\n const handleEdit = async () => {\n if (!menuRequest) return;\n\n const request = menuRequest;\n handleMenuClose();\n\n // Fetch available plans for this API\n try {\n const apiProductResponse = await fetchApi.fetch(\n `${backendUrl}/api/kuadrant/apiproducts/${request.spec.apiNamespace}/${request.spec.apiName}`\n );\n\n if (apiProductResponse.ok) {\n const apiProduct = await apiProductResponse.json();\n const plans = apiProduct.spec?.plans || [];\n setEditDialogState({ open: true, request, plans });\n } else {\n console.error('Failed to fetch API product');\n setEditDialogState({ open: true, request, plans: [] });\n }\n } catch (err) {\n console.error('Error fetching plans:', err);\n setEditDialogState({ open: true, request, plans: [] });\n }\n };\n\n const handleDelete = async () => {\n if (!menuRequest) return;\n\n const request = menuRequest;\n handleMenuClose();\n\n if (!window.confirm('Are you sure you want to delete this request?')) {\n return;\n }\n\n try {\n const response = await fetchApi.fetch(\n `${backendUrl}/api/kuadrant/requests/${request.metadata.namespace}/${request.metadata.name}`,\n { method: 'DELETE' }\n );\n\n if (!response.ok) {\n throw new Error('Failed to delete request');\n }\n\n setRefresh(r => r + 1);\n } catch (err) {\n console.error('Error deleting request:', err);\n alert('Failed to delete request');\n }\n };\n\n const columns: TableColumn<APIKeyRequest>[] = [\n {\n title: 'API Product',\n field: 'spec.apiName',\n render: (row: APIKeyRequest) => (\n <Link to={`/catalog/default/api/${row.spec.apiName}/api-keys`}>\n <strong>{row.spec.apiName}</strong>\n </Link>\n ),\n },\n {\n title: 'Plan',\n field: 'spec.planTier',\n render: (row: APIKeyRequest) => {\n const color = row.spec.planTier === 'gold' ? 'primary' :\n row.spec.planTier === 'silver' ? 'default' : 'secondary';\n return <Chip label={row.spec.planTier} color={color} 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: 'Status',\n field: 'status.phase',\n render: (row: APIKeyRequest) => {\n const phase = row.status?.phase || 'Pending';\n const color = phase === 'Approved' ? 'primary' :\n phase === 'Rejected' ? 'secondary' : 'default';\n return <Chip label={phase} color={color} size=\"small\" />;\n },\n },\n {\n title: 'Reason',\n render: (row: APIKeyRequest) => {\n if (row.status?.reason) {\n const color = row.status.phase === 'Rejected' ? 'error' : 'textPrimary';\n return (\n <Tooltip title={row.status.reason} placement=\"top\">\n <Typography\n variant=\"body2\"\n color={color}\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 return <Typography variant=\"body2\" color=\"textSecondary\">-</Typography>;\n },\n },\n {\n title: 'Reviewed By',\n render: (row: APIKeyRequest) => {\n if ((row.status?.phase === 'Approved' || row.status?.phase === 'Rejected') && row.status.reviewedBy) {\n const reviewedDate = row.status.reviewedAt ? new Date(row.status.reviewedAt).toLocaleDateString() : '';\n return (\n <Box>\n <Typography variant=\"body2\">{row.status.reviewedBy}</Typography>\n {reviewedDate && (\n <Typography variant=\"caption\" color=\"textSecondary\">\n {reviewedDate}\n </Typography>\n )}\n </Box>\n );\n }\n return <Typography variant=\"body2\" color=\"textSecondary\">-</Typography>;\n },\n },\n {\n title: 'API Key',\n render: (row: APIKeyRequest) => {\n if (row.status?.phase === 'Approved' && row.status.apiKey) {\n const isVisible = visibleKeys.has(row.metadata.name);\n return (\n <Box display=\"flex\" alignItems=\"center\" style={{ gap: 8 }}>\n <Box fontFamily=\"monospace\" fontSize=\"0.875rem\">\n {isVisible ? row.status.apiKey : '•'.repeat(20) + '...'}\n </Box>\n <Tooltip title={isVisible ? 'hide key' : 'show key'}>\n <IconButton\n size=\"small\"\n onClick={() => toggleKeyVisibility(row.metadata.name)}\n >\n {isVisible ? <VisibilityOffIcon fontSize=\"small\" /> : <VisibilityIcon fontSize=\"small\" />}\n </IconButton>\n </Tooltip>\n </Box>\n );\n }\n return <Typography variant=\"body2\" color=\"textSecondary\">-</Typography>;\n },\n },\n {\n title: 'Requested',\n field: 'metadata.creationTimestamp',\n render: (row: APIKeyRequest) => {\n if (!row.metadata.creationTimestamp) {\n return <Typography variant=\"body2\">-</Typography>;\n }\n const date = new Date(row.metadata.creationTimestamp);\n return <Typography variant=\"body2\">{date.toLocaleDateString()}</Typography>;\n },\n },\n {\n title: '',\n render: (row: APIKeyRequest) => {\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 aria-controls={menuAnchor ? 'myapikeys-menu' : undefined}\n aria-haspopup=\"true\"\n >\n <MoreVertIcon />\n </IconButton>\n );\n },\n },\n ];\n\n const getTabData = () => {\n switch (selectedTab) {\n case 0:\n return approvedRequests;\n case 1:\n return pendingRequests;\n case 2:\n return rejectedRequests;\n default:\n return allRequests;\n }\n };\n\n const getTabColumns = () => {\n switch (selectedTab) {\n case 0: // Active - no Reason\n return columns.filter(col => col.title !== 'Reason');\n case 1: // Pending - no Reason, Reviewed By, API Key\n return columns.filter(col =>\n col.title !== 'Reason' &&\n col.title !== 'Reviewed By' &&\n col.title !== 'API Key'\n );\n case 2: // Rejected - no API Key\n return columns.filter(col => col.title !== 'API Key');\n default:\n return columns;\n }\n };\n\n const tabData = getTabData();\n const tabColumns = getTabColumns();\n const isPending = (row: APIKeyRequest) => !row.status || row.status.phase === 'Pending';\n\n return (\n <>\n <InfoCard\n title=\"My API Keys\"\n subheader={`${approvedRequests.length} active, ${pendingRequests.length} pending`}\n >\n <Box mb={2}>\n <Tabs\n value={selectedTab}\n onChange={(_, newValue) => setSelectedTab(newValue)}\n indicatorColor=\"primary\"\n textColor=\"primary\"\n >\n <Tab label={`Active (${approvedRequests.length})`} />\n <Tab label={`Pending (${pendingRequests.length})`} />\n <Tab label={`Rejected (${rejectedRequests.length})`} />\n </Tabs>\n </Box>\n {tabData.length === 0 ? (\n <Box p={3} textAlign=\"center\">\n <Typography variant=\"body1\" color=\"textSecondary\">\n {selectedTab === 0 && 'No active API keys. Request access to an API to get started.'}\n {selectedTab === 1 && 'No pending requests.'}\n {selectedTab === 2 && 'No rejected requests.'}\n </Typography>\n </Box>\n ) : (\n <Table\n options={{\n paging: tabData.length > 10,\n pageSize: 10,\n search: false,\n toolbar: false,\n }}\n columns={tabColumns}\n data={tabData.map((item: APIKeyRequest) => ({\n ...item,\n id: item.metadata.name,\n }))}\n />\n )}\n </InfoCard>\n\n <Menu\n id=\"myapikeys-menu\"\n open={Boolean(menuAnchor)}\n onClose={handleMenuClose}\n anchorReference=\"anchorPosition\"\n anchorPosition={menuAnchor || { top: 0, left: 0 }}\n >\n {menuRequest && (() => {\n const items = [];\n if (isPending(menuRequest)) {\n items.push(<MenuItem key=\"edit\" onClick={handleEdit}>Edit</MenuItem>);\n }\n items.push(<MenuItem key=\"delete\" onClick={handleDelete}>Delete</MenuItem>);\n return items;\n })()}\n </Menu>\n\n {editDialogState.request && (\n <EditAPIKeyRequestDialog\n open={editDialogState.open}\n request={editDialogState.request}\n availablePlans={editDialogState.plans}\n onClose={() => setEditDialogState({ open: false, request: null, plans: [] })}\n onSuccess={() => {\n setEditDialogState({ open: false, request: null, plans: [] });\n setRefresh(r => r + 1);\n }}\n />\n )}\n </>\n );\n};\n","import React from 'react';\nimport { Typography, Box } from '@material-ui/core';\nimport { Progress } from '@backstage/core-components';\nimport { Permission } from '@backstage/plugin-permission-common';\nimport { useKuadrantPermission } from '../../utils/permissions';\n\ninterface PermissionGateProps {\n children: React.ReactNode;\n permission: Permission;\n fallback?: React.ReactNode;\n errorMessage?: string;\n}\n\nexport const PermissionGate = ({ children, permission, fallback, errorMessage }: PermissionGateProps) => {\n const { allowed, loading, error } = useKuadrantPermission(permission);\n\n if (loading) {\n return <Progress />;\n }\n\n if (error) {\n return (\n <Box p={4}>\n <Typography color=\"error\">\n Unable to check permissions: {error.message}\n </Typography>\n <Typography variant=\"body2\" color=\"textSecondary\">\n Please try again or contact your administrator\n </Typography>\n </Box>\n );\n }\n\n if (!allowed) {\n if (fallback) {\n return <>{fallback}</>;\n }\n return (\n <Box p={4}>\n <Typography color=\"textSecondary\">\n {errorMessage || 'You don\\'t have permission to view this page'}\n </Typography>\n <Box mt={1}>\n <Typography variant=\"caption\" color=\"textSecondary\">\n Required permission: {permission.name}\n </Typography>\n </Box>\n </Box>\n );\n }\n\n return <>{children}</>;\n};\n","import React from 'react';\nimport { Box, Typography, Chip } from '@material-ui/core';\nimport { Alert } from '@material-ui/lab';\n\ninterface PlanPolicyDetailsProps {\n selectedPolicy: {\n metadata: {\n name: string;\n };\n plans?: Array<{\n tier: string;\n description?: string;\n limits?: {\n daily?: number;\n monthly?: number;\n yearly?: number;\n };\n }>;\n } | null;\n alertSeverity?: 'warning' | 'info';\n alertMessage?: string;\n includeTopMargin?: boolean;\n}\n\nexport const PlanPolicyDetails: React.FC<PlanPolicyDetailsProps> = ({\n selectedPolicy,\n alertSeverity = 'warning',\n alertMessage = 'No PlanPolicy found for this HTTPRoute. API keys and rate limiting may not be available.',\n includeTopMargin = true,\n}) => {\n return (\n <Box\n mt={includeTopMargin ? 1 : 0}\n p={2}\n bgcolor=\"#f5f5f5\"\n borderRadius={1}\n border=\"1px solid #e0e0e0\"\n >\n {selectedPolicy ? (\n <>\n <Typography variant=\"subtitle2\" gutterBottom style={{ fontWeight: 600 }}>\n Associated PlanPolicy: <strong>{selectedPolicy.metadata.name}</strong>\n </Typography>\n\n {selectedPolicy.plans && selectedPolicy.plans.length > 0 ? (\n <>\n <Typography\n variant=\"caption\"\n display=\"block\"\n gutterBottom\n color=\"textSecondary\"\n style={{ marginTop: 8 }}\n >\n Available Plans:\n </Typography>\n <Box display=\"flex\" flexWrap=\"wrap\" mt={1} style={{ gap: 8 }}>\n {selectedPolicy.plans.map((plan: any, idx: number) => {\n const limitText = plan.limits?.daily\n ? `${plan.limits.daily}/day`\n : plan.limits?.monthly\n ? `${plan.limits.monthly}/month`\n : plan.limits?.yearly\n ? `${plan.limits.yearly}/year`\n : 'No limit';\n\n return (\n <Chip\n key={idx}\n label={`${plan.tier}: ${limitText}`}\n size=\"small\"\n variant=\"outlined\"\n color=\"primary\"\n />\n );\n })}\n </Box>\n {selectedPolicy.plans.some((p: any) => p.description) && (\n <Box mt={1}>\n {selectedPolicy.plans.filter((p: any) => p.description).map((plan: any, idx: number) => (\n <Typography key={idx} variant=\"caption\" display=\"block\" color=\"textSecondary\">\n • <strong>{plan.tier}:</strong> {plan.description}\n </Typography>\n ))}\n </Box>\n )}\n </>\n ) : (\n <Typography variant=\"caption\" color=\"textSecondary\">\n No plans defined in this PlanPolicy\n </Typography>\n )}\n </>\n ) : (\n <Alert severity={alertSeverity}>{alertMessage}</Alert>\n )}\n </Box>\n );\n};\n\n","import React, { useState } from 'react';\nimport {\n Dialog,\n DialogTitle,\n DialogContent,\n DialogActions,\n Button,\n TextField,\n Box,\n Typography,\n Chip,\n Grid,\n MenuItem,\n makeStyles,\n} from '@material-ui/core';\nimport { useApi, configApiRef, fetchApiRef } from '@backstage/core-plugin-api';\nimport { Alert } from '@material-ui/lab';\nimport useAsync from 'react-use/lib/useAsync';\nimport { PlanPolicyDetails } from '../PlanPolicyDetailsCard';\n\nconst useStyles = makeStyles({\n asterisk: {\n color: '#f44336',\n },\n});\n\ninterface CreateAPIProductDialogProps {\n open: boolean;\n onClose: () => void;\n onSuccess: () => void;\n}\n\nexport const CreateAPIProductDialog = ({ open, onClose, onSuccess }: CreateAPIProductDialogProps) => {\n const classes = useStyles();\n const config = useApi(configApiRef);\n const fetchApi = useApi(fetchApiRef);\n const backendUrl = config.getString('backend.baseUrl');\n\n const [name, setName] = useState('');\n const [displayName, setDisplayName] = useState('');\n const [description, setDescription] = useState('');\n const [version, setVersion] = useState('v1');\n const [approvalMode, setApprovalMode] = useState<'automatic' | 'manual'>('manual');\n const [publishStatus, setPublishStatus] = useState<'Draft' | 'Published'>('Published');\n const [tags, setTags] = useState<string[]>([]);\n const [tagInput, setTagInput] = useState('');\n const [selectedHTTPRoute, setSelectedHTTPRoute] = useState('');\n const [contactEmail, setContactEmail] = useState('');\n const [contactTeam, setContactTeam] = useState('');\n const [docsURL, setDocsURL] = useState('');\n const [openAPISpec, setOpenAPISpec] = useState('');\n const [error, setError] = useState('');\n const [creating, setCreating] = useState(false);\n\n const { value: httpRoutes, loading: httpRoutesLoading } = useAsync(async () => {\n const response = await fetchApi.fetch(`${backendUrl}/api/kuadrant/httproutes`);\n const data = await response.json();\n // filter to only show httproutes annotated for backstage exposure\n return (data.items || []).filter((route: any) =>\n route.metadata.annotations?.['backstage.io/expose'] === 'true'\n );\n }, [backendUrl, fetchApi, open]);\n\n // load planpolicies with full details to show associated plans\n const { value: planPolicies } = useAsync(async () => {\n const response = await fetchApi.fetch(`${backendUrl}/api/kuadrant/planpolicies`);\n return await response.json();\n }, [backendUrl, fetchApi, open]);\n\n // find planpolicy associated with selected httproute\n const getPlanPolicyForRoute = (routeNamespace: string, routeName: string) => {\n if (!planPolicies?.items) return null;\n\n return planPolicies.items.find((pp: any) => {\n const ref = pp.targetRef;\n return (\n ref?.kind === 'HTTPRoute' &&\n ref?.name === routeName &&\n (!ref?.namespace || ref?.namespace === routeNamespace)\n );\n });\n };\n\n const selectedRouteInfo = selectedHTTPRoute ? selectedHTTPRoute.split('/') : null;\n const selectedPolicy = selectedRouteInfo\n ? getPlanPolicyForRoute(selectedRouteInfo[0], selectedRouteInfo[1])\n : null;\n\n const handleAddTag = () => {\n if (tagInput.trim() && !tags.includes(tagInput.trim())) {\n setTags([...tags, tagInput.trim()]);\n setTagInput('');\n }\n };\n\n const handleDeleteTag = (tagToDelete: string) => {\n setTags(tags.filter(tag => tag !== tagToDelete));\n };\n\n const handleCreate = async () => {\n setError('');\n setCreating(true);\n\n try {\n if (!selectedHTTPRoute) {\n throw new Error('Please select an HTTPRoute');\n }\n\n const [selectedRouteNamespace, selectedRouteName] = selectedHTTPRoute.split('/');\n\n // derive namespace from selected httproute\n const namespace = selectedRouteNamespace;\n\n const apiProduct = {\n apiVersion: 'extensions.kuadrant.io/v1alpha1',\n kind: 'APIProduct',\n metadata: {\n name,\n namespace,\n },\n spec: {\n displayName,\n description,\n version,\n approvalMode,\n publishStatus,\n tags,\n targetRef: {\n group: 'gateway.networking.k8s.io',\n kind: 'HTTPRoute',\n name: selectedRouteName,\n namespace: selectedRouteNamespace,\n },\n ...(contactEmail || contactTeam ? {\n contact: {\n ...(contactEmail && { email: contactEmail }),\n ...(contactTeam && { team: contactTeam }),\n },\n } : {}),\n ...(docsURL || openAPISpec ? {\n documentation: {\n ...(docsURL && { docsURL }),\n ...(openAPISpec && { openAPISpec }),\n },\n } : {}),\n },\n };\n\n const response = await fetchApi.fetch(`${backendUrl}/api/kuadrant/apiproducts`, {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n },\n body: JSON.stringify(apiProduct),\n });\n\n if (!response.ok) {\n const errorData = await response.json();\n throw new Error(errorData.error || 'failed to create apiproduct');\n }\n\n onSuccess();\n handleClose();\n } catch (err) {\n setError(err instanceof Error ? err.message : String(err));\n } finally {\n setCreating(false);\n }\n };\n\n const handleClose = () => {\n setName('');\n setDisplayName('');\n setDescription('');\n setVersion('v1');\n setApprovalMode('manual');\n setPublishStatus('Published');\n setTags([]);\n setTagInput('');\n setSelectedHTTPRoute('');\n setContactEmail('');\n setContactTeam('');\n setDocsURL('');\n setOpenAPISpec('');\n setError('');\n onClose();\n };\n\n return (\n <Dialog open={open} onClose={handleClose} maxWidth=\"md\" fullWidth>\n <DialogTitle>Create API Product</DialogTitle>\n <DialogContent>\n {error && (\n <Alert severity=\"error\" style={{ marginBottom: 16 }}>\n {error}\n </Alert>\n )}\n\n <Grid container spacing={2}>\n <Grid item xs={6}>\n <TextField\n fullWidth\n label=\"Name\"\n value={name}\n onChange={e => setName(e.target.value)}\n placeholder=\"my-api\"\n helperText=\"Kubernetes resource name (lowercase, hyphens)\"\n margin=\"normal\"\n required\n InputLabelProps={{\n classes: {\n asterisk: classes.asterisk,\n },\n }}\n />\n </Grid>\n <Grid item xs={6}>\n <TextField\n fullWidth\n label=\"Display Name\"\n value={displayName}\n onChange={e => setDisplayName(e.target.value)}\n placeholder=\"My API\"\n margin=\"normal\"\n required\n InputLabelProps={{\n classes: {\n asterisk: classes.asterisk,\n },\n }}\n />\n </Grid>\n <Grid item xs={6}>\n <TextField\n fullWidth\n label=\"Version\"\n value={version}\n onChange={e => setVersion(e.target.value)}\n placeholder=\"v1\"\n margin=\"normal\"\n />\n </Grid>\n <Grid item xs={12}>\n <TextField\n fullWidth\n select\n label=\"Approval Mode\"\n value={approvalMode}\n onChange={e => setApprovalMode(e.target.value as 'automatic' | 'manual')}\n margin=\"normal\"\n helperText=\"Automatic: keys are created immediately. Manual: requires approval.\"\n >\n <MenuItem value=\"manual\">Manual</MenuItem>\n <MenuItem value=\"automatic\">Automatic</MenuItem>\n </TextField>\n </Grid>\n <Grid item xs={12}>\n <TextField\n fullWidth\n select\n label=\"Publish Status\"\n value={publishStatus}\n onChange={e => setPublishStatus(e.target.value as 'Draft' | 'Published')}\n margin=\"normal\"\n helperText=\"Draft: hidden from catalog. Published: visible to consumers.\"\n >\n <MenuItem value=\"Draft\">Draft</MenuItem>\n <MenuItem value=\"Published\">Published</MenuItem>\n </TextField>\n </Grid>\n <Grid item xs={12}>\n <TextField\n fullWidth\n label=\"Description\"\n value={description}\n onChange={e => setDescription(e.target.value)}\n placeholder=\"API description\"\n margin=\"normal\"\n multiline\n rows={2}\n required\n InputLabelProps={{\n classes: {\n asterisk: classes.asterisk,\n },\n }}\n />\n </Grid>\n\n <Grid item xs={12}>\n <Typography variant=\"subtitle2\" gutterBottom style={{ marginTop: 16 }}>\n Tags\n </Typography>\n <Box display=\"flex\" flexWrap=\"wrap\" marginBottom={1} style={{ gap: 8 }}>\n {tags.map(tag => (\n <Chip\n key={tag}\n label={tag}\n onDelete={() => handleDeleteTag(tag)}\n size=\"small\"\n />\n ))}\n </Box>\n <Box display=\"flex\" style={{ gap: 8 }}>\n <TextField\n fullWidth\n size=\"small\"\n value={tagInput}\n onChange={e => setTagInput(e.target.value)}\n onKeyPress={e => e.key === 'Enter' && handleAddTag()}\n placeholder=\"Add tag\"\n />\n <Button onClick={handleAddTag} variant=\"outlined\" size=\"small\">\n Add\n </Button>\n </Box>\n </Grid>\n\n <Grid item xs={12}>\n <TextField\n fullWidth\n select\n label=\"HTTPRoute\"\n value={selectedHTTPRoute}\n onChange={e => setSelectedHTTPRoute(e.target.value)}\n margin=\"normal\"\n required\n helperText=\"Select an HTTPRoute (backstage.io/expose: true). APIProduct will be created in the same namespace.\"\n disabled={httpRoutesLoading}\n InputLabelProps={{\n classes: {\n asterisk: classes.asterisk,\n },\n }}\n >\n {httpRoutesLoading && (\n <MenuItem value=\"\">Loading...</MenuItem>\n )}\n {!httpRoutesLoading && httpRoutes && httpRoutes.length === 0 && (\n <MenuItem value=\"\">No HTTPRoutes available</MenuItem>\n )}\n {!httpRoutesLoading && httpRoutes && httpRoutes.map((route: any) => (\n <MenuItem\n key={`${route.metadata.namespace}/${route.metadata.name}`}\n value={`${route.metadata.namespace}/${route.metadata.name}`}\n >\n {route.metadata.name} ({route.metadata.namespace})\n </MenuItem>\n ))}\n </TextField>\n </Grid>\n {selectedHTTPRoute && (\n <Grid item xs={12}>\n <PlanPolicyDetails\n selectedPolicy={selectedPolicy}\n alertSeverity=\"warning\"\n alertMessage=\"No PlanPolicy found for this HTTPRoute. API keys and rate limiting may not be available.\"\n includeTopMargin={true}\n />\n </Grid>\n )}\n\n <Grid item xs={6}>\n <TextField\n fullWidth\n label=\"Contact Email\"\n value={contactEmail}\n onChange={e => setContactEmail(e.target.value)}\n placeholder=\"api-team@example.com\"\n margin=\"normal\"\n />\n </Grid>\n <Grid item xs={6}>\n <TextField\n fullWidth\n label=\"Contact Team\"\n value={contactTeam}\n onChange={e => setContactTeam(e.target.value)}\n placeholder=\"platform-team\"\n margin=\"normal\"\n />\n </Grid>\n <Grid item xs={6}>\n <TextField\n fullWidth\n label=\"Docs URL\"\n value={docsURL}\n onChange={e => setDocsURL(e.target.value)}\n placeholder=\"https://api.example.com/docs\"\n margin=\"normal\"\n />\n </Grid>\n <Grid item xs={6}>\n <TextField\n fullWidth\n label=\"OpenAPI Spec URL\"\n value={openAPISpec}\n onChange={e => setOpenAPISpec(e.target.value)}\n placeholder=\"https://api.example.com/openapi.json\"\n margin=\"normal\"\n />\n </Grid>\n </Grid>\n </DialogContent>\n <DialogActions>\n <Button onClick={handleClose}>Cancel</Button>\n <Button\n onClick={handleCreate}\n color=\"primary\"\n variant=\"contained\"\n disabled={creating || !name || !displayName || !description || !selectedHTTPRoute}\n >\n {creating ? 'Creating...' : 'Create'}\n </Button>\n </DialogActions>\n </Dialog>\n );\n};\n","import React, { useState, useEffect } from 'react';\nimport {\n Dialog,\n DialogTitle,\n DialogContent,\n DialogActions,\n Button,\n TextField,\n Box,\n Typography,\n Chip,\n Grid,\n MenuItem,\n makeStyles,\n} from '@material-ui/core';\nimport { useApi, configApiRef, fetchApiRef } from '@backstage/core-plugin-api';\nimport { Alert } from '@material-ui/lab';\nimport { Progress } from '@backstage/core-components';\nimport useAsync from 'react-use/lib/useAsync';\nimport { PlanPolicyDetails } from '../PlanPolicyDetailsCard';\n\nconst useStyles = makeStyles({\n asterisk: {\n color: '#f44336',\n },\n});\n\ninterface EditAPIProductDialogProps {\n open: boolean;\n onClose: () => void;\n onSuccess: () => void;\n namespace: string;\n name: string;\n}\n\nexport const EditAPIProductDialog = ({open, onClose, onSuccess, namespace, name}: EditAPIProductDialogProps) => {\n const classes = useStyles();\n const config = useApi(configApiRef);\n const fetchApi = useApi(fetchApiRef);\n const backendUrl = config.getString('backend.baseUrl');\n const [loading, setLoading] = useState(false);\n const [displayName, setDisplayName] = useState('');\n const [description, setDescription] = useState('');\n const [version, setVersion] = useState('v1');\n const [publishStatus, setPublishStatus] = useState<'Draft' | 'Published'>('Draft');\n const [approvalMode, setApprovalMode] = useState<'automatic' | 'manual'>('manual');\n const [tags, setTags] = useState<string[]>([]);\n const [targetRef, setTargetRef] = useState<any>(null);\n const [tagInput, setTagInput] = useState('');\n const [contactEmail, setContactEmail] = useState('');\n const [contactTeam, setContactTeam] = useState('');\n const [docsURL, setDocsURL] = useState('');\n const [openAPISpec, setOpenAPISpec] = useState('');\n const [error, setError] = useState('');\n const [saving, setSaving] = useState(false);\n\n // Load APIProduct data when dialog opens\n useEffect(() => {\n if (open && namespace && name) {\n setLoading(true);\n setError('');\n\n fetchApi.fetch(`${backendUrl}/api/kuadrant/apiproducts/${namespace}/${name}`)\n .then(async res => {\n if (!res.ok) {\n const errorData = await res.json();\n throw new Error(errorData.error || `Failed to fetch API product: ${res.status}`);\n }\n return res.json();\n })\n .then(data => {\n setDisplayName(data.spec.displayName || '');\n setDescription(data.spec.description || '');\n setVersion(data.spec.version || 'v1');\n setPublishStatus(data.spec.publishStatus || 'Draft');\n setApprovalMode(data.spec.approvalMode || 'manual');\n setTags(data.spec.tags || []);\n setTargetRef(data.spec.targetRef || null);\n setContactEmail(data.spec.contact?.email || '');\n setContactTeam(data.spec.contact?.team || '');\n setDocsURL(data.spec.documentation?.docsURL || '');\n setOpenAPISpec(data.spec.documentation?.openAPISpec || '');\n setLoading(false);\n })\n .catch(err => {\n setError(err.message || 'Failed to load API product');\n setLoading(false);\n });\n }\n }, [open, namespace, name, backendUrl, fetchApi]);\n\n // load planpolicies with full details to show associated plans\n const { value: planPolicies } = useAsync(async () => {\n if (!open) return null;\n const response = await fetchApi.fetch(`${backendUrl}/api/kuadrant/planpolicies`);\n return await response.json();\n }, [backendUrl, fetchApi, open]);\n\n // find planpolicy associated with targetRef\n const selectedPolicy = React.useMemo(() => {\n if (!planPolicies?.items || !targetRef) return null;\n\n return planPolicies.items.find((pp: any) => {\n const ref = pp.targetRef;\n return (\n ref?.kind === 'HTTPRoute' &&\n ref?.name === targetRef.name &&\n (!ref?.namespace || ref?.namespace === (targetRef.namespace || namespace))\n );\n });\n }, [planPolicies, targetRef, namespace]);\n\n const handleAddTag = () => {\n if (tagInput.trim() && !tags.includes(tagInput.trim())) {\n setTags([...tags, tagInput.trim()]);\n setTagInput('');\n }\n };\n\n const handleDeleteTag = (tagToDelete: string) => {\n setTags(tags.filter(tag => tag !== tagToDelete));\n };\n\n const handleSave = async () => {\n setError('');\n setSaving(true);\n\n try {\n const patch = {\n spec: {\n displayName,\n description,\n version,\n publishStatus,\n approvalMode,\n tags,\n targetRef,\n ...(contactEmail || contactTeam ? {\n contact: {\n ...(contactEmail && { email: contactEmail }),\n ...(contactTeam && { team: contactTeam }),\n },\n } : {}),\n ...(docsURL || openAPISpec ? {\n documentation: {\n ...(docsURL && { docsURL }),\n ...(openAPISpec && { openAPISpec }),\n },\n } : {}),\n },\n };\n\n const response = await fetchApi.fetch(\n `${backendUrl}/api/kuadrant/apiproducts/${namespace}/${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();\n throw new Error(errorData.error || 'failed to update apiproduct');\n }\n\n onSuccess();\n onClose();\n } catch (err) {\n setError(err instanceof Error ? err.message : String(err));\n } finally {\n setSaving(false);\n }\n };\n\n return (\n <Dialog open={open} onClose={onClose} maxWidth=\"md\" fullWidth>\n <DialogTitle>Edit API Product</DialogTitle>\n <DialogContent>\n {error && (\n <Alert severity=\"error\" style={{ marginBottom: 16 }}>\n {error}\n </Alert>\n )}\n\n {loading ? (\n <Progress />\n ) : (\n <Grid container spacing={2}>\n <Grid item xs={6}>\n <TextField\n fullWidth\n label=\"Name\"\n value={name}\n disabled\n helperText=\"Kubernetes resource name (immutable)\"\n margin=\"normal\"\n />\n </Grid>\n <Grid item xs={6}>\n <TextField\n fullWidth\n label=\"Namespace\"\n value={namespace}\n disabled\n helperText=\"Derived from HTTPRoute (immutable)\"\n margin=\"normal\"\n />\n </Grid>\n <Grid item xs={6}>\n <TextField\n fullWidth\n label=\"Display Name\"\n value={displayName}\n onChange={e => setDisplayName(e.target.value)}\n placeholder=\"My API\"\n margin=\"normal\"\n required\n InputLabelProps={{\n classes: {\n asterisk: classes.asterisk,\n },\n }}\n />\n </Grid>\n <Grid item xs={6}>\n <TextField\n fullWidth\n label=\"Version\"\n value={version}\n onChange={e => setVersion(e.target.value)}\n placeholder=\"v1\"\n margin=\"normal\"\n />\n </Grid>\n <Grid item xs={6}>\n <TextField\n fullWidth\n select\n label=\"Publish Status\"\n value={publishStatus}\n onChange={e => setPublishStatus(e.target.value as 'Draft' | 'Published')}\n margin=\"normal\"\n helperText=\"Draft: Hidden from catalog. Published: Visible in catalog.\"\n >\n <MenuItem value=\"Draft\">Draft (Hidden)</MenuItem>\n <MenuItem value=\"Published\">Published (Visible)</MenuItem>\n </TextField>\n </Grid>\n <Grid item xs={6}>\n <TextField\n fullWidth\n select\n label=\"Approval Mode\"\n value={approvalMode}\n onChange={e => setApprovalMode(e.target.value as 'automatic' | 'manual')}\n margin=\"normal\"\n helperText=\"Automatic: keys created immediately. Manual: requires approval.\"\n >\n <MenuItem value=\"manual\">Manual</MenuItem>\n <MenuItem value=\"automatic\">Automatic</MenuItem>\n </TextField>\n </Grid>\n <Grid item xs={12}>\n <TextField\n fullWidth\n label=\"Description\"\n value={description}\n onChange={e => setDescription(e.target.value)}\n placeholder=\"API description\"\n margin=\"normal\"\n multiline\n rows={2}\n required\n InputLabelProps={{\n classes: {\n asterisk: classes.asterisk,\n },\n }}\n />\n </Grid>\n\n <Grid item xs={12}>\n <Typography variant=\"subtitle2\" gutterBottom style={{ marginTop: 16 }}>\n Tags\n </Typography>\n <Box display=\"flex\" flexWrap=\"wrap\" marginBottom={1} style={{ gap: 8 }}>\n {tags.map(tag => (\n <Chip\n key={tag}\n label={tag}\n onDelete={() => handleDeleteTag(tag)}\n size=\"small\"\n />\n ))}\n </Box>\n <Box display=\"flex\" style={{ gap: 8 }}>\n <TextField\n fullWidth\n size=\"small\"\n value={tagInput}\n onChange={e => setTagInput(e.target.value)}\n onKeyPress={e => e.key === 'Enter' && handleAddTag()}\n placeholder=\"Add tag\"\n />\n <Button onClick={handleAddTag} variant=\"outlined\" size=\"small\">\n Add\n </Button>\n </Box>\n </Grid>\n {targetRef && (\n <>\n <Grid item xs={12}>\n <TextField\n fullWidth\n label=\"HTTPRoute\"\n value={`${targetRef.namespace || namespace}/${targetRef.name}`}\n disabled\n helperText=\"Target HTTPRoute (immutable)\"\n margin=\"normal\"\n />\n </Grid>\n\n <Grid item xs={12}>\n <PlanPolicyDetails\n selectedPolicy={selectedPolicy}\n alertSeverity=\"info\"\n alertMessage=\"No PlanPolicy found for this HTTPRoute.\"\n includeTopMargin={false}\n />\n </Grid>\n </>\n )}\n\n <Grid item xs={6}>\n <TextField\n fullWidth\n label=\"Contact Email\"\n value={contactEmail}\n onChange={e => setContactEmail(e.target.value)}\n placeholder=\"api-team@example.com\"\n margin=\"normal\"\n />\n </Grid>\n <Grid item xs={6}>\n <TextField\n fullWidth\n label=\"Contact Team\"\n value={contactTeam}\n onChange={e => setContactTeam(e.target.value)}\n placeholder=\"platform-team\"\n margin=\"normal\"\n />\n </Grid>\n <Grid item xs={6}>\n <TextField\n fullWidth\n label=\"Docs URL\"\n value={docsURL}\n onChange={e => setDocsURL(e.target.value)}\n placeholder=\"https://api.example.com/docs\"\n margin=\"normal\"\n />\n </Grid>\n <Grid item xs={6}>\n <TextField\n fullWidth\n label=\"OpenAPI Spec URL\"\n value={openAPISpec}\n onChange={e => setOpenAPISpec(e.target.value)}\n placeholder=\"https://api.example.com/openapi.json\"\n margin=\"normal\"\n />\n </Grid>\n </Grid>\n )}\n </DialogContent>\n <DialogActions>\n <Button onClick={onClose}>Cancel</Button>\n <Button\n onClick={handleSave}\n color=\"primary\"\n variant=\"contained\"\n disabled={saving || loading || !displayName || !description}\n >\n {saving ? 'Saving...' : 'Save'}\n </Button>\n </DialogActions>\n </Dialog>\n );\n};\n","import React, { useState } from 'react';\nimport { Typography, Grid, Box, Chip, Button, IconButton, Dialog, DialogTitle, DialogContent, DialogContentText, DialogActions } from '@material-ui/core';\nimport AddIcon from '@material-ui/icons/Add';\nimport DeleteIcon from '@material-ui/icons/Delete';\nimport EditIcon from '@material-ui/icons/Edit';\nimport {\n InfoCard,\n Header,\n Page,\n Content,\n SupportButton,\n Progress,\n ResponseErrorPanel,\n Link,\n Table,\n TableColumn,\n} from '@backstage/core-components';\nimport useAsync from 'react-use/lib/useAsync';\nimport { useApi, configApiRef, fetchApiRef } from '@backstage/core-plugin-api';\nimport { ApprovalQueueCard } from '../ApprovalQueueCard';\nimport { MyApiKeysCard } from '../MyApiKeysCard';\nimport { PermissionGate } from '../PermissionGate';\nimport { CreateAPIProductDialog } from '../CreateAPIProductDialog';\nimport {\n kuadrantApiProductCreatePermission,\n kuadrantApiProductDeleteOwnPermission,\n kuadrantApiProductDeleteAllPermission,\n kuadrantApiProductUpdateOwnPermission,\n kuadrantApiProductUpdateAllPermission,\n kuadrantApiProductListPermission,\n kuadrantApiKeyRequestReadAllPermission,\n kuadrantApiKeyRequestReadOwnPermission,\n kuadrantPlanPolicyListPermission,\n} from '../../permissions';\nimport { useKuadrantPermission } from '../../utils/permissions';\nimport { EditAPIProductDialog } from '../EditAPIProductDialog';\n\ntype KuadrantResource = {\n metadata: {\n name: string;\n namespace: string;\n creationTimestamp: string;\n };\n spec?: any;\n};\n\ntype KuadrantList = {\n items: KuadrantResource[];\n};\n\nexport const ResourceList = () => {\n const config = useApi(configApiRef);\n const fetchApi = useApi(fetchApiRef);\n const backendUrl = config.getString('backend.baseUrl');\n const [createDialogOpen, setCreateDialogOpen] = useState(false);\n const [editDialogOpen, setEditDialogOpen] = useState(false);\n const [refreshTrigger, setRefreshTrigger] = useState(0);\n const [deleteDialogOpen, setDeleteDialogOpen] = useState(false);\n const [apiProductToDelete, setApiProductToDelete] = useState<{ namespace: string; name: string } | null>(null);\n const [apiProductToEdit, setApiProductToEdit] = useState<{ namespace: string; name: string } | null>(null);\n const [deleting, setDeleting] = useState(false);\n\n const {\n allowed: canCreateApiProduct,\n loading: createPermissionLoading,\n error: createPermissionError,\n } = useKuadrantPermission(kuadrantApiProductCreatePermission);\n\n const {\n allowed: canViewAllRequests,\n loading: approvalQueueAllPermissionLoading,\n } = useKuadrantPermission(kuadrantApiKeyRequestReadAllPermission);\n\n const {\n allowed: canViewOwnRequests,\n loading: approvalQueueOwnPermissionLoading,\n error: approvalQueuePermissionError,\n } = useKuadrantPermission(kuadrantApiKeyRequestReadOwnPermission);\n\n const canViewApprovalQueue = canViewAllRequests || canViewOwnRequests;\n const approvalQueuePermissionLoading = approvalQueueAllPermissionLoading || approvalQueueOwnPermissionLoading;\n\n const {\n allowed: canDeleteOwnApiProduct,\n loading: deleteOwnPermissionLoading,\n } = useKuadrantPermission(kuadrantApiProductDeleteOwnPermission);\n\n const {\n allowed: canDeleteAllApiProducts,\n loading: deleteAllPermissionLoading,\n error: deletePermissionError,\n } = useKuadrantPermission(kuadrantApiProductDeleteAllPermission);\n\n const {\n allowed: canUpdateOwnApiProduct,\n } = useKuadrantPermission(kuadrantApiProductUpdateOwnPermission);\n\n const {\n allowed: canUpdateAllApiProducts,\n } = useKuadrantPermission(kuadrantApiProductUpdateAllPermission);\n\n const canDeleteApiProduct = canDeleteOwnApiProduct || canDeleteAllApiProducts;\n const canUpdateApiProduct = canUpdateOwnApiProduct || canUpdateAllApiProducts;\n const deletePermissionLoading = deleteOwnPermissionLoading || deleteAllPermissionLoading;\n\n const {\n allowed: canListPlanPolicies,\n loading: planPolicyPermissionLoading,\n error: planPolicyPermissionError,\n } = useKuadrantPermission(kuadrantPlanPolicyListPermission);\n\n const { value: apiProducts, loading: apiProductsLoading, error: apiProductsError } = useAsync(async (): Promise<KuadrantList> => {\n const response = await fetchApi.fetch(`${backendUrl}/api/kuadrant/apiproducts`);\n return await response.json();\n }, [backendUrl, fetchApi, refreshTrigger]);\n\n const { value: planPolicies, loading: planPoliciesLoading, error: planPoliciesError } = useAsync(async (): Promise<KuadrantList> => {\n const response = await fetchApi.fetch(`${backendUrl}/api/kuadrant/planpolicies`);\n return await response.json();\n }, [backendUrl, fetchApi, refreshTrigger]);\n\n const loading = apiProductsLoading || planPoliciesLoading || createPermissionLoading || approvalQueuePermissionLoading || deletePermissionLoading || planPolicyPermissionLoading;\n const error = apiProductsError || planPoliciesError;\n const permissionError = createPermissionError || approvalQueuePermissionError || deletePermissionError || planPolicyPermissionError;\n\n const handleCreateSuccess = () => {\n setRefreshTrigger(prev => prev + 1);\n };\n\n const handleEditClick = (namespace: string, name: string) => {\n setApiProductToEdit({ namespace, name });\n setEditDialogOpen(true);\n };\n\n const handleEditSuccess = () => {\n setRefreshTrigger(prev => prev + 1);\n };\n\n const handleDeleteClick = (namespace: string, name: string) => {\n setApiProductToDelete({ namespace, name });\n setDeleteDialogOpen(true);\n };\n\n const handleDeleteConfirm = async () => {\n if (!apiProductToDelete) return;\n\n setDeleting(true);\n try {\n const response = await fetchApi.fetch(\n `${backendUrl}/api/kuadrant/apiproducts/${apiProductToDelete.namespace}/${apiProductToDelete.name}`,\n { method: 'DELETE' }\n );\n\n if (!response.ok) {\n throw new Error('failed to delete apiproduct');\n }\n\n setRefreshTrigger(prev => prev + 1);\n } catch (err) {\n console.error('error deleting apiproduct:', err);\n } finally {\n setDeleting(false);\n setDeleteDialogOpen(false);\n setApiProductToDelete(null);\n }\n };\n\n const handleDeleteCancel = () => {\n setDeleteDialogOpen(false);\n setApiProductToDelete(null);\n };\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 });\n };\n\n const columns: TableColumn[] = [\n {\n title: 'Name',\n field: 'name',\n render: (row: any) => (\n <Link to={`/catalog/default/api/${row.metadata.name}/api-product`}>\n <strong>{row.spec?.displayName || row.metadata.name}</strong>\n </Link>\n ),\n },\n {\n title: 'Resource Name',\n field: 'metadata.name',\n },\n {\n title: 'Version',\n field: 'spec.version',\n render: (row: any) => row.spec?.version || '-',\n },\n {\n title: 'HTTPRoute',\n field: 'spec.targetRef.name',\n render: (row: any) => row.spec?.targetRef?.name || '-',\n },\n {\n title: 'Publish Status',\n field: 'spec.publishStatus',\n render: (row: any) => {\n const status = row.spec?.publishStatus || 'Draft';\n return (\n <Chip\n label={status}\n size=\"small\"\n color={status === 'Published' ? 'primary' : 'default'}\n />\n );\n },\n },\n {\n title: 'Approval Mode',\n field: 'spec.approvalMode',\n render: (row: any) => {\n const mode = row.spec?.approvalMode || 'manual';\n return (\n <Chip\n label={mode}\n size=\"small\"\n color={mode === 'automatic' ? 'secondary' : 'default'}\n />\n );\n },\n },\n {\n title: 'Namespace',\n field: 'metadata.namespace',\n },\n {\n title: 'Created',\n field: 'metadata.creationTimestamp',\n render: (row: any) => formatDate(row.metadata.creationTimestamp),\n },\n {\n title: 'Actions',\n field: 'actions',\n render: (row: any) => (\n <Box display=\"flex\" style={{ gap: 4 }}>\n {canUpdateApiProduct && (\n <IconButton\n size=\"small\"\n onClick={() => handleEditClick(row.metadata.namespace, row.metadata.name)}\n title=\"Edit API Product\"\n >\n <EditIcon fontSize=\"small\" />\n </IconButton>\n )}\n \n {canDeleteApiProduct && (\n <IconButton\n size=\"small\"\n onClick={() => handleDeleteClick(row.metadata.namespace, row.metadata.name)}\n title=\"Delete API Product\"\n >\n <DeleteIcon fontSize=\"small\" />\n </IconButton>\n )}\n </Box>\n ),\n },\n ];\n\n const planPolicyColumns: TableColumn[] = [\n {\n title: 'Name',\n field: 'metadata.name',\n render: (row: any) => (\n <Link to={`/kuadrant/planpolicy/${row.metadata.namespace}/${row.metadata.name}`}>\n <strong>{row.metadata.name}</strong>\n </Link>\n ),\n },\n {\n title: 'Namespace',\n field: 'metadata.namespace',\n },\n ];\n\n const renderResources = (resources: KuadrantResource[] | undefined) => {\n if (!resources || resources.length === 0) {\n return <Typography variant=\"body2\" color=\"textSecondary\">No API products found</Typography>;\n }\n return (\n <Table\n options={{ paging: false, search: false, toolbar: false }}\n columns={columns}\n data={resources}\n />\n );\n };\n\n const renderPlanPolicies = (resources: KuadrantResource[] | undefined) => {\n if (!resources || resources.length === 0) {\n return <Typography variant=\"body2\" color=\"textSecondary\">No plan policies found</Typography>;\n }\n return (\n <Table\n options={{ paging: false, search: false, toolbar: false }}\n columns={planPolicyColumns}\n data={resources}\n />\n );\n };\n\n return (\n <Page themeId=\"tool\">\n <Header title=\"Kuadrant\" subtitle=\"API management for Kubernetes\">\n <SupportButton>Manage API products and access requests</SupportButton>\n </Header>\n <Content>\n {loading && <Progress />}\n {error && <ResponseErrorPanel error={error} />}\n {permissionError && (\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: {createPermissionError ? 'kuadrant.apiproduct.create' :\n deletePermissionError ? 'kuadrant.apiproduct.delete' :\n approvalQueuePermissionError ? 'kuadrant.apikeyrequest.read.all' :\n planPolicyPermissionError ? 'kuadrant.planpolicy.list' : 'unknown'}\n </Typography>\n <Typography variant=\"body2\" color=\"textSecondary\">\n please try again or contact your administrator\n </Typography>\n </Box>\n )}\n {!loading && !error && !permissionError && (\n <Grid container spacing={3} direction=\"column\">\n <Grid item>\n <MyApiKeysCard />\n </Grid>\n\n <Grid item>\n <InfoCard\n title=\"API Products\"\n action={\n canCreateApiProduct ? (\n <Box display=\"flex\" alignItems=\"center\" height=\"100%\" mt={1}>\n <Button\n variant=\"contained\"\n color=\"primary\"\n size=\"small\"\n startIcon={<AddIcon />}\n onClick={() => setCreateDialogOpen(true)}\n >\n Create API Product\n </Button>\n </Box>\n ) : undefined\n }\n >\n {renderResources(apiProducts?.items)}\n </InfoCard>\n </Grid>\n\n {canListPlanPolicies && (\n <Grid item>\n <InfoCard title=\"Plan Policies\">\n {renderPlanPolicies(planPolicies?.items)}\n </InfoCard>\n </Grid>\n )}\n\n {canViewApprovalQueue && (\n <Grid item>\n <ApprovalQueueCard />\n </Grid>\n )}\n </Grid>\n )}\n <CreateAPIProductDialog\n open={createDialogOpen}\n onClose={() => setCreateDialogOpen(false)}\n onSuccess={handleCreateSuccess}\n />\n <EditAPIProductDialog\n open={editDialogOpen}\n onClose={() => setEditDialogOpen(false)}\n onSuccess={handleEditSuccess}\n namespace={apiProductToEdit?.namespace || ''}\n name={apiProductToEdit?.name || ''}\n />\n <Dialog open={deleteDialogOpen} onClose={handleDeleteCancel}>\n <DialogTitle>Delete API Product</DialogTitle>\n <DialogContent>\n <DialogContentText>\n Are you sure you want to delete {apiProductToDelete?.name} from namespace {apiProductToDelete?.namespace}?\n This will permanently remove the API Product from Kubernetes.\n </DialogContentText>\n </DialogContent>\n <DialogActions>\n <Button onClick={handleDeleteCancel} color=\"primary\">\n Cancel\n </Button>\n <Button onClick={handleDeleteConfirm} color=\"secondary\" disabled={deleting}>\n {deleting ? 'Deleting...' : 'Delete'}\n </Button>\n </DialogActions>\n </Dialog>\n </Content>\n </Page>\n );\n};\n\nexport const KuadrantPage = () => {\n return (\n <PermissionGate\n permission={kuadrantApiProductListPermission}\n errorMessage=\"you don't have permission to view the Kuadrant page\"\n >\n <ResourceList />\n </PermissionGate>\n );\n};\n"],"names":["EditAPIKeyRequestDialog","open","onClose","onSuccess","request","availablePlans","config","useApi","configApiRef","fetchApi","fetchApiRef","backendUrl","getString","planTier","setPlanTier","useState","useCase","setUseCase","saving","setSaving","error","setError","useEffect","spec","handleClose","Dialog","maxWidth","fullWidth","DialogTitle","DialogContent","Box","mb","p","bgcolor","color","borderRadius","Typography","variant","FormControl","margin","InputLabel","Select","value","onChange","e","target","disabled","map","plan","limitDesc","Object","entries","limits","key","val","join","MenuItem","tier","TextField","label","placeholder","multiline","rows","helperText","DialogActions","Button","onClick","async","patch","trim","response","fetch","metadata","namespace","name","method","headers","body","JSON","stringify","ok","errorData","json","catch","Error","status","err","console","message","MyApiKeysCard","identityApi","identityApiRef","selectedTab","setSelectedTab","setUserId","visibleKeys","setVisibleKeys","Set","menuAnchor","setMenuAnchor","menuRequest","setMenuRequest","editDialogState","setEditDialogState","plans","refresh","setRefresh","useAsync","identity","getBackstageIdentity","extractedUserId","userEntityRef","split","log","requests","loading","items","InfoCard","title","allRequests","approvedRequests","filter","r","phase","pendingRequests","rejectedRequests","handleMenuClose","handleEdit","apiProductResponse","apiNamespace","apiName","apiProduct","handleDelete","window","confirm","alert","columns","field","render","row","Link","to","strong","Chip","size","Tooltip","placement","style","overflow","textOverflow","whiteSpace","reason","reviewedBy","reviewedDate","reviewedAt","Date","toLocaleDateString","apiKey","isVisible","has","display","alignItems","gap","fontFamily","fontSize","repeat","IconButton","toggleKeyVisibility","keyName","prev","newSet","delete","add","VisibilityOffIcon","VisibilityIcon","creationTimestamp","date","stopPropagation","rect","currentTarget","getBoundingClientRect","top","bottom","left","aria-controls","undefined","aria-haspopup","MoreVertIcon","tabData","getTabData","tabColumns","col","getTabColumns","subheader","length","Tabs","_","newValue","indicatorColor","textColor","Tab","textAlign","Table","options","paging","pageSize","search","toolbar","data","item","id","Menu","Boolean","anchorReference","anchorPosition","push","PermissionGate","children","permission","fallback","errorMessage","allowed","useKuadrantPermission","Progress","mt","PlanPolicyDetails","selectedPolicy","alertSeverity","alertMessage","includeTopMargin","border","gutterBottom","fontWeight","marginTop","flexWrap","idx","limitText","daily","monthly","yearly","some","description","Alert","severity","useStyles","makeStyles","asterisk","CreateAPIProductDialog","classes","setName","displayName","setDisplayName","setDescription","version","setVersion","approvalMode","setApprovalMode","publishStatus","setPublishStatus","tags","setTags","tagInput","setTagInput","selectedHTTPRoute","setSelectedHTTPRoute","contactEmail","setContactEmail","contactTeam","setContactTeam","docsURL","setDocsURL","openAPISpec","setOpenAPISpec","creating","setCreating","httpRoutes","httpRoutesLoading","route","annotations","planPolicies","selectedRouteInfo","routeNamespace","routeName","find","pp","ref","targetRef","kind","handleAddTag","includes","marginBottom","Grid","container","spacing","xs","required","InputLabelProps","select","tag","onDelete","handleDeleteTag","tagToDelete","onKeyPress","selectedRouteNamespace","selectedRouteName","apiVersion","group","contact","email","team","documentation","String","EditAPIProductDialog","setLoading","setTargetRef","then","res","React","ResourceList","createDialogOpen","setCreateDialogOpen","editDialogOpen","setEditDialogOpen","refreshTrigger","setRefreshTrigger","deleteDialogOpen","setDeleteDialogOpen","apiProductToDelete","setApiProductToDelete","apiProductToEdit","setApiProductToEdit","deleting","setDeleting","canCreateApiProduct","createPermissionLoading","createPermissionError","kuadrantApiProductCreatePermission","canViewAllRequests","approvalQueueAllPermissionLoading","kuadrantApiKeyRequestReadAllPermission","canViewOwnRequests","approvalQueueOwnPermissionLoading","approvalQueuePermissionError","kuadrantApiKeyRequestReadOwnPermission","canViewApprovalQueue","approvalQueuePermissionLoading","canDeleteOwnApiProduct","deleteOwnPermissionLoading","kuadrantApiProductDeleteOwnPermission","canDeleteAllApiProducts","deleteAllPermissionLoading","deletePermissionError","kuadrantApiProductDeleteAllPermission","canUpdateOwnApiProduct","kuadrantApiProductUpdateOwnPermission","canUpdateAllApiProducts","kuadrantApiProductUpdateAllPermission","canDeleteApiProduct","canUpdateApiProduct","deletePermissionLoading","canListPlanPolicies","planPolicyPermissionLoading","planPolicyPermissionError","kuadrantPlanPolicyListPermission","apiProducts","apiProductsLoading","apiProductsError","planPoliciesLoading","planPoliciesError","permissionError","handleDeleteCancel","mode","formatDate","dateString","year","month","day","handleEditClick","EditIcon","handleDeleteClick","DeleteIcon","planPolicyColumns","Page","themeId","Header","subtitle","SupportButton","Content","ResponseErrorPanel","direction","action","height","startIcon","AddIcon","resources","renderPlanPolicies","ApprovalQueueCard","DialogContentText","KuadrantPage","kuadrantApiProductListPermission"],"sourceRoot":""}