@kuadrant/kuadrant-backstage-plugin-frontend 0.0.2-dev-724df72 → 0.0.2-dev-d9a9178

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.
@@ -1 +0,0 @@
1
- {"version":3,"file":"static/3459.a7c29521.chunk.js","mappings":"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","sources":["webpack://internal.plugin-kuadrant/./src/components/FilterPanel/FilterPanel.tsx","webpack://internal.plugin-kuadrant/./src/components/PermissionGate/PermissionGate.tsx","webpack://internal.plugin-kuadrant/./src/components/CreateAPIProductDialog/CreateAPIProductDialog.tsx","webpack://internal.plugin-kuadrant/./src/components/KuadrantPage/KuadrantPage.tsx"],"sourcesContent":["import React from 'react';\nimport {\n Box,\n Typography,\n Checkbox,\n FormControlLabel,\n FormGroup,\n Divider,\n Button,\n Collapse,\n makeStyles,\n} from '@material-ui/core';\nimport ExpandMoreIcon from '@material-ui/icons/ExpandMore';\nimport ExpandLessIcon from '@material-ui/icons/ExpandLess';\n\nconst useStyles = makeStyles(theme => ({\n root: {\n width: 240,\n minWidth: 240,\n padding: theme.spacing(2),\n borderRight: `1px solid ${theme.palette.divider}`,\n backgroundColor: theme.palette.background.paper,\n height: '100%',\n overflowY: 'auto',\n },\n sectionTitle: {\n fontWeight: 600,\n fontSize: '0.75rem',\n textTransform: 'uppercase',\n letterSpacing: '0.05em',\n color: theme.palette.text.secondary,\n marginBottom: theme.spacing(1),\n display: 'flex',\n alignItems: 'center',\n justifyContent: 'space-between',\n cursor: 'pointer',\n userSelect: 'none',\n },\n filterSection: {\n marginBottom: theme.spacing(2),\n },\n checkbox: {\n padding: theme.spacing(0.5),\n },\n checkboxLabel: {\n fontSize: '0.875rem',\n },\n clearButton: {\n marginTop: theme.spacing(2),\n },\n count: {\n fontSize: '0.75rem',\n color: theme.palette.text.secondary,\n marginLeft: theme.spacing(1),\n },\n}));\n\nexport interface FilterOption {\n value: string;\n label: string;\n count?: number;\n}\n\nexport interface FilterSection {\n id: string;\n title: string;\n options: FilterOption[];\n collapsed?: boolean;\n}\n\nexport interface FilterState {\n [sectionId: string]: string[];\n}\n\ninterface FilterPanelProps {\n sections: FilterSection[];\n filters: FilterState;\n onChange: (filters: FilterState) => void;\n onClear?: () => void;\n}\n\nexport const FilterPanel = ({\n sections,\n filters,\n onChange,\n onClear,\n}: FilterPanelProps) => {\n const classes = useStyles();\n const [collapsedSections, setCollapsedSections] = React.useState<Set<string>>(\n new Set(sections.filter(s => s.collapsed).map(s => s.id)),\n );\n\n const toggleSection = (sectionId: string) => {\n setCollapsedSections(prev => {\n const next = new Set(prev);\n if (next.has(sectionId)) {\n next.delete(sectionId);\n } else {\n next.add(sectionId);\n }\n return next;\n });\n };\n\n const handleCheckboxChange = (sectionId: string, value: string) => {\n const currentValues = filters[sectionId] || [];\n const newValues = currentValues.includes(value)\n ? currentValues.filter(v => v !== value)\n : [...currentValues, value];\n\n onChange({\n ...filters,\n [sectionId]: newValues,\n });\n };\n\n const hasActiveFilters = Object.values(filters).some(\n values => values.length > 0,\n );\n\n const handleClear = () => {\n const clearedFilters: FilterState = {};\n sections.forEach(section => {\n clearedFilters[section.id] = [];\n });\n onChange(clearedFilters);\n onClear?.();\n };\n\n return (\n <Box className={classes.root}>\n <Box display=\"flex\" justifyContent=\"space-between\" alignItems=\"center\" mb={2}>\n <Typography variant=\"subtitle2\">Filters</Typography>\n {hasActiveFilters && (\n <Button\n size=\"small\"\n color=\"primary\"\n onClick={handleClear}\n >\n Clear all\n </Button>\n )}\n </Box>\n\n <Divider />\n\n {sections.map(section => {\n const isCollapsed = collapsedSections.has(section.id);\n const selectedCount = (filters[section.id] || []).length;\n\n return (\n <Box key={section.id} className={classes.filterSection} mt={2}>\n <Box\n className={classes.sectionTitle}\n onClick={() => toggleSection(section.id)}\n >\n <Box display=\"flex\" alignItems=\"center\">\n <span>{section.title}</span>\n {selectedCount > 0 && (\n <span className={classes.count}>({selectedCount})</span>\n )}\n </Box>\n {isCollapsed ? (\n <ExpandMoreIcon fontSize=\"small\" />\n ) : (\n <ExpandLessIcon fontSize=\"small\" />\n )}\n </Box>\n\n <Collapse in={!isCollapsed}>\n <FormGroup>\n {section.options.map(option => (\n <FormControlLabel\n key={option.value}\n control={\n <Checkbox\n checked={(filters[section.id] || []).includes(option.value)}\n onChange={() =>\n handleCheckboxChange(section.id, option.value)\n }\n size=\"small\"\n className={classes.checkbox}\n color=\"primary\"\n />\n }\n label={\n <Box display=\"flex\" alignItems=\"center\">\n <span className={classes.checkboxLabel}>\n {option.label}\n </span>\n {option.count !== undefined && (\n <span className={classes.count}>({option.count})</span>\n )}\n </Box>\n }\n />\n ))}\n </FormGroup>\n </Collapse>\n </Box>\n );\n })}\n </Box>\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, {useEffect, 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 FormControl,\n RadioGroup,\n FormControlLabel,\n Radio,\n Tooltip,\n IconButton,\n InputAdornment,\n} from '@material-ui/core';\nimport InfoOutlinedIcon from '@material-ui/icons/InfoOutlined';\nimport AddIcon from '@material-ui/icons/Add';\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';\nimport { validateKubernetesName, validateURL } from '../../utils/validation';\n\nconst useStyles = makeStyles((theme) => ({\n asterisk: {\n color: '#f44336',\n },\n sectionHeader: {\n display: 'flex',\n alignItems: 'center',\n gap: theme.spacing(0.5),\n marginTop: theme.spacing(2),\n marginBottom: theme.spacing(1),\n },\n infoIcon: {\n fontSize: 18,\n color: theme.palette.text.secondary,\n },\n tagChip: {\n marginRight: theme.spacing(0.5),\n marginBottom: theme.spacing(0.5),\n },\n}));\n\ninterface CreateAPIProductDialogProps {\n open: boolean;\n onClose: () => void;\n onSuccess: (productInfo: { namespace: string; name: string; displayName: string }) => 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 const [nameError, setNameError] = useState<string | null>(null);\n const [openAPISpecError, setOpenAPISpecError] = useState<string | null>(null);\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 return data.items || [];\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 // format tier info for dropdown display\n const formatTierInfo = (policy: any): string => {\n if (!policy?.spec?.plans) return '';\n const tiers = Object.entries(policy.spec.plans)\n .map(([name, plan]: [string, any]) => {\n const limit = plan?.limits?.requests;\n if (!limit) return name;\n return `${name}: ${limit.count}/${limit.period}`;\n })\n .join('; ');\n return tiers ? ` (${tiers})` : '';\n };\n\n // get policy info for a route (for dropdown display)\n const getPolicyInfoForRoute = (routeNamespace: string, routeName: string): string => {\n const policy = getPlanPolicyForRoute(routeNamespace, routeName);\n if (!policy) return 'N/A';\n return `${policy.metadata.name}${formatTierInfo(policy)}`;\n };\n\n useEffect(() => {\n if (open) {\n setNameError(null);\n setOpenAPISpecError(null);\n }\n }, [open]);\n\n // validate handlers\n const handleNameChange = (value: string) => {\n setName(value);\n setNameError(validateKubernetesName(value));\n };\n\n const handleOpenAPISpecChange = (value: string) => {\n setOpenAPISpec(value);\n setOpenAPISpecError(validateURL(value));\n };\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 && { openAPISpecURL: 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({ namespace, name, displayName });\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 setNameError(null);\n setOpenAPISpecError(null);\n onClose();\n };\n\n const hasValidationErrors = !!nameError || !!openAPISpecError;\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 {/* API product info section */}\n <Box className={classes.sectionHeader}>\n <Typography variant=\"subtitle1\"><strong>API product info</strong></Typography>\n </Box>\n <Grid container spacing={2}>\n <Grid item xs={6}>\n <TextField\n fullWidth\n label=\"API product name\"\n value={displayName}\n onChange={e => setDisplayName(e.target.value)}\n placeholder=\"My API\"\n helperText=\"Give a unique name for your API product\"\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=\"Resource name\"\n value={name}\n onChange={e => handleNameChange(e.target.value)}\n placeholder=\"my-api\"\n helperText={nameError || \"Kubernetes resource name with lowercase, hyphens. Eg.flight_API\"}\n error={!!nameError}\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 helperText=\"Give a version to your API product\"\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=\"Tag\"\n value={tagInput}\n onChange={e => setTagInput(e.target.value)}\n onKeyPress={e => {\n if (e.key === 'Enter') {\n e.preventDefault();\n handleAddTag();\n }\n }}\n placeholder=\"Add tag\"\n helperText=\"Add a tag to your API product\"\n margin=\"normal\"\n disabled={creating}\n InputProps={{\n endAdornment: tagInput ? (\n <InputAdornment position=\"end\">\n <IconButton size=\"small\" onClick={handleAddTag} disabled={creating}>\n <AddIcon fontSize=\"small\" />\n </IconButton>\n </InputAdornment>\n ) : undefined,\n }}\n />\n </Grid>\n {tags.length > 0 && (\n <Grid item xs={12}>\n <Box display=\"flex\" flexWrap=\"wrap\">\n {tags.map(tag => (\n <Chip\n key={tag}\n label={tag}\n onDelete={creating ? undefined : () => handleDeleteTag(tag)}\n size=\"small\"\n className={classes.tagChip}\n disabled={creating}\n />\n ))}\n </Box>\n </Grid>\n )}\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 </Grid>\n\n {/* Add API and Associate route section */}\n <Box className={classes.sectionHeader}>\n <Typography variant=\"subtitle1\"><strong>Add API and Associate route</strong></Typography>\n <Tooltip title=\"Register an existing API and associate HTTPRoute for your API product\">\n <InfoOutlinedIcon className={classes.infoIcon} />\n </Tooltip>\n </Box>\n <Grid container spacing={2}>\n <Grid item xs={12}>\n <TextField\n fullWidth\n label=\"OpenAPI Spec URL\"\n value={openAPISpec}\n onChange={e => handleOpenAPISpecChange(e.target.value)}\n placeholder=\"https://api.example.com/openapi.json\"\n helperText={openAPISpecError || \"Enter the full path to your API spec file\"}\n error={!!openAPISpecError}\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={12}>\n <TextField\n fullWidth\n label=\"Documentation URL\"\n value={docsURL}\n onChange={e => setDocsURL(e.target.value)}\n placeholder=\"https://docs.example.com/api\"\n helperText=\"Link to external documentation for this API\"\n margin=\"normal\"\n disabled={creating}\n />\n </Grid>\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. 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 SelectProps={{\n 'data-testid': 'httproute-select',\n } as any}\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 const routeNs = route.metadata.namespace;\n const routeName = route.metadata.name;\n const policyInfo = getPolicyInfoForRoute(routeNs, routeName);\n return (\n <MenuItem\n key={`${routeNs}/${routeName}`}\n value={`${routeNs}/${routeName}`}\n >\n <Box>\n <Typography variant=\"body1\">{routeName}</Typography>\n <Typography variant=\"caption\" color=\"textSecondary\">\n Associated PlanPolicy: {policyInfo}\n </Typography>\n </Box>\n </MenuItem>\n );\n })}\n </TextField>\n </Grid>\n </Grid>\n\n {/* HTTPRoute policies section */}\n {selectedHTTPRoute && (\n <>\n <Box className={classes.sectionHeader}>\n <Typography variant=\"subtitle1\"><strong>HTTPRoute policies</strong></Typography>\n <Tooltip title=\"Shows the associated policies and rate limit tiers for the selected HTTPRoute\">\n <InfoOutlinedIcon className={classes.infoIcon} />\n </Tooltip>\n </Box>\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={false}\n />\n </>\n )}\n\n {/* API Key approval section */}\n <Box className={classes.sectionHeader}>\n <Typography variant=\"subtitle1\"><strong>API Key approval</strong></Typography>\n <Tooltip title=\"Choose how API key requests are handled for this product\">\n <InfoOutlinedIcon className={classes.infoIcon} />\n </Tooltip>\n </Box>\n <FormControl component=\"fieldset\" disabled={creating}>\n <RadioGroup\n row\n value={approvalMode}\n onChange={e => setApprovalMode(e.target.value as 'automatic' | 'manual')}\n >\n <FormControlLabel\n value=\"manual\"\n control={<Radio color=\"primary\" />}\n label={\n <Box>\n <Typography variant=\"body2\">Need manual approval</Typography>\n <Typography variant=\"caption\" color=\"textSecondary\">\n Requires approval for requesting this API\n </Typography>\n </Box>\n }\n />\n <FormControlLabel\n value=\"automatic\"\n control={<Radio color=\"primary\" />}\n label={\n <Box>\n <Typography variant=\"body2\">Automatic</Typography>\n <Typography variant=\"caption\" color=\"textSecondary\">\n Keys are created without need to be approved\n </Typography>\n </Box>\n }\n />\n </RadioGroup>\n </FormControl>\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 || hasValidationErrors}\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, useMemo, useCallback } from \"react\";\nimport {\n Typography,\n Box,\n Chip,\n Button,\n IconButton,\n CircularProgress,\n makeStyles,\n} 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 VpnKeyIcon from \"@material-ui/icons/VpnKey\";\nimport LockIcon from \"@material-ui/icons/Lock\";\nimport { FilterPanel, FilterSection, FilterState } from \"../FilterPanel\";\nimport {\n Header,\n Page,\n Content,\n SupportButton,\n ResponseErrorPanel,\n Link,\n Table,\n TableColumn,\n} from \"@backstage/core-components\";\nimport useAsync from \"react-use/lib/useAsync\";\nimport {\n useApi,\n configApiRef,\n fetchApiRef,\n alertApiRef,\n identityApiRef,\n} from \"@backstage/core-plugin-api\";\nimport { PermissionGate } from \"../PermissionGate\";\nimport { CreateAPIProductDialog } from \"../CreateAPIProductDialog\";\nimport {\n kuadrantApiProductCreatePermission,\n kuadrantApiProductDeleteOwnPermission,\n kuadrantApiProductDeleteAllPermission,\n kuadrantApiProductUpdateOwnPermission,\n kuadrantApiProductUpdateAllPermission,\n kuadrantApiProductListPermission,\n kuadrantPlanPolicyListPermission,\n} from \"../../permissions\";\nimport { useKuadrantPermission } from \"../../utils/permissions\";\nimport { EditAPIProductDialog } from \"../EditAPIProductDialog\";\nimport { ConfirmDeleteDialog } from \"../ConfirmDeleteDialog\";\nimport emptyStateIllustration from \"../../assets/empty-state-illustration.png\";\n\ntype KuadrantResource = {\n metadata: {\n name: string;\n namespace: string;\n creationTimestamp: string;\n annotations?: Record<string, string>;\n };\n spec?: any;\n status?: any;\n};\n\ntype KuadrantList = {\n items: KuadrantResource[];\n};\n\nconst useStyles = makeStyles((theme) => ({\n container: {\n display: \"flex\",\n height: \"100%\",\n minHeight: 400,\n },\n tableContainer: {\n flex: 1,\n overflow: \"auto\",\n padding: 10,\n },\n emptyState: {\n display: \"flex\",\n alignItems: \"center\",\n justifyContent: \"center\",\n padding: theme.spacing(6),\n minHeight: 400,\n },\n emptyStateContent: {\n display: \"flex\",\n alignItems: \"center\",\n gap: theme.spacing(6),\n maxWidth: 900,\n },\n emptyStateText: {\n flex: 1,\n },\n emptyStateTitle: {\n marginBottom: theme.spacing(2),\n },\n emptyStateDescription: {\n marginBottom: theme.spacing(3),\n color: theme.palette.text.secondary,\n },\n emptyStateImage: {\n maxWidth: 400,\n height: \"auto\",\n },\n}));\n\nexport const ResourceList = () => {\n const classes = useStyles();\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<{\n namespace: string;\n name: string;\n } | null>(null);\n const [apiProductToEdit, setApiProductToEdit] = useState<{\n namespace: string;\n name: string;\n } | null>(null);\n const [deleting, setDeleting] = useState(false);\n const [deleteStats, setDeleteStats] = useState<{\n requests: number;\n secrets: number;\n } | null>(null);\n const [filters, setFilters] = useState<FilterState>({\n status: [],\n policy: [],\n route: [],\n namespace: [],\n tags: [],\n authentication: [],\n });\n\n const {\n allowed: canCreateApiProduct,\n loading: createPermissionLoading,\n error: createPermissionError,\n } = useKuadrantPermission(kuadrantApiProductCreatePermission);\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 { allowed: canUpdateOwnApiProduct } = useKuadrantPermission(\n kuadrantApiProductUpdateOwnPermission,\n );\n\n const { allowed: canUpdateAllApiProducts } = useKuadrantPermission(\n kuadrantApiProductUpdateAllPermission,\n );\n\n const deletePermissionLoading =\n deleteOwnPermissionLoading || deleteAllPermissionLoading;\n\n const {\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 {\n value: apiProducts,\n loading: apiProductsLoading,\n error: apiProductsError,\n } = useAsync(async (): Promise<KuadrantList> => {\n const response = await fetchApi.fetch(\n `${backendUrl}/api/kuadrant/apiproducts`,\n );\n return await response.json();\n }, [backendUrl, fetchApi, refreshTrigger]);\n\n const {\n value: planPolicies,\n loading: planPoliciesLoading,\n error: planPoliciesError,\n } = useAsync(async (): Promise<KuadrantList> => {\n const response = await fetchApi.fetch(\n `${backendUrl}/api/kuadrant/planpolicies`,\n );\n return await response.json();\n }, [backendUrl, fetchApi, refreshTrigger]);\n\n // helper to find policy for a given route\n const getPolicyForProduct = useCallback((product: KuadrantResource): string | null => {\n if (!planPolicies?.items) return null;\n const targetRef = product.spec?.targetRef;\n if (!targetRef) return null;\n\n const policy = planPolicies.items.find((pp: KuadrantResource) => {\n const ref = (pp as any).targetRef;\n return (\n ref?.kind === \"HTTPRoute\" &&\n ref?.name === targetRef.name &&\n (!ref?.namespace || ref?.namespace === (targetRef.namespace || product.metadata.namespace))\n );\n });\n return policy?.metadata.name || null;\n }, [planPolicies]);\n\n // helper to get auth schemes for a product\n const getAuthSchemes = useCallback((product: KuadrantResource): string[] => {\n const authSchemes = product.status?.discoveredAuthScheme?.authentication || {};\n const schemeObjects = Object.values(authSchemes);\n const schemes: string[] = [];\n\n if (schemeObjects.some((scheme: any) => scheme.hasOwnProperty(\"apiKey\"))) {\n schemes.push(\"API Key\");\n }\n if (schemeObjects.some((scheme: any) => scheme.hasOwnProperty(\"jwt\"))) {\n schemes.push(\"OIDC\");\n }\n if (schemes.length === 0) {\n schemes.push(\"Unknown\");\n }\n return schemes;\n }, []);\n\n const loading =\n apiProductsLoading ||\n planPoliciesLoading ||\n createPermissionLoading ||\n deletePermissionLoading ||\n planPolicyPermissionLoading;\n const error = apiProductsError || planPoliciesError;\n const permissionError =\n createPermissionError || deletePermissionError || planPolicyPermissionError;\n\n const allProducts = useMemo(() => {\n return apiProducts?.items || [];\n }, [apiProducts]);\n\n const filterSections: FilterSection[] = useMemo(() => {\n const statusCounts = { Draft: 0, Published: 0 };\n const policyCounts = new Map<string, number>();\n const routeCounts = new Map<string, number>();\n const namespaceCounts = new Map<string, number>();\n const tagCounts = new Map<string, number>();\n const authCounts = new Map<string, number>();\n\n allProducts.forEach((p: KuadrantResource) => {\n const status = p.spec?.publishStatus || \"Draft\";\n statusCounts[status as keyof typeof statusCounts]++;\n\n const policy = getPolicyForProduct(p) || \"N/A\";\n policyCounts.set(policy, (policyCounts.get(policy) || 0) + 1);\n\n const route = p.spec?.targetRef?.name || \"unknown\";\n routeCounts.set(route, (routeCounts.get(route) || 0) + 1);\n\n const ns = p.metadata.namespace;\n namespaceCounts.set(ns, (namespaceCounts.get(ns) || 0) + 1);\n\n const tags = p.spec?.tags || [];\n tags.forEach((tag: string) => {\n tagCounts.set(tag, (tagCounts.get(tag) || 0) + 1);\n });\n\n const authSchemes = getAuthSchemes(p);\n authSchemes.forEach((scheme: string) => {\n authCounts.set(scheme, (authCounts.get(scheme) || 0) + 1);\n });\n });\n\n return [\n {\n id: \"status\",\n title: \"Status\",\n options: [\n { value: \"Draft\", label: \"Draft\", count: statusCounts.Draft },\n { value: \"Published\", label: \"Published\", count: statusCounts.Published },\n ],\n },\n {\n id: \"authentication\",\n title: \"Authentication\",\n options: Array.from(authCounts.entries()).map(([scheme, count]) => ({\n value: scheme,\n label: scheme,\n count,\n })),\n },\n {\n id: \"policy\",\n title: \"Policy\",\n options: Array.from(policyCounts.entries()).map(([name, count]) => ({\n value: name,\n label: name,\n count,\n })),\n collapsed: policyCounts.size > 5,\n },\n {\n id: \"route\",\n title: \"Route\",\n options: Array.from(routeCounts.entries()).map(([name, count]) => ({\n value: name,\n label: name,\n count,\n })),\n collapsed: routeCounts.size > 5,\n },\n {\n id: \"namespace\",\n title: \"Namespace\",\n options: Array.from(namespaceCounts.entries()).map(([ns, count]) => ({\n value: ns,\n label: ns,\n count,\n })),\n collapsed: namespaceCounts.size > 5,\n },\n {\n id: \"tags\",\n title: \"Tags\",\n options: Array.from(tagCounts.entries()).map(([tag, count]) => ({\n value: tag,\n label: tag,\n count,\n })),\n collapsed: tagCounts.size > 5,\n },\n ];\n }, [allProducts, getPolicyForProduct, getAuthSchemes]);\n\n const filteredProducts = useMemo(() => {\n return allProducts.filter((p: KuadrantResource) => {\n if (filters.status.length > 0) {\n const status = p.spec?.publishStatus || \"Draft\";\n if (!filters.status.includes(status)) return false;\n }\n\n if (filters.authentication.length > 0) {\n const authSchemes = getAuthSchemes(p);\n if (!filters.authentication.some((a: string) => authSchemes.includes(a))) return false;\n }\n\n if (filters.policy.length > 0) {\n const policy = getPolicyForProduct(p) || \"N/A\";\n if (!filters.policy.includes(policy)) return false;\n }\n\n if (filters.route.length > 0) {\n const route = p.spec?.targetRef?.name || \"unknown\";\n if (!filters.route.includes(route)) return false;\n }\n\n if (filters.namespace.length > 0) {\n if (!filters.namespace.includes(p.metadata.namespace)) return false;\n }\n\n if (filters.tags.length > 0) {\n const tags = p.spec?.tags || [];\n if (!filters.tags.some((t: string) => tags.includes(t))) return false;\n }\n\n return true;\n });\n }, [allProducts, filters, getPolicyForProduct, getAuthSchemes]);\n\n const handleCreateSuccess = (productInfo: { namespace: string; name: string; displayName: string }) => {\n setRefreshTrigger((prev) => prev + 1);\n alertApi.post({\n message: `\"${productInfo.displayName}\" created successfully`,\n severity: \"success\",\n display: \"transient\",\n });\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 const productName = apiProductToEdit?.name || \"API Product\";\n alertApi.post({\n message: `\"${productName}\" updated successfully`,\n severity: \"success\",\n display: \"transient\",\n });\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(\n `${backendUrl}/api/kuadrant/requests?namespace=${namespace}`,\n );\n if (response.ok) {\n const data = await response.json();\n const related = (data.items || []).filter(\n (r: any) =>\n r.spec.apiName === name && r.spec.apiNamespace === namespace,\n );\n const approved = related.filter(\n (r: any) => r.status?.phase === \"Approved\",\n ).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 const deletedName = apiProductToDelete?.name || \"API Product\";\n setRefreshTrigger((prev) => prev + 1);\n alertApi.post({\n message: `\"${deletedName}\" deleted successfully`,\n severity: \"success\",\n display: \"transient\",\n });\n } catch (err) {\n console.error(\"error deleting apiproduct:\", err);\n alertApi.post({\n message: \"Failed to delete API Product\",\n severity: \"error\",\n display: \"transient\",\n });\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 handlePublishToggle = async (row: any) => {\n const namespace = row.metadata.namespace;\n const name = row.metadata.name;\n const displayName = row.spec?.displayName || name;\n const currentStatus = row.spec?.publishStatus || \"Draft\";\n const newStatus = currentStatus === \"Published\" ? \"Draft\" : \"Published\";\n\n try {\n const response = await fetchApi.fetch(\n `${backendUrl}/api/kuadrant/apiproducts/${namespace}/${name}`,\n {\n method: \"PATCH\",\n headers: { \"Content-Type\": \"application/json\" },\n body: JSON.stringify({\n spec: { publishStatus: newStatus },\n }),\n },\n );\n\n if (!response.ok) {\n throw new Error(\"failed to update publish status\");\n }\n\n setRefreshTrigger((prev) => prev + 1);\n alertApi.post({\n message: `\"${displayName}\" ${newStatus === \"Published\" ? \"published\" : \"unpublished\"} successfully`,\n severity: \"success\",\n display: \"transient\",\n });\n } catch (err) {\n console.error(\"error updating publish status:\", err);\n alertApi.post({\n message: \"Failed to update publish status\",\n severity: \"error\",\n display: \"transient\",\n });\n }\n };\n\n const columns: TableColumn[] = [\n {\n title: \"Name\",\n field: \"spec.displayName\",\n render: (row: any) => {\n const displayName = row.spec?.displayName ?? row.metadata.name;\n return (\n <Link to={`/kuadrant/api-products/${row.metadata.namespace}/${row.metadata.name}`}>\n <strong>{displayName}</strong>\n </Link>\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: \"Version\",\n field: \"spec.version\",\n render: (row: any) => row.spec?.version || \"-\",\n },\n {\n title: \"Route\",\n field: \"spec.targetRef.name\",\n render: (row: any) => row.spec?.targetRef?.name || \"-\",\n },\n {\n title: \"Policy\",\n field: \"policy\",\n render: (row: any) => getPolicyForProduct(row) || \"N/A\",\n },\n {\n title: \"Tags\",\n field: \"spec.tags\",\n render: (row: any) => {\n const tags = row.spec?.tags || [];\n if (tags.length === 0) return \"-\";\n return (\n <Box display=\"flex\" style={{ gap: 4, flexWrap: \"wrap\" }}>\n {tags.map((tag: string) => (\n <Chip key={tag} label={tag} size=\"small\" variant=\"outlined\" />\n ))}\n </Box>\n );\n },\n },\n {\n title: \"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: \"Authentication\",\n field: \"status.discoveredAuthScheme\",\n render: (row: any) => {\n const authSchemes =\n row.status?.discoveredAuthScheme?.authentication || {};\n const schemeObjects = Object.values(authSchemes);\n\n const hasApiKey = schemeObjects.some((scheme: any) =>\n scheme.hasOwnProperty(\"apiKey\"),\n );\n const hasJwt = schemeObjects.some((scheme: any) =>\n scheme.hasOwnProperty(\"jwt\"),\n );\n\n if (!hasApiKey && !hasJwt) {\n return (\n <Typography variant=\"body2\" style={{ fontStyle: \"italic\" }}>\n unknown\n </Typography>\n );\n }\n\n return (\n <Box display=\"flex\" style={{ gap: 4 }}>\n {hasApiKey && (\n <Chip\n icon={<VpnKeyIcon />}\n label=\"API Key\"\n size=\"small\"\n color=\"primary\"\n />\n )}\n {hasJwt && (\n <Chip\n icon={<LockIcon />}\n label=\"OIDC\"\n size=\"small\"\n color=\"secondary\"\n />\n )}\n </Box>\n );\n },\n },\n {\n title: \"Namespace\",\n field: \"metadata.namespace\",\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 =\n canUpdateAllApiProducts || (canUpdateOwnApiProduct && isOwner);\n const canDelete =\n canDeleteAllApiProducts || (canDeleteOwnApiProduct && isOwner);\n const isPublished = row.spec?.publishStatus === \"Published\";\n\n return (\n <Box display=\"flex\" alignItems=\"center\" style={{ gap: 4 }}>\n {canEdit && (\n <Button\n size=\"small\"\n color=\"primary\"\n onClick={() => handlePublishToggle(row)}\n style={{ marginRight: 4, textTransform: \"none\" }}\n >\n {isPublished ? \"Unpublish\" : \"Publish\"}\n </Button>\n )}\n {canEdit && (\n <IconButton\n size=\"small\"\n onClick={() =>\n handleEditClick(row.metadata.namespace, row.metadata.name)\n }\n title=\"Edit API Product\"\n >\n <EditIcon fontSize=\"small\" />\n </IconButton>\n )}\n {canDelete && (\n <IconButton\n size=\"small\"\n onClick={() =>\n handleDeleteClick(row.metadata.namespace, row.metadata.name)\n }\n title=\"Delete API Product\"\n >\n <DeleteIcon fontSize=\"small\" />\n </IconButton>\n )}\n </Box>\n );\n },\n },\n ];\n\n return (\n <Page themeId=\"tool\">\n <Header\n title=\"API Products\"\n subtitle=\"Manage API products for Kubernetes\"\n >\n <SupportButton>Manage API products and plan policies</SupportButton>\n </Header>\n <Content>\n {loading && (\n <Box\n display=\"flex\"\n flexDirection=\"column\"\n alignItems=\"center\"\n justifyContent=\"center\"\n minHeight={300}\n >\n <CircularProgress />\n <Typography variant=\"h6\" style={{ marginTop: 16 }}>\n Loading data...\n </Typography>\n <Typography variant=\"body2\" color=\"textSecondary\">\n Preparing your data... This should only take a moment.\n </Typography>\n </Box>\n )}\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:{\" \"}\n {createPermissionError\n ? \"kuadrant.apiproduct.create\"\n : deletePermissionError\n ? \"kuadrant.apiproduct.delete\"\n : planPolicyPermissionError\n ? \"kuadrant.planpolicy.list\"\n : \"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 && allProducts.length === 0 && (\n <Box className={classes.emptyState}>\n <Box className={classes.emptyStateContent}>\n <Box className={classes.emptyStateText}>\n <Typography variant=\"h4\" className={classes.emptyStateTitle}>\n API Product\n </Typography>\n <Typography\n variant=\"body1\"\n className={classes.emptyStateDescription}\n >\n Create API product by registering existing API, associate\n route and policy\n </Typography>\n {canCreateApiProduct && (\n <Button\n variant=\"contained\"\n color=\"primary\"\n startIcon={<AddIcon />}\n onClick={() => setCreateDialogOpen(true)}\n >\n Create API Product\n </Button>\n )}\n </Box>\n <img\n src={emptyStateIllustration}\n alt=\"API Product illustration\"\n className={classes.emptyStateImage}\n />\n </Box>\n </Box>\n )}\n {!loading && !error && !permissionError && allProducts.length > 0 && (\n <Box className={classes.container}>\n <FilterPanel\n sections={filterSections}\n filters={filters}\n onChange={setFilters}\n />\n <Box className={classes.tableContainer}>\n <Box display=\"flex\" justifyContent=\"flex-end\" mb={2}>\n {canCreateApiProduct && (\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 )}\n </Box>\n {filteredProducts.length === 0 ? (\n <Box p={4} textAlign=\"center\">\n <Typography variant=\"body1\" color=\"textSecondary\">\n No API products match the selected filters.\n </Typography>\n </Box>\n ) : (\n <Table\n options={{\n paging: filteredProducts.length > 10,\n pageSize: 20,\n search: true,\n filtering: false,\n debounceInterval: 300,\n toolbar: true,\n emptyRowsWhenPaging: false,\n }}\n columns={columns}\n data={filteredProducts}\n />\n )}\n </Box>\n </Box>\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(s)\n• ${deleteStats.secrets} API Key Secret(s)\n\nThis action cannot be undone.`\n : `Deleting \"${apiProductToDelete?.name}\" will also remove all associated API Keys 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":["useStyles","makeStyles","theme","root","width","minWidth","padding","spacing","borderRight","palette","divider","backgroundColor","background","paper","height","overflowY","sectionTitle","fontWeight","fontSize","textTransform","letterSpacing","color","text","secondary","marginBottom","display","alignItems","justifyContent","cursor","userSelect","filterSection","checkbox","checkboxLabel","clearButton","marginTop","count","marginLeft","FilterPanel","sections","filters","onChange","onClear","classes","collapsedSections","setCollapsedSections","React","Set","filter","s","collapsed","map","id","hasActiveFilters","Object","values","some","length","Box","className","mb","Typography","variant","Button","size","onClick","clearedFilters","forEach","section","Divider","isCollapsed","has","selectedCount","mt","toggleSection","sectionId","prev","next","delete","add","span","title","ExpandMoreIcon","ExpandLessIcon","Collapse","in","FormGroup","options","option","FormControlLabel","control","Checkbox","checked","includes","value","currentValues","newValues","v","handleCheckboxChange","label","undefined","PermissionGate","children","permission","fallback","errorMessage","allowed","loading","error","useKuadrantPermission","Progress","p","message","name","asterisk","sectionHeader","gap","infoIcon","tagChip","marginRight","CreateAPIProductDialog","open","onClose","onSuccess","config","useApi","configApiRef","fetchApi","fetchApiRef","backendUrl","getString","setName","useState","displayName","setDisplayName","description","setDescription","version","setVersion","approvalMode","setApprovalMode","publishStatus","setPublishStatus","tags","setTags","tagInput","setTagInput","selectedHTTPRoute","setSelectedHTTPRoute","contactEmail","setContactEmail","contactTeam","setContactTeam","docsURL","setDocsURL","openAPISpec","setOpenAPISpec","setError","creating","setCreating","httpRoutesRetry","setHttpRoutesRetry","nameError","setNameError","openAPISpecError","setOpenAPISpecError","httpRoutes","httpRoutesLoading","httpRoutesError","useAsync","async","response","fetch","json","items","planPolicies","planPoliciesError","getPlanPolicyForRoute","routeNamespace","routeName","find","pp","ref","targetRef","kind","namespace","selectedRouteInfo","split","selectedPolicy","useEffect","handleAddTag","trim","handleClose","hasValidationErrors","Dialog","maxWidth","fullWidth","DialogTitle","DialogContent","Alert","severity","style","strong","Grid","container","item","xs","TextField","e","target","placeholder","helperText","margin","required","disabled","InputLabelProps","handleNameChange","validateKubernetesName","onKeyPress","key","preventDefault","InputProps","endAdornment","InputAdornment","position","IconButton","AddIcon","flexWrap","tag","Chip","onDelete","handleDeleteTag","tagToDelete","multiline","rows","Tooltip","InfoOutlinedIcon","handleOpenAPISpecChange","validateURL","select","SelectProps","MenuItem","route","routeNs","metadata","policyInfo","policy","spec","plans","tiers","entries","plan","limit","limits","requests","period","join","formatTierInfo","getPolicyInfoForRoute","PlanPolicyDetails","alertSeverity","alertMessage","includeTopMargin","FormControl","component","RadioGroup","row","Radio","DialogActions","Error","selectedRouteNamespace","selectedRouteName","apiProduct","apiVersion","group","contact","email","team","documentation","openAPISpecURL","method","headers","body","JSON","stringify","ok","errorData","err","String","startIcon","CircularProgress","minHeight","tableContainer","flex","overflow","emptyState","emptyStateContent","emptyStateText","emptyStateTitle","emptyStateDescription","emptyStateImage","ResourceList","alertApi","alertApiRef","identityApi","identityApiRef","userEntityRef","setUserEntityRef","createDialogOpen","setCreateDialogOpen","editDialogOpen","setEditDialogOpen","refreshTrigger","setRefreshTrigger","deleteDialogOpen","setDeleteDialogOpen","apiProductToDelete","setApiProductToDelete","apiProductToEdit","setApiProductToEdit","deleting","setDeleting","deleteStats","setDeleteStats","setFilters","status","authentication","canCreateApiProduct","createPermissionLoading","createPermissionError","kuadrantApiProductCreatePermission","canDeleteOwnApiProduct","deleteOwnPermissionLoading","kuadrantApiProductDeleteOwnPermission","canDeleteAllApiProducts","deleteAllPermissionLoading","deletePermissionError","kuadrantApiProductDeleteAllPermission","canUpdateOwnApiProduct","kuadrantApiProductUpdateOwnPermission","canUpdateAllApiProducts","kuadrantApiProductUpdateAllPermission","deletePermissionLoading","planPolicyPermissionLoading","planPolicyPermissionError","kuadrantPlanPolicyListPermission","identity","getBackstageIdentity","apiProducts","apiProductsLoading","apiProductsError","planPoliciesLoading","getPolicyForProduct","useCallback","product","getAuthSchemes","authSchemes","discoveredAuthScheme","schemeObjects","schemes","scheme","hasOwnProperty","push","permissionError","allProducts","useMemo","filterSections","statusCounts","Draft","Published","policyCounts","Map","routeCounts","namespaceCounts","tagCounts","authCounts","set","get","ns","Array","from","filteredProducts","a","t","columns","field","render","Link","to","customFilterAndSearch","term","toLowerCase","hasApiKey","hasJwt","icon","VpnKeyIcon","LockIcon","fontStyle","filtering","isOwner","annotations","canEdit","canDelete","isPublished","newStatus","post","console","handlePublishToggle","handleEditClick","EditIcon","related","r","apiName","apiNamespace","approved","phase","secrets","warn","handleDeleteClick","DeleteIcon","Page","themeId","Header","subtitle","SupportButton","Content","flexDirection","ResponseErrorPanel","img","src","emptyStateIllustration","alt","textAlign","Table","paging","pageSize","search","debounceInterval","toolbar","emptyRowsWhenPaging","data","productInfo","EditAPIProductDialog","productName","ConfirmDeleteDialog","confirmText","onConfirm","deletedName","onCancel","KuadrantPage","kuadrantApiProductListPermission"],"sourceRoot":""}
@@ -1,2 +0,0 @@
1
- "use strict";(self.webpackChunkinternal_plugin_kuadrant=self.webpackChunkinternal_plugin_kuadrant||[]).push([[8799],{24170:(e,a,t)=>{t.d(a,{o:()=>i,q:()=>n});const i=e=>e&&e.trim()?e.length>253?"Must be 253 characters or less":/^[a-z0-9]([-a-z0-9]*[a-z0-9])?$/.test(e)?null:"Must be lowercase alphanumeric with hyphens, start and end with alphanumeric":"Name is required",n=e=>{if(!e)return null;try{const a=new URL(e);return["http:","https:"].includes(a.protocol)?null:"Must be a valid HTTP or HTTPS URL"}catch{return"Must be a valid URL"}}},26997:(e,a,t)=>{t.d(a,{C:()=>$});var i=t(31085),n=t(95478),r=t.n(n),l=t(76891),s=t(61477),o=t(46805),d=t(72501),c=t(10394),u=t(42899),p=t(16249),h=t(34839),m=t(29365),x=t(67720),A=t(71677),g=t(95061),v=t(29635),j=t(86901),y=t(30285),b=t(93453),f=t(64947),k=t(78467),P=t(58837),T=t(89031),S=t(18466),w=t(22097),I=t(84441),C=t(86687),R=t(91638),H=t(65867),N=t(24170);const W=(0,P.A)(e=>({asterisk:{color:"#f44336"},sectionHeader:{display:"flex",alignItems:"center",gap:e.spacing(.5),marginTop:e.spacing(2),marginBottom:e.spacing(1)},infoIcon:{fontSize:18,color:e.palette.text.secondary},tagChip:{marginRight:e.spacing(.5),marginBottom:e.spacing(.5)}})),$=({open:e,onClose:a,onSuccess:t,namespace:P,name:$})=>{const L=W(),M=(0,w.useApi)(w.configApiRef),z=(0,w.useApi)(w.fetchApiRef),U=M.getString("backend.baseUrl"),[q,E]=(0,n.useState)(!1),[D,F]=(0,n.useState)(""),[B,K]=(0,n.useState)(""),[_,J]=(0,n.useState)("v1"),[G,O]=(0,n.useState)("Draft"),[V,Q]=(0,n.useState)("manual"),[X,Y]=(0,n.useState)([]),[Z,ee]=(0,n.useState)(null),[ae,te]=(0,n.useState)(""),[ie,ne]=(0,n.useState)(""),[re,le]=(0,n.useState)(""),[se,oe]=(0,n.useState)(""),[de,ce]=(0,n.useState)(""),[ue,pe]=(0,n.useState)(""),[he,me]=(0,n.useState)(!1),[xe,Ae]=(0,n.useState)(null);(0,n.useEffect)(()=>{e&&P&&$&&(E(!0),pe(""),z.fetch(`${U}/api/kuadrant/apiproducts/${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,i,n;F(e.spec.displayName||""),K(e.spec.description||""),J(e.spec.version||"v1"),O(e.spec.publishStatus||"Draft"),Q(e.spec.approvalMode||"manual"),Y(e.spec.tags||[]),ee(e.spec.targetRef||null),ne((null===(a=e.spec.contact)||void 0===a?void 0:a.email)||""),le((null===(t=e.spec.contact)||void 0===t?void 0:t.team)||""),oe((null===(i=e.spec.documentation)||void 0===i?void 0:i.docsURL)||""),ce((null===(n=e.spec.documentation)||void 0===n?void 0:n.openAPISpecURL)||""),Ae(null),E(!1)}).catch(e=>{pe(e.message||"Failed to load API product"),E(!1)}))},[e,P,$,U,z]);const{value:ge,error:ve}=(0,R.A)(async()=>{if(!e)return null;const a=await z.fetch(`${U}/api/kuadrant/planpolicies`);return await a.json()},[U,z,e]),je=r().useMemo(()=>(null==ge?void 0:ge.items)&&Z?ge.items.find(e=>{const a=e.targetRef;return"HTTPRoute"===(null==a?void 0:a.kind)&&(null==a?void 0:a.name)===Z.name&&(!(null==a?void 0:a.namespace)||(null==a?void 0:a.namespace)===(Z.namespace||P))}):null,[ge,Z,P]);(0,n.useEffect)(()=>{e&&Ae(null)},[e]);const ye=()=>{ae.trim()&&!X.includes(ae.trim())&&(Y([...X,ae.trim()]),te(""))};return(0,i.jsxs)(l.A,{open:e,onClose:a,maxWidth:"md",fullWidth:!0,children:[(0,i.jsx)(s.A,{children:"Edit API Product"}),(0,i.jsxs)(o.A,{children:[ue&&(0,i.jsx)(I.A,{severity:"error",style:{marginBottom:16},children:ue}),ve&&(0,i.jsxs)(I.A,{severity:"warning",style:{marginBottom:16},children:[(0,i.jsx)("strong",{children:"Failed to load PlanPolicies:"})," ",ve.message,(0,i.jsx)(d.A,{variant:"body2",style:{marginTop:8},children:"Plan information may be incomplete."})]}),q?(0,i.jsx)(C.k,{}):(0,i.jsxs)(i.Fragment,{children:[(0,i.jsx)(c.A,{className:L.sectionHeader,children:(0,i.jsx)(d.A,{variant:"subtitle1",children:(0,i.jsx)("strong",{children:"API product info"})})}),(0,i.jsxs)(u.A,{container:!0,spacing:2,children:[(0,i.jsx)(u.A,{item:!0,xs:6,children:(0,i.jsx)(p.A,{fullWidth:!0,label:"API product name",value:D,onChange:e=>F(e.target.value),placeholder:"My API",helperText:"Give a unique name for your API product",margin:"normal",required:!0,disabled:he,InputLabelProps:{classes:{asterisk:L.asterisk}}})}),(0,i.jsx)(u.A,{item:!0,xs:6,children:(0,i.jsx)(p.A,{fullWidth:!0,label:"Resource name",value:$,disabled:!0,helperText:"Kubernetes resource name (immutable)",margin:"normal"})}),(0,i.jsx)(u.A,{item:!0,xs:6,children:(0,i.jsx)(p.A,{fullWidth:!0,label:"Version",value:_,onChange:e=>J(e.target.value),placeholder:"v1",helperText:"Give a version to your API product",margin:"normal",disabled:he})}),(0,i.jsx)(u.A,{item:!0,xs:6,children:(0,i.jsx)(p.A,{fullWidth:!0,label:"Tag",value:ae,onChange:e=>te(e.target.value),onKeyPress:e=>{"Enter"===e.key&&(e.preventDefault(),ye())},placeholder:"Add tag",helperText:"Add a tag to your API product",margin:"normal",disabled:he,InputProps:{endAdornment:ae?(0,i.jsx)(h.A,{position:"end",children:(0,i.jsx)(m.A,{size:"small",onClick:ye,disabled:he,children:(0,i.jsx)(S.A,{fontSize:"small"})})}):void 0}})}),X.length>0&&(0,i.jsx)(u.A,{item:!0,xs:12,children:(0,i.jsx)(c.A,{display:"flex",flexWrap:"wrap",children:X.map(e=>(0,i.jsx)(x.A,{label:e,onDelete:he?void 0:()=>{return a=e,void Y(X.filter(e=>e!==a));var a},size:"small",className:L.tagChip,disabled:he},e))})}),(0,i.jsx)(u.A,{item:!0,xs:12,children:(0,i.jsx)(p.A,{fullWidth:!0,label:"Description",value:B,onChange:e=>K(e.target.value),placeholder:"API description",margin:"normal",multiline:!0,rows:2,required:!0,disabled:he,InputLabelProps:{classes:{asterisk:L.asterisk}}})})]}),(0,i.jsxs)(c.A,{className:L.sectionHeader,children:[(0,i.jsx)(d.A,{variant:"subtitle1",children:(0,i.jsx)("strong",{children:"Associated route"})}),(0,i.jsx)(A.Ay,{title:"The HTTPRoute this API product is associated with",children:(0,i.jsx)(T.A,{className:L.infoIcon})})]}),(0,i.jsxs)(u.A,{container:!0,spacing:2,children:[(0,i.jsx)(u.A,{item:!0,xs:12,children:(0,i.jsx)(p.A,{fullWidth:!0,label:"OpenAPI Spec URL",value:de,onChange:e=>{return a=e.target.value,ce(a),void Ae((0,N.q)(a));var a},placeholder:"https://api.example.com/openapi.json",helperText:xe||"Enter the full path to your API spec file",error:!!xe,margin:"normal",disabled:he})}),(0,i.jsx)(u.A,{item:!0,xs:12,children:(0,i.jsx)(p.A,{fullWidth:!0,label:"Documentation URL",value:se,onChange:e=>oe(e.target.value),placeholder:"https://docs.example.com/api",helperText:"Link to external documentation for this API",margin:"normal",disabled:he})}),Z&&(0,i.jsx)(u.A,{item:!0,xs:12,children:(0,i.jsx)(p.A,{fullWidth:!0,label:"HTTPRoute",value:`${Z.namespace||P}/${Z.name}`,disabled:!0,helperText:"Target HTTPRoute (immutable)",margin:"normal"})})]}),Z&&(0,i.jsxs)(i.Fragment,{children:[(0,i.jsxs)(c.A,{className:L.sectionHeader,children:[(0,i.jsx)(d.A,{variant:"subtitle1",children:(0,i.jsx)("strong",{children:"HTTPRoute policies"})}),(0,i.jsx)(A.Ay,{title:"Shows the associated policies and rate limit tiers for the HTTPRoute",children:(0,i.jsx)(T.A,{className:L.infoIcon})})]}),(0,i.jsx)(H.g,{selectedPolicy:je,alertSeverity:"info",alertMessage:"No PlanPolicy found for this HTTPRoute.",includeTopMargin:!1})]}),(0,i.jsxs)(c.A,{className:L.sectionHeader,children:[(0,i.jsx)(d.A,{variant:"subtitle1",children:(0,i.jsx)("strong",{children:"API Key approval"})}),(0,i.jsx)(A.Ay,{title:"Choose how API key requests are handled for this product",children:(0,i.jsx)(T.A,{className:L.infoIcon})})]}),(0,i.jsx)(g.A,{component:"fieldset",disabled:he,children:(0,i.jsxs)(v.A,{row:!0,value:V,onChange:e=>Q(e.target.value),children:[(0,i.jsx)(j.A,{value:"manual",control:(0,i.jsx)(y.A,{color:"primary"}),label:(0,i.jsxs)(c.A,{children:[(0,i.jsx)(d.A,{variant:"body2",children:"Need manual approval"}),(0,i.jsx)(d.A,{variant:"caption",color:"textSecondary",children:"Requires approval for requesting this API"})]})}),(0,i.jsx)(j.A,{value:"automatic",control:(0,i.jsx)(y.A,{color:"primary"}),label:(0,i.jsxs)(c.A,{children:[(0,i.jsx)(d.A,{variant:"body2",children:"Automatic"}),(0,i.jsx)(d.A,{variant:"caption",color:"textSecondary",children:"Keys are created without need to be approved"})]})})]})})]})]}),(0,i.jsxs)(b.A,{children:[(0,i.jsx)(f.A,{onClick:a,disabled:he,children:"Cancel"}),(0,i.jsx)(f.A,{onClick:async()=>{pe(""),me(!0);try{const e={spec:{displayName:D,description:B,version:_,publishStatus:G,approvalMode:V,tags:X,targetRef:Z,...ie||re?{contact:{...ie&&{email:ie},...re&&{team:re}}}:{},...se||de?{documentation:{...se&&{docsURL:se},...de&&{openAPISpecURL:de}}}:{}}},i=await z.fetch(`${U}/api/kuadrant/apiproducts/${P}/${$}`,{method:"PATCH",headers:{"Content-Type":"application/json"},body:JSON.stringify(e)});if(!i.ok){const e=await i.json();throw new Error(e.error||"failed to update apiproduct")}t(),a()}catch(e){pe(e instanceof Error?e.message:String(e))}finally{me(!1)}},color:"primary",variant:"contained",disabled:he||q||!D||!B||!!xe,startIcon:he?(0,i.jsx)(k.A,{size:16,color:"inherit"}):void 0,children:he?"Saving...":"Save"})]})]})}},34955:(e,a,t)=>{t.d(a,{Al:()=>l,DS:()=>h,EM:()=>d,FL:()=>r,J:()=>n,KV:()=>v,R_:()=>c,U3:()=>s,dp:()=>p,jH:()=>g,q0:()=>m,uL:()=>A,v_:()=>o,vs:()=>u,z4:()=>x});var i=t(83572);(0,i.i)({name:"kuadrant.planpolicy.create",attributes:{action:"create"}}),(0,i.i)({name:"kuadrant.planpolicy.read",attributes:{action:"read"}}),(0,i.i)({name:"kuadrant.planpolicy.update",attributes:{action:"update"}}),(0,i.i)({name:"kuadrant.planpolicy.delete",attributes:{action:"delete"}});const n=(0,i.i)({name:"kuadrant.planpolicy.list",attributes:{action:"read"}}),r=(0,i.i)({name:"kuadrant.apiproduct.create",attributes:{action:"create"}}),l=((0,i.i)({name:"kuadrant.apiproduct.read.own",attributes:{action:"read"}}),(0,i.i)({name:"kuadrant.apiproduct.read.all",attributes:{action:"read"}})),s=(0,i.i)({name:"kuadrant.apiproduct.update.own",attributes:{action:"update"}}),o=(0,i.i)({name:"kuadrant.apiproduct.update.all",attributes:{action:"update"}}),d=(0,i.i)({name:"kuadrant.apiproduct.delete.own",attributes:{action:"delete"}}),c=(0,i.i)({name:"kuadrant.apiproduct.delete.all",attributes:{action:"delete"}}),u=(0,i.i)({name:"kuadrant.apiproduct.list",attributes:{action:"read"}}),p=(0,i.i)({name:"kuadrant.apikey.create",attributes:{action:"create"},resourceType:"apiproduct"}),h=(0,i.i)({name:"kuadrant.apikey.read.own",attributes:{action:"read"}}),m=((0,i.i)({name:"kuadrant.apikey.read.all",attributes:{action:"read"}}),(0,i.i)({name:"kuadrant.apikey.update.own",attributes:{action:"update"}})),x=(0,i.i)({name:"kuadrant.apikey.update.all",attributes:{action:"update"}}),A=(0,i.i)({name:"kuadrant.apikey.delete.own",attributes:{action:"delete"}}),g=(0,i.i)({name:"kuadrant.apikey.delete.all",attributes:{action:"delete"}}),v=(0,i.i)({name:"kuadrant.apikey.approve",attributes:{action:"update"}})},46205:(e,a,t)=>{t.d(a,{W:()=>r,l:()=>n});var i=t(24217);function n(e,a){const t="resourceType"in e?{permission:e,resourceRef:a}:{permission:e},n=(0,i.J)(t);return{allowed:n.allowed,loading:n.loading,error:n.error}}function r(e,a,t,i){return!!i||!(!t||e!==a)}},63221:(e,a,t)=>{t.d(a,{K:()=>A});var i=t(31085),n=t(95478),r=t(76891),l=t(61477),s=t(10394),o=t(46805),d=t(59461),c=t(72501),u=t(16249),p=t(93453),h=t(64947),m=t(78467),x=t(77225);const A=({open:e,title:a,description:t,confirmText:A,severity:g="normal",deleting:v=!1,onConfirm:j,onCancel:y})=>{const[b,f]=(0,n.useState)("");(0,n.useEffect)(()=>{e||f("")},[e]);const k="high"===g&&A,P=!k||b===A;return(0,i.jsxs)(r.A,{open:e,onClose:v?void 0:y,maxWidth:"sm",fullWidth:!0,children:[(0,i.jsxs)(l.A,{children:["high"===g&&(0,i.jsxs)(s.A,{display:"flex",alignItems:"center",style:{gap:8},children:[(0,i.jsx)(x.A,{color:"error"}),(0,i.jsx)("span",{children:a})]}),"high"!==g&&a]}),(0,i.jsxs)(o.A,{children:[(0,i.jsx)(d.A,{style:{whiteSpace:"pre-line"},children:t}),k&&(0,i.jsxs)(s.A,{mt:2,children:[(0,i.jsxs)(c.A,{variant:"body2",color:"textSecondary",gutterBottom:!0,children:["Type ",(0,i.jsx)("strong",{children:A})," to confirm:"]}),(0,i.jsx)(u.A,{fullWidth:!0,variant:"outlined",size:"small",value:b,onChange:e=>f(e.target.value),disabled:v,autoFocus:!0,placeholder:A})]})]}),(0,i.jsxs)(p.A,{children:[(0,i.jsx)(h.A,{onClick:y,disabled:v,children:"Cancel"}),(0,i.jsx)(h.A,{onClick:()=>{P&&j()},color:"secondary",variant:"contained",disabled:v||!P,startIcon:v?(0,i.jsx)(m.A,{size:16,color:"inherit"}):void 0,children:v?"Deleting...":"Delete"})]})]})}},65867:(e,a,t)=>{t.d(a,{g:()=>d});var i=t(31085),n=(t(95478),t(10394)),r=t(72501),l=t(67720),s=t(54917),o=t(84441);const d=({selectedPolicy:e,alertSeverity:a="warning",alertMessage:t="No PlanPolicy found for this HTTPRoute. API keys and rate limiting may not be available.",includeTopMargin:d=!0})=>{const c=(0,s.A)();return(0,i.jsx)(n.A,{mt:d?1:0,p:2,bgcolor:c.palette.background.default,borderRadius:1,border:`1px solid ${c.palette.divider}`,children:e?(0,i.jsxs)(i.Fragment,{children:[(0,i.jsxs)(r.A,{variant:"subtitle2",gutterBottom:!0,style:{fontWeight:600},children:["Associated PlanPolicy: ",(0,i.jsx)("strong",{children:e.metadata.name})]}),e.plans&&e.plans.length>0?(0,i.jsxs)(i.Fragment,{children:[(0,i.jsx)(r.A,{variant:"caption",display:"block",gutterBottom:!0,color:"textSecondary",style:{marginTop:8},children:"Available Tiers:"}),(0,i.jsx)(n.A,{display:"flex",flexWrap:"wrap",mt:1,style:{gap:8},children:e.plans.map((e,a)=>{var t,n,r;const s=(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===(r=e.limits)||void 0===r?void 0:r.yearly)?`${e.limits.yearly}/year`:"No limit";return(0,i.jsx)(l.A,{label:`${e.tier}: ${s}`,size:"small",variant:"outlined",color:"primary"},a)})})]}):(0,i.jsx)(r.A,{variant:"caption",color:"textSecondary",children:"No plans defined in this PlanPolicy"})]}):(0,i.jsx)(o.A,{severity:a,children:t})})}}}]);
2
- //# sourceMappingURL=8799.83d049f3.chunk.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"static/8799.83d049f3.chunk.js","mappings":"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","sources":["webpack://internal.plugin-kuadrant/./src/utils/validation.ts","webpack://internal.plugin-kuadrant/./src/components/EditAPIProductDialog/EditAPIProductDialog.tsx","webpack://internal.plugin-kuadrant/./src/permissions.ts","webpack://internal.plugin-kuadrant/./src/utils/permissions.ts","webpack://internal.plugin-kuadrant/./src/components/ConfirmDeleteDialog/ConfirmDeleteDialog.tsx","webpack://internal.plugin-kuadrant/./src/components/PlanPolicyDetailsCard/PlanPolicyDetails.tsx"],"sourcesContent":["// Kubernetes name validation\nexport const validateKubernetesName = (value: string): string | null => {\n if (!value || !value.trim()) {\n return 'Name is required';\n }\n if (value.length > 253) {\n return 'Must be 253 characters or less';\n }\n\n const dns1123Regex = /^[a-z0-9]([-a-z0-9]*[a-z0-9])?$/;\n\n if (!dns1123Regex.test(value)) {\n return 'Must be lowercase alphanumeric with hyphens, start and end with alphanumeric';\n }\n\n return null;\n};\n\n// URL validation\nexport const validateURL = (value: string): string | null => {\n if (!value) {\n return null;\n }\n\n try {\n const url = new URL(value);\n if (!['http:', 'https:'].includes(url.protocol)) {\n return 'Must be a valid HTTP or HTTPS URL';\n }\n return null;\n } catch {\n return 'Must be a valid URL';\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 CircularProgress,\n makeStyles,\n FormControl,\n RadioGroup,\n FormControlLabel,\n Radio,\n Tooltip,\n IconButton,\n InputAdornment,\n} from '@material-ui/core';\nimport InfoOutlinedIcon from '@material-ui/icons/InfoOutlined';\nimport AddIcon from '@material-ui/icons/Add';\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';\nimport { validateURL } from '../../utils/validation';\n\nconst useStyles = makeStyles((theme) => ({\n asterisk: {\n color: '#f44336',\n },\n sectionHeader: {\n display: 'flex',\n alignItems: 'center',\n gap: theme.spacing(0.5),\n marginTop: theme.spacing(2),\n marginBottom: theme.spacing(1),\n },\n infoIcon: {\n fontSize: 18,\n color: theme.palette.text.secondary,\n },\n tagChip: {\n marginRight: theme.spacing(0.5),\n marginBottom: theme.spacing(0.5),\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 const [openAPISpecError, setOpenAPISpecError] = useState<string | null>(null);\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?.openAPISpecURL || '');\n setOpenAPISpecError(null);\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 useEffect(() => {\n if (open) {\n setOpenAPISpecError(null);\n }\n }, [open]);\n\n const handleOpenAPISpecChange = (value: string) => {\n setOpenAPISpec(value);\n setOpenAPISpecError(validateURL(value));\n };\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 && { openAPISpecURL: 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 <>\n {/* API product info section */}\n <Box className={classes.sectionHeader}>\n <Typography variant=\"subtitle1\"><strong>API product info</strong></Typography>\n </Box>\n <Grid container spacing={2}>\n <Grid item xs={6}>\n <TextField\n fullWidth\n label=\"API product name\"\n value={displayName}\n onChange={e => setDisplayName(e.target.value)}\n placeholder=\"My API\"\n helperText=\"Give a unique name for your API product\"\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=\"Resource 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=\"Version\"\n value={version}\n onChange={e => setVersion(e.target.value)}\n placeholder=\"v1\"\n helperText=\"Give a version to your API product\"\n margin=\"normal\"\n disabled={saving}\n />\n </Grid>\n <Grid item xs={6}>\n <TextField\n fullWidth\n label=\"Tag\"\n value={tagInput}\n onChange={e => setTagInput(e.target.value)}\n onKeyPress={e => {\n if (e.key === 'Enter') {\n e.preventDefault();\n handleAddTag();\n }\n }}\n placeholder=\"Add tag\"\n helperText=\"Add a tag to your API product\"\n margin=\"normal\"\n disabled={saving}\n InputProps={{\n endAdornment: tagInput ? (\n <InputAdornment position=\"end\">\n <IconButton size=\"small\" onClick={handleAddTag} disabled={saving}>\n <AddIcon fontSize=\"small\" />\n </IconButton>\n </InputAdornment>\n ) : undefined,\n }}\n />\n </Grid>\n {tags.length > 0 && (\n <Grid item xs={12}>\n <Box display=\"flex\" flexWrap=\"wrap\">\n {tags.map(tag => (\n <Chip\n key={tag}\n label={tag}\n onDelete={saving ? undefined : () => handleDeleteTag(tag)}\n size=\"small\"\n className={classes.tagChip}\n disabled={saving}\n />\n ))}\n </Box>\n </Grid>\n )}\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 </Grid>\n\n {/* Associated route section */}\n <Box className={classes.sectionHeader}>\n <Typography variant=\"subtitle1\"><strong>Associated route</strong></Typography>\n <Tooltip title=\"The HTTPRoute this API product is associated with\">\n <InfoOutlinedIcon className={classes.infoIcon} />\n </Tooltip>\n </Box>\n <Grid container spacing={2}>\n <Grid item xs={12}>\n <TextField\n fullWidth\n label=\"OpenAPI Spec URL\"\n value={openAPISpec}\n onChange={e => handleOpenAPISpecChange(e.target.value)}\n placeholder=\"https://api.example.com/openapi.json\"\n helperText={openAPISpecError || \"Enter the full path to your API spec file\"}\n error={!!openAPISpecError}\n margin=\"normal\"\n disabled={saving}\n />\n </Grid>\n <Grid item xs={12}>\n <TextField\n fullWidth\n label=\"Documentation URL\"\n value={docsURL}\n onChange={e => setDocsURL(e.target.value)}\n placeholder=\"https://docs.example.com/api\"\n helperText=\"Link to external documentation for this API\"\n margin=\"normal\"\n disabled={saving}\n />\n </Grid>\n {targetRef && (\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>\n\n {/* HTTPRoute policies section */}\n {targetRef && (\n <>\n <Box className={classes.sectionHeader}>\n <Typography variant=\"subtitle1\"><strong>HTTPRoute policies</strong></Typography>\n <Tooltip title=\"Shows the associated policies and rate limit tiers for the HTTPRoute\">\n <InfoOutlinedIcon className={classes.infoIcon} />\n </Tooltip>\n </Box>\n <PlanPolicyDetails\n selectedPolicy={selectedPolicy}\n alertSeverity=\"info\"\n alertMessage=\"No PlanPolicy found for this HTTPRoute.\"\n includeTopMargin={false}\n />\n </>\n )}\n\n {/* API Key approval section */}\n <Box className={classes.sectionHeader}>\n <Typography variant=\"subtitle1\"><strong>API Key approval</strong></Typography>\n <Tooltip title=\"Choose how API key requests are handled for this product\">\n <InfoOutlinedIcon className={classes.infoIcon} />\n </Tooltip>\n </Box>\n <FormControl component=\"fieldset\" disabled={saving}>\n <RadioGroup\n row\n value={approvalMode}\n onChange={e => setApprovalMode(e.target.value as 'automatic' | 'manual')}\n >\n <FormControlLabel\n value=\"manual\"\n control={<Radio color=\"primary\" />}\n label={\n <Box>\n <Typography variant=\"body2\">Need manual approval</Typography>\n <Typography variant=\"caption\" color=\"textSecondary\">\n Requires approval for requesting this API\n </Typography>\n </Box>\n }\n />\n <FormControlLabel\n value=\"automatic\"\n control={<Radio color=\"primary\" />}\n label={\n <Box>\n <Typography variant=\"body2\">Automatic</Typography>\n <Typography variant=\"caption\" color=\"textSecondary\">\n Keys are created without need to be approved\n </Typography>\n </Box>\n }\n />\n </RadioGroup>\n </FormControl>\n </>\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 || !!openAPISpecError}\n startIcon={saving ? <CircularProgress size={16} color=\"inherit\" /> : undefined}\n >\n {saving ? 'Saving...' : 'Save'}\n </Button>\n </DialogActions>\n </Dialog>\n );\n};\n","import { createPermission } from '@backstage/plugin-permission-common';\n\n/**\n * permission definitions for the kuadrant plugin\n *\n * these permissions control access to kuadrant resources and operations.\n * they must match the permissions defined in the backend plugin.\n *\n * permission types:\n * - BasicPermission: standard permission that applies globally\n * - ResourcePermission: permission scoped to specific resource types (e.g., apiproduct)\n *\n * permission patterns:\n * - `.create` - create new resources\n * - `.read` - read resource details\n * - `.read.own` - read only resources owned by the user\n * - `.read.all` - read all resources regardless of ownership\n * - `.update` - modify existing resources\n * - `.delete` - delete resources\n * - `.delete.own` - delete only resources owned by the user\n * - `.delete.all` - delete any resource regardless of ownership\n * - `.list` - list/view collections of resources\n */\n\n// planpolicy permissions\nexport const kuadrantPlanPolicyCreatePermission = createPermission({\n name: 'kuadrant.planpolicy.create',\n attributes: { action: 'create' },\n});\n\nexport const kuadrantPlanPolicyReadPermission = createPermission({\n name: 'kuadrant.planpolicy.read',\n attributes: { action: 'read' },\n});\n\nexport const kuadrantPlanPolicyUpdatePermission = createPermission({\n name: 'kuadrant.planpolicy.update',\n attributes: { action: 'update' },\n});\n\nexport const kuadrantPlanPolicyDeletePermission = createPermission({\n name: 'kuadrant.planpolicy.delete',\n attributes: { action: 'delete' },\n});\n\nexport const kuadrantPlanPolicyListPermission = createPermission({\n name: 'kuadrant.planpolicy.list',\n attributes: { action: 'read' },\n});\n\n// apiproduct permissions\n\n/**\n * permission to create new API products\n * granted to api owners and admins\n */\nexport const kuadrantApiProductCreatePermission = createPermission({\n name: 'kuadrant.apiproduct.create',\n attributes: { action: 'create' },\n});\n\n/**\n * permission to read API products owned by the current user\n * for api owners to view their own products\n */\nexport const kuadrantApiProductReadOwnPermission = createPermission({\n name: 'kuadrant.apiproduct.read.own',\n attributes: { action: 'read' },\n});\n\n/**\n * permission to read all API products regardless of ownership\n * for platform engineers/admins who need to view all products\n */\nexport const kuadrantApiProductReadAllPermission = createPermission({\n name: 'kuadrant.apiproduct.read.all',\n attributes: { action: 'read' },\n});\n\n/**\n * permission to update API products owned by the current user\n * for api owners to modify their own products\n */\nexport const kuadrantApiProductUpdateOwnPermission = createPermission({\n name: 'kuadrant.apiproduct.update.own',\n attributes: { action: 'update' },\n});\n\n/**\n * permission to update any API product regardless of ownership\n * for platform engineers/admins\n */\nexport const kuadrantApiProductUpdateAllPermission = createPermission({\n name: 'kuadrant.apiproduct.update.all',\n attributes: { action: 'update' },\n});\n\n/**\n * permission to delete API products owned by the current user\n * for api owners to remove their own products\n */\nexport const kuadrantApiProductDeleteOwnPermission = createPermission({\n name: 'kuadrant.apiproduct.delete.own',\n attributes: { action: 'delete' },\n});\n\n/**\n * permission to delete any API product regardless of ownership\n * for platform engineers/admins\n */\nexport const kuadrantApiProductDeleteAllPermission = createPermission({\n name: 'kuadrant.apiproduct.delete.all',\n attributes: { action: 'delete' },\n});\n\n/**\n * permission to list API products\n * backend filters results based on .own vs .all read permissions\n */\nexport const kuadrantApiProductListPermission = createPermission({\n name: 'kuadrant.apiproduct.list',\n attributes: { action: 'read' },\n});\n\n// apikey permissions\n\n/**\n * permission to create API keys (request API access)\n *\n * this is a ResourcePermission scoped to 'apiproduct', allowing\n * fine-grained control over which API products users can request access to.\n *\n * use in frontend: useKuadrantPermission(kuadrantApiKeyCreatePermission)\n * use in backend with resource: { permission, resourceRef: 'apiproduct:namespace/name' }\n */\nexport const kuadrantApiKeyCreatePermission = createPermission({\n name: 'kuadrant.apikey.create',\n attributes: { action: 'create' },\n resourceType: 'apiproduct',\n});\n\n/**\n * permission to read API keys owned by the current user\n * allows users to view their own API keys and request history\n */\nexport const kuadrantApiKeyReadOwnPermission = createPermission({\n name: 'kuadrant.apikey.read.own',\n attributes: { action: 'read' },\n});\n\n/**\n * permission to read all API keys regardless of ownership\n * for platform engineers/admins who need to view the approval queue and audit keys\n */\nexport const kuadrantApiKeyReadAllPermission = createPermission({\n name: 'kuadrant.apikey.read.all',\n attributes: { action: 'read' },\n});\n\n/**\n * permission to update API keys owned by the current user\n * allows users to edit their own pending requests (change plan tier, use case)\n */\nexport const kuadrantApiKeyUpdateOwnPermission = createPermission({\n name: 'kuadrant.apikey.update.own',\n attributes: { action: 'update' },\n});\n\n/**\n * permission to update any API key regardless of ownership\n * typically granted to API owners and platform engineers for approving/rejecting requests\n */\nexport const kuadrantApiKeyUpdateAllPermission = createPermission({\n name: 'kuadrant.apikey.update.all',\n attributes: { action: 'update' },\n});\n\n/**\n * permission to delete API keys owned by the current user\n * allows users to cancel their own requests or revoke their own access\n */\nexport const kuadrantApiKeyDeleteOwnPermission = createPermission({\n name: 'kuadrant.apikey.delete.own',\n attributes: { action: 'delete' },\n});\n\n/**\n * permission to delete any API key regardless of ownership\n * for platform engineers/admins who need to revoke access\n */\nexport const kuadrantApiKeyDeleteAllPermission = createPermission({\n name: 'kuadrant.apikey.delete.all',\n attributes: { action: 'delete' },\n});\n\n/**\n * permission to approve/reject API key requests\n * grants access to the approval queue - for API owners and admins only\n * separate from update.own which consumers use to edit their pending requests\n */\nexport const kuadrantApiKeyApprovePermission = createPermission({\n name: 'kuadrant.apikey.approve',\n attributes: { action: 'update' },\n});\n\nexport const kuadrantPermissions = [\n kuadrantPlanPolicyCreatePermission,\n kuadrantPlanPolicyReadPermission,\n kuadrantPlanPolicyUpdatePermission,\n kuadrantPlanPolicyDeletePermission,\n kuadrantPlanPolicyListPermission,\n kuadrantApiProductCreatePermission,\n kuadrantApiProductReadOwnPermission,\n kuadrantApiProductReadAllPermission,\n kuadrantApiProductUpdateOwnPermission,\n kuadrantApiProductUpdateAllPermission,\n kuadrantApiProductDeleteOwnPermission,\n kuadrantApiProductDeleteAllPermission,\n kuadrantApiProductListPermission,\n kuadrantApiKeyCreatePermission,\n kuadrantApiKeyReadOwnPermission,\n kuadrantApiKeyReadAllPermission,\n kuadrantApiKeyUpdateOwnPermission,\n kuadrantApiKeyUpdateAllPermission,\n kuadrantApiKeyDeleteOwnPermission,\n kuadrantApiKeyDeleteAllPermission,\n kuadrantApiKeyApprovePermission,\n];\n","import { usePermission } from '@backstage/plugin-permission-react';\nimport { Permission, ResourcePermission } from '@backstage/plugin-permission-common';\n\n/**\n * result of a permission check including error state\n */\nexport interface PermissionCheckResult {\n allowed: boolean;\n loading: boolean;\n error?: Error;\n}\n\n/**\n * custom hook for checking kuadrant permissions that handles both\n * BasicPermission and ResourcePermission types without type bypasses\n *\n * @param permission - the permission to check\n * @param resourceRef - optional resource reference for ResourcePermissions\n * @returns permission check result with error handling\n *\n * @example\n * // basic permission\n * const { allowed, loading, error } = useKuadrantPermission(\n * kuadrantApiProductListPermission\n * );\n *\n * @example\n * // resource permission\n * const { allowed, loading, error } = useKuadrantPermission(\n * kuadrantApiKeyCreatePermission,\n * 'apiproduct:namespace/name'\n * );\n */\nexport function useKuadrantPermission(\n permission: Permission,\n resourceRef?: string,\n): PermissionCheckResult {\n // construct the permission request based on whether it's a ResourcePermission\n const permissionRequest = 'resourceType' in permission\n ? { permission: permission as ResourcePermission, resourceRef }\n : { permission };\n\n const result = usePermission(permissionRequest as any);\n\n return {\n allowed: result.allowed,\n loading: result.loading,\n error: result.error,\n };\n}\n\n/**\n * helper to determine if a user can delete a specific API key or request\n *\n * @param ownerId - the user id who owns the key/request\n * @param currentUserId - the current user's id\n * @param canDeleteOwn - whether user has permission to delete their own keys\n * @param canDeleteAll - whether user has permission to delete all keys\n * @returns true if user can delete this specific key/request\n */\nexport function canDeleteResource(\n ownerId: string,\n currentUserId: string,\n canDeleteOwn: boolean,\n canDeleteAll: boolean,\n): boolean {\n if (canDeleteAll) return true;\n if (canDeleteOwn && ownerId === currentUserId) return true;\n return false;\n}\n","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 from 'react';\nimport { Box, Typography, Chip, useTheme } 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 const theme = useTheme();\n return (\n <Box\n mt={includeTopMargin ? 1 : 0}\n p={2}\n bgcolor={theme.palette.background.default}\n borderRadius={1}\n border={`1px solid ${theme.palette.divider}`}\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 Tiers:\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 </>\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"],"names":["validateKubernetesName","value","trim","length","test","validateURL","url","URL","includes","protocol","useStyles","makeStyles","theme","asterisk","color","sectionHeader","display","alignItems","gap","spacing","marginTop","marginBottom","infoIcon","fontSize","palette","text","secondary","tagChip","marginRight","EditAPIProductDialog","open","onClose","onSuccess","namespace","name","classes","config","useApi","configApiRef","fetchApi","fetchApiRef","backendUrl","getString","loading","setLoading","useState","displayName","setDisplayName","description","setDescription","version","setVersion","publishStatus","setPublishStatus","approvalMode","setApprovalMode","tags","setTags","targetRef","setTargetRef","tagInput","setTagInput","contactEmail","setContactEmail","contactTeam","setContactTeam","docsURL","setDocsURL","openAPISpec","setOpenAPISpec","error","setError","saving","setSaving","openAPISpecError","setOpenAPISpecError","useEffect","fetch","then","async","res","ok","errorData","json","Error","status","data","spec","contact","email","team","documentation","openAPISpecURL","catch","err","message","planPolicies","planPoliciesError","useAsync","response","selectedPolicy","React","items","find","pp","ref","kind","handleAddTag","Dialog","maxWidth","fullWidth","DialogTitle","DialogContent","Alert","severity","style","strong","Typography","variant","Progress","Box","className","Grid","container","item","xs","TextField","label","onChange","e","target","placeholder","helperText","margin","required","disabled","InputLabelProps","onKeyPress","key","preventDefault","InputProps","endAdornment","InputAdornment","position","IconButton","size","onClick","AddIcon","undefined","flexWrap","map","tag","Chip","onDelete","handleDeleteTag","tagToDelete","filter","multiline","rows","Tooltip","title","InfoOutlinedIcon","handleOpenAPISpecChange","PlanPolicyDetails","alertSeverity","alertMessage","includeTopMargin","FormControl","component","RadioGroup","row","FormControlLabel","control","Radio","DialogActions","Button","patch","method","headers","body","JSON","stringify","String","startIcon","CircularProgress","createPermission","attributes","action","kuadrantPlanPolicyListPermission","kuadrantApiProductCreatePermission","kuadrantApiProductReadAllPermission","kuadrantApiProductUpdateOwnPermission","kuadrantApiProductUpdateAllPermission","kuadrantApiProductDeleteOwnPermission","kuadrantApiProductDeleteAllPermission","kuadrantApiProductListPermission","kuadrantApiKeyCreatePermission","resourceType","kuadrantApiKeyReadOwnPermission","kuadrantApiKeyUpdateOwnPermission","kuadrantApiKeyUpdateAllPermission","kuadrantApiKeyDeleteOwnPermission","kuadrantApiKeyDeleteAllPermission","kuadrantApiKeyApprovePermission","useKuadrantPermission","permission","resourceRef","permissionRequest","result","usePermission","allowed","canDeleteResource","ownerId","currentUserId","canDeleteOwn","canDeleteAll","ConfirmDeleteDialog","confirmText","deleting","onConfirm","onCancel","inputValue","setInputValue","requiresTextConfirmation","canConfirm","WarningIcon","span","DialogContentText","whiteSpace","mt","gutterBottom","autoFocus","useTheme","p","bgcolor","background","default","borderRadius","border","divider","fontWeight","metadata","plans","plan","idx","limitText","limits","daily","monthly","yearly","tier"],"sourceRoot":""}