@kuadrant/kuadrant-backstage-plugin-frontend 0.0.2-dev-06b7d4e → 0.0.2-dev-e57ae53
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/components/CreateAPIProductDialog/CreateAPIProductDialog.esm.js +27 -9
- package/dist/components/CreateAPIProductDialog/CreateAPIProductDialog.esm.js.map +1 -1
- package/dist/components/EditAPIProductDialog/EditAPIProductDialog.esm.js +6 -3
- package/dist/components/EditAPIProductDialog/EditAPIProductDialog.esm.js.map +1 -1
- package/dist-scalprum/{internal.plugin-kuadrant.6fccbd5914b453589217.js → internal.plugin-kuadrant.d57df6840acced29678f.js} +2 -2
- package/dist-scalprum/{internal.plugin-kuadrant.6fccbd5914b453589217.js.map → internal.plugin-kuadrant.d57df6840acced29678f.js.map} +1 -1
- package/dist-scalprum/plugin-manifest.json +2 -2
- package/dist-scalprum/static/7632.a43cc47c.chunk.js +2 -0
- package/dist-scalprum/static/7632.a43cc47c.chunk.js.map +1 -0
- package/package.json +1 -1
- package/dist-scalprum/static/7632.3141cc8f.chunk.js +0 -2
- package/dist-scalprum/static/7632.3141cc8f.chunk.js.map +0 -1
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"static/7632.a43cc47c.chunk.js","mappings":"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","sources":["webpack://internal.plugin-kuadrant/./src/components/ConfirmDeleteDialog/ConfirmDeleteDialog.tsx","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 DialogContentText,\n DialogActions,\n Button,\n TextField,\n Typography,\n Box,\n CircularProgress,\n} from '@material-ui/core';\nimport WarningIcon from '@material-ui/icons/Warning';\n\nexport interface ConfirmDeleteDialogProps {\n open: boolean;\n title: string;\n description: string;\n // for dangerous deletes, require typing this text to confirm\n confirmText?: string;\n // severity affects styling - 'high' shows warning icon and requires text confirmation\n severity?: 'normal' | 'high';\n deleting?: boolean;\n onConfirm: () => void;\n onCancel: () => void;\n}\n\nexport const ConfirmDeleteDialog = ({\n open,\n title,\n description,\n confirmText,\n severity = 'normal',\n deleting = false,\n onConfirm,\n onCancel,\n}: ConfirmDeleteDialogProps) => {\n const [inputValue, setInputValue] = useState('');\n\n // reset input when dialog opens/closes\n useEffect(() => {\n if (!open) {\n setInputValue('');\n }\n }, [open]);\n\n const requiresTextConfirmation = severity === 'high' && confirmText;\n const canConfirm = requiresTextConfirmation ? inputValue === confirmText : true;\n\n const handleConfirm = () => {\n if (canConfirm) {\n onConfirm();\n }\n };\n\n return (\n <Dialog\n open={open}\n onClose={deleting ? undefined : onCancel}\n maxWidth=\"sm\"\n fullWidth\n >\n <DialogTitle>\n {severity === 'high' && (\n <Box display=\"flex\" alignItems=\"center\" style={{ gap: 8 }}>\n <WarningIcon color=\"error\" />\n <span>{title}</span>\n </Box>\n )}\n {severity !== 'high' && title}\n </DialogTitle>\n <DialogContent>\n <DialogContentText style={{ whiteSpace: 'pre-line' }}>\n {description}\n </DialogContentText>\n {requiresTextConfirmation && (\n <Box mt={2}>\n <Typography variant=\"body2\" color=\"textSecondary\" gutterBottom>\n Type <strong>{confirmText}</strong> to confirm:\n </Typography>\n <TextField\n fullWidth\n variant=\"outlined\"\n size=\"small\"\n value={inputValue}\n onChange={e => setInputValue(e.target.value)}\n disabled={deleting}\n autoFocus\n placeholder={confirmText}\n />\n </Box>\n )}\n </DialogContent>\n <DialogActions>\n <Button onClick={onCancel} disabled={deleting}>\n Cancel\n </Button>\n <Button\n onClick={handleConfirm}\n color=\"secondary\"\n variant=\"contained\"\n disabled={deleting || !canConfirm}\n startIcon={deleting ? <CircularProgress size={16} color=\"inherit\" /> : undefined}\n >\n {deleting ? 'Deleting...' : 'Delete'}\n </Button>\n </DialogActions>\n </Dialog>\n );\n};\n","import React, { useState, useEffect } from 'react';\nimport {\n Dialog,\n DialogTitle,\n DialogContent,\n DialogActions,\n Button,\n TextField,\n Box,\n Typography,\n FormControl,\n InputLabel,\n Select,\n MenuItem,\n CircularProgress,\n} from '@material-ui/core';\nimport { useApi, configApiRef, fetchApiRef } from '@backstage/core-plugin-api';\nimport { APIKey } from '../../types/api-management';\n\ninterface EditAPIKeyDialogProps {\n open: boolean;\n onClose: () => void;\n onSuccess: () => void;\n request: APIKey;\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}: EditAPIKeyDialogProps) => {\n const config = useApi(configApiRef);\n const fetchApi = useApi(fetchApiRef);\n const backendUrl = config.getString('backend.baseUrl');\n\n const [planTier, setPlanTier] = useState('');\n const [useCase, setUseCase] = useState('');\n const [saving, setSaving] = useState(false);\n const [error, setError] = useState('');\n\n useEffect(() => {\n if (open && request) {\n setPlanTier(request.spec.planTier || '');\n setUseCase(request.spec.useCase || '');\n setError('');\n }\n }, [open, request]);\n\n const handleSave = async () => {\n if (!planTier) {\n setError('Please select a tier');\n return;\n }\n\n setError('');\n setSaving(true);\n\n try {\n const patch = {\n spec: {\n planTier,\n useCase: useCase.trim(),\n },\n };\n\n const response = await fetchApi.fetch(\n `${backendUrl}/api/kuadrant/requests/${request.metadata.namespace}/${request.metadata.name}`,\n {\n method: 'PATCH',\n headers: {\n 'Content-Type': 'application/json',\n },\n body: JSON.stringify(patch),\n }\n );\n\n if (!response.ok) {\n const errorData = await response.json().catch(() => ({}));\n throw new Error(errorData.error || `Failed to update request: ${response.status}`);\n }\n\n onSuccess();\n onClose();\n } catch (err) {\n console.error('Error updating API key request:', err);\n setError(err instanceof Error ? err.message : 'Unknown error occurred');\n } finally {\n setSaving(false);\n }\n };\n\n const handleClose = () => {\n if (!saving) {\n setError('');\n onClose();\n }\n };\n\n return (\n <Dialog open={open} onClose={handleClose} maxWidth=\"sm\" fullWidth>\n <DialogTitle>Edit API Access Request</DialogTitle>\n <DialogContent>\n {error && (\n <Box mb={2} p={2} bgcolor=\"error.main\" color=\"error.contrastText\" borderRadius={1}>\n <Typography variant=\"body2\">{error}</Typography>\n </Box>\n )}\n\n <FormControl fullWidth margin=\"normal\">\n <InputLabel>Tier</InputLabel>\n <Select\n value={planTier}\n onChange={(e) => setPlanTier(e.target.value as string)}\n disabled={saving}\n >\n {availablePlans.map((plan) => {\n const limitDesc = Object.entries(plan.limits || {})\n .map(([key, val]) => `${val} per ${key}`)\n .join(', ');\n return (\n <MenuItem key={plan.tier} value={plan.tier}>\n {plan.tier} {limitDesc ? `(${limitDesc})` : ''}\n </MenuItem>\n );\n })}\n </Select>\n </FormControl>\n\n <TextField\n label=\"Use Case\"\n placeholder=\"Describe how you plan to use this API\"\n multiline\n rows={3}\n fullWidth\n margin=\"normal\"\n value={useCase}\n onChange={(e) => setUseCase(e.target.value)}\n disabled={saving}\n helperText=\"Explain your intended use of this API for admin review\"\n />\n </DialogContent>\n <DialogActions>\n <Button onClick={handleClose} disabled={saving}>\n Cancel\n </Button>\n <Button\n onClick={handleSave}\n color=\"primary\"\n variant=\"contained\"\n disabled={!planTier || saving}\n startIcon={saving ? <CircularProgress size={16} color=\"inherit\" /> : undefined}\n >\n {saving ? 'Saving...' : 'Save Changes'}\n </Button>\n </DialogActions>\n </Dialog>\n );\n};\n","import React, { useState } from 'react';\nimport { InfoCard, Table, TableColumn, Link, Progress } from '@backstage/core-components';\nimport { useApi, configApiRef, fetchApiRef, identityApiRef, alertApiRef } from '@backstage/core-plugin-api';\nimport useAsync from 'react-use/lib/useAsync';\nimport { Box, Chip, Typography, Tabs, Tab, IconButton, Tooltip, Menu, MenuItem, CircularProgress } 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 { ConfirmDeleteDialog } from '../ConfirmDeleteDialog';\nimport { APIKey } 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 alertApi = useApi(alertApiRef);\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<APIKey | null>(null);\n const [editDialogState, setEditDialogState] = useState<{ open: boolean; request: APIKey | null; plans: any[] }>({\n open: false,\n request: null,\n plans: [],\n });\n const [refresh, setRefresh] = useState(0);\n const [deleting, setDeleting] = useState<string | null>(null);\n const [deleteDialogState, setDeleteDialogState] = useState<{\n open: boolean;\n request: APIKey | null;\n }>({ open: false, request: null });\n const [apiKeyValues, setApiKeyValues] = useState<Map<string, string>>(new Map());\n const [apiKeyLoading, setApiKeyLoading] = useState<Set<string>>(new Set());\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 [optimisticallyDeleted, setOptimisticallyDeleted] = useState<Set<string>>(new Set());\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 <Progress />\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 || []).filter(\n (r: APIKey) => !optimisticallyDeleted.has(r.metadata.name)\n );\n const approvedRequests = allRequests.filter((r: APIKey) => r.status?.phase === 'Approved');\n const pendingRequests = allRequests.filter((r: APIKey) => !r.status?.phase || r.status.phase === 'Pending');\n const rejectedRequests = allRequests.filter((r: APIKey) => 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 fetchApiKeyFromSecret = async (requestNamespace: string, requestName: string) => {\n const key = `${requestNamespace}/${requestName}`;\n if (apiKeyLoading.has(key)) {\n return;\n }\n\n setApiKeyLoading(prev => new Set(prev).add(key));\n try {\n const response = await fetchApi.fetch(\n `${backendUrl}/api/kuadrant/requests/${requestNamespace}/${requestName}/secret`\n );\n if (response.ok) {\n const data = await response.json();\n setApiKeyValues(prev => new Map(prev).set(key, data.apiKey));\n }\n } catch (err) {\n console.error('failed to fetch api key:', err);\n } finally {\n setApiKeyLoading(prev => {\n const next = new Set(prev);\n next.delete(key);\n return next;\n });\n }\n };\n\n const clearApiKeyValue = (requestNamespace: string, requestName: string) => {\n const key = `${requestNamespace}/${requestName}`;\n setApiKeyValues(prev => {\n const next = new Map(prev);\n next.delete(key);\n return next;\n });\n };\n\n const 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 apiProductName = request.spec.apiProductRef?.name;\n const apiProductNamespace = request.metadata.namespace;\n const apiProductResponse = await fetchApi.fetch(\n `${backendUrl}/api/kuadrant/apiproducts/${apiProductNamespace}/${apiProductName}`\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 handleDeleteClick = () => {\n if (!menuRequest) return;\n const request = menuRequest;\n handleMenuClose();\n setDeleteDialogState({ open: true, request });\n };\n\n const handleDeleteConfirm = async () => {\n if (!deleteDialogState.request) return;\n\n const request = deleteDialogState.request;\n const requestName = request.metadata.name;\n\n // optimistic update - remove from UI immediately\n setOptimisticallyDeleted(prev => new Set(prev).add(requestName));\n setDeleting(requestName);\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 alertApi.post({ message: 'Request deleted', severity: 'success', display: 'transient' });\n setDeleteDialogState({ open: false, request: null });\n } catch (err) {\n console.error('Error deleting request:', err);\n // rollback optimistic update on error\n setOptimisticallyDeleted(prev => {\n const next = new Set(prev);\n next.delete(requestName);\n return next;\n });\n alertApi.post({ message: 'Failed to delete request', severity: 'error', display: 'transient' });\n } finally {\n setDeleting(null);\n }\n };\n\n const handleDeleteCancel = () => {\n setDeleteDialogState({ open: false, request: null });\n };\n\n const columns: TableColumn<APIKey>[] = [\n {\n title: 'API Product',\n field: 'spec.apiProductRef.name',\n render: (row: APIKey) => {\n const apiProductName = row.spec.apiProductRef?.name || 'unknown';\n return (\n <Link to={`/catalog/default/api/${apiProductName}/api-keys`}>\n <strong>{apiProductName}</strong>\n </Link>\n );\n },\n },\n {\n title: 'Tier',\n field: 'spec.planTier',\n render: (row: APIKey) => {\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: APIKey) => {\n if (!row.spec.useCase) {\n return <Typography variant=\"body2\">-</Typography>;\n }\n return (\n <Tooltip title={row.spec.useCase} placement=\"top\">\n <Typography\n variant=\"body2\"\n style={{\n maxWidth: '200px',\n overflow: 'hidden',\n textOverflow: 'ellipsis',\n whiteSpace: 'nowrap',\n }}\n >\n {row.spec.useCase}\n </Typography>\n </Tooltip>\n );\n },\n },\n {\n title: 'Status',\n field: 'status.phase',\n render: (row: APIKey) => {\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: 'Reviewed By',\n field: 'status.reviewedBy',\n render: (row: APIKey) => {\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 field: 'status.secretRef',\n filtering: false,\n render: (row: APIKey) => {\n if (row.status?.phase !== 'Approved') {\n return <Typography variant=\"body2\" color=\"textSecondary\">-</Typography>;\n }\n\n const key = `${row.metadata.namespace}/${row.metadata.name}`;\n const hasSecretRef = row.status?.secretRef?.name;\n const isVisible = visibleKeys.has(row.metadata.name);\n const isLoading = apiKeyLoading.has(key);\n const apiKeyValue = apiKeyValues.get(key);\n\n if (!hasSecretRef) {\n return (\n <Typography variant=\"body2\" color=\"textSecondary\">\n Awaiting secret...\n </Typography>\n );\n }\n\n const handleToggle = () => {\n if (isVisible) {\n // hiding - clear the value from memory\n clearApiKeyValue(row.metadata.namespace, row.metadata.name);\n toggleKeyVisibility(row.metadata.name);\n } else {\n // showing - fetch fresh value\n fetchApiKeyFromSecret(row.metadata.namespace, row.metadata.name);\n toggleKeyVisibility(row.metadata.name);\n }\n };\n\n return (\n <Box display=\"flex\" alignItems=\"center\" style={{ gap: 8 }}>\n <Box fontFamily=\"monospace\" fontSize=\"0.875rem\">\n {isLoading ? 'Loading...' : isVisible && apiKeyValue ? apiKeyValue : '•'.repeat(20) + '...'}\n </Box>\n <Tooltip title={isVisible ? 'hide key' : 'show key'}>\n <IconButton\n size=\"small\"\n onClick={handleToggle}\n disabled={isLoading}\n >\n {isVisible ? <VisibilityOffIcon fontSize=\"small\" /> : <VisibilityIcon fontSize=\"small\" />}\n </IconButton>\n </Tooltip>\n </Box>\n );\n },\n },\n {\n title: 'Requested',\n field: 'metadata.creationTimestamp',\n render: (row: APIKey) => {\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 filtering: false,\n render: (row: APIKey) => {\n const isDeleting = deleting === row.metadata.name;\n if (isDeleting) {\n return <CircularProgress size={20} />;\n }\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: APIKey) => !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 > 5,\n pageSize: 20,\n search: true,\n filtering: true,\n debounceInterval: 300,\n toolbar: true,\n emptyRowsWhenPaging: false,\n }}\n columns={tabColumns}\n data={tabData.map((item: APIKey) => ({\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={handleDeleteClick}>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 <ConfirmDeleteDialog\n open={deleteDialogState.open}\n title=\"Delete API Key Request\"\n description={`Are you sure you want to delete the API key request for ${deleteDialogState.request?.spec.apiProductRef?.name || 'this API'}?`}\n deleting={deleting !== null}\n onConfirm={handleDeleteConfirm}\n onCancel={handleDeleteCancel}\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 CircularProgress,\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 const [httpRoutesRetry, setHttpRoutesRetry] = useState(0);\n\n const {\n value: httpRoutes,\n loading: httpRoutesLoading,\n error: httpRoutesError\n } = 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, httpRoutesRetry]);\n\n // load planpolicies with full details to show associated plans\n const {\n value: planPolicies,\n error: planPoliciesError\n } = 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: 'devportal.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 {httpRoutesError && (\n <Alert severity=\"error\" style={{ marginBottom: 16 }}>\n <strong>Failed to load HTTPRoutes:</strong> {httpRoutesError.message}\n <Box mt={1}>\n <Button\n size=\"small\"\n variant=\"outlined\"\n onClick={() => setHttpRoutesRetry(prev => prev + 1)}\n >\n Retry\n </Button>\n </Box>\n </Alert>\n )}\n\n {planPoliciesError && (\n <Alert severity=\"warning\" style={{ marginBottom: 16 }}>\n <strong>Failed to load PlanPolicies:</strong> {planPoliciesError.message}\n <Typography variant=\"body2\" style={{ marginTop: 8 }}>\n You can still create the API Product, but plan information may be incomplete.\n </Typography>\n </Alert>\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 disabled={creating}\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 disabled={creating}\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 disabled={creating}\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 disabled={creating}\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 disabled={creating}\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 disabled={creating}\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={creating ? undefined : () => handleDeleteTag(tag)}\n size=\"small\"\n disabled={creating}\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 disabled={creating}\n />\n <Button onClick={handleAddTag} variant=\"outlined\" size=\"small\" disabled={creating}>\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={\n httpRoutesError\n ? \"Unable to load HTTPRoutes. Please retry.\"\n : \"Select an HTTPRoute (backstage.io/expose: true). APIProduct will be created in the same namespace.\"\n }\n error={!!httpRoutesError}\n disabled={httpRoutesLoading || creating || !!httpRoutesError}\n InputLabelProps={{\n classes: {\n asterisk: classes.asterisk,\n },\n }}\n >\n {httpRoutesLoading && (\n <MenuItem value=\"\">Loading...</MenuItem>\n )}\n {httpRoutesError && (\n <MenuItem value=\"\">Error loading routes</MenuItem>\n )}\n {!httpRoutesLoading && !httpRoutesError && httpRoutes && httpRoutes.length === 0 && (\n <MenuItem value=\"\">No HTTPRoutes available</MenuItem>\n )}\n {!httpRoutesLoading && !httpRoutesError && 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 disabled={creating}\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 disabled={creating}\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 disabled={creating}\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 disabled={creating}\n />\n </Grid>\n </Grid>\n </DialogContent>\n <DialogActions>\n <Button onClick={handleClose} disabled={creating}>Cancel</Button>\n <Button\n onClick={handleCreate}\n color=\"primary\"\n variant=\"contained\"\n disabled={creating || !name || !displayName || !description || !selectedHTTPRoute}\n startIcon={creating ? <CircularProgress size={16} color=\"inherit\" /> : undefined}\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 CircularProgress,\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 {\n value: planPolicies,\n error: planPoliciesError\n } = 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 {planPoliciesError && (\n <Alert severity=\"warning\" style={{ marginBottom: 16 }}>\n <strong>Failed to load PlanPolicies:</strong> {planPoliciesError.message}\n <Typography variant=\"body2\" style={{ marginTop: 8 }}>\n Plan information may be incomplete.\n </Typography>\n </Alert>\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 disabled={saving}\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 disabled={saving}\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 disabled={saving}\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 disabled={saving}\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 disabled={saving}\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={saving ? undefined : () => handleDeleteTag(tag)}\n size=\"small\"\n disabled={saving}\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 disabled={saving}\n />\n <Button onClick={handleAddTag} variant=\"outlined\" size=\"small\" disabled={saving}>\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 disabled={saving}\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 disabled={saving}\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 disabled={saving}\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 disabled={saving}\n />\n </Grid>\n </Grid>\n )}\n </DialogContent>\n <DialogActions>\n <Button onClick={onClose} disabled={saving}>Cancel</Button>\n <Button\n onClick={handleSave}\n color=\"primary\"\n variant=\"contained\"\n disabled={saving || loading || !displayName || !description}\n startIcon={saving ? <CircularProgress size={16} color=\"inherit\" /> : undefined}\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 } 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, alertApiRef, identityApiRef } 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';\nimport { ConfirmDeleteDialog } from '../ConfirmDeleteDialog';\n\ntype KuadrantResource = {\n metadata: {\n name: string;\n namespace: string;\n creationTimestamp: string;\n annotations?: Record<string, 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 alertApi = useApi(alertApiRef);\n const identityApi = useApi(identityApiRef);\n const backendUrl = config.getString('backend.baseUrl');\n const [userEntityRef, setUserEntityRef] = useState<string>('');\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 const [deleteStats, setDeleteStats] = useState<{requests: number; secrets: number} | null>(null);\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 deletePermissionLoading = deleteOwnPermissionLoading || deleteAllPermissionLoading;\n\n const {\n allowed: canListPlanPolicies,\n loading: planPolicyPermissionLoading,\n error: planPolicyPermissionError,\n } = useKuadrantPermission(kuadrantPlanPolicyListPermission);\n\n useAsync(async () => {\n const identity = await identityApi.getBackstageIdentity();\n setUserEntityRef(identity.userEntityRef);\n }, [identityApi]);\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 alertApi.post({ message: 'API Product created', severity: 'success', display: 'transient' });\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 alertApi.post({ message: 'API Product updated', severity: 'success', display: 'transient' });\n };\n\n const handleDeleteClick = async (namespace: string, name: string) => {\n setApiProductToDelete({ namespace, name });\n setDeleteStats(null);\n\n try {\n const response = await fetchApi.fetch(`${backendUrl}/api/kuadrant/requests?namespace=${namespace}`);\n if (response.ok) {\n const data = await response.json();\n const related = (data.items || []).filter(\n (r: any) => r.spec.apiName === name && r.spec.apiNamespace === namespace\n );\n const approved = related.filter((r: any) => r.status?.phase === 'Approved').length;\n setDeleteStats({ requests: related.length, secrets: approved });\n }\n } catch (err) {\n console.warn('Failed to fetch delete stats:', err);\n }\n\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 alertApi.post({ message: 'API Product deleted', severity: 'success', display: 'transient' });\n } catch (err) {\n console.error('error deleting apiproduct:', err);\n alertApi.post({ message: 'Failed to delete API Product', severity: 'error', display: 'transient' });\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: 'spec.displayName',\n render: (row: any) => {\n const publishStatus = row.spec?.publishStatus;\n const isPublished = publishStatus === 'Published';\n const displayName = row.spec?.displayName ?? row.metadata.name;\n\n if (isPublished) {\n return (\n <Link to={`/catalog/default/api/${row.metadata.name}/api-product`}>\n <strong>{displayName}</strong>\n </Link>\n );\n }\n\n return (\n <span className=\"text-muted\">\n <strong>{displayName}</strong>\n </span>\n );\n },\n customFilterAndSearch: (term, row: any) => {\n const displayName = row.spec?.displayName || row.metadata.name || '';\n return displayName.toLowerCase().includes(term.toLowerCase());\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 filtering: false,\n render: (row: any) => {\n const owner = row.metadata?.annotations?.['backstage.io/owner'];\n const isOwner = owner === userEntityRef;\n const canEdit = canUpdateAllApiProducts || (canUpdateOwnApiProduct && isOwner);\n const canDelete = canDeleteAllApiProducts || (canDeleteOwnApiProduct && isOwner);\n\n if (!canEdit && !canDelete) return null;\n\n return (\n <Box display=\"flex\" style={{ gap: 4 }}>\n {canEdit && (\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 {canDelete && (\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\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={{\n paging: resources.length > 5,\n pageSize: 20,\n search: true,\n filtering: true,\n debounceInterval: 300,\n toolbar: true,\n emptyRowsWhenPaging: false,\n }}\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 <ConfirmDeleteDialog\n open={deleteDialogOpen}\n title=\"Delete API Product\"\n description={\n deleteStats\n ? `Deleting \"${apiProductToDelete?.name}\" will also remove:\n\n• ${deleteStats.requests} API Key Request(s)\n• ${deleteStats.secrets} API Key Secret(s)\n\nThis action cannot be undone.`\n : `Deleting \"${apiProductToDelete?.name}\" will also remove all associated API Key Requests and Secrets.\nThis action cannot be undone.`\n }\n confirmText={apiProductToDelete?.name}\n severity=\"high\"\n deleting={deleting}\n onConfirm={handleDeleteConfirm}\n onCancel={handleDeleteCancel}\n />\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":["ConfirmDeleteDialog","open","title","description","confirmText","severity","deleting","onConfirm","onCancel","inputValue","setInputValue","useState","useEffect","requiresTextConfirmation","canConfirm","Dialog","onClose","undefined","maxWidth","fullWidth","DialogTitle","Box","display","alignItems","style","gap","WarningIcon","color","span","DialogContent","DialogContentText","whiteSpace","mt","Typography","variant","gutterBottom","strong","TextField","size","value","onChange","e","target","disabled","autoFocus","placeholder","DialogActions","Button","onClick","startIcon","CircularProgress","EditAPIKeyRequestDialog","onSuccess","request","availablePlans","config","useApi","configApiRef","fetchApi","fetchApiRef","backendUrl","getString","planTier","setPlanTier","useCase","setUseCase","saving","setSaving","error","setError","spec","handleClose","mb","p","bgcolor","borderRadius","FormControl","margin","InputLabel","Select","map","plan","limitDesc","Object","entries","limits","key","val","join","MenuItem","tier","label","multiline","rows","helperText","async","patch","trim","response","fetch","metadata","namespace","name","method","headers","body","JSON","stringify","ok","errorData","json","catch","Error","status","err","console","message","MyApiKeysCard","deleteDialogState","identityApi","identityApiRef","alertApi","alertApiRef","selectedTab","setSelectedTab","setUserId","visibleKeys","setVisibleKeys","Set","menuAnchor","setMenuAnchor","menuRequest","setMenuRequest","editDialogState","setEditDialogState","plans","refresh","setRefresh","setDeleting","setDeleteDialogState","apiKeyValues","setApiKeyValues","Map","apiKeyLoading","setApiKeyLoading","useAsync","identity","getBackstageIdentity","extractedUserId","userEntityRef","split","log","optimisticallyDeleted","setOptimisticallyDeleted","requests","loading","items","InfoCard","Progress","allRequests","filter","r","has","approvedRequests","phase","pendingRequests","rejectedRequests","toggleKeyVisibility","keyName","prev","newSet","delete","add","handleMenuClose","handleEdit","apiProductName","apiProductRef","apiProductNamespace","apiProductResponse","apiProduct","handleDeleteClick","columns","field","render","row","Link","to","Chip","Tooltip","placement","overflow","textOverflow","reviewedBy","reviewedDate","reviewedAt","Date","toLocaleDateString","filtering","hasSecretRef","secretRef","isVisible","isLoading","apiKeyValue","get","fontFamily","fontSize","repeat","IconButton","requestNamespace","requestName","next","clearApiKeyValue","data","set","apiKey","fetchApiKeyFromSecret","VisibilityOffIcon","VisibilityIcon","creationTimestamp","date","stopPropagation","rect","currentTarget","getBoundingClientRect","top","bottom","left","aria-controls","aria-haspopup","MoreVertIcon","tabData","getTabData","tabColumns","col","getTabColumns","subheader","length","Tabs","_","newValue","indicatorColor","textColor","Tab","textAlign","Table","options","paging","pageSize","search","debounceInterval","toolbar","emptyRowsWhenPaging","item","id","Menu","Boolean","anchorReference","anchorPosition","push","post","PermissionGate","children","permission","fallback","errorMessage","allowed","useKuadrantPermission","PlanPolicyDetails","selectedPolicy","alertSeverity","alertMessage","includeTopMargin","border","fontWeight","marginTop","flexWrap","idx","limitText","daily","monthly","yearly","some","Alert","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","httpRoutesRetry","setHttpRoutesRetry","httpRoutes","httpRoutesLoading","httpRoutesError","route","annotations","planPolicies","planPoliciesError","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","setUserEntityRef","createDialogOpen","setCreateDialogOpen","editDialogOpen","setEditDialogOpen","refreshTrigger","setRefreshTrigger","deleteDialogOpen","setDeleteDialogOpen","apiProductToDelete","setApiProductToDelete","apiProductToEdit","setApiProductToEdit","deleteStats","setDeleteStats","canCreateApiProduct","createPermissionLoading","createPermissionError","kuadrantApiProductCreatePermission","canViewAllRequests","approvalQueueAllPermissionLoading","kuadrantApiKeyRequestReadAllPermission","canViewOwnRequests","approvalQueueOwnPermissionLoading","approvalQueuePermissionError","kuadrantApiKeyRequestReadOwnPermission","canViewApprovalQueue","approvalQueuePermissionLoading","canDeleteOwnApiProduct","deleteOwnPermissionLoading","kuadrantApiProductDeleteOwnPermission","canDeleteAllApiProducts","deleteAllPermissionLoading","deletePermissionError","kuadrantApiProductDeleteAllPermission","canUpdateOwnApiProduct","kuadrantApiProductUpdateOwnPermission","canUpdateAllApiProducts","kuadrantApiProductUpdateAllPermission","deletePermissionLoading","canListPlanPolicies","planPolicyPermissionLoading","planPolicyPermissionError","kuadrantPlanPolicyListPermission","apiProducts","apiProductsLoading","apiProductsError","planPoliciesLoading","permissionError","isPublished","className","customFilterAndSearch","term","toLowerCase","mode","formatDate","dateString","year","month","day","isOwner","canEdit","canDelete","handleEditClick","EditIcon","related","apiName","apiNamespace","approved","secrets","warn","DeleteIcon","planPolicyColumns","Page","themeId","Header","subtitle","SupportButton","Content","ResponseErrorPanel","direction","action","height","AddIcon","resources","renderPlanPolicies","ApprovalQueueCard","KuadrantPage","kuadrantApiProductListPermission"],"sourceRoot":""}
|
package/package.json
CHANGED
|
@@ -1,2 +0,0 @@
|
|
|
1
|
-
"use strict";(self.webpackChunkinternal_plugin_kuadrant=self.webpackChunkinternal_plugin_kuadrant||[]).push([[7632],{63221:(e,a,t)=>{t.d(a,{K:()=>v});var s=t(31085),n=t(95478),l=t(76891),i=t(61477),r=t(10394),o=t(46805),d=t(59461),c=t(72501),u=t(16249),p=t(93453),m=t(64947),h=t(78467),x=t(77225);const v=({open:e,title:a,description:t,confirmText:v,severity:A="normal",deleting:g=!1,onConfirm:j,onCancel:f})=>{const[y,b]=(0,n.useState)("");(0,n.useEffect)(()=>{e||b("")},[e]);const P="high"===A&&v,S=!P||y===v;return(0,s.jsxs)(l.A,{open:e,onClose:g?void 0:f,maxWidth:"sm",fullWidth:!0,children:[(0,s.jsxs)(i.A,{children:["high"===A&&(0,s.jsxs)(r.A,{display:"flex",alignItems:"center",style:{gap:8},children:[(0,s.jsx)(x.A,{color:"error"}),(0,s.jsx)("span",{children:a})]}),"high"!==A&&a]}),(0,s.jsxs)(o.A,{children:[(0,s.jsx)(d.A,{style:{whiteSpace:"pre-line"},children:t}),P&&(0,s.jsxs)(r.A,{mt:2,children:[(0,s.jsxs)(c.A,{variant:"body2",color:"textSecondary",gutterBottom:!0,children:["Type ",(0,s.jsx)("strong",{children:v})," to confirm:"]}),(0,s.jsx)(u.A,{fullWidth:!0,variant:"outlined",size:"small",value:y,onChange:e=>b(e.target.value),disabled:g,autoFocus:!0,placeholder:v})]})]}),(0,s.jsxs)(p.A,{children:[(0,s.jsx)(m.A,{onClick:f,disabled:g,children:"Cancel"}),(0,s.jsx)(m.A,{onClick:()=>{S&&j()},color:"secondary",variant:"contained",disabled:g||!S,startIcon:g?(0,s.jsx)(h.A,{size:16,color:"inherit"}):void 0,children:g?"Deleting...":"Delete"})]})]})}},84189:(e,a,t)=>{t.d(a,{n:()=>j});var s=t(31085),n=t(95478),l=t(76891),i=t(61477),r=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),v=t(64947),A=t(78467),g=t(22097);const j=({open:e,onClose:a,onSuccess:t,request:j,availablePlans:f})=>{const y=(0,g.useApi)(g.configApiRef),b=(0,g.useApi)(g.fetchApiRef),P=y.getString("backend.baseUrl"),[S,w]=(0,n.useState)(""),[k,C]=(0,n.useState)(""),[T,R]=(0,n.useState)(!1),[I,$]=(0,n.useState)("");(0,n.useEffect)(()=>{e&&j&&(w(j.spec.planTier||""),C(j.spec.useCase||""),$(""))},[e,j]);const q=()=>{T||($(""),a())};return(0,s.jsxs)(l.A,{open:e,onClose:q,maxWidth:"sm",fullWidth:!0,children:[(0,s.jsx)(i.A,{children:"Edit API Access Request"}),(0,s.jsxs)(r.A,{children:[I&&(0,s.jsx)(o.A,{mb:2,p:2,bgcolor:"error.main",color:"error.contrastText",borderRadius:1,children:(0,s.jsx)(d.A,{variant:"body2",children:I})}),(0,s.jsxs)(c.A,{fullWidth:!0,margin:"normal",children:[(0,s.jsx)(u.A,{children:"Tier"}),(0,s.jsx)(p.A,{value:S,onChange:e=>w(e.target.value),disabled:T,children:f.map(e=>{const a=Object.entries(e.limits||{}).map(([e,a])=>`${a} per ${e}`).join(", ");return(0,s.jsxs)(m.A,{value:e.tier,children:[e.tier," ",a?`(${a})`:""]},e.tier)})})]}),(0,s.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=>C(e.target.value),disabled:T,helperText:"Explain your intended use of this API for admin review"})]}),(0,s.jsxs)(x.A,{children:[(0,s.jsx)(v.A,{onClick:q,disabled:T,children:"Cancel"}),(0,s.jsx)(v.A,{onClick:async()=>{if(S){$(""),R(!0);try{const e={spec:{planTier:S,useCase:k.trim()}},s=await b.fetch(`${P}/api/kuadrant/requests/${j.metadata.namespace}/${j.metadata.name}`,{method:"PATCH",headers:{"Content-Type":"application/json"},body:JSON.stringify(e)});if(!s.ok){const e=await s.json().catch(()=>({}));throw new Error(e.error||`Failed to update request: ${s.status}`)}t(),a()}catch(e){console.error("Error updating API key request:",e),$(e instanceof Error?e.message:"Unknown error occurred")}finally{R(!1)}}else $("Please select a tier")},color:"primary",variant:"contained",disabled:!S||T,startIcon:T?(0,s.jsx)(A.A,{size:16,color:"inherit"}):void 0,children:T?"Saving...":"Save Changes"})]})]})}},95251:(e,a,t)=>{t.r(a),t.d(a,{KuadrantPage:()=>ae});var s=t(31085),n=t(95478),l=t.n(n),i=t(67720),r=t(10394),o=t(29365),d=t(72501),c=t(42899),u=t(64947),p=t(18466),m=t(39590),h=t(75625),x=t(37725),v=t(25010),A=t(289),g=t(55639),j=t(45210),f=t(46681),y=t(86687),b=t(42367),P=t(96040),S=t(91638),w=t(22097),k=t(16281),C=t(71677),T=t(78467),R=t(31653),I=t(38605),$=t(37757),q=t(26343),W=t(32269),E=t(61524),N=t(71407),D=t(84189),M=t(63221);const z=()=>{var e,a;const t=(0,w.useApi)(w.configApiRef),l=(0,w.useApi)(w.fetchApiRef),c=(0,w.useApi)(w.identityApiRef),u=(0,w.useApi)(w.alertApiRef),p=t.getString("backend.baseUrl"),[m,h]=(0,n.useState)(0),[,A]=(0,n.useState)(""),[g,j]=(0,n.useState)(new Set),[f,b]=(0,n.useState)(null),[k,z]=(0,n.useState)(null),[K,L]=(0,n.useState)({open:!1,request:null,plans:[]}),[B,U]=(0,n.useState)(0),[F,H]=(0,n.useState)(null),[_,O]=(0,n.useState)({open:!1,request:null}),[V,J]=(0,n.useState)(new Map),[Y,X]=(0,n.useState)(new Set);(0,S.A)(async()=>{const e=await c.getBackstageIdentity(),a=e.userEntityRef.split("/")[1]||"guest";console.log(`MyApiKeysCard: setting userId from userEntityRef: ${e.userEntityRef} -> "${a}"`),A(a)},[c]);const[G,Q]=(0,n.useState)(new Set),{value:Z,loading:ee,error:ae}=(0,S.A)(async()=>{const e=await l.fetch(`${p}/api/kuadrant/requests/my`);if(!e.ok)throw new Error("failed to fetch requests");return(await e.json()).items||[]},[p,l,B]);if(ee)return(0,s.jsx)(P.n,{title:"My API Keys",children:(0,s.jsx)(y.k,{})});if(ae)return(0,s.jsx)(P.n,{title:"My API Keys",children:(0,s.jsxs)(d.A,{color:"error",children:["Error loading API keys: ",ae.message]})});const te=(Z||[]).filter(e=>!G.has(e.metadata.name)),se=te.filter(e=>{var a;return"Approved"===(null===(a=e.status)||void 0===a?void 0:a.phase)}),ne=te.filter(e=>{var a;return!(null===(a=e.status)||void 0===a?void 0:a.phase)||"Pending"===e.status.phase}),le=te.filter(e=>{var a;return"Rejected"===(null===(a=e.status)||void 0===a?void 0:a.phase)}),ie=e=>{j(a=>{const t=new Set(a);return t.has(e)?t.delete(e):t.add(e),t})},re=()=>{b(null),z(null)},oe=async()=>{if(!k)return;const e=k;re();try{var a;const s=null===(a=e.spec.apiProductRef)||void 0===a?void 0:a.name,n=e.metadata.namespace,i=await l.fetch(`${p}/api/kuadrant/apiproducts/${n}/${s}`);if(i.ok){var t;const a=(null===(t=(await i.json()).spec)||void 0===t?void 0:t.plans)||[];L({open:!0,request:e,plans:a})}else console.error("Failed to fetch API product"),L({open:!0,request:e,plans:[]})}catch(a){console.error("Error fetching plans:",a),L({open:!0,request:e,plans:[]})}},de=()=>{if(!k)return;const e=k;re(),O({open:!0,request:e})},ce=[{title:"API Product",field:"spec.apiProductRef.name",render:e=>{var a;const t=(null===(a=e.spec.apiProductRef)||void 0===a?void 0:a.name)||"unknown";return(0,s.jsx)(x.N_,{to:`/catalog/default/api/${t}/api-keys`,children:(0,s.jsx)("strong",{children:t})})}},{title:"Tier",field:"spec.planTier",render:e=>{const a="gold"===e.spec.planTier?"primary":"silver"===e.spec.planTier?"default":"secondary";return(0,s.jsx)(i.A,{label:e.spec.planTier,color:a,size:"small"})}},{title:"Use Case",field:"spec.useCase",render:e=>e.spec.useCase?(0,s.jsx)(C.Ay,{title:e.spec.useCase,placement:"top",children:(0,s.jsx)(d.A,{variant:"body2",style:{maxWidth:"200px",overflow:"hidden",textOverflow:"ellipsis",whiteSpace:"nowrap"},children:e.spec.useCase})}):(0,s.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",n="Approved"===t?"primary":"Rejected"===t?"secondary":"default";return(0,s.jsx)(i.A,{label:t,color:n,size:"small"})}},{title:"Reviewed By",field:"status.reviewedBy",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,s.jsxs)(r.A,{children:[(0,s.jsx)(d.A,{variant:"body2",children:e.status.reviewedBy}),a&&(0,s.jsx)(d.A,{variant:"caption",color:"textSecondary",children:a})]})}return(0,s.jsx)(d.A,{variant:"body2",color:"textSecondary",children:"-"})}},{title:"API Key",field:"status.secretRef",filtering:!1,render:e=>{var a,t,n;if("Approved"!==(null===(a=e.status)||void 0===a?void 0:a.phase))return(0,s.jsx)(d.A,{variant:"body2",color:"textSecondary",children:"-"});const i=`${e.metadata.namespace}/${e.metadata.name}`,c=null===(n=e.status)||void 0===n||null===(t=n.secretRef)||void 0===t?void 0:t.name,u=g.has(e.metadata.name),m=Y.has(i),h=V.get(i);return c?(0,s.jsxs)(r.A,{display:"flex",alignItems:"center",style:{gap:8},children:[(0,s.jsx)(r.A,{fontFamily:"monospace",fontSize:"0.875rem",children:m?"Loading...":u&&h?h:"•".repeat(20)+"..."}),(0,s.jsx)(C.Ay,{title:u?"hide key":"show key",children:(0,s.jsx)(o.A,{size:"small",onClick:()=>{u?(((e,a)=>{const t=`${e}/${a}`;J(e=>{const a=new Map(e);return a.delete(t),a})})(e.metadata.namespace,e.metadata.name),ie(e.metadata.name)):((async(e,a)=>{const t=`${e}/${a}`;if(!Y.has(t)){X(e=>new Set(e).add(t));try{const s=await l.fetch(`${p}/api/kuadrant/requests/${e}/${a}/secret`);if(s.ok){const e=await s.json();J(a=>new Map(a).set(t,e.apiKey))}}catch(e){console.error("failed to fetch api key:",e)}finally{X(e=>{const a=new Set(e);return a.delete(t),a})}}})(e.metadata.namespace,e.metadata.name),ie(e.metadata.name))},disabled:m,children:u?(0,s.jsx)(E.A,{fontSize:"small"}):(0,s.jsx)(W.A,{fontSize:"small"})})})]}):(0,s.jsx)(d.A,{variant:"body2",color:"textSecondary",children:"Awaiting secret..."})}},{title:"Requested",field:"metadata.creationTimestamp",render:e=>{if(!e.metadata.creationTimestamp)return(0,s.jsx)(d.A,{variant:"body2",children:"-"});const a=new Date(e.metadata.creationTimestamp);return(0,s.jsx)(d.A,{variant:"body2",children:a.toLocaleDateString()})}},{title:"",filtering:!1,render:e=>F===e.metadata.name?(0,s.jsx)(T.A,{size:20}):(0,s.jsx)(o.A,{size:"small",onClick:a=>{a.stopPropagation();const t=a.currentTarget.getBoundingClientRect();b({top:t.bottom,left:t.left}),z(e)},"aria-controls":f?"myapikeys-menu":void 0,"aria-haspopup":"true",children:(0,s.jsx)(N.A,{})})}],ue=(()=>{switch(m){case 0:return se;case 1:return ne;case 2:return le;default:return te}})(),pe=(()=>{switch(m){case 0:return ce.filter(e=>"Reason"!==e.title);case 1:return ce.filter(e=>"Reason"!==e.title&&"Reviewed By"!==e.title&&"API Key"!==e.title);case 2:return ce.filter(e=>"API Key"!==e.title);default:return ce}})();return(0,s.jsxs)(s.Fragment,{children:[(0,s.jsxs)(P.n,{title:"My API Keys",subheader:`${se.length} active, ${ne.length} pending`,children:[(0,s.jsx)(r.A,{mb:2,children:(0,s.jsxs)(R.A,{value:m,onChange:(e,a)=>h(a),indicatorColor:"primary",textColor:"primary",children:[(0,s.jsx)(I.A,{label:`Active (${se.length})`}),(0,s.jsx)(I.A,{label:`Pending (${ne.length})`}),(0,s.jsx)(I.A,{label:`Rejected (${le.length})`})]})}),0===ue.length?(0,s.jsx)(r.A,{p:3,textAlign:"center",children:(0,s.jsxs)(d.A,{variant:"body1",color:"textSecondary",children:[0===m&&"No active API keys. Request access to an API to get started.",1===m&&"No pending requests.",2===m&&"No rejected requests."]})}):(0,s.jsx)(v.X,{options:{paging:ue.length>5,pageSize:20,search:!0,filtering:!0,debounceInterval:300,toolbar:!0,emptyRowsWhenPaging:!1},columns:pe,data:ue.map(e=>({...e,id:e.metadata.name}))})]}),(0,s.jsx)($.A,{id:"myapikeys-menu",open:Boolean(f),onClose:re,anchorReference:"anchorPosition",anchorPosition:f||{top:0,left:0},children:k&&(()=>{const e=[];var a;return(a=k).status&&"Pending"!==a.status.phase||e.push((0,s.jsx)(q.A,{onClick:oe,children:"Edit"},"edit")),e.push((0,s.jsx)(q.A,{onClick:de,children:"Delete"},"delete")),e})()}),K.request&&(0,s.jsx)(D.n,{open:K.open,request:K.request,availablePlans:K.plans,onClose:()=>L({open:!1,request:null,plans:[]}),onSuccess:()=>{L({open:!1,request:null,plans:[]}),U(e=>e+1)}}),(0,s.jsx)(M.K,{open:_.open,title:"Delete API Key Request",description:`Are you sure you want to delete the API key request for ${(null===(a=_.request)||void 0===a||null===(e=a.spec.apiProductRef)||void 0===e?void 0:e.name)||"this API"}?`,deleting:null!==F,onConfirm:async()=>{if(!_.request)return;const e=_.request,a=e.metadata.name;Q(e=>new Set(e).add(a)),H(a);try{if(!(await l.fetch(`${p}/api/kuadrant/requests/${e.metadata.namespace}/${e.metadata.name}`,{method:"DELETE"})).ok)throw new Error("Failed to delete request");U(e=>e+1),u.post({message:"Request deleted",severity:"success",display:"transient"}),O({open:!1,request:null})}catch(e){console.error("Error deleting request:",e),Q(e=>{const t=new Set(e);return t.delete(a),t}),u.post({message:"Failed to delete request",severity:"error",display:"transient"})}finally{H(null)}},onCancel:()=>{O({open:!1,request:null})}})]})};var K=t(46205);const L=({children:e,permission:a,fallback:t,errorMessage:n})=>{const{allowed:l,loading:i,error:o}=(0,K.l)(a);return i?(0,s.jsx)(y.k,{}):o?(0,s.jsxs)(r.A,{p:4,children:[(0,s.jsxs)(d.A,{color:"error",children:["Unable to check permissions: ",o.message]}),(0,s.jsx)(d.A,{variant:"body2",color:"textSecondary",children:"Please try again or contact your administrator"})]}):l?(0,s.jsx)(s.Fragment,{children:e}):t?(0,s.jsx)(s.Fragment,{children:t}):(0,s.jsxs)(r.A,{p:4,children:[(0,s.jsx)(d.A,{color:"textSecondary",children:n||"You don't have permission to view this page"}),(0,s.jsx)(r.A,{mt:1,children:(0,s.jsxs)(d.A,{variant:"caption",color:"textSecondary",children:["Required permission: ",a.name]})})]})};var B=t(58837),U=t(76891),F=t(61477),H=t(46805),_=t(16249),O=t(93453),V=t(84441);const J=({selectedPolicy:e,alertSeverity:a="warning",alertMessage:t="No PlanPolicy found for this HTTPRoute. API keys and rate limiting may not be available.",includeTopMargin:n=!0})=>(0,s.jsx)(r.A,{mt:n?1:0,p:2,bgcolor:"#f5f5f5",borderRadius:1,border:"1px solid #e0e0e0",children:e?(0,s.jsxs)(s.Fragment,{children:[(0,s.jsxs)(d.A,{variant:"subtitle2",gutterBottom:!0,style:{fontWeight:600},children:["Associated PlanPolicy: ",(0,s.jsx)("strong",{children:e.metadata.name})]}),e.plans&&e.plans.length>0?(0,s.jsxs)(s.Fragment,{children:[(0,s.jsx)(d.A,{variant:"caption",display:"block",gutterBottom:!0,color:"textSecondary",style:{marginTop:8},children:"Available Plans:"}),(0,s.jsx)(r.A,{display:"flex",flexWrap:"wrap",mt:1,style:{gap:8},children:e.plans.map((e,a)=>{var t,n,l;const r=(null===(t=e.limits)||void 0===t?void 0:t.daily)?`${e.limits.daily}/day`:(null===(n=e.limits)||void 0===n?void 0:n.monthly)?`${e.limits.monthly}/month`:(null===(l=e.limits)||void 0===l?void 0:l.yearly)?`${e.limits.yearly}/year`:"No limit";return(0,s.jsx)(i.A,{label:`${e.tier}: ${r}`,size:"small",variant:"outlined",color:"primary"},a)})}),e.plans.some(e=>e.description)&&(0,s.jsx)(r.A,{mt:1,children:e.plans.filter(e=>e.description).map((e,a)=>(0,s.jsxs)(d.A,{variant:"caption",display:"block",color:"textSecondary",children:["• ",(0,s.jsxs)("strong",{children:[e.tier,":"]})," ",e.description]},a))})]}):(0,s.jsx)(d.A,{variant:"caption",color:"textSecondary",children:"No plans defined in this PlanPolicy"})]}):(0,s.jsx)(V.A,{severity:a,children:t})}),Y=(0,B.A)({asterisk:{color:"#f44336"}}),X=({open:e,onClose:a,onSuccess:t})=>{const l=Y(),o=(0,w.useApi)(w.configApiRef),p=(0,w.useApi)(w.fetchApiRef),m=o.getString("backend.baseUrl"),[h,x]=(0,n.useState)(""),[v,A]=(0,n.useState)(""),[g,j]=(0,n.useState)(""),[f,y]=(0,n.useState)("v1"),[b,P]=(0,n.useState)("manual"),[k,C]=(0,n.useState)("Published"),[R,I]=(0,n.useState)([]),[$,W]=(0,n.useState)(""),[E,N]=(0,n.useState)(""),[D,M]=(0,n.useState)(""),[z,K]=(0,n.useState)(""),[L,B]=(0,n.useState)(""),[X,G]=(0,n.useState)(""),[Q,Z]=(0,n.useState)(""),[ee,ae]=(0,n.useState)(!1),{value:te,loading:se}=(0,S.A)(async()=>{const e=await p.fetch(`${m}/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"])})},[m,p,e]),{value:ne}=(0,S.A)(async()=>{const e=await p.fetch(`${m}/api/kuadrant/planpolicies`);return await e.json()},[m,p,e]),le=E?E.split("/"):null,ie=le?(re=le[0],oe=le[1],(null==ne?void 0:ne.items)?ne.items.find(e=>{const a=e.targetRef;return"HTTPRoute"===(null==a?void 0:a.kind)&&(null==a?void 0:a.name)===oe&&(!(null==a?void 0:a.namespace)||(null==a?void 0:a.namespace)===re)}):null):null;var re,oe;const de=()=>{$.trim()&&!R.includes($.trim())&&(I([...R,$.trim()]),W(""))},ce=()=>{x(""),A(""),j(""),y("v1"),P("manual"),C("Published"),I([]),W(""),N(""),M(""),K(""),B(""),G(""),Z(""),a()};return(0,s.jsxs)(U.A,{open:e,onClose:ce,maxWidth:"md",fullWidth:!0,children:[(0,s.jsx)(F.A,{children:"Create API Product"}),(0,s.jsxs)(H.A,{children:[Q&&(0,s.jsx)(V.A,{severity:"error",style:{marginBottom:16},children:Q}),(0,s.jsxs)(c.A,{container:!0,spacing:2,children:[(0,s.jsx)(c.A,{item:!0,xs:6,children:(0,s.jsx)(_.A,{fullWidth:!0,label:"Name",value:h,onChange:e=>x(e.target.value),placeholder:"my-api",helperText:"Kubernetes resource name (lowercase, hyphens)",margin:"normal",required:!0,disabled:ee,InputLabelProps:{classes:{asterisk:l.asterisk}}})}),(0,s.jsx)(c.A,{item:!0,xs:6,children:(0,s.jsx)(_.A,{fullWidth:!0,label:"Display Name",value:v,onChange:e=>A(e.target.value),placeholder:"My API",margin:"normal",required:!0,disabled:ee,InputLabelProps:{classes:{asterisk:l.asterisk}}})}),(0,s.jsx)(c.A,{item:!0,xs:6,children:(0,s.jsx)(_.A,{fullWidth:!0,label:"Version",value:f,onChange:e=>y(e.target.value),placeholder:"v1",margin:"normal",disabled:ee})}),(0,s.jsx)(c.A,{item:!0,xs:12,children:(0,s.jsxs)(_.A,{fullWidth:!0,select:!0,label:"Approval Mode",value:b,onChange:e=>P(e.target.value),margin:"normal",helperText:"Automatic: keys are created immediately. Manual: requires approval.",disabled:ee,children:[(0,s.jsx)(q.A,{value:"manual",children:"Manual"}),(0,s.jsx)(q.A,{value:"automatic",children:"Automatic"})]})}),(0,s.jsx)(c.A,{item:!0,xs:12,children:(0,s.jsxs)(_.A,{fullWidth:!0,select:!0,label:"Publish Status",value:k,onChange:e=>C(e.target.value),margin:"normal",helperText:"Draft: hidden from catalog. Published: visible to consumers.",disabled:ee,children:[(0,s.jsx)(q.A,{value:"Draft",children:"Draft"}),(0,s.jsx)(q.A,{value:"Published",children:"Published"})]})}),(0,s.jsx)(c.A,{item:!0,xs:12,children:(0,s.jsx)(_.A,{fullWidth:!0,label:"Description",value:g,onChange:e=>j(e.target.value),placeholder:"API description",margin:"normal",multiline:!0,rows:2,required:!0,disabled:ee,InputLabelProps:{classes:{asterisk:l.asterisk}}})}),(0,s.jsxs)(c.A,{item:!0,xs:12,children:[(0,s.jsx)(d.A,{variant:"subtitle2",gutterBottom:!0,style:{marginTop:16},children:"Tags"}),(0,s.jsx)(r.A,{display:"flex",flexWrap:"wrap",marginBottom:1,style:{gap:8},children:R.map(e=>(0,s.jsx)(i.A,{label:e,onDelete:ee?void 0:()=>{return a=e,void I(R.filter(e=>e!==a));var a},size:"small",disabled:ee},e))}),(0,s.jsxs)(r.A,{display:"flex",style:{gap:8},children:[(0,s.jsx)(_.A,{fullWidth:!0,size:"small",value:$,onChange:e=>W(e.target.value),onKeyPress:e=>"Enter"===e.key&&de(),placeholder:"Add tag",disabled:ee}),(0,s.jsx)(u.A,{onClick:de,variant:"outlined",size:"small",disabled:ee,children:"Add"})]})]}),(0,s.jsx)(c.A,{item:!0,xs:12,children:(0,s.jsxs)(_.A,{fullWidth:!0,select:!0,label:"HTTPRoute",value:E,onChange:e=>N(e.target.value),margin:"normal",required:!0,helperText:"Select an HTTPRoute (backstage.io/expose: true). APIProduct will be created in the same namespace.",disabled:se||ee,InputLabelProps:{classes:{asterisk:l.asterisk}},children:[se&&(0,s.jsx)(q.A,{value:"",children:"Loading..."}),!se&&te&&0===te.length&&(0,s.jsx)(q.A,{value:"",children:"No HTTPRoutes available"}),!se&&te&&te.map(e=>(0,s.jsxs)(q.A,{value:`${e.metadata.namespace}/${e.metadata.name}`,children:[e.metadata.name," (",e.metadata.namespace,")"]},`${e.metadata.namespace}/${e.metadata.name}`))]})}),E&&(0,s.jsx)(c.A,{item:!0,xs:12,children:(0,s.jsx)(J,{selectedPolicy:ie,alertSeverity:"warning",alertMessage:"No PlanPolicy found for this HTTPRoute. API keys and rate limiting may not be available.",includeTopMargin:!0})}),(0,s.jsx)(c.A,{item:!0,xs:6,children:(0,s.jsx)(_.A,{fullWidth:!0,label:"Contact Email",value:D,onChange:e=>M(e.target.value),placeholder:"api-team@example.com",margin:"normal",disabled:ee})}),(0,s.jsx)(c.A,{item:!0,xs:6,children:(0,s.jsx)(_.A,{fullWidth:!0,label:"Contact Team",value:z,onChange:e=>K(e.target.value),placeholder:"platform-team",margin:"normal",disabled:ee})}),(0,s.jsx)(c.A,{item:!0,xs:6,children:(0,s.jsx)(_.A,{fullWidth:!0,label:"Docs URL",value:L,onChange:e=>B(e.target.value),placeholder:"https://api.example.com/docs",margin:"normal",disabled:ee})}),(0,s.jsx)(c.A,{item:!0,xs:6,children:(0,s.jsx)(_.A,{fullWidth:!0,label:"OpenAPI Spec URL",value:X,onChange:e=>G(e.target.value),placeholder:"https://api.example.com/openapi.json",margin:"normal",disabled:ee})})]})]}),(0,s.jsxs)(O.A,{children:[(0,s.jsx)(u.A,{onClick:ce,disabled:ee,children:"Cancel"}),(0,s.jsx)(u.A,{onClick:async()=>{Z(""),ae(!0);try{if(!E)throw new Error("Please select an HTTPRoute");const[e,a]=E.split("/"),s={apiVersion:"devportal.kuadrant.io/v1alpha1",kind:"APIProduct",metadata:{name:h,namespace:e},spec:{displayName:v,description:g,version:f,approvalMode:b,publishStatus:k,tags:R,targetRef:{group:"gateway.networking.k8s.io",kind:"HTTPRoute",name:a,namespace:e},...D||z?{contact:{...D&&{email:D},...z&&{team:z}}}:{},...L||X?{documentation:{...L&&{docsURL:L},...X&&{openAPISpec:X}}}:{}}},n=await p.fetch(`${m}/api/kuadrant/apiproducts`,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify(s)});if(!n.ok){const e=await n.json();throw new Error(e.error||"failed to create apiproduct")}t(),ce()}catch(e){Z(e instanceof Error?e.message:String(e))}finally{ae(!1)}},color:"primary",variant:"contained",disabled:ee||!h||!v||!g||!E,startIcon:ee?(0,s.jsx)(T.A,{size:16,color:"inherit"}):void 0,children:ee?"Creating...":"Create"})]})]})};var G=t(34955);const Q=(0,B.A)({asterisk:{color:"#f44336"}}),Z=({open:e,onClose:a,onSuccess:t,namespace:o,name:p})=>{const m=Q(),h=(0,w.useApi)(w.configApiRef),x=(0,w.useApi)(w.fetchApiRef),v=h.getString("backend.baseUrl"),[A,g]=(0,n.useState)(!1),[j,f]=(0,n.useState)(""),[b,P]=(0,n.useState)(""),[k,C]=(0,n.useState)("v1"),[R,I]=(0,n.useState)("Draft"),[$,W]=(0,n.useState)("manual"),[E,N]=(0,n.useState)([]),[D,M]=(0,n.useState)(null),[z,K]=(0,n.useState)(""),[L,B]=(0,n.useState)(""),[Y,X]=(0,n.useState)(""),[G,Z]=(0,n.useState)(""),[ee,ae]=(0,n.useState)(""),[te,se]=(0,n.useState)(""),[ne,le]=(0,n.useState)(!1);(0,n.useEffect)(()=>{e&&o&&p&&(g(!0),se(""),x.fetch(`${v}/api/kuadrant/apiproducts/${o}/${p}`).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,s,n;f(e.spec.displayName||""),P(e.spec.description||""),C(e.spec.version||"v1"),I(e.spec.publishStatus||"Draft"),W(e.spec.approvalMode||"manual"),N(e.spec.tags||[]),M(e.spec.targetRef||null),B((null===(a=e.spec.contact)||void 0===a?void 0:a.email)||""),X((null===(t=e.spec.contact)||void 0===t?void 0:t.team)||""),Z((null===(s=e.spec.documentation)||void 0===s?void 0:s.docsURL)||""),ae((null===(n=e.spec.documentation)||void 0===n?void 0:n.openAPISpec)||""),g(!1)}).catch(e=>{se(e.message||"Failed to load API product"),g(!1)}))},[e,o,p,v,x]);const{value:ie}=(0,S.A)(async()=>{if(!e)return null;const a=await x.fetch(`${v}/api/kuadrant/planpolicies`);return await a.json()},[v,x,e]),re=l().useMemo(()=>(null==ie?void 0:ie.items)&&D?ie.items.find(e=>{const a=e.targetRef;return"HTTPRoute"===(null==a?void 0:a.kind)&&(null==a?void 0:a.name)===D.name&&(!(null==a?void 0:a.namespace)||(null==a?void 0:a.namespace)===(D.namespace||o))}):null,[ie,D,o]),oe=()=>{z.trim()&&!E.includes(z.trim())&&(N([...E,z.trim()]),K(""))};return(0,s.jsxs)(U.A,{open:e,onClose:a,maxWidth:"md",fullWidth:!0,children:[(0,s.jsx)(F.A,{children:"Edit API Product"}),(0,s.jsxs)(H.A,{children:[te&&(0,s.jsx)(V.A,{severity:"error",style:{marginBottom:16},children:te}),A?(0,s.jsx)(y.k,{}):(0,s.jsxs)(c.A,{container:!0,spacing:2,children:[(0,s.jsx)(c.A,{item:!0,xs:6,children:(0,s.jsx)(_.A,{fullWidth:!0,label:"Name",value:p,disabled:!0,helperText:"Kubernetes resource name (immutable)",margin:"normal"})}),(0,s.jsx)(c.A,{item:!0,xs:6,children:(0,s.jsx)(_.A,{fullWidth:!0,label:"Namespace",value:o,disabled:!0,helperText:"Derived from HTTPRoute (immutable)",margin:"normal"})}),(0,s.jsx)(c.A,{item:!0,xs:6,children:(0,s.jsx)(_.A,{fullWidth:!0,label:"Display Name",value:j,onChange:e=>f(e.target.value),placeholder:"My API",margin:"normal",required:!0,disabled:ne,InputLabelProps:{classes:{asterisk:m.asterisk}}})}),(0,s.jsx)(c.A,{item:!0,xs:6,children:(0,s.jsx)(_.A,{fullWidth:!0,label:"Version",value:k,onChange:e=>C(e.target.value),placeholder:"v1",margin:"normal",disabled:ne})}),(0,s.jsx)(c.A,{item:!0,xs:6,children:(0,s.jsxs)(_.A,{fullWidth:!0,select:!0,label:"Publish Status",value:R,onChange:e=>I(e.target.value),margin:"normal",helperText:"Draft: Hidden from catalog. Published: Visible in catalog.",disabled:ne,children:[(0,s.jsx)(q.A,{value:"Draft",children:"Draft (Hidden)"}),(0,s.jsx)(q.A,{value:"Published",children:"Published (Visible)"})]})}),(0,s.jsx)(c.A,{item:!0,xs:6,children:(0,s.jsxs)(_.A,{fullWidth:!0,select:!0,label:"Approval Mode",value:$,onChange:e=>W(e.target.value),margin:"normal",helperText:"Automatic: keys created immediately. Manual: requires approval.",disabled:ne,children:[(0,s.jsx)(q.A,{value:"manual",children:"Manual"}),(0,s.jsx)(q.A,{value:"automatic",children:"Automatic"})]})}),(0,s.jsx)(c.A,{item:!0,xs:12,children:(0,s.jsx)(_.A,{fullWidth:!0,label:"Description",value:b,onChange:e=>P(e.target.value),placeholder:"API description",margin:"normal",multiline:!0,rows:2,required:!0,disabled:ne,InputLabelProps:{classes:{asterisk:m.asterisk}}})}),(0,s.jsxs)(c.A,{item:!0,xs:12,children:[(0,s.jsx)(d.A,{variant:"subtitle2",gutterBottom:!0,style:{marginTop:16},children:"Tags"}),(0,s.jsx)(r.A,{display:"flex",flexWrap:"wrap",marginBottom:1,style:{gap:8},children:E.map(e=>(0,s.jsx)(i.A,{label:e,onDelete:ne?void 0:()=>{return a=e,void N(E.filter(e=>e!==a));var a},size:"small",disabled:ne},e))}),(0,s.jsxs)(r.A,{display:"flex",style:{gap:8},children:[(0,s.jsx)(_.A,{fullWidth:!0,size:"small",value:z,onChange:e=>K(e.target.value),onKeyPress:e=>"Enter"===e.key&&oe(),placeholder:"Add tag",disabled:ne}),(0,s.jsx)(u.A,{onClick:oe,variant:"outlined",size:"small",disabled:ne,children:"Add"})]})]}),D&&(0,s.jsxs)(s.Fragment,{children:[(0,s.jsx)(c.A,{item:!0,xs:12,children:(0,s.jsx)(_.A,{fullWidth:!0,label:"HTTPRoute",value:`${D.namespace||o}/${D.name}`,disabled:!0,helperText:"Target HTTPRoute (immutable)",margin:"normal"})}),(0,s.jsx)(c.A,{item:!0,xs:12,children:(0,s.jsx)(J,{selectedPolicy:re,alertSeverity:"info",alertMessage:"No PlanPolicy found for this HTTPRoute.",includeTopMargin:!1})})]}),(0,s.jsx)(c.A,{item:!0,xs:6,children:(0,s.jsx)(_.A,{fullWidth:!0,label:"Contact Email",value:L,onChange:e=>B(e.target.value),placeholder:"api-team@example.com",margin:"normal",disabled:ne})}),(0,s.jsx)(c.A,{item:!0,xs:6,children:(0,s.jsx)(_.A,{fullWidth:!0,label:"Contact Team",value:Y,onChange:e=>X(e.target.value),placeholder:"platform-team",margin:"normal",disabled:ne})}),(0,s.jsx)(c.A,{item:!0,xs:6,children:(0,s.jsx)(_.A,{fullWidth:!0,label:"Docs URL",value:G,onChange:e=>Z(e.target.value),placeholder:"https://api.example.com/docs",margin:"normal",disabled:ne})}),(0,s.jsx)(c.A,{item:!0,xs:6,children:(0,s.jsx)(_.A,{fullWidth:!0,label:"OpenAPI Spec URL",value:ee,onChange:e=>ae(e.target.value),placeholder:"https://api.example.com/openapi.json",margin:"normal",disabled:ne})})]})]}),(0,s.jsxs)(O.A,{children:[(0,s.jsx)(u.A,{onClick:a,disabled:ne,children:"Cancel"}),(0,s.jsx)(u.A,{onClick:async()=>{se(""),le(!0);try{const e={spec:{displayName:j,description:b,version:k,publishStatus:R,approvalMode:$,tags:E,targetRef:D,...L||Y?{contact:{...L&&{email:L},...Y&&{team:Y}}}:{},...G||ee?{documentation:{...G&&{docsURL:G},...ee&&{openAPISpec:ee}}}:{}}},s=await x.fetch(`${v}/api/kuadrant/apiproducts/${o}/${p}`,{method:"PATCH",headers:{"Content-Type":"application/json"},body:JSON.stringify(e)});if(!s.ok){const e=await s.json();throw new Error(e.error||"failed to update apiproduct")}t(),a()}catch(e){se(e instanceof Error?e.message:String(e))}finally{le(!1)}},color:"primary",variant:"contained",disabled:ne||A||!j||!b,startIcon:ne?(0,s.jsx)(T.A,{size:16,color:"inherit"}):void 0,children:ne?"Saving...":"Save"})]})]})},ee=()=>{const e=(0,w.useApi)(w.configApiRef),a=(0,w.useApi)(w.fetchApiRef),t=(0,w.useApi)(w.alertApiRef),l=(0,w.useApi)(w.identityApiRef),C=e.getString("backend.baseUrl"),[T,R]=(0,n.useState)(""),[I,$]=(0,n.useState)(!1),[q,W]=(0,n.useState)(!1),[E,N]=(0,n.useState)(0),[D,L]=(0,n.useState)(!1),[B,U]=(0,n.useState)(null),[F,H]=(0,n.useState)(null),[_,O]=(0,n.useState)(!1),[V,J]=(0,n.useState)(null),{allowed:Y,loading:Q,error:ee}=(0,K.l)(G.FL),{allowed:ae,loading:te}=(0,K.l)(G.TE),{allowed:se,loading:ne,error:le}=(0,K.l)(G.SU),ie=ae||se,re=te||ne,{allowed:oe,loading:de}=(0,K.l)(G.EM),{allowed:ce,loading:ue,error:pe}=(0,K.l)(G.R_),{allowed:me}=(0,K.l)(G.U3),{allowed:he}=(0,K.l)(G.v_),xe=de||ue,{allowed:ve,loading:Ae,error:ge}=(0,K.l)(G.J);(0,S.A)(async()=>{const e=await l.getBackstageIdentity();R(e.userEntityRef)},[l]);const{value:je,loading:fe,error:ye}=(0,S.A)(async()=>{const e=await a.fetch(`${C}/api/kuadrant/apiproducts`);return await e.json()},[C,a,E]),{value:be,loading:Pe,error:Se}=(0,S.A)(async()=>{const e=await a.fetch(`${C}/api/kuadrant/planpolicies`);return await e.json()},[C,a,E]),we=fe||Pe||Q||re||xe||Ae,ke=ye||Se,Ce=ee||le||pe||ge,Te=[{title:"Name",field:"spec.displayName",render:e=>{var a,t;const n="Published"===(null===(a=e.spec)||void 0===a?void 0:a.publishStatus);var l;const i=null!==(l=null===(t=e.spec)||void 0===t?void 0:t.displayName)&&void 0!==l?l:e.metadata.name;return n?(0,s.jsx)(x.N_,{to:`/catalog/default/api/${e.metadata.name}/api-product`,children:(0,s.jsx)("strong",{children:i})}):(0,s.jsx)("span",{className:"text-muted",children:(0,s.jsx)("strong",{children:i})})},customFilterAndSearch:(e,a)=>{var t;return((null===(t=a.spec)||void 0===t?void 0:t.displayName)||a.metadata.name||"").toLowerCase().includes(e.toLowerCase())}},{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,s.jsx)(i.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,s.jsx)(i.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",filtering:!1,render:e=>{var t,n;const l=(null===(n=e.metadata)||void 0===n||null===(t=n.annotations)||void 0===t?void 0:t["backstage.io/owner"])===T,i=he||me&&l,d=ce||oe&&l;return i||d?(0,s.jsxs)(r.A,{display:"flex",style:{gap:4},children:[i&&(0,s.jsx)(o.A,{size:"small",onClick:()=>{return a=e.metadata.namespace,t=e.metadata.name,H({namespace:a,name:t}),void W(!0);var a,t},title:"Edit API Product",children:(0,s.jsx)(h.A,{fontSize:"small"})}),d&&(0,s.jsx)(o.A,{size:"small",onClick:()=>(async(e,t)=>{U({namespace:e,name:t}),J(null);try{const s=await a.fetch(`${C}/api/kuadrant/requests?namespace=${e}`);if(s.ok){const a=((await s.json()).items||[]).filter(a=>a.spec.apiName===t&&a.spec.apiNamespace===e),n=a.filter(e=>{var a;return"Approved"===(null===(a=e.status)||void 0===a?void 0:a.phase)}).length;J({requests:a.length,secrets:n})}}catch(e){console.warn("Failed to fetch delete stats:",e)}L(!0)})(e.metadata.namespace,e.metadata.name),title:"Delete API Product",children:(0,s.jsx)(m.A,{fontSize:"small"})})]}):null}}],Re=[{title:"Name",field:"metadata.name",render:e=>(0,s.jsx)(x.N_,{to:`/kuadrant/planpolicy/${e.metadata.namespace}/${e.metadata.name}`,children:(0,s.jsx)("strong",{children:e.metadata.name})})},{title:"Namespace",field:"metadata.namespace"}];return(0,s.jsxs)(A.Y,{themeId:"tool",children:[(0,s.jsx)(g.Y,{title:"Kuadrant",subtitle:"API management for Kubernetes",children:(0,s.jsx)(j.Y,{children:"Manage API products and access requests"})}),(0,s.jsxs)(f.U,{children:[we&&(0,s.jsx)(y.k,{}),ke&&(0,s.jsx)(b._,{error:ke}),Ce&&(0,s.jsxs)(r.A,{p:2,children:[(0,s.jsxs)(d.A,{color:"error",children:["unable to check permissions: ",Ce.message]}),(0,s.jsxs)(d.A,{variant:"body2",color:"textSecondary",children:["permission: ",ee?"kuadrant.apiproduct.create":pe?"kuadrant.apiproduct.delete":le?"kuadrant.apikeyrequest.read.all":ge?"kuadrant.planpolicy.list":"unknown"]}),(0,s.jsx)(d.A,{variant:"body2",color:"textSecondary",children:"please try again or contact your administrator"})]}),!we&&!ke&&!Ce&&(0,s.jsxs)(c.A,{container:!0,spacing:3,direction:"column",children:[(0,s.jsx)(c.A,{item:!0,children:(0,s.jsx)(z,{})}),(0,s.jsx)(c.A,{item:!0,children:(0,s.jsx)(P.n,{title:"API Products",action:Y?(0,s.jsx)(r.A,{display:"flex",alignItems:"center",height:"100%",mt:1,children:(0,s.jsx)(u.A,{variant:"contained",color:"primary",size:"small",startIcon:(0,s.jsx)(p.A,{}),onClick:()=>$(!0),children:"Create API Product"})}):void 0,children:(Ie=null==je?void 0:je.items,Ie&&0!==Ie.length?(0,s.jsx)(v.X,{options:{paging:Ie.length>5,pageSize:20,search:!0,filtering:!0,debounceInterval:300,toolbar:!0,emptyRowsWhenPaging:!1},columns:Te,data:Ie}):(0,s.jsx)(d.A,{variant:"body2",color:"textSecondary",children:"No API products found"}))})}),ve&&(0,s.jsx)(c.A,{item:!0,children:(0,s.jsx)(P.n,{title:"Plan Policies",children:(e=>e&&0!==e.length?(0,s.jsx)(v.X,{options:{paging:!1,search:!1,toolbar:!1},columns:Re,data:e}):(0,s.jsx)(d.A,{variant:"body2",color:"textSecondary",children:"No plan policies found"}))(null==be?void 0:be.items)})}),ie&&(0,s.jsx)(c.A,{item:!0,children:(0,s.jsx)(k.d,{})})]}),(0,s.jsx)(X,{open:I,onClose:()=>$(!1),onSuccess:()=>{N(e=>e+1),t.post({message:"API Product created",severity:"success",display:"transient"})}}),(0,s.jsx)(Z,{open:q,onClose:()=>W(!1),onSuccess:()=>{N(e=>e+1),t.post({message:"API Product updated",severity:"success",display:"transient"})},namespace:(null==F?void 0:F.namespace)||"",name:(null==F?void 0:F.name)||""}),(0,s.jsx)(M.K,{open:D,title:"Delete API Product",description:V?`Deleting "${null==B?void 0:B.name}" will also remove:\n\n• ${V.requests} API Key Request(s)\n• ${V.secrets} API Key Secret(s)\n\nThis action cannot be undone.`:`Deleting "${null==B?void 0:B.name}" will also remove all associated API Key Requests and Secrets.\nThis action cannot be undone.`,confirmText:null==B?void 0:B.name,severity:"high",deleting:_,onConfirm:async()=>{if(B){O(!0);try{if(!(await a.fetch(`${C}/api/kuadrant/apiproducts/${B.namespace}/${B.name}`,{method:"DELETE"})).ok)throw new Error("failed to delete apiproduct");N(e=>e+1),t.post({message:"API Product deleted",severity:"success",display:"transient"})}catch(e){console.error("error deleting apiproduct:",e),t.post({message:"Failed to delete API Product",severity:"error",display:"transient"})}finally{O(!1),L(!1),U(null)}}},onCancel:()=>{L(!1),U(null)}})]})]});var Ie},ae=()=>(0,s.jsx)(L,{permission:G.OP,errorMessage:"you don't have permission to view the Kuadrant page",children:(0,s.jsx)(ee,{})})}}]);
|
|
2
|
-
//# sourceMappingURL=7632.3141cc8f.chunk.js.map
|