@kuadrant/kuadrant-backstage-plugin-frontend 0.0.2-dev-9be93a3 → 0.0.2-dev-8189c1c

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (73) hide show
  1. package/dist/api.esm.js +16 -0
  2. package/dist/api.esm.js.map +1 -1
  3. package/dist/components/ApiAccessCard/ApiAccessCard.esm.js.map +1 -1
  4. package/dist/components/ApiKeyManagementTab/ApiKeyManagementTab.esm.js.map +1 -1
  5. package/dist/components/ApiProductDetails/ApiProductDetails.esm.js.map +1 -1
  6. package/dist/components/ApiProductPolicies/ApiProductPolicies.esm.js +113 -0
  7. package/dist/components/ApiProductPolicies/ApiProductPolicies.esm.js.map +1 -0
  8. package/dist/components/CreateAPIProductDialog/CreateAPIProductDialog.esm.js +57 -13
  9. package/dist/components/CreateAPIProductDialog/CreateAPIProductDialog.esm.js.map +1 -1
  10. package/dist/components/EditAPIProductDialog/EditAPIProductDialog.esm.js +49 -8
  11. package/dist/components/EditAPIProductDialog/EditAPIProductDialog.esm.js.map +1 -1
  12. package/dist/components/KuadrantPage/ApiProductsPage.esm.js +1 -1
  13. package/dist/components/KuadrantPage/ApiProductsPage.esm.js.map +1 -1
  14. package/dist/components/RequestAccessDialog/RequestAccessDialog.esm.js.map +1 -1
  15. package/dist/index.d.ts +59 -8
  16. package/dist/index.esm.js +1 -1
  17. package/dist/permissions.esm.js +12 -2
  18. package/dist/permissions.esm.js.map +1 -1
  19. package/dist/utils/policies.esm.js +11 -0
  20. package/dist/utils/policies.esm.js.map +1 -0
  21. package/dist-scalprum/{internal.plugin-kuadrant.8bf02d5482a5a3ed6337.js → internal.plugin-kuadrant.1ed3b3b414caa08ac5be.js} +2 -2
  22. package/dist-scalprum/{internal.plugin-kuadrant.8bf02d5482a5a3ed6337.js.map → internal.plugin-kuadrant.1ed3b3b414caa08ac5be.js.map} +1 -1
  23. package/dist-scalprum/plugin-manifest.json +3 -3
  24. package/dist-scalprum/static/{2967.28e7c7f8.chunk.js → 2967.17aec2fb.chunk.js} +1 -1
  25. package/dist-scalprum/static/2967.17aec2fb.chunk.js.map +1 -0
  26. package/dist-scalprum/static/{3947.ff1c25cf.chunk.js → 3947.ad129ba4.chunk.js} +1 -1
  27. package/dist-scalprum/static/{3947.ff1c25cf.chunk.js.map → 3947.ad129ba4.chunk.js.map} +1 -1
  28. package/dist-scalprum/static/{3976.cf3ec7be.chunk.js → 3976.c8138b52.chunk.js} +1 -1
  29. package/dist-scalprum/static/{3976.cf3ec7be.chunk.js.map → 3976.c8138b52.chunk.js.map} +1 -1
  30. package/dist-scalprum/static/4447.d924bc59.chunk.js +2 -0
  31. package/dist-scalprum/static/4447.d924bc59.chunk.js.map +1 -0
  32. package/dist-scalprum/static/{5203.e92a3353.chunk.js → 5203.11ef2211.chunk.js} +1 -1
  33. package/dist-scalprum/static/5203.11ef2211.chunk.js.map +1 -0
  34. package/dist-scalprum/static/{6371.6ee1f11d.chunk.js → 6371.d849c508.chunk.js} +1 -1
  35. package/dist-scalprum/static/6371.d849c508.chunk.js.map +1 -0
  36. package/dist-scalprum/static/6387.79be6155.chunk.js +2 -0
  37. package/dist-scalprum/static/6387.79be6155.chunk.js.map +1 -0
  38. package/dist-scalprum/static/{6800.a7645f4c.chunk.js → 6800.6faeb7c6.chunk.js} +1 -1
  39. package/dist-scalprum/static/{6800.a7645f4c.chunk.js.map → 6800.6faeb7c6.chunk.js.map} +1 -1
  40. package/dist-scalprum/static/69.b6afd1fe.chunk.js +2 -0
  41. package/dist-scalprum/static/69.b6afd1fe.chunk.js.map +1 -0
  42. package/dist-scalprum/static/7005.98c5e400.chunk.js +2 -0
  43. package/dist-scalprum/static/7005.98c5e400.chunk.js.map +1 -0
  44. package/dist-scalprum/static/{7270.b0e185f1.chunk.js → 7270.4a71807b.chunk.js} +1 -1
  45. package/dist-scalprum/static/{7270.b0e185f1.chunk.js.map → 7270.4a71807b.chunk.js.map} +1 -1
  46. package/dist-scalprum/static/{7791.5c1eea8a.chunk.js → 7791.0a1d55bc.chunk.js} +1 -1
  47. package/dist-scalprum/static/{7791.5c1eea8a.chunk.js.map → 7791.0a1d55bc.chunk.js.map} +1 -1
  48. package/dist-scalprum/static/8789.2f437443.chunk.js +2 -0
  49. package/dist-scalprum/static/8789.2f437443.chunk.js.map +1 -0
  50. package/dist-scalprum/static/{9051.db875198.chunk.js → 9051.d45ac154.chunk.js} +2 -2
  51. package/dist-scalprum/static/9051.d45ac154.chunk.js.map +1 -0
  52. package/dist-scalprum/static/{9838.5df9b1de.chunk.js → 9838.966ce0a0.chunk.js} +1 -1
  53. package/dist-scalprum/static/9838.966ce0a0.chunk.js.map +1 -0
  54. package/dist-scalprum/static/{exposed-PluginRoot.8a8dd91f.chunk.js → exposed-PluginRoot.3b24e5af.chunk.js} +2 -2
  55. package/dist-scalprum/static/{exposed-PluginRoot.8a8dd91f.chunk.js.map → exposed-PluginRoot.3b24e5af.chunk.js.map} +1 -1
  56. package/package.json +1 -1
  57. package/dist/components/PlanPolicyDetailsCard/PlanPolicyDetails.esm.js +0 -48
  58. package/dist/components/PlanPolicyDetailsCard/PlanPolicyDetails.esm.js.map +0 -1
  59. package/dist-scalprum/static/2967.28e7c7f8.chunk.js.map +0 -1
  60. package/dist-scalprum/static/4447.222bb58d.chunk.js +0 -2
  61. package/dist-scalprum/static/4447.222bb58d.chunk.js.map +0 -1
  62. package/dist-scalprum/static/5203.e92a3353.chunk.js.map +0 -1
  63. package/dist-scalprum/static/6371.6ee1f11d.chunk.js.map +0 -1
  64. package/dist-scalprum/static/69.1decc5e8.chunk.js +0 -2
  65. package/dist-scalprum/static/69.1decc5e8.chunk.js.map +0 -1
  66. package/dist-scalprum/static/7005.a9f73153.chunk.js +0 -2
  67. package/dist-scalprum/static/7005.a9f73153.chunk.js.map +0 -1
  68. package/dist-scalprum/static/8789.84963cbc.chunk.js +0 -2
  69. package/dist-scalprum/static/8789.84963cbc.chunk.js.map +0 -1
  70. package/dist-scalprum/static/8804.63919453.chunk.js +0 -2
  71. package/dist-scalprum/static/8804.63919453.chunk.js.map +0 -1
  72. package/dist-scalprum/static/9051.db875198.chunk.js.map +0 -1
  73. package/dist-scalprum/static/9838.5df9b1de.chunk.js.map +0 -1
@@ -1 +1 @@
1
- {"version":3,"file":"static/7270.b0e185f1.chunk.js","mappings":"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","sources":["webpack://internal.plugin-kuadrant/../../node_modules/react-use/lib/useAsyncFn.js","webpack://internal.plugin-kuadrant/../../node_modules/react-use/lib/useMountedState.js","webpack://internal.plugin-kuadrant/./src/components/FilterPanel/FilterPanel.tsx","webpack://internal.plugin-kuadrant/./src/components/EditAPIKeyDialog/EditAPIKeyDialog.tsx","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/icons/ExpandLess.js","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/icons/BrokenImage.js","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/icons/Visibility.js","webpack://internal.plugin-kuadrant/./src/components/SimpleRequestAccessDialog/SimpleRequestAccessDialog.tsx","webpack://internal.plugin-kuadrant/./src/components/MyApiKeysTable/MyApiKeysTable.tsx","webpack://internal.plugin-kuadrant/./src/components/MyApiKeysPage/MyApiKeysPage.tsx","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/icons/Delete.js","webpack://internal.plugin-kuadrant/./node_modules/@backstage/core-components/dist/hooks/useSupportConfig.esm.js","webpack://internal.plugin-kuadrant/./node_modules/@backstage/core-components/dist/icons/icons.esm.js","webpack://internal.plugin-kuadrant/./node_modules/@backstage/core-components/dist/components/SupportButton/SupportButton.esm.js","webpack://internal.plugin-kuadrant/./src/utils/styles.ts","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/icons/VisibilityOff.js","webpack://internal.plugin-kuadrant/./src/components/ConfirmDeleteDialog/ConfirmDeleteDialog.tsx","webpack://internal.plugin-kuadrant/./src/api.ts","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/icons/Warning.js","webpack://internal.plugin-kuadrant/../../node_modules/react-use/lib/useAsync.js"],"sourcesContent":["\"use strict\";\nObject.defineProperty(exports, \"__esModule\", { value: true });\nvar tslib_1 = require(\"tslib\");\nvar react_1 = require(\"react\");\nvar useMountedState_1 = tslib_1.__importDefault(require(\"./useMountedState\"));\nfunction useAsyncFn(fn, deps, initialState) {\n if (deps === void 0) { deps = []; }\n if (initialState === void 0) { initialState = { loading: false }; }\n var lastCallId = react_1.useRef(0);\n var isMounted = useMountedState_1.default();\n var _a = react_1.useState(initialState), state = _a[0], set = _a[1];\n var callback = react_1.useCallback(function () {\n var args = [];\n for (var _i = 0; _i < arguments.length; _i++) {\n args[_i] = arguments[_i];\n }\n var callId = ++lastCallId.current;\n if (!state.loading) {\n set(function (prevState) { return (tslib_1.__assign(tslib_1.__assign({}, prevState), { loading: true })); });\n }\n return fn.apply(void 0, args).then(function (value) {\n isMounted() && callId === lastCallId.current && set({ value: value, loading: false });\n return value;\n }, function (error) {\n isMounted() && callId === lastCallId.current && set({ error: error, loading: false });\n return error;\n });\n }, deps);\n return [state, callback];\n}\nexports.default = useAsyncFn;\n","\"use strict\";\nObject.defineProperty(exports, \"__esModule\", { value: true });\nvar react_1 = require(\"react\");\nfunction useMountedState() {\n var mountedRef = react_1.useRef(false);\n var get = react_1.useCallback(function () { return mountedRef.current; }, []);\n react_1.useEffect(function () {\n mountedRef.current = true;\n return function () {\n mountedRef.current = false;\n };\n }, []);\n return get;\n}\nexports.default = useMountedState;\n","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, { useState, useEffect } from \"react\";\nimport {\n Dialog,\n DialogTitle,\n DialogContent,\n DialogActions,\n Button,\n TextField,\n Box,\n Typography,\n FormControl,\n InputLabel,\n Select,\n MenuItem,\n CircularProgress,\n} from \"@material-ui/core\";\nimport { useApi } from \"@backstage/core-plugin-api\";\nimport { kuadrantApiRef } from '../../api';\nimport { APIKey } from \"../../types/api-management\";\n\ninterface EditAPIKeyDialogProps {\n open: boolean;\n onClose: () => void;\n onSuccess: () => void;\n request: APIKey;\n availablePlans: Array<{\n tier: string;\n description?: string;\n limits?: any;\n }>;\n}\n\nexport const EditAPIKeyDialog = ({\n open,\n onClose,\n onSuccess,\n request,\n availablePlans,\n}: EditAPIKeyDialogProps) => {\n const kuadrantApi = useApi(kuadrantApiRef);\n\n const [planTier, setPlanTier] = useState(\"\");\n const [useCase, setUseCase] = useState(\"\");\n const [saving, setSaving] = useState(false);\n const [error, setError] = useState(\"\");\n\n useEffect(() => {\n if (open && request) {\n setPlanTier(request.spec.planTier || \"\");\n setUseCase(request.spec.useCase || \"\");\n setError(\"\");\n }\n }, [open, request]);\n\n const handleSave = async () => {\n if (!planTier) {\n setError(\"Please select a tier\");\n return;\n }\n\n setError(\"\");\n setSaving(true);\n\n try {\n const patch = {\n spec: {\n planTier,\n useCase: useCase.trim(),\n },\n };\n\n await kuadrantApi.updateRequest(\n request.metadata.name,\n request.metadata.namespace,\n // @ts-ignore Applying a partial obj\n patch,\n );\n\n onSuccess();\n onClose();\n } catch (err) {\n console.error(\"Error updating API key request:\", err);\n setError(err instanceof Error ? err.message : \"Unknown error occurred\");\n } finally {\n setSaving(false);\n }\n };\n\n const handleClose = () => {\n if (!saving) {\n setError(\"\");\n onClose();\n }\n };\n\n return (\n <Dialog open={open} onClose={handleClose} maxWidth=\"sm\" fullWidth>\n <DialogTitle>Edit API Key</DialogTitle>\n <DialogContent>\n {error && (\n <Box\n mb={2}\n p={2}\n bgcolor=\"error.main\"\n color=\"error.contrastText\"\n borderRadius={1}\n >\n <Typography variant=\"body2\">{error}</Typography>\n </Box>\n )}\n\n <FormControl fullWidth margin=\"normal\">\n <InputLabel>Tier</InputLabel>\n <Select\n value={planTier}\n onChange={(e) => setPlanTier(e.target.value as string)}\n disabled={saving}\n >\n {availablePlans.map((plan) => {\n const limitDesc = Object.entries(plan.limits || {})\n .map(([key, val]) => `${val} per ${key}`)\n .join(\", \");\n return (\n <MenuItem key={plan.tier} value={plan.tier}>\n {plan.tier} {limitDesc ? `(${limitDesc})` : \"\"}\n </MenuItem>\n );\n })}\n </Select>\n </FormControl>\n\n <TextField\n label=\"Use Case\"\n placeholder=\"Describe how you plan to use this API\"\n multiline\n rows={3}\n fullWidth\n margin=\"normal\"\n value={useCase}\n onChange={(e) => setUseCase(e.target.value)}\n disabled={saving}\n helperText=\"Explain your intended use of this API for admin review\"\n />\n </DialogContent>\n <DialogActions>\n <Button onClick={handleClose} disabled={saving}>\n Cancel\n </Button>\n <Button\n onClick={handleSave}\n color=\"primary\"\n variant=\"contained\"\n disabled={!planTier || saving}\n startIcon={\n saving ? <CircularProgress size={16} color=\"inherit\" /> : undefined\n }\n >\n {saving ? \"Saving...\" : \"Save Changes\"}\n </Button>\n </DialogActions>\n </Dialog>\n );\n};\n","\"use strict\";\n\nvar _interopRequireDefault = require(\"@babel/runtime/helpers/interopRequireDefault\");\n\nvar _interopRequireWildcard = require(\"@babel/runtime/helpers/interopRequireWildcard\");\n\nObject.defineProperty(exports, \"__esModule\", {\n value: true\n});\nexports.default = void 0;\n\nvar React = _interopRequireWildcard(require(\"react\"));\n\nvar _createSvgIcon = _interopRequireDefault(require(\"./utils/createSvgIcon\"));\n\nvar _default = (0, _createSvgIcon.default)( /*#__PURE__*/React.createElement(\"path\", {\n d: \"M12 8l-6 6 1.41 1.41L12 10.83l4.59 4.58L18 14z\"\n}), 'ExpandLess');\n\nexports.default = _default;","\"use strict\";\n\nvar _interopRequireDefault = require(\"@babel/runtime/helpers/interopRequireDefault\");\n\nvar _interopRequireWildcard = require(\"@babel/runtime/helpers/interopRequireWildcard\");\n\nObject.defineProperty(exports, \"__esModule\", {\n value: true\n});\nexports.default = void 0;\n\nvar React = _interopRequireWildcard(require(\"react\"));\n\nvar _createSvgIcon = _interopRequireDefault(require(\"./utils/createSvgIcon\"));\n\nvar _default = (0, _createSvgIcon.default)( /*#__PURE__*/React.createElement(\"path\", {\n d: \"M21 5v6.59l-3-3.01-4 4.01-4-4-4 4-3-3.01V5c0-1.1.9-2 2-2h14c1.1 0 2 .9 2 2zm-3 6.42l3 3.01V19c0 1.1-.9 2-2 2H5c-1.1 0-2-.9-2-2v-6.58l3 2.99 4-4 4 4 4-3.99z\"\n}), 'BrokenImage');\n\nexports.default = _default;","\"use strict\";\n\nvar _interopRequireDefault = require(\"@babel/runtime/helpers/interopRequireDefault\");\n\nvar _interopRequireWildcard = require(\"@babel/runtime/helpers/interopRequireWildcard\");\n\nObject.defineProperty(exports, \"__esModule\", {\n value: true\n});\nexports.default = void 0;\n\nvar React = _interopRequireWildcard(require(\"react\"));\n\nvar _createSvgIcon = _interopRequireDefault(require(\"./utils/createSvgIcon\"));\n\nvar _default = (0, _createSvgIcon.default)( /*#__PURE__*/React.createElement(\"path\", {\n d: \"M12 4.5C7 4.5 2.73 7.61 1 12c1.73 4.39 6 7.5 11 7.5s9.27-3.11 11-7.5c-1.73-4.39-6-7.5-11-7.5zM12 17c-2.76 0-5-2.24-5-5s2.24-5 5-5 5 2.24 5 5-2.24 5-5 5zm0-8c-1.66 0-3 1.34-3 3s1.34 3 3 3 3-1.34 3-3-1.34-3-3-3z\"\n}), 'Visibility');\n\nexports.default = _default;","import React, { useState } from 'react';\nimport {\n Dialog,\n DialogTitle,\n DialogContent,\n DialogActions,\n Button,\n TextField,\n Select,\n MenuItem,\n FormControl,\n InputLabel,\n Typography,\n CircularProgress,\n FormHelperText,\n} from '@material-ui/core';\nimport {\n useApi,\n configApiRef,\n fetchApiRef,\n alertApiRef,\n identityApiRef,\n} from '@backstage/core-plugin-api';\nimport useAsync from 'react-use/lib/useAsync';\n\nexport interface SimpleRequestAccessDialogProps {\n open: boolean;\n onClose: () => void;\n onSuccess: () => void;\n}\n\ntype ApiProduct = {\n metadata: {\n name: string;\n namespace: string;\n };\n spec?: {\n publishStatus?: string;\n };\n status?: {\n discoveredPlans?: Array<{\n tier: string;\n limits?: Record<string, number>;\n }>;\n };\n};\n\nexport const SimpleRequestAccessDialog = ({\n open,\n onClose,\n onSuccess,\n}: SimpleRequestAccessDialogProps) => {\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\n const [selectedApi, setSelectedApi] = useState('');\n const [selectedTier, setSelectedTier] = useState('');\n const [useCase, setUseCase] = useState('');\n const [creating, setCreating] = useState(false);\n\n // Fetch all published API products\n const {\n value: apiProducts,\n loading: loadingProducts,\n } = useAsync(async () => {\n if (!open) return [];\n const response = await fetchApi.fetch(\n `${backendUrl}/api/kuadrant/apiproducts`,\n );\n if (!response.ok) {\n throw new Error('Failed to fetch API products');\n }\n const data = await response.json();\n // Filter to only show Published products\n return (data.items || []).filter(\n (p: ApiProduct) => p.spec?.publishStatus === 'Published',\n );\n }, [backendUrl, fetchApi, open]);\n\n // Get user email\n const { value: userEmail } = useAsync(async () => {\n const identity = await identityApi.getBackstageIdentity();\n\n // Extract username from userEntityRef (e.g., \"user:default/john\" -> \"john\")\n const username = identity.userEntityRef.split('/')[1] || 'unknown';\n\n // Construct email from username\n return `${username}@example.com`;\n }, [identityApi]);\n\n // Get available tiers for selected API\n const selectedProduct = apiProducts?.find(\n (p: ApiProduct) =>\n `${p.metadata.namespace}/${p.metadata.name}` === selectedApi,\n );\n const availableTiers = selectedProduct?.status?.discoveredPlans || [];\n\n const handleClose = () => {\n setSelectedApi('');\n setSelectedTier('');\n setUseCase('');\n onClose();\n };\n\n const handleSubmit = async () => {\n if (!selectedApi || !selectedTier) return;\n\n const [namespace, apiProductName] = selectedApi.split('/');\n\n setCreating(true);\n try {\n const response = await fetchApi.fetch(\n `${backendUrl}/api/kuadrant/requests`,\n {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n },\n body: JSON.stringify({\n apiProductName,\n namespace,\n planTier: selectedTier,\n useCase: useCase.trim() || '',\n userEmail: userEmail || 'unknown',\n }),\n },\n );\n\n if (!response.ok) {\n let errorData;\n try {\n errorData = await response.json();\n } catch {\n errorData = { error: `Server returned ${response.status} ${response.statusText}` };\n }\n\n console.error('Backend error response:', {\n status: response.status,\n statusText: response.statusText,\n errorData,\n requestBody: {\n apiProductName,\n namespace,\n planTier: selectedTier,\n useCase: useCase.trim() || '',\n userEmail: userEmail || 'unknown',\n }\n });\n\n // Extract user-friendly error message\n const rawError = errorData.error || errorData.message || `Server returned ${response.status}`;\n\n // Try to extract validation error details from Kubernetes error messages\n // Example: \"failed to create apikeys: APIKey.devportal.kuadrant.io \"name\" is invalid: spec.requestedBy.email: Invalid value: \"admin\": spec.requestedBy.email in body should match '^[a-zA-Z0-9._%+-]+@[a-zA-Z0-9.-]+\\.[a-zA-Z]{2,}$'\"\n let errorMsg = rawError;\n\n if (rawError.includes('spec.requestedBy.email')) {\n errorMsg = 'Invalid email format. Please contact your administrator to update your profile email.';\n } else if (rawError.includes('is invalid:')) {\n // Extract the validation message after \"is invalid:\"\n const match = rawError.match(/is invalid: (.+?)(?:\\s+\\(|$)/);\n if (match) {\n errorMsg = `Validation error: ${match[1]}`;\n }\n } else if (response.status === 403) {\n errorMsg = 'You do not have permission to request access to this API.';\n } else if (response.status === 404) {\n errorMsg = 'The selected API or tier was not found.';\n } else if (response.status >= 500) {\n errorMsg = `Server error: ${rawError}`;\n }\n\n throw new Error(errorMsg);\n }\n\n alertApi.post({\n message: 'API key requested successfully',\n severity: 'success',\n display: 'transient',\n });\n\n handleClose();\n onSuccess();\n } catch (err) {\n const errorMessage =\n err instanceof Error ? err.message : 'unknown error occurred';\n console.error('Failed to create API key request:', {\n error: err,\n selectedApi,\n selectedTier,\n userEmail,\n });\n alertApi.post({\n message: `Failed to request API key: ${errorMessage}`,\n severity: 'error',\n display: 'permanent',\n });\n } finally {\n setCreating(false);\n }\n };\n\n return (\n <Dialog open={open} onClose={handleClose} maxWidth=\"sm\" fullWidth>\n <DialogTitle>Request API key</DialogTitle>\n <DialogContent>\n <Typography variant=\"body2\" color=\"textSecondary\">\n Submit your request to generate an API key.\n </Typography>\n\n <FormControl fullWidth margin=\"normal\" disabled={creating || loadingProducts}>\n <InputLabel id=\"api-select-label\" required>\n API\n </InputLabel>\n <Select\n labelId=\"api-select-label\"\n value={selectedApi}\n onChange={e => {\n setSelectedApi(e.target.value as string);\n setSelectedTier(''); // Reset tier when API changes\n }}\n disabled={creating || loadingProducts}\n data-testid=\"api-select\"\n >\n {loadingProducts ? (\n <MenuItem disabled>Loading...</MenuItem>\n ) : (\n apiProducts?.map((product: ApiProduct) => (\n <MenuItem\n key={`${product.metadata.namespace}/${product.metadata.name}`}\n value={`${product.metadata.namespace}/${product.metadata.name}`}\n >\n {product.metadata.name}\n </MenuItem>\n ))\n )}\n </Select>\n <FormHelperText>\n Select one API. Please submit separate requests for multiple APIs.\n </FormHelperText>\n </FormControl>\n\n <FormControl\n fullWidth\n margin=\"normal\"\n disabled={creating || !selectedApi}\n required\n >\n <InputLabel id=\"tier-select-label\">Tiers</InputLabel>\n <Select\n labelId=\"tier-select-label\"\n value={selectedTier}\n onChange={e => setSelectedTier(e.target.value as string)}\n disabled={creating || !selectedApi}\n data-testid=\"tier-select\"\n >\n {!selectedApi ? (\n <MenuItem disabled>Select an API first</MenuItem>\n ) : availableTiers.length === 0 ? (\n <MenuItem disabled>No tiers available</MenuItem>\n ) : (\n availableTiers.map((plan: { tier: string; limits?: Record<string, number> }) => {\n const limitDesc = Object.entries(plan.limits || {})\n .map(([key, val]) => `${val} per ${key}`)\n .join(', ');\n return (\n <MenuItem key={plan.tier} value={plan.tier}>\n {plan.tier} {limitDesc ? `(${limitDesc})` : ''}\n </MenuItem>\n );\n })\n )}\n </Select>\n <FormHelperText>Select an API to view available tiers.</FormHelperText>\n </FormControl>\n\n <TextField\n label=\"Use case\"\n placeholder=\"Briefly describe your specific use case of using this API key\"\n multiline\n rows={2}\n fullWidth\n margin=\"normal\"\n value={useCase}\n onChange={e => setUseCase(e.target.value)}\n disabled={creating}\n inputProps={{ 'data-testid': 'usecase-input' }}\n />\n </DialogContent>\n <DialogActions>\n <Button onClick={handleClose} disabled={creating} data-testid=\"cancel-button\">\n Cancel\n </Button>\n <Button\n onClick={handleSubmit}\n color=\"primary\"\n variant=\"contained\"\n disabled={!selectedApi || !selectedTier || creating}\n startIcon={\n creating ? (\n <CircularProgress size={16} color=\"inherit\" />\n ) : undefined\n }\n data-testid=\"submit-button\"\n >\n {creating ? 'Submitting...' : 'Request'}\n </Button>\n </DialogActions>\n </Dialog>\n );\n};\n","import React, { useState, useMemo } from \"react\";\nimport { useNavigate } from \"react-router-dom\";\nimport {\n Table,\n TableColumn,\n Link,\n Progress,\n ResponseErrorPanel,\n} from \"@backstage/core-components\";\nimport {\n useApi,\n alertApiRef,\n} from \"@backstage/core-plugin-api\";\nimport { kuadrantApiRef } from '../../api';\nimport useAsync from \"react-use/lib/useAsync\";\nimport {\n Box,\n Chip,\n Typography,\n IconButton,\n Tooltip,\n Menu,\n MenuItem,\n CircularProgress,\n Dialog,\n DialogTitle,\n DialogContent,\n DialogActions,\n Button,\n makeStyles,\n} from \"@material-ui/core\";\nimport { SimpleRequestAccessDialog } from \"../SimpleRequestAccessDialog\";\nimport VisibilityIcon from \"@material-ui/icons/Visibility\";\nimport VisibilityOffIcon from \"@material-ui/icons/VisibilityOff\";\nimport FileCopyIcon from \"@material-ui/icons/FileCopy\";\nimport WarningIcon from \"@material-ui/icons/Warning\";\nimport DeleteIcon from \"@material-ui/icons/Delete\";\nimport { EditAPIKeyDialog } from \"../EditAPIKeyDialog\";\nimport { ConfirmDeleteDialog } from \"../ConfirmDeleteDialog\";\nimport { FilterPanel, FilterSection, FilterState } from \"../FilterPanel\";\nimport { APIKey, APIProduct } from \"../../types/api-management\";\nimport { getMyApiKeysStatusChipStyle } from \"../../utils/styles\";\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 useCasePanel: {\n padding: theme.spacing(2),\n backgroundColor: theme.palette.background.default,\n },\n useCaseLabel: {\n fontWeight: 600,\n marginBottom: theme.spacing(1),\n color: theme.palette.text.secondary,\n textTransform: \"uppercase\",\n fontSize: \"0.75rem\",\n },\n rejectedBanner: {\n backgroundColor: theme.palette.error.light,\n border: `1px solid ${theme.palette.error.main}`,\n borderRadius: theme.shape.borderRadius,\n padding: theme.spacing(1.5, 2),\n marginBottom: theme.spacing(2),\n display: \"flex\",\n alignItems: \"center\",\n gap: theme.spacing(1),\n },\n}));\n\ninterface ExpandedRowProps {\n request: APIKey;\n}\n\nconst ExpandedRowContent = ({ request }: ExpandedRowProps) => {\n const classes = useStyles();\n const isRejected = request.status?.phase === \"Rejected\";\n const apiProductName = request.spec.apiProductRef?.name || \"unknown\";\n\n return (\n <Box className={classes.useCasePanel} onClick={(e) => e.stopPropagation()}>\n {isRejected && (\n <Box className={classes.rejectedBanner}>\n <WarningIcon color=\"error\" fontSize=\"small\" />\n <Typography variant=\"body2\">\n This API key was rejected.{\" \"}\n <Link to={`/catalog/default/api/${apiProductName}/api-keys`}>\n Request a new API key\n </Link>\n </Typography>\n </Box>\n )}\n <Typography className={classes.useCaseLabel}>Use Case</Typography>\n <Typography variant=\"body2\">\n {request.spec.useCase || \"No use case provided\"}\n </Typography>\n </Box>\n );\n};\n\nexport const MyApiKeysTable = () => {\n const classes = useStyles();\n const navigate = useNavigate();\n const kuadrantApi = useApi(kuadrantApiRef);\n const alertApi = useApi(alertApiRef);\n\n const [requestDialogOpen, setRequestDialogOpen] = useState(false);\n const [visibleKeys, setVisibleKeys] = useState<Set<string>>(new Set());\n const [menuAnchor, setMenuAnchor] = useState<{\n top: number;\n left: number;\n } | null>(null);\n const [menuRequest, setMenuRequest] = useState<APIKey | null>(null);\n const [editDialogState, setEditDialogState] = useState<{\n open: boolean;\n request: APIKey | null;\n plans: any[];\n }>({ open: false, request: null, plans: [] });\n const [refresh, setRefresh] = useState(0);\n const [deleting, setDeleting] = useState<string | null>(null);\n const [deleteDialogState, setDeleteDialogState] = useState<{\n open: boolean;\n request: APIKey | null;\n }>({ open: false, request: null });\n const [apiKeyValues, setApiKeyValues] = useState<Map<string, string>>(\n new Map(),\n );\n const [apiKeyLoading, setApiKeyLoading] = useState<Set<string>>(new Set());\n const [alreadyReadKeys, setAlreadyReadKeys] = useState<Set<string>>(\n new Set(),\n );\n const [showOnceWarningOpen, setShowOnceWarningOpen] = useState(false);\n const [pendingKeyReveal, setPendingKeyReveal] = useState<{\n namespace: string;\n name: string;\n } | null>(null);\n const [optimisticallyDeleted, setOptimisticallyDeleted] = useState<\n Set<string>\n >(new Set());\n\n const [filters, setFilters] = useState<FilterState>({\n status: [],\n apiProduct: [],\n tier: [],\n });\n\n const {\n value: data,\n loading,\n error,\n } = useAsync(async () => {\n const [requestsData, productsData] = await Promise.all([\n kuadrantApi.getRequests(),\n kuadrantApi.getApiProducts(),\n ]);\n\n const requests: APIKey[] = requestsData.items || [];\n const products: APIProduct[] = productsData.items || [];\n\n // build owner map from products\n const ownerMap = new Map<string, string>();\n products.forEach((p) => {\n const key = `${p.metadata.namespace}/${p.metadata.name}`;\n const owner = p.metadata.annotations?.[\"backstage.io/owner\"] || \"unknown\";\n ownerMap.set(key, owner);\n });\n\n return { requests, products, ownerMap };\n }, [kuadrantApi, refresh]);\n\n const allRequests = useMemo(() => {\n if (!data?.requests) return [];\n return data.requests.filter(\n (r: APIKey) => !optimisticallyDeleted.has(r.metadata.name),\n );\n }, [data?.requests, optimisticallyDeleted]);\n\n // filter options from data\n const filterSections: FilterSection[] = useMemo(() => {\n const statusCounts = { Approved: 0, Pending: 0, Rejected: 0 };\n const apiProductCounts = new Map<string, number>();\n const tierCounts = new Map<string, number>();\n\n allRequests.forEach((r: APIKey) => {\n const status = r.status?.phase || \"Pending\";\n statusCounts[status as keyof typeof statusCounts]++;\n\n const apiProduct = r.spec.apiProductRef?.name || \"unknown\";\n apiProductCounts.set(\n apiProduct,\n (apiProductCounts.get(apiProduct) || 0) + 1,\n );\n\n const tier = r.spec.planTier || \"unknown\";\n tierCounts.set(tier, (tierCounts.get(tier) || 0) + 1);\n });\n\n return [\n {\n id: \"status\",\n title: \"Status\",\n options: [\n { value: \"Approved\", label: \"Active\", count: statusCounts.Approved },\n { value: \"Pending\", label: \"Pending\", count: statusCounts.Pending },\n {\n value: \"Rejected\",\n label: \"Rejected\",\n count: statusCounts.Rejected,\n },\n ],\n },\n {\n id: \"apiProduct\",\n title: \"API Product\",\n options: Array.from(apiProductCounts.entries()).map(\n ([name, count]) => ({\n value: name,\n label: name,\n count,\n }),\n ),\n collapsed: apiProductCounts.size > 5,\n },\n {\n id: \"tier\",\n title: \"Tier\",\n options: Array.from(tierCounts.entries()).map(([tier, count]) => ({\n value: tier,\n label: tier.charAt(0).toUpperCase() + tier.slice(1),\n count,\n })),\n },\n ];\n }, [allRequests]);\n\n // filtered requests\n const filteredRequests = useMemo(() => {\n return allRequests.filter((r: APIKey) => {\n // status filter\n if (filters.status.length > 0) {\n const status = r.status?.phase || \"Pending\";\n if (!filters.status.includes(status)) return false;\n }\n\n // api product filter\n if (filters.apiProduct.length > 0) {\n const apiProduct = r.spec.apiProductRef?.name || \"unknown\";\n if (!filters.apiProduct.includes(apiProduct)) return false;\n }\n\n // tier filter\n if (filters.tier.length > 0) {\n const tier = r.spec.planTier || \"unknown\";\n if (!filters.tier.includes(tier)) return false;\n }\n\n return true;\n });\n }, [allRequests, filters]);\n\n const toggleKeyVisibility = (keyName: string) => {\n setVisibleKeys((prev) => {\n const newSet = new Set(prev);\n if (newSet.has(keyName)) {\n newSet.delete(keyName);\n } else {\n newSet.add(keyName);\n }\n return newSet;\n });\n };\n\n const fetchApiKeyFromSecret = async (\n requestNamespace: string,\n requestName: string,\n ) => {\n const key = `${requestNamespace}/${requestName}`;\n if (apiKeyLoading.has(key)) return;\n\n setApiKeyLoading((prev) => new Set(prev).add(key));\n try {\n const result = await kuadrantApi.getApiKeySecret(requestNamespace, requestName);\n setApiKeyValues((prev) => new Map(prev).set(key, result.apiKey));\n setAlreadyReadKeys((prev) => new Set(prev).add(key));\n } catch (err) {\n const errorMessage =\n err instanceof Error ? err.message : \"unknown error occurred\";\n if (errorMessage.includes(\"403\") || errorMessage.includes(\"already been viewed\")) {\n setAlreadyReadKeys((prev) => new Set(prev).add(key));\n alertApi.post({\n message:\n \"This API key has already been viewed and cannot be retrieved again.\",\n severity: \"warning\",\n display: \"transient\",\n });\n } else {\n alertApi.post({\n message: `Failed to fetch api key: ${errorMessage}`,\n severity: \"error\",\n display: \"transient\",\n });\n }\n } finally {\n setApiKeyLoading((prev) => {\n const next = new Set(prev);\n next.delete(key);\n return next;\n });\n }\n };\n\n const clearApiKeyValue = (requestNamespace: string, requestName: string) => {\n const key = `${requestNamespace}/${requestName}`;\n setApiKeyValues((prev) => {\n const next = new Map(prev);\n next.delete(key);\n return next;\n });\n };\n\n const handleMenuClose = () => {\n setMenuAnchor(null);\n setMenuRequest(null);\n };\n\n const handleEdit = async () => {\n if (!menuRequest) return;\n\n const request = menuRequest;\n handleMenuClose();\n\n try {\n const apiProductName = request.spec.apiProductRef?.name;\n const apiProductNamespace = request.metadata.namespace;\n const apiProduct = await kuadrantApi.getApiProduct(apiProductNamespace, apiProductName);\n const plans = apiProduct.status?.discoveredPlans || [];\n setEditDialogState({ open: true, request, plans });\n } catch (err) {\n console.error(\"Error fetching plans:\", err);\n setEditDialogState({ open: true, request: menuRequest, plans: [] });\n const errorMessage = err instanceof Error ? err.message : \"unknown error occurred\";\n alertApi.post({\n message: `Failed to fetch Plans. ${errorMessage}`,\n severity: 'error',\n display: 'transient',\n });\n }\n };\n\n const handleDeleteClick = () => {\n if (!menuRequest) return;\n const request = menuRequest;\n handleMenuClose();\n setDeleteDialogState({ open: true, request });\n };\n\n const handleDeleteConfirm = async () => {\n if (!deleteDialogState.request) return;\n\n const request = deleteDialogState.request;\n const requestName = request.metadata.name;\n\n setOptimisticallyDeleted((prev) => new Set(prev).add(requestName));\n setDeleting(requestName);\n\n try {\n await kuadrantApi.deleteRequest(request.metadata.namespace, request.metadata.name);\n setRefresh((r) => r + 1);\n alertApi.post({\n message: \"API key deleted\",\n severity: \"success\",\n display: \"transient\",\n });\n setDeleteDialogState({ open: false, request: null });\n } catch (err) {\n console.error(\"Error deleting request:\", err);\n setOptimisticallyDeleted((prev) => {\n const next = new Set(prev);\n next.delete(requestName);\n return next;\n });\n const errorMessage = err instanceof Error ? err.message : \"unknown error occurred\";\n alertApi.post({\n message: `Failed to delete APIKey. ${errorMessage}`,\n severity: 'error',\n display: 'transient',\n });\n } finally {\n setDeleting(null);\n }\n };\n\n const handleDeleteCancel = () => {\n setDeleteDialogState({ open: false, request: null });\n };\n\n const columns: TableColumn<APIKey>[] = [\n {\n title: \"API Product\",\n field: \"spec.apiProductRef.name\",\n render: (row: APIKey) => {\n const apiProductName = row.spec.apiProductRef?.name || \"unknown\";\n return (\n <Link to={`/catalog/default/api/${apiProductName}/api-keys`}>\n <strong>{apiProductName}</strong>\n </Link>\n );\n },\n },\n {\n title: \"Owner\",\n field: \"owner\",\n render: (row: APIKey) => {\n const key = `${row.metadata.namespace}/${row.spec.apiProductRef?.name}`;\n const owner = data?.ownerMap?.get(key) || \"unknown\";\n // strip 'user:default/' prefix if present\n const displayOwner = owner.replace(/^user:default\\//, \"\");\n return <Typography variant=\"body2\">{displayOwner}</Typography>;\n },\n },\n {\n title: \"Status\",\n field: \"status.phase\",\n render: (row: APIKey) => {\n const phase = row.status?.phase || \"Pending\";\n const label = phase === \"Approved\" ? \"Active\" : phase;\n return (\n <Chip label={label} size=\"small\" style={getMyApiKeysStatusChipStyle(phase)} />\n );\n },\n },\n {\n title: \"Tier\",\n field: \"spec.planTier\",\n render: (row: APIKey) => (\n <Chip label={row.spec.planTier} size=\"small\" variant=\"outlined\" />\n ),\n },\n {\n title: \"API Key\",\n field: \"status.secretRef\",\n filtering: false,\n render: (row: APIKey) => {\n if (row.status?.phase !== \"Approved\") {\n return (\n <Typography variant=\"body2\" color=\"textSecondary\">\n -\n </Typography>\n );\n }\n\n const key = `${row.metadata.namespace}/${row.metadata.name}`;\n const hasSecretRef = row.status?.secretRef?.name;\n const isVisible = visibleKeys.has(row.metadata.name);\n const isLoading = apiKeyLoading.has(key);\n const apiKeyValue = apiKeyValues.get(key);\n const canReadSecret = row.status?.canReadSecret !== false;\n const isAlreadyRead = alreadyReadKeys.has(key) || !canReadSecret;\n\n if (!hasSecretRef) {\n return (\n <Typography variant=\"body2\" color=\"textSecondary\">\n Awaiting secret...\n </Typography>\n );\n }\n\n if (isAlreadyRead && !apiKeyValue) {\n return (\n <Tooltip title=\"This API key has already been viewed and cannot be retrieved again\">\n <Box display=\"flex\" alignItems=\"center\">\n <Typography\n variant=\"body2\"\n color=\"textSecondary\"\n style={{ fontFamily: \"monospace\", marginRight: 8 }}\n >\n Already viewed\n </Typography>\n <VisibilityOffIcon fontSize=\"small\" color=\"disabled\" />\n </Box>\n </Tooltip>\n );\n }\n\n const handleRevealClick = () => {\n if (isVisible) {\n clearApiKeyValue(row.metadata.namespace, row.metadata.name);\n toggleKeyVisibility(row.metadata.name);\n } else if (!isAlreadyRead) {\n setPendingKeyReveal({\n namespace: row.metadata.namespace,\n name: row.metadata.name,\n });\n setShowOnceWarningOpen(true);\n }\n };\n\n const handleCopy = async () => {\n if (apiKeyValue) {\n await navigator.clipboard.writeText(apiKeyValue);\n alertApi.post({\n message: \"API key copied to clipboard\",\n severity: \"success\",\n display: \"transient\",\n });\n }\n };\n\n return (\n <Box display=\"flex\" alignItems=\"center\" style={{ gap: 8 }}>\n <Box fontFamily=\"monospace\" fontSize=\"0.875rem\">\n {isLoading\n ? \"Loading...\"\n : isVisible && apiKeyValue\n ? apiKeyValue\n : \"•\".repeat(20) + \"...\"}\n </Box>\n {isVisible && apiKeyValue && (\n <Tooltip title=\"Copy to clipboard\">\n <IconButton size=\"small\" onClick={handleCopy}>\n <FileCopyIcon fontSize=\"small\" />\n </IconButton>\n </Tooltip>\n )}\n <Tooltip\n title={\n isVisible ? \"Hide API key\" : \"Reveal API key (one-time only)\"\n }\n >\n <span>\n <IconButton\n size=\"small\"\n onClick={handleRevealClick}\n disabled={isLoading || (isAlreadyRead && !apiKeyValue)}\n >\n {isVisible ? (\n <VisibilityOffIcon fontSize=\"small\" />\n ) : (\n <VisibilityIcon fontSize=\"small\" />\n )}\n </IconButton>\n </span>\n </Tooltip>\n </Box>\n );\n },\n },\n {\n title: \"Requested\",\n field: \"metadata.creationTimestamp\",\n render: (row: APIKey) => {\n if (!row.metadata.creationTimestamp) {\n return <Typography variant=\"body2\">-</Typography>;\n }\n const date = new Date(row.metadata.creationTimestamp);\n return (\n <Typography variant=\"body2\">{date.toLocaleDateString()}</Typography>\n );\n },\n },\n {\n title: \"Actions\",\n filtering: false,\n width: \"100px\",\n render: (row: APIKey) => {\n const isDeleting = deleting === row.metadata.name;\n if (isDeleting) {\n return <CircularProgress size={20} />;\n }\n return (\n <Box display=\"flex\" style={{ gap: 4 }}>\n <Tooltip title=\"View details\">\n <IconButton\n size=\"small\"\n onClick={(e) => {\n e.stopPropagation();\n navigate(\n `/kuadrant/api-keys/${row.metadata.namespace}/${row.metadata.name}`,\n );\n }}\n >\n <VisibilityIcon fontSize=\"small\" />\n </IconButton>\n </Tooltip>\n <Tooltip title=\"Delete\">\n <IconButton\n size=\"small\"\n onClick={(e) => {\n e.stopPropagation();\n setDeleteDialogState({ open: true, request: row });\n }}\n >\n <DeleteIcon fontSize=\"small\" />\n </IconButton>\n </Tooltip>\n </Box>\n );\n },\n },\n ];\n\n const detailPanelConfig = useMemo(\n () => [\n {\n render: (data: any) => {\n const request = data.rowData as APIKey;\n if (!request?.metadata?.name) {\n return <Box />;\n }\n return <ExpandedRowContent request={request} />;\n },\n },\n ],\n [],\n );\n\n if (loading) {\n return <Progress />;\n }\n\n if (error) {\n return <ResponseErrorPanel error={error} />;\n }\n\n const isPending = (row: APIKey) =>\n !row.status || row.status.phase === \"Pending\";\n\n return (\n <>\n <Box display=\"flex\" justifyContent=\"flex-end\" mb={2} px={2}>\n <Button\n variant=\"contained\"\n color=\"primary\"\n onClick={() => setRequestDialogOpen(true)}\n data-testid=\"request-access-button\"\n >\n Request Access\n </Button>\n </Box>\n <Box className={classes.container}>\n <FilterPanel\n sections={filterSections}\n filters={filters}\n onChange={setFilters}\n />\n <Box className={classes.tableContainer}>\n {allRequests.length === 0 ? (\n <Box p={4} textAlign=\"center\">\n <Typography variant=\"body1\" color=\"textSecondary\" gutterBottom>\n No API keys found. Request access to an API to get started.\n </Typography>\n </Box>\n ) : filteredRequests.length === 0 ? (\n <Box p={4} textAlign=\"center\">\n <Typography variant=\"body1\" color=\"textSecondary\">\n No API keys match the selected filters.\n </Typography>\n </Box>\n ) : (\n <Table\n options={{\n paging: filteredRequests.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={filteredRequests.map((item: APIKey) => ({\n ...item,\n id: item.metadata.name,\n }))}\n detailPanel={detailPanelConfig}\n />\n )}\n </Box>\n </Box>\n\n <Menu\n id=\"myapikeys-menu\"\n open={Boolean(menuAnchor)}\n onClose={handleMenuClose}\n anchorReference=\"anchorPosition\"\n anchorPosition={menuAnchor || { top: 0, left: 0 }}\n >\n {menuRequest &&\n (() => {\n const items = [];\n items.push(\n <MenuItem\n key=\"view\"\n onClick={() => {\n navigate(\n `/kuadrant/api-keys/${menuRequest.metadata.namespace}/${menuRequest.metadata.name}`,\n );\n handleMenuClose();\n }}\n >\n View Details\n </MenuItem>,\n );\n if (isPending(menuRequest)) {\n items.push(\n <MenuItem key=\"edit\" onClick={handleEdit}>\n Edit\n </MenuItem>,\n );\n }\n items.push(\n <MenuItem key=\"delete\" onClick={handleDeleteClick}>\n Delete\n </MenuItem>,\n );\n return items;\n })()}\n </Menu>\n\n {editDialogState.request && (\n <EditAPIKeyDialog\n open={editDialogState.open}\n request={editDialogState.request}\n availablePlans={editDialogState.plans}\n onClose={() =>\n setEditDialogState({ open: false, request: null, plans: [] })\n }\n onSuccess={() => {\n setEditDialogState({ open: false, request: null, plans: [] });\n setRefresh((r) => r + 1);\n }}\n />\n )}\n\n <ConfirmDeleteDialog\n open={deleteDialogState.open}\n title=\"Delete API Key\"\n description={`Are you sure you want to delete this API key for ${deleteDialogState.request?.spec.apiProductRef?.name || \"this API\"}?`}\n deleting={deleting !== null}\n onConfirm={handleDeleteConfirm}\n onCancel={handleDeleteCancel}\n />\n\n <Dialog\n open={showOnceWarningOpen}\n onClose={() => {\n setShowOnceWarningOpen(false);\n setPendingKeyReveal(null);\n }}\n maxWidth=\"sm\"\n >\n <DialogTitle>\n <Box display=\"flex\" alignItems=\"center\">\n <WarningIcon color=\"primary\" style={{ marginRight: 8 }} />\n View API Key\n </Box>\n </DialogTitle>\n <DialogContent>\n <Typography variant=\"body1\" paragraph>\n This API key can only be viewed <strong>once</strong>. After you\n reveal it, you will not be able to retrieve it again.\n </Typography>\n <Typography variant=\"body2\" color=\"textSecondary\">\n Make sure to copy and store it securely before closing this view.\n </Typography>\n </DialogContent>\n <DialogActions>\n <Button\n onClick={() => {\n setShowOnceWarningOpen(false);\n setPendingKeyReveal(null);\n }}\n >\n Cancel\n </Button>\n <Button\n variant=\"contained\"\n color=\"primary\"\n onClick={() => {\n if (pendingKeyReveal) {\n fetchApiKeyFromSecret(\n pendingKeyReveal.namespace,\n pendingKeyReveal.name,\n );\n toggleKeyVisibility(pendingKeyReveal.name);\n }\n setShowOnceWarningOpen(false);\n setPendingKeyReveal(null);\n }}\n >\n Reveal API Key\n </Button>\n </DialogActions>\n </Dialog>\n\n <SimpleRequestAccessDialog\n open={requestDialogOpen}\n onClose={() => setRequestDialogOpen(false)}\n onSuccess={() => {\n setRequestDialogOpen(false);\n setRefresh((prev) => prev + 1);\n }}\n />\n </>\n );\n};\n","import React from 'react';\nimport {\n Header,\n Page,\n Content,\n SupportButton,\n} from '@backstage/core-components';\nimport { MyApiKeysTable } from '../MyApiKeysTable';\n\nexport const MyApiKeysPage = () => {\n return (\n <Page themeId=\"tool\">\n <Header title=\"My API Keys\" subtitle=\"View and manage your API keys\">\n <SupportButton>Manage your API keys and access requests</SupportButton>\n </Header>\n <Content>\n <MyApiKeysTable />\n </Content>\n </Page>\n );\n};\n","\"use strict\";\n\nvar _interopRequireDefault = require(\"@babel/runtime/helpers/interopRequireDefault\");\n\nvar _interopRequireWildcard = require(\"@babel/runtime/helpers/interopRequireWildcard\");\n\nObject.defineProperty(exports, \"__esModule\", {\n value: true\n});\nexports.default = void 0;\n\nvar React = _interopRequireWildcard(require(\"react\"));\n\nvar _createSvgIcon = _interopRequireDefault(require(\"./utils/createSvgIcon\"));\n\nvar _default = (0, _createSvgIcon.default)( /*#__PURE__*/React.createElement(\"path\", {\n d: \"M6 19c0 1.1.9 2 2 2h8c1.1 0 2-.9 2-2V7H6v12zM19 4h-3.5l-1-1h-5l-1 1H5v2h14V4z\"\n}), 'Delete');\n\nexports.default = _default;","import { useApiHolder, configApiRef } from '@backstage/core-plugin-api';\nimport { coreComponentsTranslationRef } from '../translation.esm.js';\nimport { useTranslationRef } from '@backstage/core-plugin-api/alpha';\n\nconst useDefaultSupportConfig = () => {\n const { t } = useTranslationRef(coreComponentsTranslationRef);\n return {\n url: \"https://github.com/backstage/backstage/issues\",\n items: [\n {\n title: t(\"supportConfig.default.title\"),\n icon: \"warning\",\n links: [\n {\n // TODO: Update to dedicated support page on backstage.io/docs\n title: t(\"supportConfig.default.linkTitle\"),\n url: \"https://github.com/backstage/backstage/blob/master/app-config.yaml\"\n }\n ]\n }\n ]\n };\n};\nfunction useSupportConfig() {\n const apiHolder = useApiHolder();\n const config = apiHolder.get(configApiRef);\n const supportConfig = config?.getOptionalConfig(\"app.support\");\n const defaultSupportConfig = useDefaultSupportConfig();\n if (!supportConfig) {\n return defaultSupportConfig;\n }\n return {\n url: supportConfig.getString(\"url\"),\n items: supportConfig.getConfigArray(\"items\").flatMap((itemConf) => ({\n title: itemConf.getString(\"title\"),\n icon: itemConf.getOptionalString(\"icon\"),\n links: (itemConf.getOptionalConfigArray(\"links\") ?? []).flatMap(\n (linkConf) => ({\n url: linkConf.getString(\"url\"),\n title: linkConf.getOptionalString(\"title\") ?? \"\"\n })\n )\n }))\n };\n}\n\nexport { useSupportConfig };\n//# sourceMappingURL=useSupportConfig.esm.js.map\n","import { jsx } from 'react/jsx-runtime';\nimport { useApp } from '@backstage/core-plugin-api';\nimport MuiBrokenImageIcon from '@material-ui/icons/BrokenImage';\n\nfunction AppIcon(props) {\n const { id: key, Fallback = MuiBrokenImageIcon, ...rest } = props;\n const app = useApp();\n const Icon = app.getSystemIcon(key) ?? Fallback;\n return /* @__PURE__ */ jsx(Icon, { ...rest });\n}\nfunction BrokenImageIcon(props) {\n return /* @__PURE__ */ jsx(AppIcon, { id: \"brokenImage\", ...props });\n}\nfunction CatalogIcon(props) {\n return /* @__PURE__ */ jsx(AppIcon, { id: \"catalog\", ...props });\n}\nfunction ChatIcon(props) {\n return /* @__PURE__ */ jsx(AppIcon, { id: \"chat\", ...props });\n}\nfunction DashboardIcon(props) {\n return /* @__PURE__ */ jsx(AppIcon, { id: \"dashboard\", ...props });\n}\nfunction DocsIcon(props) {\n return /* @__PURE__ */ jsx(AppIcon, { id: \"docs\", ...props });\n}\nfunction EmailIcon(props) {\n return /* @__PURE__ */ jsx(AppIcon, { id: \"email\", ...props });\n}\nfunction GitHubIcon(props) {\n return /* @__PURE__ */ jsx(AppIcon, { id: \"github\", ...props });\n}\nfunction GroupIcon(props) {\n return /* @__PURE__ */ jsx(AppIcon, { id: \"group\", ...props });\n}\nfunction HelpIcon(props) {\n return /* @__PURE__ */ jsx(AppIcon, { id: \"help\", ...props });\n}\nfunction UserIcon(props) {\n return /* @__PURE__ */ jsx(AppIcon, { id: \"user\", ...props });\n}\nfunction WarningIcon(props) {\n return /* @__PURE__ */ jsx(AppIcon, { id: \"warning\", ...props });\n}\nfunction StarIcon(props) {\n return /* @__PURE__ */ jsx(AppIcon, { id: \"star\", ...props });\n}\nfunction UnstarredIcon(props) {\n return /* @__PURE__ */ jsx(AppIcon, { id: \"unstarred\", ...props });\n}\n\nexport { AppIcon, BrokenImageIcon, CatalogIcon, ChatIcon, DashboardIcon, DocsIcon, EmailIcon, GitHubIcon, GroupIcon, HelpIcon, StarIcon, UnstarredIcon, UserIcon, WarningIcon };\n//# sourceMappingURL=icons.esm.js.map\n","import { jsxs, Fragment, jsx } from 'react/jsx-runtime';\nimport { useApi, configApiRef, useApp } from '@backstage/core-plugin-api';\nimport Box from '@material-ui/core/Box';\nimport Button from '@material-ui/core/Button';\nimport DialogActions from '@material-ui/core/DialogActions';\nimport IconButton from '@material-ui/core/IconButton';\nimport ListItemIcon from '@material-ui/core/ListItemIcon';\nimport ListItemText from '@material-ui/core/ListItemText';\nimport MenuItem from '@material-ui/core/MenuItem';\nimport MenuList from '@material-ui/core/MenuList';\nimport Popover from '@material-ui/core/Popover';\nimport { makeStyles } from '@material-ui/core/styles';\nimport Typography from '@material-ui/core/Typography';\nimport useMediaQuery from '@material-ui/core/useMediaQuery';\nimport { useState, Children } from 'react';\nimport 'lodash';\nimport 'qs';\nimport 'react-router-dom';\nimport '@react-hookz/web';\nimport { useSupportConfig } from '../../hooks/useSupportConfig.esm.js';\nimport { HelpIcon } from '../../icons/icons.esm.js';\nimport { Link } from '../Link/Link.esm.js';\nimport { coreComponentsTranslationRef } from '../../translation.esm.js';\nimport { useTranslationRef } from '@backstage/core-plugin-api/alpha';\n\nconst useStyles = makeStyles(\n {\n popoverList: {\n minWidth: 260,\n maxWidth: 400\n },\n menuItem: {\n whiteSpace: \"normal\"\n }\n },\n { name: \"BackstageSupportButton\" }\n);\nconst SupportIcon = ({ icon }) => {\n const app = useApp();\n const Icon = icon ? app.getSystemIcon(icon) ?? HelpIcon : HelpIcon;\n return /* @__PURE__ */ jsx(Icon, {});\n};\nconst SupportLink = ({ link }) => /* @__PURE__ */ jsx(Link, { to: link.url, children: link.title ?? link.url });\nconst SupportListItem = ({ item }) => {\n return /* @__PURE__ */ jsxs(MenuItem, { button: false, children: [\n /* @__PURE__ */ jsx(ListItemIcon, { children: /* @__PURE__ */ jsx(SupportIcon, { icon: item.icon }) }),\n /* @__PURE__ */ jsx(\n ListItemText,\n {\n primary: item.title,\n secondary: item.links?.reduce(\n (prev, link, idx) => [\n ...prev,\n idx > 0 && /* @__PURE__ */ jsx(\"br\", {}, idx),\n /* @__PURE__ */ jsx(SupportLink, { link }, link.url)\n ],\n []\n )\n }\n )\n ] });\n};\nfunction SupportButton(props) {\n const { t } = useTranslationRef(coreComponentsTranslationRef);\n const { title, items, children } = props;\n const { items: configItems } = useSupportConfig();\n const [popoverOpen, setPopoverOpen] = useState(false);\n const [anchorEl, setAnchorEl] = useState(null);\n const classes = useStyles();\n const supportConfig = useApi(configApiRef).getOptionalConfig(\"app.support\");\n const isSmallScreen = useMediaQuery(\n (theme) => theme.breakpoints.down(\"sm\")\n );\n const onClickHandler = (event) => {\n setAnchorEl(event.currentTarget);\n setPopoverOpen(true);\n };\n const popoverCloseHandler = () => {\n setPopoverOpen(false);\n };\n if (!supportConfig) {\n return null;\n }\n return /* @__PURE__ */ jsxs(Fragment, { children: [\n /* @__PURE__ */ jsx(Box, { display: \"flex\", ml: 1, children: isSmallScreen ? /* @__PURE__ */ jsx(\n IconButton,\n {\n color: \"primary\",\n size: \"small\",\n onClick: onClickHandler,\n \"data-testid\": \"support-button\",\n \"aria-label\": \"Support\",\n children: /* @__PURE__ */ jsx(HelpIcon, {})\n }\n ) : /* @__PURE__ */ jsx(\n Button,\n {\n \"data-testid\": \"support-button\",\n \"aria-label\": \"Support\",\n color: \"primary\",\n onClick: onClickHandler,\n startIcon: /* @__PURE__ */ jsx(HelpIcon, {}),\n children: t(\"supportButton.title\")\n }\n ) }),\n /* @__PURE__ */ jsxs(\n Popover,\n {\n \"data-testid\": \"support-button-popover\",\n open: popoverOpen,\n anchorEl,\n anchorOrigin: {\n vertical: \"bottom\",\n horizontal: \"right\"\n },\n transformOrigin: {\n vertical: \"top\",\n horizontal: \"right\"\n },\n onClose: popoverCloseHandler,\n children: [\n /* @__PURE__ */ jsxs(\n MenuList,\n {\n className: classes.popoverList,\n autoFocusItem: Boolean(anchorEl),\n children: [\n title && /* @__PURE__ */ jsx(\n MenuItem,\n {\n button: false,\n alignItems: \"flex-start\",\n className: classes.menuItem,\n children: /* @__PURE__ */ jsx(Typography, { variant: \"subtitle1\", children: title })\n }\n ),\n Children.map(children, (child, i) => /* @__PURE__ */ jsx(\n MenuItem,\n {\n button: false,\n alignItems: \"flex-start\",\n className: classes.menuItem,\n children: child\n },\n `child-${i}`\n )),\n (items ?? configItems).map((item, i) => /* @__PURE__ */ jsx(SupportListItem, { item }, `item-${i}`))\n ]\n }\n ),\n /* @__PURE__ */ jsx(DialogActions, { children: /* @__PURE__ */ jsx(\n Button,\n {\n color: \"primary\",\n onClick: popoverCloseHandler,\n \"aria-label\": \"Close\",\n children: t(\"supportButton.close\")\n }\n ) })\n ]\n }\n )\n ] });\n}\n\nexport { SupportButton };\n//# sourceMappingURL=SupportButton.esm.js.map\n","import { CSSProperties } from \"react\";\n\n/**\n * Returns inline styles for API key status chips on the My API Keys page.\n */\nexport const getMyApiKeysStatusChipStyle = (phase: string): CSSProperties => {\n const base = { border: \"none\" };\n switch (phase) {\n case \"Approved\":\n return { ...base, backgroundColor: \"#1976d2\", color: \"#fff\" }; // Blue\n case \"Rejected\":\n return { ...base, backgroundColor: \"#d32f2f\", color: \"#fff\" }; // Red\n case \"Pending\":\n return { ...base, backgroundColor: \"#9c27b0\", color: \"#fff\" }; // Purple\n default:\n return { ...base, backgroundColor: \"#9c27b0\", color: \"#fff\" }; // Purple (fallback for Pending)\n }\n};\n\n/**\n * Returns inline styles for API key status chips on the Approval Queue page.\n * Uses inline styles to ensure proper specificity with Material-UI Chip.\n */\nexport const getApprovalQueueStatusChipStyle = (phase: string): CSSProperties => {\n const base = { border: \"none\" };\n switch (phase) {\n case \"Approved\":\n return { ...base, backgroundColor: \"#2e7d32\", color: \"#fff\" }; // Green\n case \"Rejected\":\n return { ...base, backgroundColor: \"#d32f2f\", color: \"#fff\" }; // Red\n case \"Pending\":\n return { ...base, backgroundColor: \"#ed6c02\", color: \"#fff\" }; // Orange\n default:\n return { ...base, backgroundColor: \"#ed6c02\", color: \"#fff\" }; // Orange (fallback for Pending)\n }\n};\n\n/**\n * Returns inline styles for lifecycle chips.\n * Uses inline styles to ensure proper specificity with Material-UI Chip.\n */\nexport const getLifecycleChipStyle = (lifecycle: string): CSSProperties => {\n switch (lifecycle) {\n case \"production\":\n return { backgroundColor: \"#1976d2\", color: \"#fff\" }; // Blue\n case \"experimental\":\n return { backgroundColor: \"#9c27b0\", color: \"#fff\" }; // Purple\n case \"deprecated\":\n return { backgroundColor: \"#ff9800\", color: \"#fff\" }; // Orange\n case \"retired\":\n return { backgroundColor: \"#d32f2f\", color: \"#fff\" }; // Red\n default:\n return {};\n }\n};\n","\"use strict\";\n\nvar _interopRequireDefault = require(\"@babel/runtime/helpers/interopRequireDefault\");\n\nvar _interopRequireWildcard = require(\"@babel/runtime/helpers/interopRequireWildcard\");\n\nObject.defineProperty(exports, \"__esModule\", {\n value: true\n});\nexports.default = void 0;\n\nvar React = _interopRequireWildcard(require(\"react\"));\n\nvar _createSvgIcon = _interopRequireDefault(require(\"./utils/createSvgIcon\"));\n\nvar _default = (0, _createSvgIcon.default)( /*#__PURE__*/React.createElement(\"path\", {\n d: \"M12 7c2.76 0 5 2.24 5 5 0 .65-.13 1.26-.36 1.83l2.92 2.92c1.51-1.26 2.7-2.89 3.43-4.75-1.73-4.39-6-7.5-11-7.5-1.4 0-2.74.25-3.98.7l2.16 2.16C10.74 7.13 11.35 7 12 7zM2 4.27l2.28 2.28.46.46C3.08 8.3 1.78 10.02 1 12c1.73 4.39 6 7.5 11 7.5 1.55 0 3.03-.3 4.38-.84l.42.42L19.73 22 21 20.73 3.27 3 2 4.27zM7.53 9.8l1.55 1.55c-.05.21-.08.43-.08.65 0 1.66 1.34 3 3 3 .22 0 .44-.03.65-.08l1.55 1.55c-.67.33-1.41.53-2.2.53-2.76 0-5-2.24-5-5 0-.79.2-1.53.53-2.2zm4.31-.78l3.15 3.15.02-.16c0-1.66-1.34-3-3-3l-.17.01z\"\n}), 'VisibilityOff');\n\nexports.default = _default;","import React, { useState, useEffect } from 'react';\nimport {\n Dialog,\n DialogTitle,\n DialogContent,\n DialogContentText,\n DialogActions,\n Button,\n TextField,\n Typography,\n Box,\n CircularProgress,\n} from '@material-ui/core';\nimport WarningIcon from '@material-ui/icons/Warning';\n\nexport interface ConfirmDeleteDialogProps {\n open: boolean;\n title: string;\n description: string;\n // for dangerous deletes, require typing this text to confirm\n confirmText?: string;\n // severity affects styling - 'high' shows warning icon and requires text confirmation\n severity?: 'normal' | 'high';\n deleting?: boolean;\n onConfirm: () => void;\n onCancel: () => void;\n}\n\nexport const ConfirmDeleteDialog = ({\n open,\n title,\n description,\n confirmText,\n severity = 'normal',\n deleting = false,\n onConfirm,\n onCancel,\n}: ConfirmDeleteDialogProps) => {\n const [inputValue, setInputValue] = useState('');\n\n // reset input when dialog opens/closes\n useEffect(() => {\n if (!open) {\n setInputValue('');\n }\n }, [open]);\n\n const requiresTextConfirmation = severity === 'high' && confirmText;\n const canConfirm = requiresTextConfirmation ? inputValue === confirmText : true;\n\n const handleConfirm = () => {\n if (canConfirm) {\n onConfirm();\n }\n };\n\n return (\n <Dialog\n open={open}\n onClose={deleting ? undefined : onCancel}\n maxWidth=\"sm\"\n fullWidth\n >\n <DialogTitle>\n {severity === 'high' && (\n <Box display=\"flex\" alignItems=\"center\" style={{ gap: 8 }}>\n <WarningIcon color=\"error\" />\n <span>{title}</span>\n </Box>\n )}\n {severity !== 'high' && title}\n </DialogTitle>\n <DialogContent>\n <DialogContentText 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 {\n createApiRef,\n DiscoveryApi,\n FetchApi,\n IdentityApi,\n} from '@backstage/core-plugin-api';\nimport retry from 'async-retry';\nimport { handleFetchError } from './utils/errors';\nimport {\n APIKey, APIKeyRequest,\n APIKeySpec,\n APIProduct,\n BulkOperationResult, ExtractedSecret, K8sList, K8sResource,\n PlanPolicy,\n} from './types/api-management';\n\n/**\n * Generic Kuadrant list type for API responses\n */\nexport interface KuadrantList<T = any> {\n items: T[];\n}\n\n/**\n * Options for constructing the KuadrantApiClient\n */\nexport type Options = {\n discoveryApi: DiscoveryApi;\n fetchApi: FetchApi;\n identityApi: IdentityApi;\n};\n\n/**\n * Retry configuration for read operations (GET requests only)\n * Conservative strategy: 3 retries with exponential backoff\n */\nconst RETRY_OPTIONS: retry.Options = {\n retries: 3,\n factor: 2,\n minTimeout: 300, // 300ms, 600ms, 1200ms\n maxTimeout: 3000,\n randomize: true,\n};\n\n/**\n * Kuadrant API interface defining all operations for managing\n * API products, API keys, and related resources\n */\nexport interface KuadrantAPI {\n // ===== APIKey Requests =====\n\n /**\n * Fetch all API key requests per user\n * @returns Promise with list of all API key requests\n */\n getRequests(): Promise<KuadrantList<APIKey>>;\n\n /**\n * Fetch all API key requests\n * @returns Promise with list of all API key requests\n */\n getAllRequests(): Promise<KuadrantList<APIKey>>;\n\n /**\n * Fetch API key requests for a specific namespace\n * @param namespace - Kubernetes namespace\n * @returns Promise with list of requests in the namespace\n */\n getRequestsByNamespace(namespace: string): Promise<KuadrantList<APIKey>>;\n\n /**\n * Fetch a single API key request\n * @param namespace - API key request name\n * @param name - Kubernetes namespace\n * @returns Promise with the API key request\n */\n getRequest(namespace: string, name: string): Promise<APIKey>;\n\n /**\n * Create a new API key request\n * @param request - APIKeyRequest specification\n * @returns Promise with the created API key\n */\n createRequest(request: APIKeyRequest): Promise<APIKey>;\n\n /**\n * Update an existing API key request\n * @param namespace - Kubernetes namespace\n * @param name - API key request name\n * @param patch - Partial API key spec with fields to update\n * @returns Promise with the updated API key\n */\n updateRequest(\n namespace: string,\n name: string,\n patch: Partial<APIKeySpec>,\n ): Promise<APIKey>;\n\n /**\n * Delete an API key request\n * @param namespace - Kubernetes namespace\n * @param name - API key request name\n * @returns Promise that resolves when deletion completes\n */\n deleteRequest(namespace: string, name: string): Promise<void>;\n\n /**\n * Approve an API key request\n * @param namespace - Kubernetes namespace\n * @param name - API key request name\n * @param reviewedBy - Reviewed By User / System\n * @returns Promise with the approved API key\n */\n approveRequest(namespace: string, name: string, reviewedBy: string): Promise<APIKey>;\n\n /**\n * Reject an API key request\n * @param namespace - Kubernetes namespace\n * @param name - API key request name\n * @param reviewedBy - Reviewed By User / System\n * @returns Promise with the rejected API key\n */\n rejectRequest(namespace: string, name: string, reviewedBy: string): Promise<APIKey>;\n\n /**\n * Bulk approve multiple API key requests\n * @param requests - Array of namespace/name pairs to approve\n * @param reviewedBy - Reviewed By User / System\n * @returns Promise that resolves when all approvals complete\n */\n bulkApproveRequests(\n requests: Array<{ namespace: string; name: string }>,\n reviewedBy: string): Promise<Array<BulkOperationResult>>;\n\n /**\n * Bulk reject multiple API key requests\n * @param requests - Array of namespace/name pairs to reject\n * @param reviewedBy - Reviewed By User / System\n * @returns Promise that resolves when all rejections complete\n */\n bulkRejectRequests(\n requests: Array<{ namespace: string; name: string }>,\n reviewedBy: string): Promise<Array<BulkOperationResult>>;\n\n // ===== API Keys/Secrets =====\n\n /**\n * Fetch an API key resource\n * @param namespace - Kubernetes namespace\n * @param name - API key name\n * @returns Promise with the API key\n */\n getApiKey(namespace: string, name: string): Promise<APIKey>;\n\n /**\n * Retrieve the secret value for an API key (one-time operation)\n * @param namespace - Kubernetes namespace\n * @param name - API key name\n * @returns Promise with the secret value\n */\n getApiKeySecret(namespace: string, name: string): Promise<ExtractedSecret>;\n\n // ===== API Products =====\n\n /**\n * Fetch all API products\n * @returns Promise with list of all API products\n */\n getApiProducts(): Promise<KuadrantList<APIProduct>>;\n\n /**\n * Fetch a single API product\n * @param namespace - Kubernetes namespace\n * @param name - API product name\n * @returns Promise with the API product\n */\n getApiProduct(namespace: string, name: string): Promise<APIProduct>;\n\n /**\n * Create a new API product\n * @param product - API product specification\n * @returns Promise with the created API product\n */\n createApiProduct(product: APIProduct): Promise<APIProduct>;\n\n /**\n * Update an existing API product\n * @param namespace - Kubernetes namespace\n * @param name - API product name\n * @param patch - Partial API product spec with fields to update\n * @returns Promise with the updated API product\n */\n updateApiProduct(\n namespace: string,\n name: string,\n patch: Partial<APIProduct>,\n ): Promise<APIProduct>;\n\n /**\n * Delete an API product\n * @param namespace - Kubernetes namespace\n * @param name - API product name\n * @returns Promise that resolves when deletion completes\n */\n deleteApiProduct(namespace: string, name: string): Promise<void>;\n\n // ===== HTTP Routes & Policies =====\n\n /**\n * Fetch all HTTPRoute(s)\n * @returns Promise with list of all HTTP routes\n */\n getHttpRoutes(): Promise<K8sList>;\n\n /**\n * Fetch a specific HTTPRoute\n * @param namespace - Kubernetes namespace\n * @param name - HTTPRoute name\n * @returns Promise with an HTTPRoute\n */\n getHttpRoute(namespace: string, name: string): Promise<K8sResource>;\n\n // ===== Plan Policies =====\n\n /**\n * Fetch all plan policies\n * @returns Promise with list of all plan policies\n */\n getPlanPolicies(): Promise<KuadrantList<PlanPolicy>>;\n}\n\n/**\n * API reference for the Kuadrant API\n */\nexport const kuadrantApiRef = createApiRef<KuadrantAPI>({\n id: 'plugin.kuadrant.service',\n});\n\n/**\n * Implementation of the Kuadrant API client\n */\nexport class KuadrantApiClient implements KuadrantAPI {\n private readonly discoveryApi: DiscoveryApi;\n private readonly fetchApi: FetchApi;\n\n constructor(options: Options) {\n this.discoveryApi = options.discoveryApi;\n this.fetchApi = options.fetchApi;\n }\n\n /**\n * Get the base URL for the backend API\n */\n private async getBaseUrl(): Promise<string> {\n return await this.discoveryApi.getBaseUrl('');\n }\n\n /**\n * Wrapper for GET requests with automatic retry logic\n * Retries on network failures or 5xx errors with exponential backoff\n */\n private async fetchWithRetry<T>(url: string, errorMsg: string = \"\"): Promise<T> {\n return retry(\n async (bail) => {\n const response = await this.fetchApi.fetch(url);\n if (response.status === 401 || response.status === 403) { // Don't retry on Unauthenticated/Unauthorized\n const error = await handleFetchError(response);\n bail(new Error(error));\n }\n else if (!response.ok) {\n const error = await handleFetchError(response);\n throw new Error(`${errorMsg} ${error}`);\n }\n return await response.json();\n },\n RETRY_OPTIONS,\n );\n }\n\n /**\n * Wrapper for mutations (POST, PATCH, DELETE) without retry\n * These operations are not retried to avoid duplicate side effects\n */\n private async fetchWithoutRetry<T>(\n url: string,\n options: RequestInit,\n errorMsg: string = \"\"): Promise<T> {\n const response = await this.fetchApi.fetch(url, {\n headers: {\n \"Content-Type\": \"application/json\",\n },\n ...options,\n });\n\n if (!response.ok) {\n const error = await handleFetchError(response);\n throw new Error(`${errorMsg} ${error}`);\n }\n\n // DELETE operations don't return a body\n if (options.method === 'DELETE') {\n return undefined as T;\n }\n\n return await response.json();\n }\n\n // ===== API Requests Implementation =====\n\n async getRequests(): Promise<KuadrantList<APIKey>> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithRetry(\n `${baseUrl}kuadrant/requests/my`,\n \"Failed to fetch API Key requests.\"\n );\n }\n\n async getAllRequests(): Promise<KuadrantList<APIKey>> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithRetry(\n `${baseUrl}kuadrant/requests`,\n \"Failed to fetch API Key requests.\"\n );\n }\n\n async getRequestsByNamespace(namespace: string): Promise<KuadrantList<APIKey>> {\n const baseUrl = await this.getBaseUrl();\n const url = namespace\n ? `${baseUrl}kuadrant/requests/my?namespace=${namespace}`\n : `${baseUrl}kuadrant/requests/my`;\n return this.fetchWithRetry(url, \"Failed to fetch API Key requests by namespace.\");\n }\n\n async getRequest(namespace: string, name: string): Promise<APIKey> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithRetry(\n `${baseUrl}kuadrant/requests/${namespace}/${name}`,\n \"Failed to fetch API Key request.\"\n );\n }\n\n async createRequest(request: APIKeyRequest): Promise<APIKey> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithoutRetry(`${baseUrl}kuadrant/requests`, {\n method: 'POST',\n body: JSON.stringify(request),\n }, \"Failed to create APIKey request.\");\n }\n\n async updateRequest(\n namespace: string,\n name: string,\n patch: Partial<APIKeySpec>,\n ): Promise<APIKey> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithoutRetry(`${baseUrl}kuadrant/requests/${namespace}/${name}`, {\n method: 'PATCH',\n body: JSON.stringify(patch),\n }, \"Failed to update APIKey request.\");\n }\n\n async deleteRequest(namespace: string, name: string): Promise<void> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithoutRetry(`${baseUrl}kuadrant/requests/${namespace}/${name}`, {\n method: 'DELETE',\n }, \"Failed to delete APIKey request.\");\n }\n\n async approveRequest(namespace: string, name: string, reviewedBy: string = \"system\"): Promise<APIKey> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithoutRetry(`${baseUrl}kuadrant/requests/${namespace}/${name}/approve`, {\n method: 'POST',\n body: JSON.stringify({ reviewedBy }),\n }, \"Failed to approve APIKey request.\");\n }\n\n async rejectRequest(namespace: string, name: string, reviewedBy: string = \"system\"): Promise<APIKey> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithoutRetry(`${baseUrl}kuadrant/requests/${namespace}/${name}/reject`, {\n method: 'POST',\n body: JSON.stringify({ reviewedBy }),\n }, \"Failed to reject APIKey request.\");\n }\n\n async bulkApproveRequests(\n requests: Array<{ namespace: string; name: string }>,\n reviewedBy: string): Promise<Array<BulkOperationResult>> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithoutRetry(`${baseUrl}kuadrant/requests/bulk-approve`, {\n method: 'POST',\n body: JSON.stringify({requests, reviewedBy}),\n }, \"Failed to bulk approve APIKey requests.\");\n }\n\n async bulkRejectRequests(\n requests: Array<{ namespace: string; name: string }>,\n reviewedBy: string): Promise<Array<BulkOperationResult>> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithoutRetry(`${baseUrl}kuadrant/requests/bulk-reject`, {\n method: 'POST',\n body: JSON.stringify({requests, reviewedBy}),\n }, \"Failed to bulk reject APIKey requests\");\n }\n\n // ===== API Keys/Secrets Implementation =====\n\n async getApiKey(namespace: string, name: string): Promise<APIKey> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithRetry(\n `${baseUrl}kuadrant/apikeys/${namespace}/${name}`,\n \"Failed to fetch API Key.\"\n );\n }\n\n async getApiKeySecret(\n namespace: string,\n name: string,\n ): Promise<ExtractedSecret> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithRetry(\n `${baseUrl}kuadrant/apikeys/${namespace}/${name}/secret`,\n \"Failed to fetch API Key Secret.\"\n );\n }\n\n // ===== API Products Implementation =====\n\n async getApiProducts(): Promise<KuadrantList<APIProduct>> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithRetry(\n `${baseUrl}kuadrant/apiproducts`,\n \"Failed to fetch API Products.\"\n );\n }\n\n async getApiProduct(namespace: string, name: string): Promise<APIProduct> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithRetry(\n `${baseUrl}kuadrant/apiproducts/${namespace}/${name}`,\n \"Failed to fetch API Product.\"\n );\n }\n\n async createApiProduct(product: APIProduct): Promise<APIProduct> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithoutRetry(`${baseUrl}kuadrant/apiproducts`, {\n method: 'POST',\n body: JSON.stringify(product),\n }, \"Failed to create API Product.\");\n }\n\n async updateApiProduct(\n namespace: string,\n name: string,\n patch: Partial<APIProduct>,\n ): Promise<APIProduct> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithoutRetry(`${baseUrl}kuadrant/apiproducts/${namespace}/${name}`, {\n method: 'PATCH',\n body: JSON.stringify(patch),\n }, \"Failed to update API Product.\");\n }\n\n async deleteApiProduct(namespace: string, name: string): Promise<void> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithoutRetry(`${baseUrl}kuadrant/apiproducts/${namespace}/${name}`, {\n method: 'DELETE',\n }, \"Failed to delete API Product.\");\n }\n\n // ===== HTTP Routes =====\n\n async getHttpRoutes(): Promise<K8sList> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithRetry(\n `${baseUrl}kuadrant/httproutes`,\n \"Failed to fetch HTTPRoutes.\"\n );\n }\n\n async getHttpRoute(namespace: string, name: string): Promise<K8sResource> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithRetry(\n `${baseUrl}kuadrant/httproutes/${namespace}/${name}`,\n `Failed to fetc HTTPRoute ${namespace}/${name}.`\n );\n }\n\n // ===== Plan Policies Implementation =====\n\n async getPlanPolicies(): Promise<KuadrantList<PlanPolicy>> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithRetry(\n `${baseUrl}kuadrant/planpolicies`,\n \"Failed to fetch PlanPolicies.\"\n );\n }\n}\n","\"use strict\";\n\nvar _interopRequireDefault = require(\"@babel/runtime/helpers/interopRequireDefault\");\n\nvar _interopRequireWildcard = require(\"@babel/runtime/helpers/interopRequireWildcard\");\n\nObject.defineProperty(exports, \"__esModule\", {\n value: true\n});\nexports.default = void 0;\n\nvar React = _interopRequireWildcard(require(\"react\"));\n\nvar _createSvgIcon = _interopRequireDefault(require(\"./utils/createSvgIcon\"));\n\nvar _default = (0, _createSvgIcon.default)( /*#__PURE__*/React.createElement(\"path\", {\n d: \"M1 21h22L12 2 1 21zm12-3h-2v-2h2v2zm0-4h-2v-4h2v4z\"\n}), 'Warning');\n\nexports.default = _default;","\"use strict\";\nObject.defineProperty(exports, \"__esModule\", { value: true });\nvar tslib_1 = require(\"tslib\");\nvar react_1 = require(\"react\");\nvar useAsyncFn_1 = tslib_1.__importDefault(require(\"./useAsyncFn\"));\nfunction useAsync(fn, deps) {\n if (deps === void 0) { deps = []; }\n var _a = useAsyncFn_1.default(fn, deps, {\n loading: true,\n }), state = _a[0], callback = _a[1];\n react_1.useEffect(function () {\n callback();\n }, [callback]);\n return state;\n}\nexports.default = useAsync;\n"],"names":["Object","defineProperty","exports","value","tslib_1","react_1","useMountedState_1","__importDefault","fn","deps","initialState","loading","lastCallId","useRef","isMounted","default","_a","useState","state","set","callback","useCallback","args","_i","arguments","length","callId","current","prevState","__assign","apply","then","error","mountedRef","get","useEffect","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","values","some","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","currentValues","newValues","v","handleCheckboxChange","label","undefined","EditAPIKeyDialog","open","onClose","onSuccess","request","availablePlans","kuadrantApi","useApi","kuadrantApiRef","planTier","setPlanTier","useCase","setUseCase","saving","setSaving","setError","spec","handleClose","Dialog","maxWidth","fullWidth","DialogTitle","DialogContent","p","bgcolor","borderRadius","FormControl","margin","InputLabel","Select","e","target","disabled","plan","limitDesc","entries","limits","key","val","join","MenuItem","tier","TextField","placeholder","multiline","rows","helperText","DialogActions","async","patch","trim","updateRequest","metadata","name","namespace","err","console","Error","message","startIcon","CircularProgress","_interopRequireDefault","_interopRequireWildcard","_default","createElement","d","SimpleRequestAccessDialog","selectedProduct","config","configApiRef","fetchApi","fetchApiRef","alertApi","alertApiRef","identityApi","identityApiRef","backendUrl","getString","selectedApi","setSelectedApi","selectedTier","setSelectedTier","creating","setCreating","apiProducts","loadingProducts","useAsync","response","fetch","ok","json","items","publishStatus","userEmail","getBackstageIdentity","userEntityRef","split","find","availableTiers","status","discoveredPlans","required","labelId","data-testid","product","FormHelperText","inputProps","apiProductName","method","headers","body","JSON","stringify","errorData","statusText","requestBody","rawError","errorMsg","match","post","severity","errorMessage","container","minHeight","tableContainer","flex","overflow","useCasePanel","useCaseLabel","rejectedBanner","light","border","main","shape","gap","ExpandedRowContent","isRejected","phase","apiProductRef","stopPropagation","WarningIcon","Link","to","MyApiKeysTable","deleteDialogState","navigate","useNavigate","requestDialogOpen","setRequestDialogOpen","visibleKeys","setVisibleKeys","menuAnchor","setMenuAnchor","menuRequest","setMenuRequest","editDialogState","setEditDialogState","plans","refresh","setRefresh","deleting","setDeleting","setDeleteDialogState","apiKeyValues","setApiKeyValues","Map","apiKeyLoading","setApiKeyLoading","alreadyReadKeys","setAlreadyReadKeys","showOnceWarningOpen","setShowOnceWarningOpen","pendingKeyReveal","setPendingKeyReveal","optimisticallyDeleted","setOptimisticallyDeleted","setFilters","apiProduct","data","requestsData","productsData","Promise","all","getRequests","getApiProducts","requests","products","ownerMap","owner","annotations","allRequests","useMemo","r","filterSections","statusCounts","Approved","Pending","Rejected","apiProductCounts","tierCounts","Array","from","charAt","toUpperCase","slice","filteredRequests","toggleKeyVisibility","keyName","newSet","handleMenuClose","handleEdit","apiProductNamespace","getApiProduct","handleDeleteClick","columns","field","render","row","strong","displayOwner","replace","Chip","style","getMyApiKeysStatusChipStyle","filtering","hasSecretRef","secretRef","isVisible","isLoading","apiKeyValue","canReadSecret","isAlreadyRead","Tooltip","fontFamily","marginRight","VisibilityOffIcon","repeat","IconButton","navigator","clipboard","writeText","FileCopyIcon","requestNamespace","requestName","clearApiKeyValue","VisibilityIcon","creationTimestamp","date","Date","toLocaleDateString","DeleteIcon","detailPanelConfig","rowData","Progress","ResponseErrorPanel","px","textAlign","gutterBottom","Table","paging","pageSize","search","debounceInterval","toolbar","emptyRowsWhenPaging","item","detailPanel","Menu","Boolean","anchorReference","anchorPosition","top","left","push","ConfirmDeleteDialog","description","onConfirm","deleteRequest","onCancel","paragraph","result","getApiKeySecret","apiKey","fetchApiKeyFromSecret","MyApiKeysPage","Page","themeId","Header","subtitle","SupportButton","Content","useDefaultSupportConfig","t","useTranslationRef","url","icon","links","AppIcon","props","Fallback","rest","Icon","useApp","getSystemIcon","jsx","HelpIcon","popoverList","menuItem","whiteSpace","SupportIcon","app","SupportLink","link","children","SupportListItem","jsxs","button","ListItemIcon","ListItemText","primary","reduce","idx","configItems","useApiHolder","supportConfig","getOptionalConfig","defaultSupportConfig","getConfigArray","flatMap","itemConf","getOptionalString","getOptionalConfigArray","linkConf","useSupportConfig","popoverOpen","setPopoverOpen","anchorEl","setAnchorEl","isSmallScreen","useMediaQuery","breakpoints","down","onClickHandler","event","currentTarget","popoverCloseHandler","Fragment","ml","Popover","anchorOrigin","vertical","horizontal","transformOrigin","MenuList","autoFocusItem","Children","child","i","base","getApprovalQueueStatusChipStyle","getLifecycleChipStyle","lifecycle","confirmText","inputValue","setInputValue","requiresTextConfirmation","canConfirm","DialogContentText","autoFocus","createApiRef","useAsyncFn_1"],"sourceRoot":""}
1
+ {"version":3,"file":"static/7270.4a71807b.chunk.js","mappings":"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","sources":["webpack://internal.plugin-kuadrant/../../node_modules/react-use/lib/useAsyncFn.js","webpack://internal.plugin-kuadrant/../../node_modules/react-use/lib/useMountedState.js","webpack://internal.plugin-kuadrant/./src/components/FilterPanel/FilterPanel.tsx","webpack://internal.plugin-kuadrant/./src/components/EditAPIKeyDialog/EditAPIKeyDialog.tsx","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/icons/ExpandLess.js","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/icons/BrokenImage.js","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/icons/Visibility.js","webpack://internal.plugin-kuadrant/./src/components/SimpleRequestAccessDialog/SimpleRequestAccessDialog.tsx","webpack://internal.plugin-kuadrant/./src/components/MyApiKeysTable/MyApiKeysTable.tsx","webpack://internal.plugin-kuadrant/./src/components/MyApiKeysPage/MyApiKeysPage.tsx","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/icons/Delete.js","webpack://internal.plugin-kuadrant/./node_modules/@backstage/core-components/dist/hooks/useSupportConfig.esm.js","webpack://internal.plugin-kuadrant/./node_modules/@backstage/core-components/dist/icons/icons.esm.js","webpack://internal.plugin-kuadrant/./node_modules/@backstage/core-components/dist/components/SupportButton/SupportButton.esm.js","webpack://internal.plugin-kuadrant/./src/utils/styles.ts","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/icons/VisibilityOff.js","webpack://internal.plugin-kuadrant/./src/components/ConfirmDeleteDialog/ConfirmDeleteDialog.tsx","webpack://internal.plugin-kuadrant/./src/api.ts","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/icons/Warning.js","webpack://internal.plugin-kuadrant/../../node_modules/react-use/lib/useAsync.js"],"sourcesContent":["\"use strict\";\nObject.defineProperty(exports, \"__esModule\", { value: true });\nvar tslib_1 = require(\"tslib\");\nvar react_1 = require(\"react\");\nvar useMountedState_1 = tslib_1.__importDefault(require(\"./useMountedState\"));\nfunction useAsyncFn(fn, deps, initialState) {\n if (deps === void 0) { deps = []; }\n if (initialState === void 0) { initialState = { loading: false }; }\n var lastCallId = react_1.useRef(0);\n var isMounted = useMountedState_1.default();\n var _a = react_1.useState(initialState), state = _a[0], set = _a[1];\n var callback = react_1.useCallback(function () {\n var args = [];\n for (var _i = 0; _i < arguments.length; _i++) {\n args[_i] = arguments[_i];\n }\n var callId = ++lastCallId.current;\n if (!state.loading) {\n set(function (prevState) { return (tslib_1.__assign(tslib_1.__assign({}, prevState), { loading: true })); });\n }\n return fn.apply(void 0, args).then(function (value) {\n isMounted() && callId === lastCallId.current && set({ value: value, loading: false });\n return value;\n }, function (error) {\n isMounted() && callId === lastCallId.current && set({ error: error, loading: false });\n return error;\n });\n }, deps);\n return [state, callback];\n}\nexports.default = useAsyncFn;\n","\"use strict\";\nObject.defineProperty(exports, \"__esModule\", { value: true });\nvar react_1 = require(\"react\");\nfunction useMountedState() {\n var mountedRef = react_1.useRef(false);\n var get = react_1.useCallback(function () { return mountedRef.current; }, []);\n react_1.useEffect(function () {\n mountedRef.current = true;\n return function () {\n mountedRef.current = false;\n };\n }, []);\n return get;\n}\nexports.default = useMountedState;\n","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, { useState, useEffect } from \"react\";\nimport {\n Dialog,\n DialogTitle,\n DialogContent,\n DialogActions,\n Button,\n TextField,\n Box,\n Typography,\n FormControl,\n InputLabel,\n Select,\n MenuItem,\n CircularProgress,\n} from \"@material-ui/core\";\nimport { useApi } from \"@backstage/core-plugin-api\";\nimport { kuadrantApiRef } from '../../api';\nimport { APIKey } from \"../../types/api-management\";\n\ninterface EditAPIKeyDialogProps {\n open: boolean;\n onClose: () => void;\n onSuccess: () => void;\n request: APIKey;\n availablePlans: Array<{\n tier: string;\n description?: string;\n limits?: any;\n }>;\n}\n\nexport const EditAPIKeyDialog = ({\n open,\n onClose,\n onSuccess,\n request,\n availablePlans,\n}: EditAPIKeyDialogProps) => {\n const kuadrantApi = useApi(kuadrantApiRef);\n\n const [planTier, setPlanTier] = useState(\"\");\n const [useCase, setUseCase] = useState(\"\");\n const [saving, setSaving] = useState(false);\n const [error, setError] = useState(\"\");\n\n useEffect(() => {\n if (open && request) {\n setPlanTier(request.spec.planTier || \"\");\n setUseCase(request.spec.useCase || \"\");\n setError(\"\");\n }\n }, [open, request]);\n\n const handleSave = async () => {\n if (!planTier) {\n setError(\"Please select a tier\");\n return;\n }\n\n setError(\"\");\n setSaving(true);\n\n try {\n const patch = {\n spec: {\n planTier,\n useCase: useCase.trim(),\n },\n };\n\n await kuadrantApi.updateRequest(\n request.metadata.name,\n request.metadata.namespace,\n // @ts-ignore Applying a partial obj\n patch,\n );\n\n onSuccess();\n onClose();\n } catch (err) {\n console.error(\"Error updating API key request:\", err);\n setError(err instanceof Error ? err.message : \"Unknown error occurred\");\n } finally {\n setSaving(false);\n }\n };\n\n const handleClose = () => {\n if (!saving) {\n setError(\"\");\n onClose();\n }\n };\n\n return (\n <Dialog open={open} onClose={handleClose} maxWidth=\"sm\" fullWidth>\n <DialogTitle>Edit API Key</DialogTitle>\n <DialogContent>\n {error && (\n <Box\n mb={2}\n p={2}\n bgcolor=\"error.main\"\n color=\"error.contrastText\"\n borderRadius={1}\n >\n <Typography variant=\"body2\">{error}</Typography>\n </Box>\n )}\n\n <FormControl fullWidth margin=\"normal\">\n <InputLabel>Tier</InputLabel>\n <Select\n value={planTier}\n onChange={(e) => setPlanTier(e.target.value as string)}\n disabled={saving}\n >\n {availablePlans.map((plan) => {\n const limitDesc = Object.entries(plan.limits || {})\n .map(([key, val]) => `${val} per ${key}`)\n .join(\", \");\n return (\n <MenuItem key={plan.tier} value={plan.tier}>\n {plan.tier} {limitDesc ? `(${limitDesc})` : \"\"}\n </MenuItem>\n );\n })}\n </Select>\n </FormControl>\n\n <TextField\n label=\"Use Case\"\n placeholder=\"Describe how you plan to use this API\"\n multiline\n rows={3}\n fullWidth\n margin=\"normal\"\n value={useCase}\n onChange={(e) => setUseCase(e.target.value)}\n disabled={saving}\n helperText=\"Explain your intended use of this API for admin review\"\n />\n </DialogContent>\n <DialogActions>\n <Button onClick={handleClose} disabled={saving}>\n Cancel\n </Button>\n <Button\n onClick={handleSave}\n color=\"primary\"\n variant=\"contained\"\n disabled={!planTier || saving}\n startIcon={\n saving ? <CircularProgress size={16} color=\"inherit\" /> : undefined\n }\n >\n {saving ? \"Saving...\" : \"Save Changes\"}\n </Button>\n </DialogActions>\n </Dialog>\n );\n};\n","\"use strict\";\n\nvar _interopRequireDefault = require(\"@babel/runtime/helpers/interopRequireDefault\");\n\nvar _interopRequireWildcard = require(\"@babel/runtime/helpers/interopRequireWildcard\");\n\nObject.defineProperty(exports, \"__esModule\", {\n value: true\n});\nexports.default = void 0;\n\nvar React = _interopRequireWildcard(require(\"react\"));\n\nvar _createSvgIcon = _interopRequireDefault(require(\"./utils/createSvgIcon\"));\n\nvar _default = (0, _createSvgIcon.default)( /*#__PURE__*/React.createElement(\"path\", {\n d: \"M12 8l-6 6 1.41 1.41L12 10.83l4.59 4.58L18 14z\"\n}), 'ExpandLess');\n\nexports.default = _default;","\"use strict\";\n\nvar _interopRequireDefault = require(\"@babel/runtime/helpers/interopRequireDefault\");\n\nvar _interopRequireWildcard = require(\"@babel/runtime/helpers/interopRequireWildcard\");\n\nObject.defineProperty(exports, \"__esModule\", {\n value: true\n});\nexports.default = void 0;\n\nvar React = _interopRequireWildcard(require(\"react\"));\n\nvar _createSvgIcon = _interopRequireDefault(require(\"./utils/createSvgIcon\"));\n\nvar _default = (0, _createSvgIcon.default)( /*#__PURE__*/React.createElement(\"path\", {\n d: \"M21 5v6.59l-3-3.01-4 4.01-4-4-4 4-3-3.01V5c0-1.1.9-2 2-2h14c1.1 0 2 .9 2 2zm-3 6.42l3 3.01V19c0 1.1-.9 2-2 2H5c-1.1 0-2-.9-2-2v-6.58l3 2.99 4-4 4 4 4-3.99z\"\n}), 'BrokenImage');\n\nexports.default = _default;","\"use strict\";\n\nvar _interopRequireDefault = require(\"@babel/runtime/helpers/interopRequireDefault\");\n\nvar _interopRequireWildcard = require(\"@babel/runtime/helpers/interopRequireWildcard\");\n\nObject.defineProperty(exports, \"__esModule\", {\n value: true\n});\nexports.default = void 0;\n\nvar React = _interopRequireWildcard(require(\"react\"));\n\nvar _createSvgIcon = _interopRequireDefault(require(\"./utils/createSvgIcon\"));\n\nvar _default = (0, _createSvgIcon.default)( /*#__PURE__*/React.createElement(\"path\", {\n d: \"M12 4.5C7 4.5 2.73 7.61 1 12c1.73 4.39 6 7.5 11 7.5s9.27-3.11 11-7.5c-1.73-4.39-6-7.5-11-7.5zM12 17c-2.76 0-5-2.24-5-5s2.24-5 5-5 5 2.24 5 5-2.24 5-5 5zm0-8c-1.66 0-3 1.34-3 3s1.34 3 3 3 3-1.34 3-3-1.34-3-3-3z\"\n}), 'Visibility');\n\nexports.default = _default;","import React, { useState } from 'react';\nimport {\n Dialog,\n DialogTitle,\n DialogContent,\n DialogActions,\n Button,\n TextField,\n Select,\n MenuItem,\n FormControl,\n InputLabel,\n Typography,\n CircularProgress,\n FormHelperText,\n} from '@material-ui/core';\nimport {\n useApi,\n configApiRef,\n fetchApiRef,\n alertApiRef,\n identityApiRef,\n} from '@backstage/core-plugin-api';\nimport useAsync from 'react-use/lib/useAsync';\n\nexport interface SimpleRequestAccessDialogProps {\n open: boolean;\n onClose: () => void;\n onSuccess: () => void;\n}\n\ntype ApiProduct = {\n metadata: {\n name: string;\n namespace: string;\n };\n spec?: {\n publishStatus?: string;\n };\n status?: {\n discoveredPlans?: Array<{\n tier: string;\n limits?: Record<string, number>;\n }>;\n };\n};\n\nexport const SimpleRequestAccessDialog = ({\n open,\n onClose,\n onSuccess,\n}: SimpleRequestAccessDialogProps) => {\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\n const [selectedApi, setSelectedApi] = useState('');\n const [selectedTier, setSelectedTier] = useState('');\n const [useCase, setUseCase] = useState('');\n const [creating, setCreating] = useState(false);\n\n // Fetch all published API products\n const {\n value: apiProducts,\n loading: loadingProducts,\n } = useAsync(async () => {\n if (!open) return [];\n const response = await fetchApi.fetch(\n `${backendUrl}/api/kuadrant/apiproducts`,\n );\n if (!response.ok) {\n throw new Error('Failed to fetch API products');\n }\n const data = await response.json();\n // Filter to only show Published products\n return (data.items || []).filter(\n (p: ApiProduct) => p.spec?.publishStatus === 'Published',\n );\n }, [backendUrl, fetchApi, open]);\n\n // Get user email\n const { value: userEmail } = useAsync(async () => {\n const identity = await identityApi.getBackstageIdentity();\n\n // Extract username from userEntityRef (e.g., \"user:default/john\" -> \"john\")\n const username = identity.userEntityRef.split('/')[1] || 'unknown';\n\n // Construct email from username\n return `${username}@example.com`;\n }, [identityApi]);\n\n // Get available tiers for selected API\n const selectedProduct = apiProducts?.find(\n (p: ApiProduct) =>\n `${p.metadata.namespace}/${p.metadata.name}` === selectedApi,\n );\n const availableTiers = selectedProduct?.status?.discoveredPlans || [];\n\n const handleClose = () => {\n setSelectedApi('');\n setSelectedTier('');\n setUseCase('');\n onClose();\n };\n\n const handleSubmit = async () => {\n if (!selectedApi || !selectedTier) return;\n\n const [namespace, apiProductName] = selectedApi.split('/');\n\n setCreating(true);\n try {\n const response = await fetchApi.fetch(\n `${backendUrl}/api/kuadrant/requests`,\n {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n },\n body: JSON.stringify({\n apiProductName,\n namespace,\n planTier: selectedTier,\n useCase: useCase.trim() || '',\n userEmail: userEmail || 'unknown',\n }),\n },\n );\n\n if (!response.ok) {\n let errorData;\n try {\n errorData = await response.json();\n } catch {\n errorData = { error: `Server returned ${response.status} ${response.statusText}` };\n }\n\n console.error('Backend error response:', {\n status: response.status,\n statusText: response.statusText,\n errorData,\n requestBody: {\n apiProductName,\n namespace,\n planTier: selectedTier,\n useCase: useCase.trim() || '',\n userEmail: userEmail || 'unknown',\n }\n });\n\n // Extract user-friendly error message\n const rawError = errorData.error || errorData.message || `Server returned ${response.status}`;\n\n // Try to extract validation error details from Kubernetes error messages\n // Example: \"failed to create apikeys: APIKey.devportal.kuadrant.io \"name\" is invalid: spec.requestedBy.email: Invalid value: \"admin\": spec.requestedBy.email in body should match '^[a-zA-Z0-9._%+-]+@[a-zA-Z0-9.-]+\\.[a-zA-Z]{2,}$'\"\n let errorMsg = rawError;\n\n if (rawError.includes('spec.requestedBy.email')) {\n errorMsg = 'Invalid email format. Please contact your administrator to update your profile email.';\n } else if (rawError.includes('is invalid:')) {\n // Extract the validation message after \"is invalid:\"\n const match = rawError.match(/is invalid: (.+?)(?:\\s+\\(|$)/);\n if (match) {\n errorMsg = `Validation error: ${match[1]}`;\n }\n } else if (response.status === 403) {\n errorMsg = 'You do not have permission to request access to this API.';\n } else if (response.status === 404) {\n errorMsg = 'The selected API or tier was not found.';\n } else if (response.status >= 500) {\n errorMsg = `Server error: ${rawError}`;\n }\n\n throw new Error(errorMsg);\n }\n\n alertApi.post({\n message: 'API key requested successfully',\n severity: 'success',\n display: 'transient',\n });\n\n handleClose();\n onSuccess();\n } catch (err) {\n const errorMessage =\n err instanceof Error ? err.message : 'unknown error occurred';\n console.error('Failed to create API key request:', {\n error: err,\n selectedApi,\n selectedTier,\n userEmail,\n });\n alertApi.post({\n message: `Failed to request API key: ${errorMessage}`,\n severity: 'error',\n display: 'permanent',\n });\n } finally {\n setCreating(false);\n }\n };\n\n return (\n <Dialog open={open} onClose={handleClose} maxWidth=\"sm\" fullWidth>\n <DialogTitle>Request API key</DialogTitle>\n <DialogContent>\n <Typography variant=\"body2\" color=\"textSecondary\">\n Submit your request to generate an API key.\n </Typography>\n\n <FormControl fullWidth margin=\"normal\" disabled={creating || loadingProducts}>\n <InputLabel id=\"api-select-label\" required>\n API\n </InputLabel>\n <Select\n labelId=\"api-select-label\"\n value={selectedApi}\n onChange={e => {\n setSelectedApi(e.target.value as string);\n setSelectedTier(''); // Reset tier when API changes\n }}\n disabled={creating || loadingProducts}\n data-testid=\"api-select\"\n >\n {loadingProducts ? (\n <MenuItem disabled>Loading...</MenuItem>\n ) : (\n apiProducts?.map((product: ApiProduct) => (\n <MenuItem\n key={`${product.metadata.namespace}/${product.metadata.name}`}\n value={`${product.metadata.namespace}/${product.metadata.name}`}\n >\n {product.metadata.name}\n </MenuItem>\n ))\n )}\n </Select>\n <FormHelperText>\n Select one API. Please submit separate requests for multiple APIs.\n </FormHelperText>\n </FormControl>\n\n <FormControl\n fullWidth\n margin=\"normal\"\n disabled={creating || !selectedApi}\n required\n >\n <InputLabel id=\"tier-select-label\">Tiers</InputLabel>\n <Select\n labelId=\"tier-select-label\"\n value={selectedTier}\n onChange={e => setSelectedTier(e.target.value as string)}\n disabled={creating || !selectedApi}\n data-testid=\"tier-select\"\n >\n {!selectedApi ? (\n <MenuItem disabled>Select an API first</MenuItem>\n ) : availableTiers.length === 0 ? (\n <MenuItem disabled>No tiers available</MenuItem>\n ) : (\n availableTiers.map((plan: { tier: string; limits?: Record<string, number> }) => {\n const limitDesc = Object.entries(plan.limits || {})\n .map(([key, val]) => `${val} per ${key}`)\n .join(', ');\n return (\n <MenuItem key={plan.tier} value={plan.tier}>\n {plan.tier} {limitDesc ? `(${limitDesc})` : ''}\n </MenuItem>\n );\n })\n )}\n </Select>\n <FormHelperText>Select an API to view available tiers.</FormHelperText>\n </FormControl>\n\n <TextField\n label=\"Use case\"\n placeholder=\"Briefly describe your specific use case of using this API key\"\n multiline\n rows={2}\n fullWidth\n margin=\"normal\"\n value={useCase}\n onChange={e => setUseCase(e.target.value)}\n disabled={creating}\n inputProps={{ 'data-testid': 'usecase-input' }}\n />\n </DialogContent>\n <DialogActions>\n <Button onClick={handleClose} disabled={creating} data-testid=\"cancel-button\">\n Cancel\n </Button>\n <Button\n onClick={handleSubmit}\n color=\"primary\"\n variant=\"contained\"\n disabled={!selectedApi || !selectedTier || creating}\n startIcon={\n creating ? (\n <CircularProgress size={16} color=\"inherit\" />\n ) : undefined\n }\n data-testid=\"submit-button\"\n >\n {creating ? 'Submitting...' : 'Request'}\n </Button>\n </DialogActions>\n </Dialog>\n );\n};\n","import React, { useState, useMemo } from \"react\";\nimport { useNavigate } from \"react-router-dom\";\nimport {\n Table,\n TableColumn,\n Link,\n Progress,\n ResponseErrorPanel,\n} from \"@backstage/core-components\";\nimport {\n useApi,\n alertApiRef,\n} from \"@backstage/core-plugin-api\";\nimport { kuadrantApiRef } from '../../api';\nimport useAsync from \"react-use/lib/useAsync\";\nimport {\n Box,\n Chip,\n Typography,\n IconButton,\n Tooltip,\n Menu,\n MenuItem,\n CircularProgress,\n Dialog,\n DialogTitle,\n DialogContent,\n DialogActions,\n Button,\n makeStyles,\n} from \"@material-ui/core\";\nimport { SimpleRequestAccessDialog } from \"../SimpleRequestAccessDialog\";\nimport VisibilityIcon from \"@material-ui/icons/Visibility\";\nimport VisibilityOffIcon from \"@material-ui/icons/VisibilityOff\";\nimport FileCopyIcon from \"@material-ui/icons/FileCopy\";\nimport WarningIcon from \"@material-ui/icons/Warning\";\nimport DeleteIcon from \"@material-ui/icons/Delete\";\nimport { EditAPIKeyDialog } from \"../EditAPIKeyDialog\";\nimport { ConfirmDeleteDialog } from \"../ConfirmDeleteDialog\";\nimport { FilterPanel, FilterSection, FilterState } from \"../FilterPanel\";\nimport { APIKey, APIProduct } from \"../../types/api-management\";\nimport { getMyApiKeysStatusChipStyle } from \"../../utils/styles\";\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 useCasePanel: {\n padding: theme.spacing(2),\n backgroundColor: theme.palette.background.default,\n },\n useCaseLabel: {\n fontWeight: 600,\n marginBottom: theme.spacing(1),\n color: theme.palette.text.secondary,\n textTransform: \"uppercase\",\n fontSize: \"0.75rem\",\n },\n rejectedBanner: {\n backgroundColor: theme.palette.error.light,\n border: `1px solid ${theme.palette.error.main}`,\n borderRadius: theme.shape.borderRadius,\n padding: theme.spacing(1.5, 2),\n marginBottom: theme.spacing(2),\n display: \"flex\",\n alignItems: \"center\",\n gap: theme.spacing(1),\n },\n}));\n\ninterface ExpandedRowProps {\n request: APIKey;\n}\n\nconst ExpandedRowContent = ({ request }: ExpandedRowProps) => {\n const classes = useStyles();\n const isRejected = request.status?.phase === \"Rejected\";\n const apiProductName = request.spec.apiProductRef?.name || \"unknown\";\n\n return (\n <Box className={classes.useCasePanel} onClick={(e) => e.stopPropagation()}>\n {isRejected && (\n <Box className={classes.rejectedBanner}>\n <WarningIcon color=\"error\" fontSize=\"small\" />\n <Typography variant=\"body2\">\n This API key was rejected.{\" \"}\n <Link to={`/catalog/default/api/${apiProductName}/api-keys`}>\n Request a new API key\n </Link>\n </Typography>\n </Box>\n )}\n <Typography className={classes.useCaseLabel}>Use Case</Typography>\n <Typography variant=\"body2\">\n {request.spec.useCase || \"No use case provided\"}\n </Typography>\n </Box>\n );\n};\n\nexport const MyApiKeysTable = () => {\n const classes = useStyles();\n const navigate = useNavigate();\n const kuadrantApi = useApi(kuadrantApiRef);\n const alertApi = useApi(alertApiRef);\n\n const [requestDialogOpen, setRequestDialogOpen] = useState(false);\n const [visibleKeys, setVisibleKeys] = useState<Set<string>>(new Set());\n const [menuAnchor, setMenuAnchor] = useState<{\n top: number;\n left: number;\n } | null>(null);\n const [menuRequest, setMenuRequest] = useState<APIKey | null>(null);\n const [editDialogState, setEditDialogState] = useState<{\n open: boolean;\n request: APIKey | null;\n plans: any[];\n }>({ open: false, request: null, plans: [] });\n const [refresh, setRefresh] = useState(0);\n const [deleting, setDeleting] = useState<string | null>(null);\n const [deleteDialogState, setDeleteDialogState] = useState<{\n open: boolean;\n request: APIKey | null;\n }>({ open: false, request: null });\n const [apiKeyValues, setApiKeyValues] = useState<Map<string, string>>(\n new Map(),\n );\n const [apiKeyLoading, setApiKeyLoading] = useState<Set<string>>(new Set());\n const [alreadyReadKeys, setAlreadyReadKeys] = useState<Set<string>>(\n new Set(),\n );\n const [showOnceWarningOpen, setShowOnceWarningOpen] = useState(false);\n const [pendingKeyReveal, setPendingKeyReveal] = useState<{\n namespace: string;\n name: string;\n } | null>(null);\n const [optimisticallyDeleted, setOptimisticallyDeleted] = useState<\n Set<string>\n >(new Set());\n\n const [filters, setFilters] = useState<FilterState>({\n status: [],\n apiProduct: [],\n tier: [],\n });\n\n const {\n value: data,\n loading,\n error,\n } = useAsync(async () => {\n const [requestsData, productsData] = await Promise.all([\n kuadrantApi.getRequests(),\n kuadrantApi.getApiProducts(),\n ]);\n\n const requests: APIKey[] = requestsData.items || [];\n const products: APIProduct[] = productsData.items || [];\n\n // build owner map from products\n const ownerMap = new Map<string, string>();\n products.forEach((p) => {\n const key = `${p.metadata.namespace}/${p.metadata.name}`;\n const owner = p.metadata.annotations?.[\"backstage.io/owner\"] || \"unknown\";\n ownerMap.set(key, owner);\n });\n\n return { requests, products, ownerMap };\n }, [kuadrantApi, refresh]);\n\n const allRequests = useMemo(() => {\n if (!data?.requests) return [];\n return data.requests.filter(\n (r: APIKey) => !optimisticallyDeleted.has(r.metadata.name),\n );\n }, [data?.requests, optimisticallyDeleted]);\n\n // filter options from data\n const filterSections: FilterSection[] = useMemo(() => {\n const statusCounts = { Approved: 0, Pending: 0, Rejected: 0 };\n const apiProductCounts = new Map<string, number>();\n const tierCounts = new Map<string, number>();\n\n allRequests.forEach((r: APIKey) => {\n const status = r.status?.phase || \"Pending\";\n statusCounts[status as keyof typeof statusCounts]++;\n\n const apiProduct = r.spec.apiProductRef?.name || \"unknown\";\n apiProductCounts.set(\n apiProduct,\n (apiProductCounts.get(apiProduct) || 0) + 1,\n );\n\n const tier = r.spec.planTier || \"unknown\";\n tierCounts.set(tier, (tierCounts.get(tier) || 0) + 1);\n });\n\n return [\n {\n id: \"status\",\n title: \"Status\",\n options: [\n { value: \"Approved\", label: \"Active\", count: statusCounts.Approved },\n { value: \"Pending\", label: \"Pending\", count: statusCounts.Pending },\n {\n value: \"Rejected\",\n label: \"Rejected\",\n count: statusCounts.Rejected,\n },\n ],\n },\n {\n id: \"apiProduct\",\n title: \"API Product\",\n options: Array.from(apiProductCounts.entries()).map(\n ([name, count]) => ({\n value: name,\n label: name,\n count,\n }),\n ),\n collapsed: apiProductCounts.size > 5,\n },\n {\n id: \"tier\",\n title: \"Tier\",\n options: Array.from(tierCounts.entries()).map(([tier, count]) => ({\n value: tier,\n label: tier.charAt(0).toUpperCase() + tier.slice(1),\n count,\n })),\n },\n ];\n }, [allRequests]);\n\n // filtered requests\n const filteredRequests = useMemo(() => {\n return allRequests.filter((r: APIKey) => {\n // status filter\n if (filters.status.length > 0) {\n const status = r.status?.phase || \"Pending\";\n if (!filters.status.includes(status)) return false;\n }\n\n // api product filter\n if (filters.apiProduct.length > 0) {\n const apiProduct = r.spec.apiProductRef?.name || \"unknown\";\n if (!filters.apiProduct.includes(apiProduct)) return false;\n }\n\n // tier filter\n if (filters.tier.length > 0) {\n const tier = r.spec.planTier || \"unknown\";\n if (!filters.tier.includes(tier)) return false;\n }\n\n return true;\n });\n }, [allRequests, filters]);\n\n const toggleKeyVisibility = (keyName: string) => {\n setVisibleKeys((prev) => {\n const newSet = new Set(prev);\n if (newSet.has(keyName)) {\n newSet.delete(keyName);\n } else {\n newSet.add(keyName);\n }\n return newSet;\n });\n };\n\n const fetchApiKeyFromSecret = async (\n requestNamespace: string,\n requestName: string,\n ) => {\n const key = `${requestNamespace}/${requestName}`;\n if (apiKeyLoading.has(key)) return;\n\n setApiKeyLoading((prev) => new Set(prev).add(key));\n try {\n const result = await kuadrantApi.getApiKeySecret(requestNamespace, requestName);\n setApiKeyValues((prev) => new Map(prev).set(key, result.apiKey));\n setAlreadyReadKeys((prev) => new Set(prev).add(key));\n } catch (err) {\n const errorMessage =\n err instanceof Error ? err.message : \"unknown error occurred\";\n if (errorMessage.includes(\"403\") || errorMessage.includes(\"already been viewed\")) {\n setAlreadyReadKeys((prev) => new Set(prev).add(key));\n alertApi.post({\n message:\n \"This API key has already been viewed and cannot be retrieved again.\",\n severity: \"warning\",\n display: \"transient\",\n });\n } else {\n alertApi.post({\n message: `Failed to fetch api key: ${errorMessage}`,\n severity: \"error\",\n display: \"transient\",\n });\n }\n } finally {\n setApiKeyLoading((prev) => {\n const next = new Set(prev);\n next.delete(key);\n return next;\n });\n }\n };\n\n const clearApiKeyValue = (requestNamespace: string, requestName: string) => {\n const key = `${requestNamespace}/${requestName}`;\n setApiKeyValues((prev) => {\n const next = new Map(prev);\n next.delete(key);\n return next;\n });\n };\n\n const handleMenuClose = () => {\n setMenuAnchor(null);\n setMenuRequest(null);\n };\n\n const handleEdit = async () => {\n if (!menuRequest) return;\n\n const request = menuRequest;\n handleMenuClose();\n\n try {\n const apiProductName = request.spec.apiProductRef?.name;\n const apiProductNamespace = request.metadata.namespace;\n const apiProduct = await kuadrantApi.getApiProduct(apiProductNamespace, apiProductName);\n const plans = apiProduct.status?.discoveredPlans || [];\n setEditDialogState({ open: true, request, plans });\n } catch (err) {\n console.error(\"Error fetching plans:\", err);\n setEditDialogState({ open: true, request: menuRequest, plans: [] });\n const errorMessage = err instanceof Error ? err.message : \"unknown error occurred\";\n alertApi.post({\n message: `Failed to fetch Plans. ${errorMessage}`,\n severity: 'error',\n display: 'transient',\n });\n }\n };\n\n const handleDeleteClick = () => {\n if (!menuRequest) return;\n const request = menuRequest;\n handleMenuClose();\n setDeleteDialogState({ open: true, request });\n };\n\n const handleDeleteConfirm = async () => {\n if (!deleteDialogState.request) return;\n\n const request = deleteDialogState.request;\n const requestName = request.metadata.name;\n\n setOptimisticallyDeleted((prev) => new Set(prev).add(requestName));\n setDeleting(requestName);\n\n try {\n await kuadrantApi.deleteRequest(request.metadata.namespace, request.metadata.name);\n setRefresh((r) => r + 1);\n alertApi.post({\n message: \"API key deleted\",\n severity: \"success\",\n display: \"transient\",\n });\n setDeleteDialogState({ open: false, request: null });\n } catch (err) {\n console.error(\"Error deleting request:\", err);\n setOptimisticallyDeleted((prev) => {\n const next = new Set(prev);\n next.delete(requestName);\n return next;\n });\n const errorMessage = err instanceof Error ? err.message : \"unknown error occurred\";\n alertApi.post({\n message: `Failed to delete APIKey. ${errorMessage}`,\n severity: 'error',\n display: 'transient',\n });\n } finally {\n setDeleting(null);\n }\n };\n\n const handleDeleteCancel = () => {\n setDeleteDialogState({ open: false, request: null });\n };\n\n const columns: TableColumn<APIKey>[] = [\n {\n title: \"API Product\",\n field: \"spec.apiProductRef.name\",\n render: (row: APIKey) => {\n const apiProductName = row.spec.apiProductRef?.name || \"unknown\";\n return (\n <Link to={`/catalog/default/api/${apiProductName}/api-keys`}>\n <strong>{apiProductName}</strong>\n </Link>\n );\n },\n },\n {\n title: \"Owner\",\n field: \"owner\",\n render: (row: APIKey) => {\n const key = `${row.metadata.namespace}/${row.spec.apiProductRef?.name}`;\n const owner = data?.ownerMap?.get(key) || \"unknown\";\n // strip 'user:default/' prefix if present\n const displayOwner = owner.replace(/^user:default\\//, \"\");\n return <Typography variant=\"body2\">{displayOwner}</Typography>;\n },\n },\n {\n title: \"Status\",\n field: \"status.phase\",\n render: (row: APIKey) => {\n const phase = row.status?.phase || \"Pending\";\n const label = phase === \"Approved\" ? \"Active\" : phase;\n return (\n <Chip label={label} size=\"small\" style={getMyApiKeysStatusChipStyle(phase)} />\n );\n },\n },\n {\n title: \"Tier\",\n field: \"spec.planTier\",\n render: (row: APIKey) => (\n <Chip label={row.spec.planTier} size=\"small\" variant=\"outlined\" />\n ),\n },\n {\n title: \"API Key\",\n field: \"status.secretRef\",\n filtering: false,\n render: (row: APIKey) => {\n if (row.status?.phase !== \"Approved\") {\n return (\n <Typography variant=\"body2\" color=\"textSecondary\">\n -\n </Typography>\n );\n }\n\n const key = `${row.metadata.namespace}/${row.metadata.name}`;\n const hasSecretRef = row.status?.secretRef?.name;\n const isVisible = visibleKeys.has(row.metadata.name);\n const isLoading = apiKeyLoading.has(key);\n const apiKeyValue = apiKeyValues.get(key);\n const canReadSecret = row.status?.canReadSecret !== false;\n const isAlreadyRead = alreadyReadKeys.has(key) || !canReadSecret;\n\n if (!hasSecretRef) {\n return (\n <Typography variant=\"body2\" color=\"textSecondary\">\n Awaiting secret...\n </Typography>\n );\n }\n\n if (isAlreadyRead && !apiKeyValue) {\n return (\n <Tooltip title=\"This API key has already been viewed and cannot be retrieved again\">\n <Box display=\"flex\" alignItems=\"center\">\n <Typography\n variant=\"body2\"\n color=\"textSecondary\"\n style={{ fontFamily: \"monospace\", marginRight: 8 }}\n >\n Already viewed\n </Typography>\n <VisibilityOffIcon fontSize=\"small\" color=\"disabled\" />\n </Box>\n </Tooltip>\n );\n }\n\n const handleRevealClick = () => {\n if (isVisible) {\n clearApiKeyValue(row.metadata.namespace, row.metadata.name);\n toggleKeyVisibility(row.metadata.name);\n } else if (!isAlreadyRead) {\n setPendingKeyReveal({\n namespace: row.metadata.namespace,\n name: row.metadata.name,\n });\n setShowOnceWarningOpen(true);\n }\n };\n\n const handleCopy = async () => {\n if (apiKeyValue) {\n await navigator.clipboard.writeText(apiKeyValue);\n alertApi.post({\n message: \"API key copied to clipboard\",\n severity: \"success\",\n display: \"transient\",\n });\n }\n };\n\n return (\n <Box display=\"flex\" alignItems=\"center\" style={{ gap: 8 }}>\n <Box fontFamily=\"monospace\" fontSize=\"0.875rem\">\n {isLoading\n ? \"Loading...\"\n : isVisible && apiKeyValue\n ? apiKeyValue\n : \"•\".repeat(20) + \"...\"}\n </Box>\n {isVisible && apiKeyValue && (\n <Tooltip title=\"Copy to clipboard\">\n <IconButton size=\"small\" onClick={handleCopy}>\n <FileCopyIcon fontSize=\"small\" />\n </IconButton>\n </Tooltip>\n )}\n <Tooltip\n title={\n isVisible ? \"Hide API key\" : \"Reveal API key (one-time only)\"\n }\n >\n <span>\n <IconButton\n size=\"small\"\n onClick={handleRevealClick}\n disabled={isLoading || (isAlreadyRead && !apiKeyValue)}\n >\n {isVisible ? (\n <VisibilityOffIcon fontSize=\"small\" />\n ) : (\n <VisibilityIcon fontSize=\"small\" />\n )}\n </IconButton>\n </span>\n </Tooltip>\n </Box>\n );\n },\n },\n {\n title: \"Requested\",\n field: \"metadata.creationTimestamp\",\n render: (row: APIKey) => {\n if (!row.metadata.creationTimestamp) {\n return <Typography variant=\"body2\">-</Typography>;\n }\n const date = new Date(row.metadata.creationTimestamp);\n return (\n <Typography variant=\"body2\">{date.toLocaleDateString()}</Typography>\n );\n },\n },\n {\n title: \"Actions\",\n filtering: false,\n width: \"100px\",\n render: (row: APIKey) => {\n const isDeleting = deleting === row.metadata.name;\n if (isDeleting) {\n return <CircularProgress size={20} />;\n }\n return (\n <Box display=\"flex\" style={{ gap: 4 }}>\n <Tooltip title=\"View details\">\n <IconButton\n size=\"small\"\n onClick={(e) => {\n e.stopPropagation();\n navigate(\n `/kuadrant/api-keys/${row.metadata.namespace}/${row.metadata.name}`,\n );\n }}\n >\n <VisibilityIcon fontSize=\"small\" />\n </IconButton>\n </Tooltip>\n <Tooltip title=\"Delete\">\n <IconButton\n size=\"small\"\n onClick={(e) => {\n e.stopPropagation();\n setDeleteDialogState({ open: true, request: row });\n }}\n >\n <DeleteIcon fontSize=\"small\" />\n </IconButton>\n </Tooltip>\n </Box>\n );\n },\n },\n ];\n\n const detailPanelConfig = useMemo(\n () => [\n {\n render: (data: any) => {\n const request = data.rowData as APIKey;\n if (!request?.metadata?.name) {\n return <Box />;\n }\n return <ExpandedRowContent request={request} />;\n },\n },\n ],\n [],\n );\n\n if (loading) {\n return <Progress />;\n }\n\n if (error) {\n return <ResponseErrorPanel error={error} />;\n }\n\n const isPending = (row: APIKey) =>\n !row.status || row.status.phase === \"Pending\";\n\n return (\n <>\n <Box display=\"flex\" justifyContent=\"flex-end\" mb={2} px={2}>\n <Button\n variant=\"contained\"\n color=\"primary\"\n onClick={() => setRequestDialogOpen(true)}\n data-testid=\"request-access-button\"\n >\n Request Access\n </Button>\n </Box>\n <Box className={classes.container}>\n <FilterPanel\n sections={filterSections}\n filters={filters}\n onChange={setFilters}\n />\n <Box className={classes.tableContainer}>\n {allRequests.length === 0 ? (\n <Box p={4} textAlign=\"center\">\n <Typography variant=\"body1\" color=\"textSecondary\" gutterBottom>\n No API keys found. Request access to an API to get started.\n </Typography>\n </Box>\n ) : filteredRequests.length === 0 ? (\n <Box p={4} textAlign=\"center\">\n <Typography variant=\"body1\" color=\"textSecondary\">\n No API keys match the selected filters.\n </Typography>\n </Box>\n ) : (\n <Table\n options={{\n paging: filteredRequests.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={filteredRequests.map((item: APIKey) => ({\n ...item,\n id: item.metadata.name,\n }))}\n detailPanel={detailPanelConfig}\n />\n )}\n </Box>\n </Box>\n\n <Menu\n id=\"myapikeys-menu\"\n open={Boolean(menuAnchor)}\n onClose={handleMenuClose}\n anchorReference=\"anchorPosition\"\n anchorPosition={menuAnchor || { top: 0, left: 0 }}\n >\n {menuRequest &&\n (() => {\n const items = [];\n items.push(\n <MenuItem\n key=\"view\"\n onClick={() => {\n navigate(\n `/kuadrant/api-keys/${menuRequest.metadata.namespace}/${menuRequest.metadata.name}`,\n );\n handleMenuClose();\n }}\n >\n View Details\n </MenuItem>,\n );\n if (isPending(menuRequest)) {\n items.push(\n <MenuItem key=\"edit\" onClick={handleEdit}>\n Edit\n </MenuItem>,\n );\n }\n items.push(\n <MenuItem key=\"delete\" onClick={handleDeleteClick}>\n Delete\n </MenuItem>,\n );\n return items;\n })()}\n </Menu>\n\n {editDialogState.request && (\n <EditAPIKeyDialog\n open={editDialogState.open}\n request={editDialogState.request}\n availablePlans={editDialogState.plans}\n onClose={() =>\n setEditDialogState({ open: false, request: null, plans: [] })\n }\n onSuccess={() => {\n setEditDialogState({ open: false, request: null, plans: [] });\n setRefresh((r) => r + 1);\n }}\n />\n )}\n\n <ConfirmDeleteDialog\n open={deleteDialogState.open}\n title=\"Delete API Key\"\n description={`Are you sure you want to delete this API key for ${deleteDialogState.request?.spec.apiProductRef?.name || \"this API\"}?`}\n deleting={deleting !== null}\n onConfirm={handleDeleteConfirm}\n onCancel={handleDeleteCancel}\n />\n\n <Dialog\n open={showOnceWarningOpen}\n onClose={() => {\n setShowOnceWarningOpen(false);\n setPendingKeyReveal(null);\n }}\n maxWidth=\"sm\"\n >\n <DialogTitle>\n <Box display=\"flex\" alignItems=\"center\">\n <WarningIcon color=\"primary\" style={{ marginRight: 8 }} />\n View API Key\n </Box>\n </DialogTitle>\n <DialogContent>\n <Typography variant=\"body1\" paragraph>\n This API key can only be viewed <strong>once</strong>. After you\n reveal it, you will not be able to retrieve it again.\n </Typography>\n <Typography variant=\"body2\" color=\"textSecondary\">\n Make sure to copy and store it securely before closing this view.\n </Typography>\n </DialogContent>\n <DialogActions>\n <Button\n onClick={() => {\n setShowOnceWarningOpen(false);\n setPendingKeyReveal(null);\n }}\n >\n Cancel\n </Button>\n <Button\n variant=\"contained\"\n color=\"primary\"\n onClick={() => {\n if (pendingKeyReveal) {\n fetchApiKeyFromSecret(\n pendingKeyReveal.namespace,\n pendingKeyReveal.name,\n );\n toggleKeyVisibility(pendingKeyReveal.name);\n }\n setShowOnceWarningOpen(false);\n setPendingKeyReveal(null);\n }}\n >\n Reveal API Key\n </Button>\n </DialogActions>\n </Dialog>\n\n <SimpleRequestAccessDialog\n open={requestDialogOpen}\n onClose={() => setRequestDialogOpen(false)}\n onSuccess={() => {\n setRequestDialogOpen(false);\n setRefresh((prev) => prev + 1);\n }}\n />\n </>\n );\n};\n","import React from 'react';\nimport {\n Header,\n Page,\n Content,\n SupportButton,\n} from '@backstage/core-components';\nimport { MyApiKeysTable } from '../MyApiKeysTable';\n\nexport const MyApiKeysPage = () => {\n return (\n <Page themeId=\"tool\">\n <Header title=\"My API Keys\" subtitle=\"View and manage your API keys\">\n <SupportButton>Manage your API keys and access requests</SupportButton>\n </Header>\n <Content>\n <MyApiKeysTable />\n </Content>\n </Page>\n );\n};\n","\"use strict\";\n\nvar _interopRequireDefault = require(\"@babel/runtime/helpers/interopRequireDefault\");\n\nvar _interopRequireWildcard = require(\"@babel/runtime/helpers/interopRequireWildcard\");\n\nObject.defineProperty(exports, \"__esModule\", {\n value: true\n});\nexports.default = void 0;\n\nvar React = _interopRequireWildcard(require(\"react\"));\n\nvar _createSvgIcon = _interopRequireDefault(require(\"./utils/createSvgIcon\"));\n\nvar _default = (0, _createSvgIcon.default)( /*#__PURE__*/React.createElement(\"path\", {\n d: \"M6 19c0 1.1.9 2 2 2h8c1.1 0 2-.9 2-2V7H6v12zM19 4h-3.5l-1-1h-5l-1 1H5v2h14V4z\"\n}), 'Delete');\n\nexports.default = _default;","import { useApiHolder, configApiRef } from '@backstage/core-plugin-api';\nimport { coreComponentsTranslationRef } from '../translation.esm.js';\nimport { useTranslationRef } from '@backstage/core-plugin-api/alpha';\n\nconst useDefaultSupportConfig = () => {\n const { t } = useTranslationRef(coreComponentsTranslationRef);\n return {\n url: \"https://github.com/backstage/backstage/issues\",\n items: [\n {\n title: t(\"supportConfig.default.title\"),\n icon: \"warning\",\n links: [\n {\n // TODO: Update to dedicated support page on backstage.io/docs\n title: t(\"supportConfig.default.linkTitle\"),\n url: \"https://github.com/backstage/backstage/blob/master/app-config.yaml\"\n }\n ]\n }\n ]\n };\n};\nfunction useSupportConfig() {\n const apiHolder = useApiHolder();\n const config = apiHolder.get(configApiRef);\n const supportConfig = config?.getOptionalConfig(\"app.support\");\n const defaultSupportConfig = useDefaultSupportConfig();\n if (!supportConfig) {\n return defaultSupportConfig;\n }\n return {\n url: supportConfig.getString(\"url\"),\n items: supportConfig.getConfigArray(\"items\").flatMap((itemConf) => ({\n title: itemConf.getString(\"title\"),\n icon: itemConf.getOptionalString(\"icon\"),\n links: (itemConf.getOptionalConfigArray(\"links\") ?? []).flatMap(\n (linkConf) => ({\n url: linkConf.getString(\"url\"),\n title: linkConf.getOptionalString(\"title\") ?? \"\"\n })\n )\n }))\n };\n}\n\nexport { useSupportConfig };\n//# sourceMappingURL=useSupportConfig.esm.js.map\n","import { jsx } from 'react/jsx-runtime';\nimport { useApp } from '@backstage/core-plugin-api';\nimport MuiBrokenImageIcon from '@material-ui/icons/BrokenImage';\n\nfunction AppIcon(props) {\n const { id: key, Fallback = MuiBrokenImageIcon, ...rest } = props;\n const app = useApp();\n const Icon = app.getSystemIcon(key) ?? Fallback;\n return /* @__PURE__ */ jsx(Icon, { ...rest });\n}\nfunction BrokenImageIcon(props) {\n return /* @__PURE__ */ jsx(AppIcon, { id: \"brokenImage\", ...props });\n}\nfunction CatalogIcon(props) {\n return /* @__PURE__ */ jsx(AppIcon, { id: \"catalog\", ...props });\n}\nfunction ChatIcon(props) {\n return /* @__PURE__ */ jsx(AppIcon, { id: \"chat\", ...props });\n}\nfunction DashboardIcon(props) {\n return /* @__PURE__ */ jsx(AppIcon, { id: \"dashboard\", ...props });\n}\nfunction DocsIcon(props) {\n return /* @__PURE__ */ jsx(AppIcon, { id: \"docs\", ...props });\n}\nfunction EmailIcon(props) {\n return /* @__PURE__ */ jsx(AppIcon, { id: \"email\", ...props });\n}\nfunction GitHubIcon(props) {\n return /* @__PURE__ */ jsx(AppIcon, { id: \"github\", ...props });\n}\nfunction GroupIcon(props) {\n return /* @__PURE__ */ jsx(AppIcon, { id: \"group\", ...props });\n}\nfunction HelpIcon(props) {\n return /* @__PURE__ */ jsx(AppIcon, { id: \"help\", ...props });\n}\nfunction UserIcon(props) {\n return /* @__PURE__ */ jsx(AppIcon, { id: \"user\", ...props });\n}\nfunction WarningIcon(props) {\n return /* @__PURE__ */ jsx(AppIcon, { id: \"warning\", ...props });\n}\nfunction StarIcon(props) {\n return /* @__PURE__ */ jsx(AppIcon, { id: \"star\", ...props });\n}\nfunction UnstarredIcon(props) {\n return /* @__PURE__ */ jsx(AppIcon, { id: \"unstarred\", ...props });\n}\n\nexport { AppIcon, BrokenImageIcon, CatalogIcon, ChatIcon, DashboardIcon, DocsIcon, EmailIcon, GitHubIcon, GroupIcon, HelpIcon, StarIcon, UnstarredIcon, UserIcon, WarningIcon };\n//# sourceMappingURL=icons.esm.js.map\n","import { jsxs, Fragment, jsx } from 'react/jsx-runtime';\nimport { useApi, configApiRef, useApp } from '@backstage/core-plugin-api';\nimport Box from '@material-ui/core/Box';\nimport Button from '@material-ui/core/Button';\nimport DialogActions from '@material-ui/core/DialogActions';\nimport IconButton from '@material-ui/core/IconButton';\nimport ListItemIcon from '@material-ui/core/ListItemIcon';\nimport ListItemText from '@material-ui/core/ListItemText';\nimport MenuItem from '@material-ui/core/MenuItem';\nimport MenuList from '@material-ui/core/MenuList';\nimport Popover from '@material-ui/core/Popover';\nimport { makeStyles } from '@material-ui/core/styles';\nimport Typography from '@material-ui/core/Typography';\nimport useMediaQuery from '@material-ui/core/useMediaQuery';\nimport { useState, Children } from 'react';\nimport 'lodash';\nimport 'qs';\nimport 'react-router-dom';\nimport '@react-hookz/web';\nimport { useSupportConfig } from '../../hooks/useSupportConfig.esm.js';\nimport { HelpIcon } from '../../icons/icons.esm.js';\nimport { Link } from '../Link/Link.esm.js';\nimport { coreComponentsTranslationRef } from '../../translation.esm.js';\nimport { useTranslationRef } from '@backstage/core-plugin-api/alpha';\n\nconst useStyles = makeStyles(\n {\n popoverList: {\n minWidth: 260,\n maxWidth: 400\n },\n menuItem: {\n whiteSpace: \"normal\"\n }\n },\n { name: \"BackstageSupportButton\" }\n);\nconst SupportIcon = ({ icon }) => {\n const app = useApp();\n const Icon = icon ? app.getSystemIcon(icon) ?? HelpIcon : HelpIcon;\n return /* @__PURE__ */ jsx(Icon, {});\n};\nconst SupportLink = ({ link }) => /* @__PURE__ */ jsx(Link, { to: link.url, children: link.title ?? link.url });\nconst SupportListItem = ({ item }) => {\n return /* @__PURE__ */ jsxs(MenuItem, { button: false, children: [\n /* @__PURE__ */ jsx(ListItemIcon, { children: /* @__PURE__ */ jsx(SupportIcon, { icon: item.icon }) }),\n /* @__PURE__ */ jsx(\n ListItemText,\n {\n primary: item.title,\n secondary: item.links?.reduce(\n (prev, link, idx) => [\n ...prev,\n idx > 0 && /* @__PURE__ */ jsx(\"br\", {}, idx),\n /* @__PURE__ */ jsx(SupportLink, { link }, link.url)\n ],\n []\n )\n }\n )\n ] });\n};\nfunction SupportButton(props) {\n const { t } = useTranslationRef(coreComponentsTranslationRef);\n const { title, items, children } = props;\n const { items: configItems } = useSupportConfig();\n const [popoverOpen, setPopoverOpen] = useState(false);\n const [anchorEl, setAnchorEl] = useState(null);\n const classes = useStyles();\n const supportConfig = useApi(configApiRef).getOptionalConfig(\"app.support\");\n const isSmallScreen = useMediaQuery(\n (theme) => theme.breakpoints.down(\"sm\")\n );\n const onClickHandler = (event) => {\n setAnchorEl(event.currentTarget);\n setPopoverOpen(true);\n };\n const popoverCloseHandler = () => {\n setPopoverOpen(false);\n };\n if (!supportConfig) {\n return null;\n }\n return /* @__PURE__ */ jsxs(Fragment, { children: [\n /* @__PURE__ */ jsx(Box, { display: \"flex\", ml: 1, children: isSmallScreen ? /* @__PURE__ */ jsx(\n IconButton,\n {\n color: \"primary\",\n size: \"small\",\n onClick: onClickHandler,\n \"data-testid\": \"support-button\",\n \"aria-label\": \"Support\",\n children: /* @__PURE__ */ jsx(HelpIcon, {})\n }\n ) : /* @__PURE__ */ jsx(\n Button,\n {\n \"data-testid\": \"support-button\",\n \"aria-label\": \"Support\",\n color: \"primary\",\n onClick: onClickHandler,\n startIcon: /* @__PURE__ */ jsx(HelpIcon, {}),\n children: t(\"supportButton.title\")\n }\n ) }),\n /* @__PURE__ */ jsxs(\n Popover,\n {\n \"data-testid\": \"support-button-popover\",\n open: popoverOpen,\n anchorEl,\n anchorOrigin: {\n vertical: \"bottom\",\n horizontal: \"right\"\n },\n transformOrigin: {\n vertical: \"top\",\n horizontal: \"right\"\n },\n onClose: popoverCloseHandler,\n children: [\n /* @__PURE__ */ jsxs(\n MenuList,\n {\n className: classes.popoverList,\n autoFocusItem: Boolean(anchorEl),\n children: [\n title && /* @__PURE__ */ jsx(\n MenuItem,\n {\n button: false,\n alignItems: \"flex-start\",\n className: classes.menuItem,\n children: /* @__PURE__ */ jsx(Typography, { variant: \"subtitle1\", children: title })\n }\n ),\n Children.map(children, (child, i) => /* @__PURE__ */ jsx(\n MenuItem,\n {\n button: false,\n alignItems: \"flex-start\",\n className: classes.menuItem,\n children: child\n },\n `child-${i}`\n )),\n (items ?? configItems).map((item, i) => /* @__PURE__ */ jsx(SupportListItem, { item }, `item-${i}`))\n ]\n }\n ),\n /* @__PURE__ */ jsx(DialogActions, { children: /* @__PURE__ */ jsx(\n Button,\n {\n color: \"primary\",\n onClick: popoverCloseHandler,\n \"aria-label\": \"Close\",\n children: t(\"supportButton.close\")\n }\n ) })\n ]\n }\n )\n ] });\n}\n\nexport { SupportButton };\n//# sourceMappingURL=SupportButton.esm.js.map\n","import { CSSProperties } from \"react\";\n\n/**\n * Returns inline styles for API key status chips on the My API Keys page.\n */\nexport const getMyApiKeysStatusChipStyle = (phase: string): CSSProperties => {\n const base = { border: \"none\" };\n switch (phase) {\n case \"Approved\":\n return { ...base, backgroundColor: \"#1976d2\", color: \"#fff\" }; // Blue\n case \"Rejected\":\n return { ...base, backgroundColor: \"#d32f2f\", color: \"#fff\" }; // Red\n case \"Pending\":\n return { ...base, backgroundColor: \"#9c27b0\", color: \"#fff\" }; // Purple\n default:\n return { ...base, backgroundColor: \"#9c27b0\", color: \"#fff\" }; // Purple (fallback for Pending)\n }\n};\n\n/**\n * Returns inline styles for API key status chips on the Approval Queue page.\n * Uses inline styles to ensure proper specificity with Material-UI Chip.\n */\nexport const getApprovalQueueStatusChipStyle = (phase: string): CSSProperties => {\n const base = { border: \"none\" };\n switch (phase) {\n case \"Approved\":\n return { ...base, backgroundColor: \"#2e7d32\", color: \"#fff\" }; // Green\n case \"Rejected\":\n return { ...base, backgroundColor: \"#d32f2f\", color: \"#fff\" }; // Red\n case \"Pending\":\n return { ...base, backgroundColor: \"#ed6c02\", color: \"#fff\" }; // Orange\n default:\n return { ...base, backgroundColor: \"#ed6c02\", color: \"#fff\" }; // Orange (fallback for Pending)\n }\n};\n\n/**\n * Returns inline styles for lifecycle chips.\n * Uses inline styles to ensure proper specificity with Material-UI Chip.\n */\nexport const getLifecycleChipStyle = (lifecycle: string): CSSProperties => {\n switch (lifecycle) {\n case \"production\":\n return { backgroundColor: \"#1976d2\", color: \"#fff\" }; // Blue\n case \"experimental\":\n return { backgroundColor: \"#9c27b0\", color: \"#fff\" }; // Purple\n case \"deprecated\":\n return { backgroundColor: \"#ff9800\", color: \"#fff\" }; // Orange\n case \"retired\":\n return { backgroundColor: \"#d32f2f\", color: \"#fff\" }; // Red\n default:\n return {};\n }\n};\n","\"use strict\";\n\nvar _interopRequireDefault = require(\"@babel/runtime/helpers/interopRequireDefault\");\n\nvar _interopRequireWildcard = require(\"@babel/runtime/helpers/interopRequireWildcard\");\n\nObject.defineProperty(exports, \"__esModule\", {\n value: true\n});\nexports.default = void 0;\n\nvar React = _interopRequireWildcard(require(\"react\"));\n\nvar _createSvgIcon = _interopRequireDefault(require(\"./utils/createSvgIcon\"));\n\nvar _default = (0, _createSvgIcon.default)( /*#__PURE__*/React.createElement(\"path\", {\n d: \"M12 7c2.76 0 5 2.24 5 5 0 .65-.13 1.26-.36 1.83l2.92 2.92c1.51-1.26 2.7-2.89 3.43-4.75-1.73-4.39-6-7.5-11-7.5-1.4 0-2.74.25-3.98.7l2.16 2.16C10.74 7.13 11.35 7 12 7zM2 4.27l2.28 2.28.46.46C3.08 8.3 1.78 10.02 1 12c1.73 4.39 6 7.5 11 7.5 1.55 0 3.03-.3 4.38-.84l.42.42L19.73 22 21 20.73 3.27 3 2 4.27zM7.53 9.8l1.55 1.55c-.05.21-.08.43-.08.65 0 1.66 1.34 3 3 3 .22 0 .44-.03.65-.08l1.55 1.55c-.67.33-1.41.53-2.2.53-2.76 0-5-2.24-5-5 0-.79.2-1.53.53-2.2zm4.31-.78l3.15 3.15.02-.16c0-1.66-1.34-3-3-3l-.17.01z\"\n}), 'VisibilityOff');\n\nexports.default = _default;","import React, { useState, useEffect } from 'react';\nimport {\n Dialog,\n DialogTitle,\n DialogContent,\n DialogContentText,\n DialogActions,\n Button,\n TextField,\n Typography,\n Box,\n CircularProgress,\n} from '@material-ui/core';\nimport WarningIcon from '@material-ui/icons/Warning';\n\nexport interface ConfirmDeleteDialogProps {\n open: boolean;\n title: string;\n description: string;\n // for dangerous deletes, require typing this text to confirm\n confirmText?: string;\n // severity affects styling - 'high' shows warning icon and requires text confirmation\n severity?: 'normal' | 'high';\n deleting?: boolean;\n onConfirm: () => void;\n onCancel: () => void;\n}\n\nexport const ConfirmDeleteDialog = ({\n open,\n title,\n description,\n confirmText,\n severity = 'normal',\n deleting = false,\n onConfirm,\n onCancel,\n}: ConfirmDeleteDialogProps) => {\n const [inputValue, setInputValue] = useState('');\n\n // reset input when dialog opens/closes\n useEffect(() => {\n if (!open) {\n setInputValue('');\n }\n }, [open]);\n\n const requiresTextConfirmation = severity === 'high' && confirmText;\n const canConfirm = requiresTextConfirmation ? inputValue === confirmText : true;\n\n const handleConfirm = () => {\n if (canConfirm) {\n onConfirm();\n }\n };\n\n return (\n <Dialog\n open={open}\n onClose={deleting ? undefined : onCancel}\n maxWidth=\"sm\"\n fullWidth\n >\n <DialogTitle>\n {severity === 'high' && (\n <Box display=\"flex\" alignItems=\"center\" style={{ gap: 8 }}>\n <WarningIcon color=\"error\" />\n <span>{title}</span>\n </Box>\n )}\n {severity !== 'high' && title}\n </DialogTitle>\n <DialogContent>\n <DialogContentText 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 {\n createApiRef,\n DiscoveryApi,\n FetchApi,\n IdentityApi,\n} from '@backstage/core-plugin-api';\nimport retry from 'async-retry';\nimport { handleFetchError } from './utils/errors';\nimport {\n APIKey, APIKeyRequest,\n APIKeySpec,\n APIProduct,\n BulkOperationResult, ExtractedSecret, K8sList, K8sResource,\n PlanPolicy,\n AuthPolicy,\n RateLimitPolicy,\n} from './types/api-management';\n\n/**\n * Generic Kuadrant list type for API responses\n */\nexport interface KuadrantList<T = any> {\n items: T[];\n}\n\n/**\n * Options for constructing the KuadrantApiClient\n */\nexport type Options = {\n discoveryApi: DiscoveryApi;\n fetchApi: FetchApi;\n identityApi: IdentityApi;\n};\n\n/**\n * Retry configuration for read operations (GET requests only)\n * Conservative strategy: 3 retries with exponential backoff\n */\nconst RETRY_OPTIONS: retry.Options = {\n retries: 3,\n factor: 2,\n minTimeout: 300, // 300ms, 600ms, 1200ms\n maxTimeout: 3000,\n randomize: true,\n};\n\n/**\n * Kuadrant API interface defining all operations for managing\n * API products, API keys, and related resources\n */\nexport interface KuadrantAPI {\n // ===== APIKey Requests =====\n\n /**\n * Fetch all API key requests per user\n * @returns Promise with list of all API key requests\n */\n getRequests(): Promise<KuadrantList<APIKey>>;\n\n /**\n * Fetch all API key requests\n * @returns Promise with list of all API key requests\n */\n getAllRequests(): Promise<KuadrantList<APIKey>>;\n\n /**\n * Fetch API key requests for a specific namespace\n * @param namespace - Kubernetes namespace\n * @returns Promise with list of requests in the namespace\n */\n getRequestsByNamespace(namespace: string): Promise<KuadrantList<APIKey>>;\n\n /**\n * Fetch a single API key request\n * @param namespace - API key request name\n * @param name - Kubernetes namespace\n * @returns Promise with the API key request\n */\n getRequest(namespace: string, name: string): Promise<APIKey>;\n\n /**\n * Create a new API key request\n * @param request - APIKeyRequest specification\n * @returns Promise with the created API key\n */\n createRequest(request: APIKeyRequest): Promise<APIKey>;\n\n /**\n * Update an existing API key request\n * @param namespace - Kubernetes namespace\n * @param name - API key request name\n * @param patch - Partial API key spec with fields to update\n * @returns Promise with the updated API key\n */\n updateRequest(\n namespace: string,\n name: string,\n patch: Partial<APIKeySpec>,\n ): Promise<APIKey>;\n\n /**\n * Delete an API key request\n * @param namespace - Kubernetes namespace\n * @param name - API key request name\n * @returns Promise that resolves when deletion completes\n */\n deleteRequest(namespace: string, name: string): Promise<void>;\n\n /**\n * Approve an API key request\n * @param namespace - Kubernetes namespace\n * @param name - API key request name\n * @param reviewedBy - Reviewed By User / System\n * @returns Promise with the approved API key\n */\n approveRequest(namespace: string, name: string, reviewedBy: string): Promise<APIKey>;\n\n /**\n * Reject an API key request\n * @param namespace - Kubernetes namespace\n * @param name - API key request name\n * @param reviewedBy - Reviewed By User / System\n * @returns Promise with the rejected API key\n */\n rejectRequest(namespace: string, name: string, reviewedBy: string): Promise<APIKey>;\n\n /**\n * Bulk approve multiple API key requests\n * @param requests - Array of namespace/name pairs to approve\n * @param reviewedBy - Reviewed By User / System\n * @returns Promise that resolves when all approvals complete\n */\n bulkApproveRequests(\n requests: Array<{ namespace: string; name: string }>,\n reviewedBy: string): Promise<Array<BulkOperationResult>>;\n\n /**\n * Bulk reject multiple API key requests\n * @param requests - Array of namespace/name pairs to reject\n * @param reviewedBy - Reviewed By User / System\n * @returns Promise that resolves when all rejections complete\n */\n bulkRejectRequests(\n requests: Array<{ namespace: string; name: string }>,\n reviewedBy: string): Promise<Array<BulkOperationResult>>;\n\n // ===== API Keys/Secrets =====\n\n /**\n * Fetch an API key resource\n * @param namespace - Kubernetes namespace\n * @param name - API key name\n * @returns Promise with the API key\n */\n getApiKey(namespace: string, name: string): Promise<APIKey>;\n\n /**\n * Retrieve the secret value for an API key (one-time operation)\n * @param namespace - Kubernetes namespace\n * @param name - API key name\n * @returns Promise with the secret value\n */\n getApiKeySecret(namespace: string, name: string): Promise<ExtractedSecret>;\n\n // ===== API Products =====\n\n /**\n * Fetch all API products\n * @returns Promise with list of all API products\n */\n getApiProducts(): Promise<KuadrantList<APIProduct>>;\n\n /**\n * Fetch a single API product\n * @param namespace - Kubernetes namespace\n * @param name - API product name\n * @returns Promise with the API product\n */\n getApiProduct(namespace: string, name: string): Promise<APIProduct>;\n\n /**\n * Create a new API product\n * @param product - API product specification\n * @returns Promise with the created API product\n */\n createApiProduct(product: APIProduct): Promise<APIProduct>;\n\n /**\n * Update an existing API product\n * @param namespace - Kubernetes namespace\n * @param name - API product name\n * @param patch - Partial API product spec with fields to update\n * @returns Promise with the updated API product\n */\n updateApiProduct(\n namespace: string,\n name: string,\n patch: Partial<APIProduct>,\n ): Promise<APIProduct>;\n\n /**\n * Delete an API product\n * @param namespace - Kubernetes namespace\n * @param name - API product name\n * @returns Promise that resolves when deletion completes\n */\n deleteApiProduct(namespace: string, name: string): Promise<void>;\n\n // ===== HTTP Routes & Policies =====\n\n /**\n * Fetch all HTTPRoute(s)\n * @returns Promise with list of all HTTP routes\n */\n getHttpRoutes(): Promise<K8sList>;\n\n /**\n * Fetch a specific HTTPRoute\n * @param namespace - Kubernetes namespace\n * @param name - HTTPRoute name\n * @returns Promise with an HTTPRoute\n */\n getHttpRoute(namespace: string, name: string): Promise<K8sResource>;\n\n // ===== Plan Policies =====\n\n /**\n * Fetch all plan policies\n * @returns Promise with list of all plan policies\n */\n getPlanPolicies(): Promise<KuadrantList<PlanPolicy>>;\n\n // ===== Auth Policies =====\n\n /**\n * Fetch all auth policies\n * @returns Promise with list of all auth policies\n */\n getAuthPolicies(): Promise<KuadrantList<AuthPolicy>>;\n\n // ===== RateLimitPolicies =====\n\n /**\n * Fetch all ratelimitpolicies\n * @returns Promise with list of all ratelimitpolicies\n */\n getRateLimitPolicies(): Promise<KuadrantList<RateLimitPolicy>>;\n}\n\n/**\n * API reference for the Kuadrant API\n */\nexport const kuadrantApiRef = createApiRef<KuadrantAPI>({\n id: 'plugin.kuadrant.service',\n});\n\n/**\n * Implementation of the Kuadrant API client\n */\nexport class KuadrantApiClient implements KuadrantAPI {\n private readonly discoveryApi: DiscoveryApi;\n private readonly fetchApi: FetchApi;\n\n constructor(options: Options) {\n this.discoveryApi = options.discoveryApi;\n this.fetchApi = options.fetchApi;\n }\n\n /**\n * Get the base URL for the backend API\n */\n private async getBaseUrl(): Promise<string> {\n return await this.discoveryApi.getBaseUrl('');\n }\n\n /**\n * Wrapper for GET requests with automatic retry logic\n * Retries on network failures or 5xx errors with exponential backoff\n */\n private async fetchWithRetry<T>(url: string, errorMsg: string = \"\"): Promise<T> {\n return retry(\n async (bail) => {\n const response = await this.fetchApi.fetch(url);\n if (response.status === 401 || response.status === 403) { // Don't retry on Unauthenticated/Unauthorized\n const error = await handleFetchError(response);\n bail(new Error(error));\n }\n else if (!response.ok) {\n const error = await handleFetchError(response);\n throw new Error(`${errorMsg} ${error}`);\n }\n return await response.json();\n },\n RETRY_OPTIONS,\n );\n }\n\n /**\n * Wrapper for mutations (POST, PATCH, DELETE) without retry\n * These operations are not retried to avoid duplicate side effects\n */\n private async fetchWithoutRetry<T>(\n url: string,\n options: RequestInit,\n errorMsg: string = \"\"): Promise<T> {\n const response = await this.fetchApi.fetch(url, {\n headers: {\n \"Content-Type\": \"application/json\",\n },\n ...options,\n });\n\n if (!response.ok) {\n const error = await handleFetchError(response);\n throw new Error(`${errorMsg} ${error}`);\n }\n\n // DELETE operations don't return a body\n if (options.method === 'DELETE') {\n return undefined as T;\n }\n\n return await response.json();\n }\n\n // ===== API Requests Implementation =====\n\n async getRequests(): Promise<KuadrantList<APIKey>> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithRetry(\n `${baseUrl}kuadrant/requests/my`,\n \"Failed to fetch API Key requests.\"\n );\n }\n\n async getAllRequests(): Promise<KuadrantList<APIKey>> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithRetry(\n `${baseUrl}kuadrant/requests`,\n \"Failed to fetch API Key requests.\"\n );\n }\n\n async getRequestsByNamespace(namespace: string): Promise<KuadrantList<APIKey>> {\n const baseUrl = await this.getBaseUrl();\n const url = namespace\n ? `${baseUrl}kuadrant/requests/my?namespace=${namespace}`\n : `${baseUrl}kuadrant/requests/my`;\n return this.fetchWithRetry(url, \"Failed to fetch API Key requests by namespace.\");\n }\n\n async getRequest(namespace: string, name: string): Promise<APIKey> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithRetry(\n `${baseUrl}kuadrant/requests/${namespace}/${name}`,\n \"Failed to fetch API Key request.\"\n );\n }\n\n async createRequest(request: APIKeyRequest): Promise<APIKey> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithoutRetry(`${baseUrl}kuadrant/requests`, {\n method: 'POST',\n body: JSON.stringify(request),\n }, \"Failed to create APIKey request.\");\n }\n\n async updateRequest(\n namespace: string,\n name: string,\n patch: Partial<APIKeySpec>,\n ): Promise<APIKey> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithoutRetry(`${baseUrl}kuadrant/requests/${namespace}/${name}`, {\n method: 'PATCH',\n body: JSON.stringify(patch),\n }, \"Failed to update APIKey request.\");\n }\n\n async deleteRequest(namespace: string, name: string): Promise<void> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithoutRetry(`${baseUrl}kuadrant/requests/${namespace}/${name}`, {\n method: 'DELETE',\n }, \"Failed to delete APIKey request.\");\n }\n\n async approveRequest(namespace: string, name: string, reviewedBy: string = \"system\"): Promise<APIKey> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithoutRetry(`${baseUrl}kuadrant/requests/${namespace}/${name}/approve`, {\n method: 'POST',\n body: JSON.stringify({ reviewedBy }),\n }, \"Failed to approve APIKey request.\");\n }\n\n async rejectRequest(namespace: string, name: string, reviewedBy: string = \"system\"): Promise<APIKey> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithoutRetry(`${baseUrl}kuadrant/requests/${namespace}/${name}/reject`, {\n method: 'POST',\n body: JSON.stringify({ reviewedBy }),\n }, \"Failed to reject APIKey request.\");\n }\n\n async bulkApproveRequests(\n requests: Array<{ namespace: string; name: string }>,\n reviewedBy: string): Promise<Array<BulkOperationResult>> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithoutRetry(`${baseUrl}kuadrant/requests/bulk-approve`, {\n method: 'POST',\n body: JSON.stringify({ requests, reviewedBy }),\n }, \"Failed to bulk approve APIKey requests.\");\n }\n\n async bulkRejectRequests(\n requests: Array<{ namespace: string; name: string }>,\n reviewedBy: string): Promise<Array<BulkOperationResult>> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithoutRetry(`${baseUrl}kuadrant/requests/bulk-reject`, {\n method: 'POST',\n body: JSON.stringify({ requests, reviewedBy }),\n }, \"Failed to bulk reject APIKey requests\");\n }\n\n // ===== API Keys/Secrets Implementation =====\n\n async getApiKey(namespace: string, name: string): Promise<APIKey> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithRetry(\n `${baseUrl}kuadrant/apikeys/${namespace}/${name}`,\n \"Failed to fetch API Key.\"\n );\n }\n\n async getApiKeySecret(\n namespace: string,\n name: string,\n ): Promise<ExtractedSecret> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithRetry(\n `${baseUrl}kuadrant/apikeys/${namespace}/${name}/secret`,\n \"Failed to fetch API Key Secret.\"\n );\n }\n\n // ===== API Products Implementation =====\n\n async getApiProducts(): Promise<KuadrantList<APIProduct>> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithRetry(\n `${baseUrl}kuadrant/apiproducts`,\n \"Failed to fetch API Products.\"\n );\n }\n\n async getApiProduct(namespace: string, name: string): Promise<APIProduct> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithRetry(\n `${baseUrl}kuadrant/apiproducts/${namespace}/${name}`,\n \"Failed to fetch API Product.\"\n );\n }\n\n async createApiProduct(product: APIProduct): Promise<APIProduct> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithoutRetry(`${baseUrl}kuadrant/apiproducts`, {\n method: 'POST',\n body: JSON.stringify(product),\n }, \"Failed to create API Product.\");\n }\n\n async updateApiProduct(\n namespace: string,\n name: string,\n patch: Partial<APIProduct>,\n ): Promise<APIProduct> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithoutRetry(`${baseUrl}kuadrant/apiproducts/${namespace}/${name}`, {\n method: 'PATCH',\n body: JSON.stringify(patch),\n }, \"Failed to update API Product.\");\n }\n\n async deleteApiProduct(namespace: string, name: string): Promise<void> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithoutRetry(`${baseUrl}kuadrant/apiproducts/${namespace}/${name}`, {\n method: 'DELETE',\n }, \"Failed to delete API Product.\");\n }\n\n // ===== HTTP Routes =====\n\n async getHttpRoutes(): Promise<K8sList> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithRetry(\n `${baseUrl}kuadrant/httproutes`,\n \"Failed to fetch HTTPRoutes.\"\n );\n }\n\n async getHttpRoute(namespace: string, name: string): Promise<K8sResource> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithRetry(\n `${baseUrl}kuadrant/httproutes/${namespace}/${name}`,\n `Failed to fetc HTTPRoute ${namespace}/${name}.`\n );\n }\n\n // ===== Plan Policies Implementation =====\n\n async getPlanPolicies(): Promise<KuadrantList<PlanPolicy>> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithRetry(\n `${baseUrl}kuadrant/planpolicies`,\n \"Failed to fetch PlanPolicies.\"\n );\n }\n\n // ===== AuthPolicies Implementation =====\n\n async getAuthPolicies(): Promise<KuadrantList<AuthPolicy>> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithRetry(\n `${baseUrl}kuadrant/authpolicies`,\n \"Failed to fetch AuthPolicies.\"\n );\n }\n\n // ===== RateLimitPolicies Implementation =====\n\n async getRateLimitPolicies(): Promise<KuadrantList<RateLimitPolicy>> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithRetry(\n `${baseUrl}kuadrant/ratelimitpolicies`,\n \"Failed to fetch RateLimitPolicies.\"\n );\n }\n}\n","\"use strict\";\n\nvar _interopRequireDefault = require(\"@babel/runtime/helpers/interopRequireDefault\");\n\nvar _interopRequireWildcard = require(\"@babel/runtime/helpers/interopRequireWildcard\");\n\nObject.defineProperty(exports, \"__esModule\", {\n value: true\n});\nexports.default = void 0;\n\nvar React = _interopRequireWildcard(require(\"react\"));\n\nvar _createSvgIcon = _interopRequireDefault(require(\"./utils/createSvgIcon\"));\n\nvar _default = (0, _createSvgIcon.default)( /*#__PURE__*/React.createElement(\"path\", {\n d: \"M1 21h22L12 2 1 21zm12-3h-2v-2h2v2zm0-4h-2v-4h2v4z\"\n}), 'Warning');\n\nexports.default = _default;","\"use strict\";\nObject.defineProperty(exports, \"__esModule\", { value: true });\nvar tslib_1 = require(\"tslib\");\nvar react_1 = require(\"react\");\nvar useAsyncFn_1 = tslib_1.__importDefault(require(\"./useAsyncFn\"));\nfunction useAsync(fn, deps) {\n if (deps === void 0) { deps = []; }\n var _a = useAsyncFn_1.default(fn, deps, {\n loading: true,\n }), state = _a[0], callback = _a[1];\n react_1.useEffect(function () {\n callback();\n }, [callback]);\n return state;\n}\nexports.default = useAsync;\n"],"names":["Object","defineProperty","exports","value","tslib_1","react_1","useMountedState_1","__importDefault","fn","deps","initialState","loading","lastCallId","useRef","isMounted","default","_a","useState","state","set","callback","useCallback","args","_i","arguments","length","callId","current","prevState","__assign","apply","then","error","mountedRef","get","useEffect","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","values","some","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","currentValues","newValues","v","handleCheckboxChange","label","undefined","EditAPIKeyDialog","open","onClose","onSuccess","request","availablePlans","kuadrantApi","useApi","kuadrantApiRef","planTier","setPlanTier","useCase","setUseCase","saving","setSaving","setError","spec","handleClose","Dialog","maxWidth","fullWidth","DialogTitle","DialogContent","p","bgcolor","borderRadius","FormControl","margin","InputLabel","Select","e","target","disabled","plan","limitDesc","entries","limits","key","val","join","MenuItem","tier","TextField","placeholder","multiline","rows","helperText","DialogActions","async","patch","trim","updateRequest","metadata","name","namespace","err","console","Error","message","startIcon","CircularProgress","_interopRequireDefault","_interopRequireWildcard","_default","createElement","d","SimpleRequestAccessDialog","selectedProduct","config","configApiRef","fetchApi","fetchApiRef","alertApi","alertApiRef","identityApi","identityApiRef","backendUrl","getString","selectedApi","setSelectedApi","selectedTier","setSelectedTier","creating","setCreating","apiProducts","loadingProducts","useAsync","response","fetch","ok","json","items","publishStatus","userEmail","getBackstageIdentity","userEntityRef","split","find","availableTiers","status","discoveredPlans","required","labelId","data-testid","product","FormHelperText","inputProps","apiProductName","method","headers","body","JSON","stringify","errorData","statusText","requestBody","rawError","errorMsg","match","post","severity","errorMessage","container","minHeight","tableContainer","flex","overflow","useCasePanel","useCaseLabel","rejectedBanner","light","border","main","shape","gap","ExpandedRowContent","isRejected","phase","apiProductRef","stopPropagation","WarningIcon","Link","to","MyApiKeysTable","deleteDialogState","navigate","useNavigate","requestDialogOpen","setRequestDialogOpen","visibleKeys","setVisibleKeys","menuAnchor","setMenuAnchor","menuRequest","setMenuRequest","editDialogState","setEditDialogState","plans","refresh","setRefresh","deleting","setDeleting","setDeleteDialogState","apiKeyValues","setApiKeyValues","Map","apiKeyLoading","setApiKeyLoading","alreadyReadKeys","setAlreadyReadKeys","showOnceWarningOpen","setShowOnceWarningOpen","pendingKeyReveal","setPendingKeyReveal","optimisticallyDeleted","setOptimisticallyDeleted","setFilters","apiProduct","data","requestsData","productsData","Promise","all","getRequests","getApiProducts","requests","products","ownerMap","owner","annotations","allRequests","useMemo","r","filterSections","statusCounts","Approved","Pending","Rejected","apiProductCounts","tierCounts","Array","from","charAt","toUpperCase","slice","filteredRequests","toggleKeyVisibility","keyName","newSet","handleMenuClose","handleEdit","apiProductNamespace","getApiProduct","handleDeleteClick","columns","field","render","row","strong","displayOwner","replace","Chip","style","getMyApiKeysStatusChipStyle","filtering","hasSecretRef","secretRef","isVisible","isLoading","apiKeyValue","canReadSecret","isAlreadyRead","Tooltip","fontFamily","marginRight","VisibilityOffIcon","repeat","IconButton","navigator","clipboard","writeText","FileCopyIcon","requestNamespace","requestName","clearApiKeyValue","VisibilityIcon","creationTimestamp","date","Date","toLocaleDateString","DeleteIcon","detailPanelConfig","rowData","Progress","ResponseErrorPanel","px","textAlign","gutterBottom","Table","paging","pageSize","search","debounceInterval","toolbar","emptyRowsWhenPaging","item","detailPanel","Menu","Boolean","anchorReference","anchorPosition","top","left","push","ConfirmDeleteDialog","description","onConfirm","deleteRequest","onCancel","paragraph","result","getApiKeySecret","apiKey","fetchApiKeyFromSecret","MyApiKeysPage","Page","themeId","Header","subtitle","SupportButton","Content","useDefaultSupportConfig","t","useTranslationRef","url","icon","links","AppIcon","props","Fallback","rest","Icon","useApp","getSystemIcon","jsx","HelpIcon","popoverList","menuItem","whiteSpace","SupportIcon","app","SupportLink","link","children","SupportListItem","jsxs","button","ListItemIcon","ListItemText","primary","reduce","idx","configItems","useApiHolder","supportConfig","getOptionalConfig","defaultSupportConfig","getConfigArray","flatMap","itemConf","getOptionalString","getOptionalConfigArray","linkConf","useSupportConfig","popoverOpen","setPopoverOpen","anchorEl","setAnchorEl","isSmallScreen","useMediaQuery","breakpoints","down","onClickHandler","event","currentTarget","popoverCloseHandler","Fragment","ml","Popover","anchorOrigin","vertical","horizontal","transformOrigin","MenuList","autoFocusItem","Children","child","i","base","getApprovalQueueStatusChipStyle","getLifecycleChipStyle","lifecycle","confirmText","inputValue","setInputValue","requiresTextConfirmation","canConfirm","DialogContentText","autoFocus","createApiRef","useAsyncFn_1"],"sourceRoot":""}