@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
@@ -0,0 +1 @@
1
+ {"version":3,"file":"static/9051.d45ac154.chunk.js","mappings":"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","sources":["webpack://internal.plugin-kuadrant/./src/components/RequestAccessDialog/RequestAccessDialog.tsx","webpack://internal.plugin-kuadrant/./src/permissions.ts","webpack://internal.plugin-kuadrant/./src/utils/permissions.ts","webpack://internal.plugin-kuadrant/./src/api.ts"],"sourcesContent":["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 Box,\n Typography,\n CircularProgress,\n} from '@material-ui/core';\nimport InfoIcon from '@material-ui/icons/Info';\nimport {\n useApi,\n alertApiRef,\n} from '@backstage/core-plugin-api';\nimport { kuadrantApiRef } from '../../api';\nimport { Plan } from \"../../types/api-management.ts\";\n\nexport interface RequestAccessDialogProps {\n open: boolean;\n onClose: () => void;\n onSuccess: () => void;\n apiProductName: string;\n namespace: string;\n userEmail: string;\n plans: Plan[];\n}\n\nexport const RequestAccessDialog = ({\n open,\n onClose,\n onSuccess,\n apiProductName,\n namespace,\n userEmail,\n plans,\n}: RequestAccessDialogProps) => {\n const kuadrantApi = useApi(kuadrantApiRef);\n const alertApi = useApi(alertApiRef);\n\n const [selectedPlan, setSelectedPlan] = useState('');\n const [useCase, setUseCase] = useState('');\n const [creating, setCreating] = useState(false);\n const [createError, setCreateError] = useState<string | null>(null);\n\n const handleClose = () => {\n setSelectedPlan('');\n setUseCase('');\n setCreateError(null);\n onClose();\n };\n\n const handleRequestAccess = async () => {\n if (!selectedPlan) return;\n\n setCreating(true);\n setCreateError(null);\n try {\n await kuadrantApi.createRequest({\n apiProductName,\n namespace,\n planTier: selectedPlan,\n useCase: useCase.trim() || '',\n userEmail,\n });\n\n alertApi.post({\n message: 'API key requested successfully',\n severity: 'success',\n display: 'transient',\n });\n\n setSelectedPlan('');\n setUseCase('');\n onSuccess();\n } catch (err) {\n const errorMessage =\n err instanceof Error ? err.message : 'unknown error occurred';\n alertApi.post({\n message: `Failed to request API key: ${errorMessage}`,\n severity: 'error',\n display: 'transient',\n });\n setCreateError(errorMessage);\n } finally {\n setCreating(false);\n }\n };\n\n return (\n <Dialog open={open} onClose={handleClose} maxWidth=\"sm\" fullWidth>\n <DialogTitle>Request API Access</DialogTitle>\n <DialogContent>\n <Box\n mb={2}\n p={1.5}\n bgcolor=\"info.light\"\n borderRadius={1}\n display=\"flex\"\n alignItems=\"flex-start\"\n style={{ gap: 8 }}\n >\n <InfoIcon\n color=\"primary\"\n fontSize=\"small\"\n style={{ marginTop: 2 }}\n />\n <Typography variant=\"body2\">\n Your request will be reviewed by an API owner before access is\n granted.\n </Typography>\n </Box>\n {createError && (\n <Box\n mb={2}\n p={2}\n bgcolor=\"error.main\"\n color=\"error.contrastText\"\n borderRadius={1}\n >\n <Typography variant=\"body2\">{createError}</Typography>\n </Box>\n )}\n <FormControl\n fullWidth\n margin=\"normal\"\n disabled={creating}\n data-testid=\"tier-select-form\"\n >\n <InputLabel id=\"tier-select-label\">Select Tier</InputLabel>\n <Select\n labelId=\"tier-select-label\"\n data-testid=\"tier-select\"\n value={selectedPlan}\n onChange={(e) => setSelectedPlan(e.target.value as string)}\n disabled={creating}\n >\n {plans.map((plan: Plan) => {\n const limitDesc = Object.entries(plan.limits || {})\n .map(([key, val]) => `${val} per ${key}`)\n .join(', ');\n return (\n <MenuItem\n key={plan.tier}\n value={plan.tier}\n data-testid={`tier-option-${plan.tier}`}\n >\n {plan.tier} {limitDesc ? `(${limitDesc})` : ''}\n </MenuItem>\n );\n })}\n </Select>\n </FormControl>\n <TextField\n label=\"Use Case (optional)\"\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 helperText=\"Explain your intended use of this API for admin review\"\n disabled={creating}\n />\n </DialogContent>\n <DialogActions>\n <Button onClick={handleClose} disabled={creating}>\n Cancel\n </Button>\n <Button\n onClick={handleRequestAccess}\n color=\"primary\"\n variant=\"contained\"\n disabled={!selectedPlan || creating}\n startIcon={\n creating ? (\n <CircularProgress size={16} color=\"inherit\" />\n ) : undefined\n }\n >\n {creating ? 'Submitting...' : 'Submit Request'}\n </Button>\n </DialogActions>\n </Dialog>\n );\n};\n","import { createPermission } from '@backstage/plugin-permission-common';\n\n/**\n * permission definitions for the kuadrant plugin\n *\n * these permissions control access to kuadrant resources and operations.\n * they must match the permissions defined in the backend plugin.\n *\n * permission types:\n * - BasicPermission: standard permission that applies globally\n * - ResourcePermission: permission scoped to specific resource types (e.g., apiproduct)\n *\n * permission patterns:\n * - `.create` - create new resources\n * - `.read` - read resource details\n * - `.read.own` - read only resources owned by the user\n * - `.read.all` - read all resources regardless of ownership\n * - `.update` - modify existing resources\n * - `.delete` - delete resources\n * - `.delete.own` - delete only resources owned by the user\n * - `.delete.all` - delete any resource regardless of ownership\n * - `.list` - list/view collections of resources\n */\n\n// planpolicy permissions\nexport const kuadrantPlanPolicyCreatePermission = createPermission({\n name: 'kuadrant.planpolicy.create',\n attributes: { action: 'create' },\n});\n\nexport const kuadrantPlanPolicyReadPermission = createPermission({\n name: 'kuadrant.planpolicy.read',\n attributes: { action: 'read' },\n});\n\nexport const kuadrantPlanPolicyUpdatePermission = createPermission({\n name: 'kuadrant.planpolicy.update',\n attributes: { action: 'update' },\n});\n\nexport const kuadrantPlanPolicyDeletePermission = createPermission({\n name: 'kuadrant.planpolicy.delete',\n attributes: { action: 'delete' },\n});\n\nexport const kuadrantPlanPolicyListPermission = createPermission({\n name: 'kuadrant.planpolicy.list',\n attributes: { action: 'read' },\n});\n\n// authpolicy permissions\nexport const kuadrantAuthPolicyListPermission = createPermission({\n name: 'kuadrant.authpolicy.list',\n attributes: { action: 'read' },\n});\n\n// ratelimitpolicy permissions\nexport const kuadrantRateLimitPolicyListPermission = createPermission({\n name: 'kuadrant.ratelimitpolicy.list',\n attributes: { action: 'read' },\n});\n\n// apiproduct permissions\n\n/**\n * permission to create new API products\n * granted to api owners and admins\n */\nexport const kuadrantApiProductCreatePermission = createPermission({\n name: 'kuadrant.apiproduct.create',\n attributes: { action: 'create' },\n});\n\n/**\n * permission to read API products owned by the current user\n * for api owners to view their own products\n */\nexport const kuadrantApiProductReadOwnPermission = createPermission({\n name: 'kuadrant.apiproduct.read.own',\n attributes: { action: 'read' },\n});\n\n/**\n * permission to read all API products regardless of ownership\n * for platform engineers/admins who need to view all products\n */\nexport const kuadrantApiProductReadAllPermission = createPermission({\n name: 'kuadrant.apiproduct.read.all',\n attributes: { action: 'read' },\n});\n\n/**\n * permission to update API products owned by the current user\n * for api owners to modify their own products\n */\nexport const kuadrantApiProductUpdateOwnPermission = createPermission({\n name: 'kuadrant.apiproduct.update.own',\n attributes: { action: 'update' },\n});\n\n/**\n * permission to update any API product regardless of ownership\n * for platform engineers/admins\n */\nexport const kuadrantApiProductUpdateAllPermission = createPermission({\n name: 'kuadrant.apiproduct.update.all',\n attributes: { action: 'update' },\n});\n\n/**\n * permission to delete API products owned by the current user\n * for api owners to remove their own products\n */\nexport const kuadrantApiProductDeleteOwnPermission = createPermission({\n name: 'kuadrant.apiproduct.delete.own',\n attributes: { action: 'delete' },\n});\n\n/**\n * permission to delete any API product regardless of ownership\n * for platform engineers/admins\n */\nexport const kuadrantApiProductDeleteAllPermission = createPermission({\n name: 'kuadrant.apiproduct.delete.all',\n attributes: { action: 'delete' },\n});\n\n/**\n * permission to list API products\n * backend filters results based on .own vs .all read permissions\n */\nexport const kuadrantApiProductListPermission = createPermission({\n name: 'kuadrant.apiproduct.list',\n attributes: { action: 'read' },\n});\n\n// apikey permissions\n\n/**\n * permission to create API keys (request API access)\n *\n * this is a ResourcePermission scoped to 'apiproduct', allowing\n * fine-grained control over which API products users can request access to.\n *\n * use in frontend: useKuadrantPermission(kuadrantApiKeyCreatePermission)\n * use in backend with resource: { permission, resourceRef: 'apiproduct:namespace/name' }\n */\nexport const kuadrantApiKeyCreatePermission = createPermission({\n name: 'kuadrant.apikey.create',\n attributes: { action: 'create' },\n resourceType: 'apiproduct',\n});\n\n/**\n * permission to read API keys owned by the current user\n * allows users to view their own API keys and request history\n */\nexport const kuadrantApiKeyReadOwnPermission = createPermission({\n name: 'kuadrant.apikey.read.own',\n attributes: { action: 'read' },\n});\n\n/**\n * permission to read all API keys regardless of ownership\n * for platform engineers/admins who need to view the approval queue and audit keys\n */\nexport const kuadrantApiKeyReadAllPermission = createPermission({\n name: 'kuadrant.apikey.read.all',\n attributes: { action: 'read' },\n});\n\n/**\n * permission to update API keys owned by the current user\n * allows users to edit their own pending requests (change plan tier, use case)\n */\nexport const kuadrantApiKeyUpdateOwnPermission = createPermission({\n name: 'kuadrant.apikey.update.own',\n attributes: { action: 'update' },\n});\n\n/**\n * permission to update any API key regardless of ownership\n * typically granted to API owners and platform engineers for approving/rejecting requests\n */\nexport const kuadrantApiKeyUpdateAllPermission = createPermission({\n name: 'kuadrant.apikey.update.all',\n attributes: { action: 'update' },\n});\n\n/**\n * permission to delete API keys owned by the current user\n * allows users to cancel their own requests or revoke their own access\n */\nexport const kuadrantApiKeyDeleteOwnPermission = createPermission({\n name: 'kuadrant.apikey.delete.own',\n attributes: { action: 'delete' },\n});\n\n/**\n * permission to delete any API key regardless of ownership\n * for platform engineers/admins who need to revoke access\n */\nexport const kuadrantApiKeyDeleteAllPermission = createPermission({\n name: 'kuadrant.apikey.delete.all',\n attributes: { action: 'delete' },\n});\n\n/**\n * permission to approve/reject API key requests\n * grants access to the approval queue - for API owners and admins only\n * separate from update.own which consumers use to edit their pending requests\n */\nexport const kuadrantApiKeyApprovePermission = createPermission({\n name: 'kuadrant.apikey.approve',\n attributes: { action: 'update' },\n});\n\nexport const kuadrantPermissions = [\n kuadrantPlanPolicyCreatePermission,\n kuadrantPlanPolicyReadPermission,\n kuadrantPlanPolicyUpdatePermission,\n kuadrantPlanPolicyDeletePermission,\n kuadrantPlanPolicyListPermission,\n kuadrantApiProductCreatePermission,\n kuadrantApiProductReadOwnPermission,\n kuadrantApiProductReadAllPermission,\n kuadrantApiProductUpdateOwnPermission,\n kuadrantApiProductUpdateAllPermission,\n kuadrantApiProductDeleteOwnPermission,\n kuadrantApiProductDeleteAllPermission,\n kuadrantApiProductListPermission,\n kuadrantApiKeyCreatePermission,\n kuadrantApiKeyReadOwnPermission,\n kuadrantApiKeyReadAllPermission,\n kuadrantApiKeyUpdateOwnPermission,\n kuadrantApiKeyUpdateAllPermission,\n kuadrantApiKeyDeleteOwnPermission,\n kuadrantApiKeyDeleteAllPermission,\n kuadrantApiKeyApprovePermission,\n kuadrantAuthPolicyListPermission,\n kuadrantRateLimitPolicyListPermission,\n];\n","import { usePermission } from '@backstage/plugin-permission-react';\nimport { Permission, ResourcePermission } from '@backstage/plugin-permission-common';\n\n/**\n * result of a permission check including error state\n */\nexport interface PermissionCheckResult {\n allowed: boolean;\n loading: boolean;\n error?: Error;\n}\n\n/**\n * custom hook for checking kuadrant permissions that handles both\n * BasicPermission and ResourcePermission types without type bypasses\n *\n * @param permission - the permission to check\n * @param resourceRef - optional resource reference for ResourcePermissions\n * @returns permission check result with error handling\n *\n * @example\n * // basic permission\n * const { allowed, loading, error } = useKuadrantPermission(\n * kuadrantApiProductListPermission\n * );\n *\n * @example\n * // resource permission\n * const { allowed, loading, error } = useKuadrantPermission(\n * kuadrantApiKeyCreatePermission,\n * 'apiproduct:namespace/name'\n * );\n */\nexport function useKuadrantPermission(\n permission: Permission,\n resourceRef?: string,\n): PermissionCheckResult {\n // construct the permission request based on whether it's a ResourcePermission\n const permissionRequest = 'resourceType' in permission\n ? { permission: permission as ResourcePermission, resourceRef }\n : { permission };\n\n const result = usePermission(permissionRequest as any);\n\n return {\n allowed: result.allowed,\n loading: result.loading,\n error: result.error,\n };\n}\n\n/**\n * helper to determine if a user can delete a specific API key or request\n *\n * @param ownerId - the user id who owns the key/request\n * @param currentUserId - the current user's id\n * @param canDeleteOwn - whether user has permission to delete their own keys\n * @param canDeleteAll - whether user has permission to delete all keys\n * @returns true if user can delete this specific key/request\n */\nexport function canDeleteResource(\n ownerId: string,\n currentUserId: string,\n canDeleteOwn: boolean,\n canDeleteAll: boolean,\n): boolean {\n if (canDeleteAll) return true;\n if (canDeleteOwn && ownerId === currentUserId) return true;\n return false;\n}\n","import {\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"],"names":["RequestAccessDialog","open","onClose","onSuccess","apiProductName","namespace","userEmail","plans","kuadrantApi","useApi","kuadrantApiRef","alertApi","alertApiRef","selectedPlan","setSelectedPlan","useState","useCase","setUseCase","creating","setCreating","createError","setCreateError","handleClose","Dialog","maxWidth","fullWidth","DialogTitle","DialogContent","Box","mb","p","bgcolor","borderRadius","display","alignItems","style","gap","InfoIcon","color","fontSize","marginTop","Typography","variant","FormControl","margin","disabled","data-testid","InputLabel","id","Select","labelId","value","onChange","e","target","map","plan","limitDesc","Object","entries","limits","key","val","join","MenuItem","tier","TextField","label","placeholder","multiline","rows","helperText","DialogActions","Button","onClick","async","createRequest","planTier","trim","post","message","severity","err","errorMessage","Error","startIcon","CircularProgress","size","undefined","createPermission","name","attributes","action","kuadrantPlanPolicyListPermission","kuadrantApiProductCreatePermission","kuadrantApiProductReadAllPermission","kuadrantApiProductUpdateOwnPermission","kuadrantApiProductUpdateAllPermission","kuadrantApiProductDeleteOwnPermission","kuadrantApiProductDeleteAllPermission","kuadrantApiProductListPermission","kuadrantApiKeyCreatePermission","resourceType","kuadrantApiKeyUpdateOwnPermission","kuadrantApiKeyUpdateAllPermission","kuadrantApiKeyDeleteOwnPermission","kuadrantApiKeyDeleteAllPermission","kuadrantApiKeyApprovePermission","useKuadrantPermission","permission","resourceRef","permissionRequest","result","usePermission","allowed","loading","error","canDeleteResource","ownerId","currentUserId","canDeleteOwn","canDeleteAll","createApiRef"],"sourceRoot":""}
@@ -1,2 +1,2 @@
1
1
  "use strict";(self.webpackChunkinternal_plugin_kuadrant=self.webpackChunkinternal_plugin_kuadrant||[]).push([[9838],{16397:(e,n,t)=>{t.d(n,{e:()=>x});var a=t(31085),r=t(95478),s=t(76891),i=t(61477),o=t(46805),l=t(10394),d=t(72501),c=t(95061),u=t(48543),p=t(81215),h=t(26343),m=t(16249),y=t(93453),v=t(64947),g=t(78467),A=t(22097),f=t(64047);const x=({open:e,onClose:n,onSuccess:t,request:x,availablePlans:j})=>{const b=(0,A.useApi)(f.s),[w,k]=(0,r.useState)(""),[C,$]=(0,r.useState)(""),[P,S]=(0,r.useState)(!1),[q,E]=(0,r.useState)("");(0,r.useEffect)(()=>{e&&x&&(k(x.spec.planTier||""),$(x.spec.useCase||""),E(""))},[e,x]);const z=()=>{P||(E(""),n())};return(0,a.jsxs)(s.A,{open:e,onClose:z,maxWidth:"sm",fullWidth:!0,children:[(0,a.jsx)(i.A,{children:"Edit API Key"}),(0,a.jsxs)(o.A,{children:[q&&(0,a.jsx)(l.A,{mb:2,p:2,bgcolor:"error.main",color:"error.contrastText",borderRadius:1,children:(0,a.jsx)(d.A,{variant:"body2",children:q})}),(0,a.jsxs)(c.A,{fullWidth:!0,margin:"normal",children:[(0,a.jsx)(u.A,{children:"Tier"}),(0,a.jsx)(p.A,{value:w,onChange:e=>k(e.target.value),disabled:P,children:j.map(e=>{const n=Object.entries(e.limits||{}).map(([e,n])=>`${n} per ${e}`).join(", ");return(0,a.jsxs)(h.A,{value:e.tier,children:[e.tier," ",n?`(${n})`:""]},e.tier)})})]}),(0,a.jsx)(m.A,{label:"Use Case",placeholder:"Describe how you plan to use this API",multiline:!0,rows:3,fullWidth:!0,margin:"normal",value:C,onChange:e=>$(e.target.value),disabled:P,helperText:"Explain your intended use of this API for admin review"})]}),(0,a.jsxs)(y.A,{children:[(0,a.jsx)(v.A,{onClick:z,disabled:P,children:"Cancel"}),(0,a.jsx)(v.A,{onClick:async()=>{if(w){E(""),S(!0);try{const e={spec:{planTier:w,useCase:C.trim()}};await b.updateRequest(x.metadata.name,x.metadata.namespace,e),t(),n()}catch(e){console.error("Error updating API key request:",e),E(e instanceof Error?e.message:"Unknown error occurred")}finally{S(!1)}}else E("Please select a tier")},color:"primary",variant:"contained",disabled:!w||P,startIcon:P?(0,a.jsx)(g.A,{size:16,color:"inherit"}):void 0,children:P?"Saving...":"Save Changes"})]})]})}},18466:(e,n,t)=>{var a=t(4293),r=t(78920);n.A=void 0;var s=r(t(95478)),i=(0,a(t(74044)).default)(s.createElement("path",{d:"M19 13h-6v6h-2v-6H5v-2h6V5h2v6h6v2z"}),"Add");n.A=i},23650:(e,n,t)=>{t.r(n),t.d(n,{ApiKeyManagementTab:()=>V});var a=t(31085),r=t(95478),s=t(35015),i=t(82300),o=t(86687),l=t(42367),d=t(25010),c=t(10394),u=t(72501),p=t(31653),h=t(38605),m=t(67720),y=t(71677),v=t(29365),g=t(78467),A=t(42899),f=t(64947),x=t(37757),j=t(26343),b=t(76891),w=t(61477),k=t(46805),C=t(93453),$=t(22097),P=t(64047),S=t(25467),q=t(32269),E=t(61524),z=t(93246),R=t(92399),T=t(18466),I=t(71407),K=t(99594),B=t(77225),_=t(34955),H=t(46205),D=t(16397),M=t(63221);function W(e,n,t){const a=t?`${t} `:"";return{curl:`curl -X GET ${e} \\\n -H "Authorization: ${t?`${t} ${n}`:n}"`,nodejs:`const fetch = require('node-fetch');\n\nconst apiKey = '${n}';\nconst endpoint = '${e}';\n\nfetch(endpoint, {\n method: 'GET',\n headers: {\n 'Authorization': '${a}' + apiKey\n }\n})\n .then(response => response.json())\n .then(data => console.log(data))\n .catch(error => console.error('Error:', error));`,python:`import requests\n\napi_key = '${n}'\nendpoint = '${e}'\n\nheaders = {\n 'Authorization': '${a}' + api_key\n}\n\nresponse = requests.get(endpoint, headers=headers)\nprint(response.json())`,go:`package main\n\nimport (\n "fmt"\n "net/http"\n "io"\n)\n\nfunc main() {\n apiKey := "${n}"\n endpoint := "${e}"\n\n client := &http.Client{}\n req, _ := http.NewRequest("GET", endpoint, nil)\n req.Header.Add("Authorization", "${a}" + apiKey)\n\n resp, err := client.Do(req)\n if err != nil {\n fmt.Println("Error:", err)\n return\n }\n defer resp.Body.Close()\n\n body, _ := io.ReadAll(resp.Body)\n fmt.Println(string(body))\n}`}}var N=t(14933);const V=({namespace:e})=>{var n,t,V,G,L;const{entity:U}=(0,S.tN)(),X=(0,$.useApi)(P.s),F=(0,$.useApi)($.identityApiRef),O=(0,$.useApi)($.alertApiRef),[J,Q]=(0,r.useState)(new Set),[Y,Z]=(0,r.useState)(0),[ee,ne]=(0,r.useState)(""),[te,ae]=(0,r.useState)(""),[re,se]=(0,r.useState)(!1),[ie,oe]=(0,r.useState)(!1),[le,de]=(0,r.useState)(null),[ce,ue]=(0,r.useState)(null),[pe,he]=(0,r.useState)(null),[me,ye]=(0,r.useState)(null),[ve,ge]=(0,r.useState)(new Set),[Ae,fe]=(0,r.useState)({open:!1,request:null}),[xe,je]=(0,r.useState)(new Map),[be,we]=(0,r.useState)(new Set),[ke,Ce]=(0,r.useState)(new Set),[$e,Pe]=(0,r.useState)(!1),[Se,qe]=(0,r.useState)(null),Ee=(null===(n=U.metadata.annotations)||void 0===n?void 0:n["kuadrant.io/apiproduct"])||U.metadata.name,ze=(null===(t=U.metadata.annotations)||void 0===t?void 0:t["kuadrant.io/namespace"])||e||"default";(0,s.A)(async()=>{const e=await F.getBackstageIdentity(),n=await F.getProfileInfo();ne(e.userEntityRef),ae(n.email||"")},[F]);const{value:Re,loading:Te,error:Ie}=(0,s.A)(async()=>((await X.getRequestsByNamespace(ze)).items||[]).filter(e=>e.spec.apiProductRef.name===Ee&&e.metadata.namespace===ze),[Ee,ze,Y,X]),{value:Ke,loading:Be,error:_e}=(0,s.A)(async()=>{var e;return null===(e=(await X.getApiProducts()).items)||void 0===e?void 0:e.find(e=>e.metadata.namespace===ze&&e.metadata.name===Ee)},[ze,Ee,X]),He=Ke?`apiproduct:${Ke.metadata.namespace}/${Ke.metadata.name}`:void 0,{allowed:De,loading:Me,error:We}=(0,H.l)(_.dp,He),{allowed:Ne,loading:Ve,error:Ge}=(0,H.l)(_.uL),{allowed:Le,loading:Ue,error:Xe}=(0,H.l)(_.jH),{allowed:Fe,loading:Oe,error:Je}=(0,H.l)(_.q0),Qe=()=>{ue(null),he(null)},Ye=()=>{pe&&(de(pe),oe(!0),Qe())},Ze=()=>{if(!pe)return;const e=pe;Qe(),fe({open:!0,request:e})},en=e=>{Q(n=>{const t=new Set(n);return t.has(e)?t.delete(e):t.add(e),t})},nn=(0,r.useMemo)(()=>[{render:e=>{var n;const t=e.rowData;if(!(null==t||null===(n=t.metadata)||void 0===n?void 0:n.name))return(0,a.jsx)(c.A,{});const r=`${t.metadata.namespace}/${t.metadata.name}`,s=xe.get(r);return(0,a.jsx)(tn,{request:t,apiName:Ee,revealedApiKey:s})}}],[Ee,xe]),tn=({request:e,apiName:n,revealedApiKey:t})=>{var s,o,l;const[d,m]=(0,r.useState)(0),y=(null===(s=e.status)||void 0===s?void 0:s.apiHostname)||`${n}.apps.example.com`,v=t||"<your-api-key>",g=function(e,n,t){const a=`https://${n}/api/v1/endpoint`;return e?e.authorizationHeader?W(a,t,e.authorizationHeader.prefix||""):e.customHeader?function(e,n,t,a){return{curl:`curl -X GET ${e} \\\n -H "${t}: ${a?`${a}${n}`:n}"`,nodejs:`const fetch = require('node-fetch');\n\nconst apiKey = '${n}';\nconst endpoint = '${e}';\n\nfetch(endpoint, {\n method: 'GET',\n headers: {\n '${t}': '${a}' + apiKey\n }\n})\n .then(response => response.json())\n .then(data => console.log(data))\n .catch(error => console.error('Error:', error));`,python:`import requests\n\napi_key = '${n}'\nendpoint = '${e}'\n\nheaders = {\n '${t}': '${a}' + api_key\n}\n\nresponse = requests.get(endpoint, headers=headers)\nprint(response.json())`,go:`package main\n\nimport (\n "fmt"\n "net/http"\n "io"\n)\n\nfunc main() {\n apiKey := "${n}"\n endpoint := "${e}"\n\n client := &http.Client{}\n req, _ := http.NewRequest("GET", endpoint, nil)\n req.Header.Add("${t}", "${a}" + apiKey)\n\n resp, err := client.Do(req)\n if err != nil {\n fmt.Println("Error:", err)\n return\n }\n defer resp.Body.Close()\n\n body, _ := io.ReadAll(resp.Body)\n fmt.Println(string(body))\n}`}}(a,t,e.customHeader.name,e.customHeader.prefix||""):e.queryString?function(e,n,t){return{curl:`curl -X GET "${e}?${t}=${n}"`,nodejs:`const fetch = require('node-fetch');\n\nconst apiKey = '${n}';\nconst endpoint = '${e}?${t}=' + apiKey;\n\nfetch(endpoint, {\n method: 'GET'\n})\n .then(response => response.json())\n .then(data => console.log(data))\n .catch(error => console.error('Error:', error));`,python:`import requests\n\napi_key = '${n}'\nendpoint = '${e}'\n\nparams = {\n '${t}': api_key\n}\n\nresponse = requests.get(endpoint, params=params)\nprint(response.json())`,go:`package main\n\nimport (\n "fmt"\n "net/http"\n "io"\n)\n\nfunc main() {\n apiKey := "${n}"\n endpoint := "${e}?${t}=" + apiKey\n\n client := &http.Client{}\n req, _ := http.NewRequest("GET", endpoint, nil)\n\n resp, err := client.Do(req)\n if err != nil {\n fmt.Println("Error:", err)\n return\n }\n defer resp.Body.Close()\n\n body, _ := io.ReadAll(resp.Body)\n fmt.Println(string(body))\n}`}}(a,t,e.queryString.name):e.cookie?function(e,n,t){return{curl:`curl -X GET ${e} \\\n --cookie "${t}=${n}"`,nodejs:`const fetch = require('node-fetch');\n\nconst apiKey = '${n}';\nconst endpoint = '${e}';\n\nfetch(endpoint, {\n method: 'GET',\n headers: {\n 'Cookie': '${t}=' + apiKey\n }\n})\n .then(response => response.json())\n .then(data => console.log(data))\n .catch(error => console.error('Error:', error));`,python:`import requests\n\napi_key = '${n}'\nendpoint = '${e}'\n\ncookies = {\n '${t}': api_key\n}\n\nresponse = requests.get(endpoint, cookies=cookies)\nprint(response.json())`,go:`package main\n\nimport (\n "fmt"\n "net/http"\n "io"\n)\n\nfunc main() {\n apiKey := "${n}"\n endpoint := "${e}"\n\n client := &http.Client{}\n req, _ := http.NewRequest("GET", endpoint, nil)\n req.AddCookie(&http.Cookie{\n Name: "${t}",\n Value: apiKey,\n })\n\n resp, err := client.Do(req)\n if err != nil {\n fmt.Println("Error:", err)\n return\n }\n defer resp.Body.Close()\n\n body, _ := io.ReadAll(resp.Body)\n fmt.Println(string(body))\n}`}}(a,t,e.cookie.name):W(a,t,"Bearer"):W(a,t,"Bearer")}(null===(l=e.status)||void 0===l||null===(o=l.authScheme)||void 0===o?void 0:o.credentials,y,v);return(0,a.jsxs)(c.A,{p:3,bgcolor:"background.default",onClick:e=>e.stopPropagation(),children:[e.spec.useCase&&(0,a.jsxs)(c.A,{mb:3,children:[(0,a.jsx)(u.A,{variant:"h6",gutterBottom:!0,children:"Use Case"}),(0,a.jsx)(c.A,{p:2,bgcolor:"background.paper",borderRadius:1,border:"1px solid rgba(0, 0, 0, 0.12)",children:(0,a.jsx)(u.A,{variant:"body2",style:{whiteSpace:"pre-wrap",wordBreak:"break-word",overflowWrap:"break-word"},children:e.spec.useCase})})]}),(0,a.jsx)(u.A,{variant:"h6",gutterBottom:!0,children:"Usage Examples"}),(0,a.jsxs)(u.A,{variant:"body2",paragraph:!0,children:["Use these code examples to test the API with your"," ",e.spec.planTier," tier key."]}),(0,a.jsx)(c.A,{onClick:e=>e.stopPropagation(),children:(0,a.jsxs)(p.A,{value:d,onChange:(e,n)=>{e.stopPropagation(),m(n)},indicatorColor:"primary",children:[(0,a.jsx)(h.A,{label:"cURL",onClick:e=>e.stopPropagation()}),(0,a.jsx)(h.A,{label:"Node.js",onClick:e=>e.stopPropagation()}),(0,a.jsx)(h.A,{label:"Python",onClick:e=>e.stopPropagation()}),(0,a.jsx)(h.A,{label:"Go",onClick:e=>e.stopPropagation()})]})}),(0,a.jsxs)(c.A,{mt:2,children:[0===d&&(0,a.jsx)(i.z,{text:g.curl,language:"bash",showCopyCodeButton:!0}),1===d&&(0,a.jsx)(i.z,{text:g.nodejs,language:"javascript",showCopyCodeButton:!0}),2===d&&(0,a.jsx)(i.z,{text:g.python,language:"python",showCopyCodeButton:!0}),3===d&&(0,a.jsx)(i.z,{text:g.go,language:"go",showCopyCodeButton:!0})]})]})},an=Ie||_e,rn=We||Ge||Xe||Je;if(Te||Be||Me||Ve||Ue||Oe)return(0,a.jsx)(o.k,{});if(an)return(0,a.jsx)(l._,{error:an});if(rn){const e=We?"kuadrant.apikey.create":Ge?"kuadrant.apikey.delete.own":Xe?"kuadrant.apikey.delete.all":Je?"kuadrant.apikey.update.own":"unknown";return(0,a.jsxs)(c.A,{p:2,children:[(0,a.jsxs)(u.A,{color:"error",children:["Unable to check permissions: ",rn.message]}),(0,a.jsxs)(u.A,{variant:"body2",color:"textSecondary",children:["Permission: ",e]}),(0,a.jsx)(u.A,{variant:"body2",color:"textSecondary",children:"Please try again or contact your administrator"})]})}const sn=(Re||[]).filter(e=>!ve.has(e.metadata.name)),on=(null==Ke||null===(V=Ke.status)||void 0===V?void 0:V.discoveredPlans)||[],ln=sn.filter(e=>{var n;return!(null===(n=e.status)||void 0===n?void 0:n.phase)||"Pending"===e.status.phase}),dn=sn.filter(e=>{var n;return"Approved"===(null===(n=e.status)||void 0===n?void 0:n.phase)}),cn=sn.filter(e=>{var n;return"Rejected"===(null===(n=e.status)||void 0===n?void 0:n.phase)}),un=[{title:"Tier",field:"spec.planTier",render:e=>(0,a.jsx)(m.A,{label:e.spec.planTier,color:"primary",size:"small"})},{title:"Approved",field:"status.reviewedAt",render:e=>{var n;return(0,a.jsx)(u.A,{variant:"body2",children:(null===(n=e.status)||void 0===n?void 0:n.reviewedAt)?new Date(e.status.reviewedAt).toLocaleDateString():"-"})}},{title:"API Key",field:"status.secretRef",searchable:!1,filtering:!1,render:e=>{var n,t,r;const s=`${e.metadata.namespace}/${e.metadata.name}`,i=J.has(e.metadata.name),o=be.has(s),l=xe.get(s),d=null===(t=e.status)||void 0===t||null===(n=t.secretRef)||void 0===n?void 0:n.name,p=!1!==(null===(r=e.status)||void 0===r?void 0:r.canReadSecret),h=ke.has(s)||!p;return d?h&&!l?(0,a.jsx)(y.Ay,{title:"This API key has already been viewed and cannot be retrieved again",children:(0,a.jsxs)(c.A,{display:"flex",alignItems:"center",children:[(0,a.jsx)(u.A,{variant:"body2",color:"textSecondary",style:{fontFamily:"monospace",marginRight:8},children:"Already viewed"}),(0,a.jsx)(E.A,{fontSize:"small",color:"disabled"})]})}):(0,a.jsxs)(c.A,{display:"flex",alignItems:"center",children:[(0,a.jsx)(u.A,{variant:"body2",style:{fontFamily:"monospace",marginRight:8},children:o?"Loading...":i&&l?l:"••••••••••••••••"}),i&&l&&(0,a.jsx)(y.Ay,{title:"Copy to clipboard",children:(0,a.jsx)(v.A,{size:"small",onClick:async()=>{l&&(await navigator.clipboard.writeText(l),O.post({message:"API key copied to clipboard",severity:"success",display:"transient"}))},children:(0,a.jsx)(K.A,{fontSize:"small"})})}),(0,a.jsx)(y.Ay,{title:i?"Hide API key":"Reveal API key (one-time only)",children:(0,a.jsx)("span",{children:(0,a.jsx)(v.A,{size:"small",onClick:()=>{i?(((e,n)=>{const t=`${e}/${n}`;je(e=>{const n=new Map(e);return n.delete(t),n})})(e.metadata.namespace,e.metadata.name),en(e.metadata.name)):h||(qe({namespace:e.metadata.namespace,name:e.metadata.name}),Pe(!0))},disabled:o||h&&!l,children:i?(0,a.jsx)(E.A,{}):(0,a.jsx)(q.A,{})})})})]}):(0,a.jsx)(u.A,{variant:"body2",color:"textSecondary",children:"Awaiting secret..."})}},{title:"",field:"actions",searchable:!1,filtering:!1,render:e=>{if(me===e.metadata.name)return(0,a.jsx)(g.A,{size:20});const n=e.spec.requestedBy.userId;return(0,H.W)(n,ee,Ne,Le)?(0,a.jsx)(v.A,{size:"small",onClick:n=>{n.stopPropagation();const t=n.currentTarget.getBoundingClientRect();ue({top:t.bottom,left:t.left}),he(e)},title:"Actions","aria-controls":ce?"actions-menu":void 0,"aria-haspopup":"true",children:(0,a.jsx)(I.A,{})}):null}}],pn=[{title:"Status",field:"status.phase",render:e=>{var n;const t=(null===(n=e.status)||void 0===n?void 0:n.phase)||"Pending",r="Pending"===t;return(0,a.jsx)(m.A,{label:t,size:"small",icon:r?(0,a.jsx)(z.A,{}):(0,a.jsx)(R.A,{}),color:r?"default":"secondary"})}},{title:"Tier",field:"spec.planTier",render:e=>(0,a.jsx)(m.A,{label:e.spec.planTier,color:"primary",size:"small"})},{title:"Use Case",field:"spec.useCase",render:e=>e.spec.useCase?(0,a.jsx)(y.Ay,{title:e.spec.useCase,placement:"top",children:(0,a.jsx)(u.A,{variant:"body2",style:{maxWidth:"200px",overflow:"hidden",textOverflow:"ellipsis",whiteSpace:"nowrap"},children:e.spec.useCase})}):(0,a.jsx)(u.A,{variant:"body2",children:"-"})},{title:"Requested",field:"metadata.creationTimestamp",render:e=>(0,a.jsx)(u.A,{variant:"body2",children:e.metadata.creationTimestamp?new Date(e.metadata.creationTimestamp).toLocaleDateString():"-"})},{title:"Reviewed",field:"status.reviewedAt",render:e=>{var n;return(null===(n=e.status)||void 0===n?void 0:n.reviewedAt)?(0,a.jsx)(u.A,{variant:"body2",children:new Date(e.status.reviewedAt).toLocaleDateString()}):(0,a.jsx)(u.A,{variant:"body2",children:"-"})}},{title:"",field:"actions",searchable:!1,filtering:!1,render:e=>{var n;if(me===e.metadata.name)return(0,a.jsx)(g.A,{size:20});const t=!(null===(n=e.status)||void 0===n?void 0:n.phase)||"Pending"===e.status.phase,r=e.spec.requestedBy.userId,s=(0,H.W)(r,ee,Ne,Le);return t&&(Fe&&r===ee||s)?(0,a.jsx)(v.A,{size:"small",onClick:n=>{n.stopPropagation();const t=n.currentTarget.getBoundingClientRect();ue({top:t.bottom,left:t.left}),he(e)},title:"Actions","aria-controls":ce?"actions-menu":void 0,"aria-haspopup":"true",children:(0,a.jsx)(I.A,{})}):null}}],hn=pn.filter(e=>"Reviewed"!==e.title&&"Reason"!==e.title);return(0,a.jsxs)(c.A,{p:2,children:[(0,a.jsxs)(A.A,{container:!0,spacing:3,direction:"column",children:[De&&(0,a.jsx)(A.A,{item:!0,children:(0,a.jsxs)(c.A,{display:"flex",flexDirection:"column",alignItems:"flex-end",mb:2,children:[(0,a.jsx)(f.A,{variant:"contained",color:"primary",startIcon:(0,a.jsx)(T.A,{}),onClick:()=>se(!0),disabled:0===on.length,"data-testid":"request-api-access-button","data-plans-count":on.length,children:"Request API Access"}),0===on.length&&(0,a.jsx)(u.A,{variant:"caption",color:"textSecondary",style:{marginTop:4},"data-testid":"no-plans-message",children:Ke?(()=>{var e,n,t,a;const r=null===(n=Ke.status)||void 0===n||null===(e=n.conditions)||void 0===e?void 0:e.find(e=>"Ready"===e.type),s=null===(a=Ke.status)||void 0===a||null===(t=a.conditions)||void 0===t?void 0:t.find(e=>"PlanPolicyDiscovered"===e.type);return"True"!==(null==r?void 0:r.status)?`HTTPRoute not ready: ${(null==r?void 0:r.message)||"unknown"}`:"True"!==(null==s?void 0:s.status)?`No plans discovered: ${(null==s?void 0:s.message)||"no PlanPolicy found"}`:"No plans available"})():"API product not found"})]})}),0===ln.length&&0===cn.length&&0===dn.length&&(0,a.jsx)(A.A,{item:!0,children:(0,a.jsx)(c.A,{p:3,textAlign:"center",children:(0,a.jsx)(u.A,{variant:"body1",color:"textSecondary",children:"No API keys yet. Request access to get started."})})}),ln.length>0&&(0,a.jsx)(A.A,{item:!0,children:(0,a.jsx)(d.X,{title:"Pending Requests",options:{paging:ln.length>5,pageSize:20,search:!0,filtering:!0,debounceInterval:300,toolbar:!0,emptyRowsWhenPaging:!1},columns:hn,data:ln})}),cn.length>0&&(0,a.jsx)(A.A,{item:!0,children:(0,a.jsx)(d.X,{title:"Rejected Requests",options:{paging:cn.length>5,pageSize:20,search:!0,filtering:!0,debounceInterval:300,toolbar:!0,emptyRowsWhenPaging:!1},columns:pn,data:cn})}),dn.length>0&&(0,a.jsx)(A.A,{item:!0,children:(0,a.jsx)(d.X,{title:"API Keys",options:{paging:dn.length>5,pageSize:20,search:!0,filtering:!0,debounceInterval:300,toolbar:!0,emptyRowsWhenPaging:!1},columns:un,data:dn,detailPanel:nn},"api-keys-table")})]}),(0,a.jsx)(N.c,{open:re,onClose:()=>se(!1),onSuccess:()=>{se(!1),Z(e=>e+1)},apiProductName:Ee,namespace:ze,userEmail:te,plans:on}),(0,a.jsx)(x.A,{id:"actions-menu",open:Boolean(ce),onClose:Qe,anchorReference:"anchorPosition",anchorPosition:ce||{top:0,left:0},children:pe&&(()=>{var e;const n=!(null===(e=pe.status)||void 0===e?void 0:e.phase)||"Pending"===pe.status.phase,t=pe.spec.requestedBy.userId,r=[];return Fe&&t===ee&&n&&r.push((0,a.jsx)(j.A,{onClick:Ye,children:"Edit"},"edit")),r.push((0,a.jsx)(j.A,{onClick:Ze,children:"Delete"},"delete")),r})()}),le&&(0,a.jsx)(D.e,{open:ie,onClose:()=>{oe(!1),de(null)},onSuccess:()=>{Z(e=>e+1),oe(!1),O.post({message:"API key updated",severity:"success",display:"transient"}),de(null)},request:le,availablePlans:on}),(0,a.jsx)(M.K,{open:Ae.open,title:"Delete Request",description:`Are you sure you want to delete this ${"Approved"===(null===(L=Ae.request)||void 0===L||null===(G=L.status)||void 0===G?void 0:G.phase)?"API key":"request"}?`,deleting:null!==me,onConfirm:async()=>{Ae.request&&(await(async e=>{ge(n=>new Set(n).add(e)),ye(e);try{await X.deleteRequest(ze,e),O.post({message:"API key deleted successfully",severity:"success",display:"transient"}),Z(e=>e+1)}catch(n){const t=n instanceof Error?n.message:"unknown error occurred";ge(n=>{const t=new Set(n);return t.delete(e),t}),O.post({message:`Failed to delete API key: ${t}`,severity:"error",display:"transient"})}finally{ye(null)}})(Ae.request.metadata.name),fe({open:!1,request:null}))},onCancel:()=>{fe({open:!1,request:null})}}),(0,a.jsxs)(b.A,{open:$e,onClose:()=>{Pe(!1),qe(null)},maxWidth:"sm",children:[(0,a.jsx)(w.A,{children:(0,a.jsxs)(c.A,{display:"flex",alignItems:"center",children:[(0,a.jsx)(B.A,{color:"primary",style:{marginRight:8}}),"View API Key"]})}),(0,a.jsxs)(k.A,{children:[(0,a.jsxs)(u.A,{variant:"body1",paragraph:!0,children:["This API key can only be viewed ",(0,a.jsx)("strong",{children:"once"}),". After you reveal it, you will not be able to retrieve it again."]}),(0,a.jsx)(u.A,{variant:"body2",color:"textSecondary",children:"Make sure to copy and store it securely before closing this view."})]}),(0,a.jsxs)(C.A,{children:[(0,a.jsx)(f.A,{onClick:()=>{Pe(!1),qe(null)},children:"Cancel"}),(0,a.jsx)(f.A,{variant:"contained",color:"primary",onClick:()=>{Se&&((async(e,n)=>{const t=`${e}/${n}`;if(!be.has(t)){we(e=>new Set(e).add(t));try{const a=await X.getApiKeySecret(e,n);je(e=>new Map(e).set(t,a.apiKey)),Ce(e=>new Set(e).add(t))}catch(e){const n=e instanceof Error?e.message:"unknown error occurred";n.includes("403")||n.includes("already been viewed")?(Ce(e=>new Set(e).add(t)),O.post({message:"This API key has already been viewed and cannot be retrieved again.",severity:"warning",display:"transient"})):O.post({message:`Failed to fetch api key: ${n}`,severity:"error",display:"transient"})}finally{we(e=>{const n=new Set(e);return n.delete(t),n})}}})(Se.namespace,Se.name),en(Se.name)),Pe(!1),qe(null)},children:"Reveal API Key"})]})]})]})}},25467:(e,n,t)=>{t.d(n,{tN:()=>r}),t(31085),t(22097);var a=t(15427);function r(){const e=(0,a.useVersionedContext)("entity-context");if(!e)throw new Error("Entity context is not available");const n=e.atVersion(1);if(!n)throw new Error("EntityContext v1 not available");if(!n.entity)throw new Error("useEntity hook is being called outside of an EntityLayout where the entity has not been loaded. If this is intentional, please use useAsyncEntity instead.");return{entity:n.entity}}(0,a.createVersionedContext)("entity-context")},32269:(e,n,t)=>{var a=t(4293),r=t(78920);n.A=void 0;var s=r(t(95478)),i=(0,a(t(74044)).default)(s.createElement("path",{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"}),"Visibility");n.A=i},35015:(e,n,t)=>{t.d(n,{A:()=>i});var a=t(95478),r=t(85608),s=t(71581);function i(e,n){void 0===n&&(n=[]);var t=function(e,n,t){void 0===n&&(n=[]),void 0===t&&(t={loading:!1});var i=(0,a.useRef)(0),o=(0,s.A)(),l=(0,a.useState)(t),d=l[0],c=l[1],u=(0,a.useCallback)(function(){for(var n=[],t=0;t<arguments.length;t++)n[t]=arguments[t];var a=++i.current;return d.loading||c(function(e){return(0,r.__assign)((0,r.__assign)({},e),{loading:!0})}),e.apply(void 0,n).then(function(e){return o()&&a===i.current&&c({value:e,loading:!1}),e},function(e){return o()&&a===i.current&&c({error:e,loading:!1}),e})},n);return[d,u]}(e,n,{loading:!0}),i=t[0],o=t[1];return(0,a.useEffect)(function(){o()},[o]),i}},39053:(e,n,t)=>{var a=t(4293),r=t(78920);n.A=void 0;var s=r(t(95478)),i=(0,a(t(74044)).default)(s.createElement("path",{d:"M12 2C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10S17.52 2 12 2zm1 15h-2v-6h2v6zm0-8h-2V7h2v2z"}),"Info");n.A=i},61524:(e,n,t)=>{var a=t(4293),r=t(78920);n.A=void 0;var s=r(t(95478)),i=(0,a(t(74044)).default)(s.createElement("path",{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"}),"VisibilityOff");n.A=i},63221:(e,n,t)=>{t.d(n,{K:()=>v});var a=t(31085),r=t(95478),s=t(76891),i=t(61477),o=t(10394),l=t(46805),d=t(59461),c=t(72501),u=t(16249),p=t(93453),h=t(64947),m=t(78467),y=t(77225);const v=({open:e,title:n,description:t,confirmText:v,severity:g="normal",deleting:A=!1,onConfirm:f,onCancel:x})=>{const[j,b]=(0,r.useState)("");(0,r.useEffect)(()=>{e||b("")},[e]);const w="high"===g&&v,k=!w||j===v;return(0,a.jsxs)(s.A,{open:e,onClose:A?void 0:x,maxWidth:"sm",fullWidth:!0,children:[(0,a.jsxs)(i.A,{children:["high"===g&&(0,a.jsxs)(o.A,{display:"flex",alignItems:"center",style:{gap:8},children:[(0,a.jsx)(y.A,{color:"error"}),(0,a.jsx)("span",{children:n})]}),"high"!==g&&n]}),(0,a.jsxs)(l.A,{children:[(0,a.jsx)(d.A,{style:{whiteSpace:"pre-line"},children:t}),w&&(0,a.jsxs)(o.A,{mt:2,children:[(0,a.jsxs)(c.A,{variant:"body2",color:"textSecondary",gutterBottom:!0,children:["Type ",(0,a.jsx)("strong",{children:v})," to confirm:"]}),(0,a.jsx)(u.A,{fullWidth:!0,variant:"outlined",size:"small",value:j,onChange:e=>b(e.target.value),disabled:A,autoFocus:!0,placeholder:v})]})]}),(0,a.jsxs)(p.A,{children:[(0,a.jsx)(h.A,{onClick:x,disabled:A,children:"Cancel"}),(0,a.jsx)(h.A,{onClick:()=>{k&&f()},color:"secondary",variant:"contained",disabled:A||!k,startIcon:A?(0,a.jsx)(m.A,{size:16,color:"inherit"}):void 0,children:A?"Deleting...":"Delete"})]})]})}},71407:(e,n,t)=>{var a=t(4293),r=t(78920);n.A=void 0;var s=r(t(95478)),i=(0,a(t(74044)).default)(s.createElement("path",{d:"M12 8c1.1 0 2-.9 2-2s-.9-2-2-2-2 .9-2 2 .9 2 2 2zm0 2c-1.1 0-2 .9-2 2s.9 2 2 2 2-.9 2-2-.9-2-2-2zm0 6c-1.1 0-2 .9-2 2s.9 2 2 2 2-.9 2-2-.9-2-2-2z"}),"MoreVert");n.A=i},77225:(e,n,t)=>{var a=t(4293),r=t(78920);n.A=void 0;var s=r(t(95478)),i=(0,a(t(74044)).default)(s.createElement("path",{d:"M1 21h22L12 2 1 21zm12-3h-2v-2h2v2zm0-4h-2v-4h2v4z"}),"Warning");n.A=i},93246:(e,n,t)=>{var a=t(4293),r=t(78920);n.A=void 0;var s=r(t(95478)),i=(0,a(t(74044)).default)(s.createElement("path",{d:"M6 2v6h.01L6 8.01 10 12l-4 4 .01.01H6V22h12v-5.99h-.01L18 16l-4-4 4-3.99-.01-.01H18V2H6zm10 14.5V20H8v-3.5l4-4 4 4zm-4-5l-4-4V4h8v3.5l-4 4z"}),"HourglassEmpty");n.A=i}}]);
2
- //# sourceMappingURL=9838.5df9b1de.chunk.js.map
2
+ //# sourceMappingURL=9838.966ce0a0.chunk.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"static/9838.966ce0a0.chunk.js","mappings":"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","sources":["webpack://internal.plugin-kuadrant/./src/components/EditAPIKeyDialog/EditAPIKeyDialog.tsx","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/icons/Add.js","webpack://internal.plugin-kuadrant/./src/utils/codeSnippets.ts","webpack://internal.plugin-kuadrant/./src/components/ApiKeyManagementTab/ApiKeyManagementTab.tsx","webpack://internal.plugin-kuadrant/./node_modules/@backstage/plugin-catalog-react/dist/hooks/useEntity.esm.js","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/icons/Visibility.js","webpack://internal.plugin-kuadrant/../../node_modules/react-use/esm/useAsync.js","webpack://internal.plugin-kuadrant/../../node_modules/react-use/esm/useAsyncFn.js","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/icons/Info.js","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/icons/VisibilityOff.js","webpack://internal.plugin-kuadrant/./src/components/ConfirmDeleteDialog/ConfirmDeleteDialog.tsx","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/icons/MoreVert.js","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/icons/Warning.js","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/icons/HourglassEmpty.js"],"sourcesContent":["import React, { useState, useEffect } from \"react\";\nimport {\n Dialog,\n DialogTitle,\n DialogContent,\n DialogActions,\n Button,\n TextField,\n Box,\n Typography,\n FormControl,\n InputLabel,\n Select,\n MenuItem,\n 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: \"M19 13h-6v6h-2v-6H5v-2h6V5h2v6h6v2z\"\n}), 'Add');\n\nexports.default = _default;","import { Credentials } from '../types/api-management';\n\nexport interface CodeSnippets {\n curl: string;\n nodejs: string;\n python: string;\n go: string;\n}\n\nexport function generateAuthCodeSnippets(\n credentials: Credentials | undefined,\n hostname: string,\n apiKey: string,\n): CodeSnippets {\n const baseUrl = `https://${hostname}/api/v1/endpoint`;\n\n if (!credentials) {\n return generateAuthorizationHeaderSnippets(baseUrl, apiKey, 'Bearer');\n }\n\n if (credentials.authorizationHeader) {\n const prefix = credentials!.authorizationHeader!.prefix || '';\n return generateAuthorizationHeaderSnippets(baseUrl, apiKey, prefix);\n }\n\n if (credentials.customHeader) {\n const headerName = credentials!.customHeader!.name;\n const prefix = credentials!.customHeader!.prefix || '';\n return generateCustomHeaderSnippets(baseUrl, apiKey, headerName, prefix);\n }\n\n\n if (credentials.queryString) {\n const paramName = credentials!.queryString!.name;\n return generateQueryStringSnippets(baseUrl, apiKey, paramName);\n }\n\n if (credentials.cookie) {\n const cookieName = credentials!.cookie!.name;\n return generateCookieSnippets(baseUrl, apiKey, cookieName);\n }\n // Default to Authorization Bearer if no authScheme specified\n return generateAuthorizationHeaderSnippets(baseUrl, apiKey, 'Bearer');\n}\n\nfunction generateAuthorizationHeaderSnippets(\n baseUrl: string,\n apiKey: string,\n prefix: string,\n): CodeSnippets {\n const authValue = prefix ? `${prefix} ${apiKey}` : apiKey;\n const prefixWithSpace = prefix ? `${prefix} ` : '';\n\n return {\n curl: `curl -X GET ${baseUrl} \\\\\n -H \"Authorization: ${authValue}\"`,\n\n nodejs: `const fetch = require('node-fetch');\n\nconst apiKey = '${apiKey}';\nconst endpoint = '${baseUrl}';\n\nfetch(endpoint, {\n method: 'GET',\n headers: {\n 'Authorization': '${prefixWithSpace}' + apiKey\n }\n})\n .then(response => response.json())\n .then(data => console.log(data))\n .catch(error => console.error('Error:', error));`,\n\n python: `import requests\n\napi_key = '${apiKey}'\nendpoint = '${baseUrl}'\n\nheaders = {\n 'Authorization': '${prefixWithSpace}' + api_key\n}\n\nresponse = requests.get(endpoint, headers=headers)\nprint(response.json())`,\n\n go: `package main\n\nimport (\n \"fmt\"\n \"net/http\"\n \"io\"\n)\n\nfunc main() {\n apiKey := \"${apiKey}\"\n endpoint := \"${baseUrl}\"\n\n client := &http.Client{}\n req, _ := http.NewRequest(\"GET\", endpoint, nil)\n req.Header.Add(\"Authorization\", \"${prefixWithSpace}\" + apiKey)\n\n resp, err := client.Do(req)\n if err != nil {\n fmt.Println(\"Error:\", err)\n return\n }\n defer resp.Body.Close()\n\n body, _ := io.ReadAll(resp.Body)\n fmt.Println(string(body))\n}`,\n };\n}\n\nfunction generateCustomHeaderSnippets(\n baseUrl: string,\n apiKey: string,\n headerName: string,\n prefix: string,\n): CodeSnippets {\n const headerValue = prefix ? `${prefix}${apiKey}` : apiKey;\n\n return {\n curl: `curl -X GET ${baseUrl} \\\\\n -H \"${headerName}: ${headerValue}\"`,\n\n nodejs: `const fetch = require('node-fetch');\n\nconst apiKey = '${apiKey}';\nconst endpoint = '${baseUrl}';\n\nfetch(endpoint, {\n method: 'GET',\n headers: {\n '${headerName}': '${prefix}' + apiKey\n }\n})\n .then(response => response.json())\n .then(data => console.log(data))\n .catch(error => console.error('Error:', error));`,\n\n python: `import requests\n\napi_key = '${apiKey}'\nendpoint = '${baseUrl}'\n\nheaders = {\n '${headerName}': '${prefix}' + api_key\n}\n\nresponse = requests.get(endpoint, headers=headers)\nprint(response.json())`,\n\n go: `package main\n\nimport (\n \"fmt\"\n \"net/http\"\n \"io\"\n)\n\nfunc main() {\n apiKey := \"${apiKey}\"\n endpoint := \"${baseUrl}\"\n\n client := &http.Client{}\n req, _ := http.NewRequest(\"GET\", endpoint, nil)\n req.Header.Add(\"${headerName}\", \"${prefix}\" + apiKey)\n\n resp, err := client.Do(req)\n if err != nil {\n fmt.Println(\"Error:\", err)\n return\n }\n defer resp.Body.Close()\n\n body, _ := io.ReadAll(resp.Body)\n fmt.Println(string(body))\n}`,\n };\n}\n\nfunction generateQueryStringSnippets(\n baseUrl: string,\n apiKey: string,\n paramName: string,\n): CodeSnippets {\n const urlWithParam = `${baseUrl}?${paramName}=${apiKey}`;\n\n return {\n curl: `curl -X GET \"${urlWithParam}\"`,\n\n nodejs: `const fetch = require('node-fetch');\n\nconst apiKey = '${apiKey}';\nconst endpoint = '${baseUrl}?${paramName}=' + apiKey;\n\nfetch(endpoint, {\n method: 'GET'\n})\n .then(response => response.json())\n .then(data => console.log(data))\n .catch(error => console.error('Error:', error));`,\n\n python: `import requests\n\napi_key = '${apiKey}'\nendpoint = '${baseUrl}'\n\nparams = {\n '${paramName}': api_key\n}\n\nresponse = requests.get(endpoint, params=params)\nprint(response.json())`,\n\n go: `package main\n\nimport (\n \"fmt\"\n \"net/http\"\n \"io\"\n)\n\nfunc main() {\n apiKey := \"${apiKey}\"\n endpoint := \"${baseUrl}?${paramName}=\" + apiKey\n\n client := &http.Client{}\n req, _ := http.NewRequest(\"GET\", endpoint, nil)\n\n resp, err := client.Do(req)\n if err != nil {\n fmt.Println(\"Error:\", err)\n return\n }\n defer resp.Body.Close()\n\n body, _ := io.ReadAll(resp.Body)\n fmt.Println(string(body))\n}`,\n };\n}\n\nfunction generateCookieSnippets(\n baseUrl: string,\n apiKey: string,\n cookieName: string,\n): CodeSnippets {\n return {\n curl: `curl -X GET ${baseUrl} \\\\\n --cookie \"${cookieName}=${apiKey}\"`,\n\n nodejs: `const fetch = require('node-fetch');\n\nconst apiKey = '${apiKey}';\nconst endpoint = '${baseUrl}';\n\nfetch(endpoint, {\n method: 'GET',\n headers: {\n 'Cookie': '${cookieName}=' + apiKey\n }\n})\n .then(response => response.json())\n .then(data => console.log(data))\n .catch(error => console.error('Error:', error));`,\n\n python: `import requests\n\napi_key = '${apiKey}'\nendpoint = '${baseUrl}'\n\ncookies = {\n '${cookieName}': api_key\n}\n\nresponse = requests.get(endpoint, cookies=cookies)\nprint(response.json())`,\n\n go: `package main\n\nimport (\n \"fmt\"\n \"net/http\"\n \"io\"\n)\n\nfunc main() {\n apiKey := \"${apiKey}\"\n endpoint := \"${baseUrl}\"\n\n client := &http.Client{}\n req, _ := http.NewRequest(\"GET\", endpoint, nil)\n req.AddCookie(&http.Cookie{\n Name: \"${cookieName}\",\n Value: apiKey,\n })\n\n resp, err := client.Do(req)\n if err != nil {\n fmt.Println(\"Error:\", err)\n return\n }\n defer resp.Body.Close()\n\n body, _ := io.ReadAll(resp.Body)\n fmt.Println(string(body))\n}`,\n };\n}\n","import React, { useState, useMemo } from \"react\";\nimport { useAsync } from \"react-use\";\nimport {\n Table,\n TableColumn,\n Progress,\n ResponseErrorPanel,\n CodeSnippet,\n} from \"@backstage/core-components\";\nimport {\n IconButton,\n Typography,\n Box,\n Chip,\n Grid,\n Button,\n Dialog,\n DialogTitle,\n DialogContent,\n DialogActions,\n Tabs,\n Tab,\n Menu,\n MenuItem,\n Tooltip,\n CircularProgress,\n} from \"@material-ui/core\";\nimport {\n useApi,\n identityApiRef,\n alertApiRef,\n} from \"@backstage/core-plugin-api\";\nimport { kuadrantApiRef } from '../../api';\nimport { useEntity } from \"@backstage/plugin-catalog-react\";\nimport VisibilityIcon from \"@material-ui/icons/Visibility\";\nimport VisibilityOffIcon from \"@material-ui/icons/VisibilityOff\";\nimport HourglassEmptyIcon from \"@material-ui/icons/HourglassEmpty\";\nimport CancelIcon from \"@material-ui/icons/Cancel\";\nimport AddIcon from \"@material-ui/icons/Add\";\nimport MoreVertIcon from \"@material-ui/icons/MoreVert\";\nimport FileCopyIcon from \"@material-ui/icons/FileCopy\";\nimport WarningIcon from \"@material-ui/icons/Warning\";\nimport { APIKey, APIProduct, Plan } from \"../../types/api-management\";\nimport {\n kuadrantApiKeyCreatePermission,\n kuadrantApiKeyDeleteOwnPermission,\n kuadrantApiKeyDeleteAllPermission,\n kuadrantApiKeyUpdateOwnPermission,\n} from \"../../permissions\";\nimport {\n useKuadrantPermission,\n canDeleteResource,\n} from \"../../utils/permissions\";\nimport { EditAPIKeyDialog } from \"../EditAPIKeyDialog\";\nimport { ConfirmDeleteDialog } from \"../ConfirmDeleteDialog\";\nimport { generateAuthCodeSnippets } from \"../../utils/codeSnippets\";\nimport { RequestAccessDialog } from \"../RequestAccessDialog\";\n\nexport interface ApiKeyManagementTabProps {\n namespace?: string;\n}\n\nexport const ApiKeyManagementTab = ({\n namespace: propNamespace,\n}: ApiKeyManagementTabProps) => {\n const { entity } = useEntity();\n const kuadrantApi = useApi(kuadrantApiRef);\n const identityApi = useApi(identityApiRef);\n const alertApi = useApi(alertApiRef);\n const [visibleKeys, setVisibleKeys] = useState<Set<string>>(new Set());\n const [refresh, setRefresh] = useState(0);\n const [userId, setUserId] = useState<string>(\"\");\n const [userEmail, setUserEmail] = useState<string>(\"\");\n const [requestDialogOpen, setRequestDialogOpen] = useState(false);\n const [editDialogOpen, setEditDialogOpen] = useState(false);\n const [requestToEdit, setRequestToEdit] = useState<APIKey | null>(null);\n const [menuAnchor, setMenuAnchor] = useState<{\n top: number;\n left: number;\n } | null>(null);\n const [menuRequest, setMenuRequest] = useState<APIKey | null>(null);\n const [deleting, setDeleting] = useState<string | null>(null);\n const [optimisticallyDeleted, setOptimisticallyDeleted] = useState<\n Set<string>\n >(new Set());\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\n // get apiproduct name from entity annotation (set by entity provider)\n const apiProductName =\n entity.metadata.annotations?.[\"kuadrant.io/apiproduct\"] ||\n entity.metadata.name;\n const namespace =\n entity.metadata.annotations?.[\"kuadrant.io/namespace\"] ||\n propNamespace ||\n \"default\";\n\n useAsync(async () => {\n const identity = await identityApi.getBackstageIdentity();\n const profile = await identityApi.getProfileInfo();\n setUserId(identity.userEntityRef);\n setUserEmail(profile.email || \"\");\n }, [identityApi]);\n\n const {\n value: requests,\n loading: requestsLoading,\n error: requestsError,\n } = useAsync(async () => {\n const data = await kuadrantApi.getRequestsByNamespace(namespace);\n // filter by apiproduct name, not httproute name\n return (data.items || []).filter(\n (r: APIKey) =>\n r.spec.apiProductRef.name === apiProductName &&\n r.metadata.namespace === namespace, // APIProducts and APIKeys (and its Secret) will be in the same NS\n );\n }, [apiProductName, namespace, refresh, kuadrantApi]);\n\n const {\n value: apiProduct,\n loading: plansLoading,\n error: plansError,\n } = useAsync(async () => {\n const data = await kuadrantApi.getApiProducts();\n\n const product = data.items?.find(\n (p: APIProduct) =>\n p.metadata.namespace === namespace &&\n p.metadata.name === apiProductName,\n );\n\n return product;\n }, [namespace, apiProductName, kuadrantApi]);\n\n // check permissions with resource reference once we have the apiproduct\n const resourceRef = apiProduct\n ? `apiproduct:${apiProduct.metadata.namespace}/${apiProduct.metadata.name}`\n : undefined;\n\n const {\n allowed: canCreateRequest,\n loading: createRequestPermissionLoading,\n error: createRequestPermissionError,\n } = useKuadrantPermission(kuadrantApiKeyCreatePermission, resourceRef);\n\n const {\n allowed: canDeleteOwnKey,\n loading: deleteOwnPermissionLoading,\n error: deleteOwnPermissionError,\n } = useKuadrantPermission(kuadrantApiKeyDeleteOwnPermission);\n\n const {\n allowed: canDeleteAllKeys,\n loading: deleteAllPermissionLoading,\n error: deleteAllPermissionError,\n } = useKuadrantPermission(kuadrantApiKeyDeleteAllPermission);\n\n const {\n allowed: canUpdateRequest,\n loading: updateRequestPermissionLoading,\n error: updateRequestPermissionError,\n } = useKuadrantPermission(kuadrantApiKeyUpdateOwnPermission);\n\n const handleDeleteRequest = async (name: string) => {\n // optimistic update - remove from UI immediately\n setOptimisticallyDeleted((prev) => new Set(prev).add(name));\n setDeleting(name);\n try {\n await kuadrantApi.deleteRequest(namespace, name);\n alertApi.post({\n message: \"API key deleted successfully\",\n severity: \"success\",\n display: \"transient\",\n });\n setRefresh((r) => r + 1);\n } catch (err) {\n const errorMessage =\n err instanceof Error ? err.message : \"unknown error occurred\";\n // rollback optimistic update on error\n setOptimisticallyDeleted((prev) => {\n const next = new Set(prev);\n next.delete(name);\n return next;\n });\n alertApi.post({\n message: `Failed to delete API key: ${errorMessage}`,\n severity: \"error\",\n display: \"transient\",\n });\n } finally {\n setDeleting(null);\n }\n };\n\n const fetchApiKeyFromSecret = async (\n requestNamespace: string,\n requestName: string,\n ) => {\n const key = `${requestNamespace}/${requestName}`;\n if (apiKeyLoading.has(key)) {\n return;\n }\n\n setApiKeyLoading((prev) => new Set(prev).add(key));\n try {\n const data = await kuadrantApi.getApiKeySecret(requestNamespace, requestName);\n setApiKeyValues((prev) => new Map(prev).set(key, data.apiKey));\n // after successful read, mark as already read (show-once behaviour)\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 // secret has already been read\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 handleEditRequest = (request: APIKey) => {\n setRequestToEdit(request);\n setEditDialogOpen(true);\n };\n\n const handleEditSuccess = () => {\n setRefresh((r) => r + 1);\n setEditDialogOpen(false);\n alertApi.post({\n message: \"API key updated\",\n severity: \"success\",\n display: \"transient\",\n });\n setRequestToEdit(null);\n };\n\n const handleMenuClose = () => {\n setMenuAnchor(null);\n setMenuRequest(null);\n };\n\n const handleMenuEdit = () => {\n if (!menuRequest) return;\n handleEditRequest(menuRequest);\n handleMenuClose();\n };\n\n const handleMenuDeleteClick = () => {\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 await handleDeleteRequest(deleteDialogState.request.metadata.name);\n setDeleteDialogState({ open: false, request: null });\n };\n\n const handleDeleteCancel = () => {\n setDeleteDialogState({ open: false, request: null });\n };\n\n const toggleVisibility = (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 detailPanelConfig = useMemo(\n () => [\n {\n render: (data: any) => {\n // backstage Table wraps the data in { rowData: actualData }\n const request = data.rowData as APIKey;\n if (!request?.metadata?.name) {\n return <Box />;\n }\n\n // pass already-revealed key from parent state (don't auto-fetch - that consumes show-once)\n const key = `${request.metadata.namespace}/${request.metadata.name}`;\n const revealedKey = apiKeyValues.get(key);\n return (\n <DetailPanelContent\n request={request}\n apiName={apiProductName}\n revealedApiKey={revealedKey}\n />\n );\n },\n },\n ],\n [apiProductName, apiKeyValues],\n );\n\n // separate component to isolate state\n const DetailPanelContent = ({\n request,\n apiName: api,\n revealedApiKey,\n }: {\n request: APIKey;\n apiName: string;\n revealedApiKey?: string;\n }) => {\n const [selectedLanguage, setSelectedLanguage] = useState(0);\n const hostname = request.status?.apiHostname || `${api}.apps.example.com`;\n\n // use revealed key if available, otherwise show placeholder\n const displayApiKey = revealedApiKey || \"<your-api-key>\";\n\n // Generate code snippets based on authScheme credentials\n const credentials = request.status?.authScheme?.credentials;\n const snippets = generateAuthCodeSnippets(credentials, hostname, displayApiKey);\n\n return (\n <Box\n p={3}\n bgcolor=\"background.default\"\n onClick={(e) => e.stopPropagation()}\n >\n {request.spec.useCase && (\n <Box mb={3}>\n <Typography variant=\"h6\" gutterBottom>\n Use Case\n </Typography>\n <Box\n p={2}\n bgcolor=\"background.paper\"\n borderRadius={1}\n border=\"1px solid rgba(0, 0, 0, 0.12)\"\n >\n <Typography\n variant=\"body2\"\n style={{\n whiteSpace: \"pre-wrap\",\n wordBreak: \"break-word\",\n overflowWrap: \"break-word\",\n }}\n >\n {request.spec.useCase}\n </Typography>\n </Box>\n </Box>\n )}\n <Typography variant=\"h6\" gutterBottom>\n Usage Examples\n </Typography>\n <Typography variant=\"body2\" paragraph>\n Use these code examples to test the API with your{\" \"}\n {request.spec.planTier} tier key.\n </Typography>\n <Box onClick={(e) => e.stopPropagation()}>\n <Tabs\n value={selectedLanguage}\n onChange={(e, newValue) => {\n e.stopPropagation();\n setSelectedLanguage(newValue);\n }}\n indicatorColor=\"primary\"\n >\n <Tab label=\"cURL\" onClick={(e) => e.stopPropagation()} />\n <Tab label=\"Node.js\" onClick={(e) => e.stopPropagation()} />\n <Tab label=\"Python\" onClick={(e) => e.stopPropagation()} />\n <Tab label=\"Go\" onClick={(e) => e.stopPropagation()} />\n </Tabs>\n </Box>\n <Box mt={2}>\n {selectedLanguage === 0 && (\n <CodeSnippet\n text={snippets.curl}\n language=\"bash\"\n showCopyCodeButton\n />\n )}\n {selectedLanguage === 1 && (\n <CodeSnippet\n text={snippets.nodejs}\n language=\"javascript\"\n showCopyCodeButton\n />\n )}\n {selectedLanguage === 2 && (\n <CodeSnippet\n text={snippets.python}\n language=\"python\"\n showCopyCodeButton\n />\n )}\n {selectedLanguage === 3 && (\n <CodeSnippet\n text={snippets.go}\n language=\"go\"\n showCopyCodeButton\n />\n )}\n </Box>\n </Box>\n );\n };\n\n const loading =\n requestsLoading ||\n plansLoading ||\n createRequestPermissionLoading ||\n deleteOwnPermissionLoading ||\n deleteAllPermissionLoading ||\n updateRequestPermissionLoading;\n const error = requestsError || plansError;\n const permissionError =\n createRequestPermissionError ||\n deleteOwnPermissionError ||\n deleteAllPermissionError ||\n updateRequestPermissionError;\n\n if (loading) {\n return <Progress />;\n }\n\n if (error) {\n return <ResponseErrorPanel error={error} />;\n }\n\n if (permissionError) {\n const failedPermission = createRequestPermissionError\n ? \"kuadrant.apikey.create\"\n : deleteOwnPermissionError\n ? \"kuadrant.apikey.delete.own\"\n : deleteAllPermissionError\n ? \"kuadrant.apikey.delete.all\"\n : updateRequestPermissionError\n ? \"kuadrant.apikey.update.own\"\n : \"unknown\";\n return (\n <Box p={2}>\n <Typography color=\"error\">\n Unable to check permissions: {permissionError.message}\n </Typography>\n <Typography variant=\"body2\" color=\"textSecondary\">\n Permission: {failedPermission}\n </Typography>\n <Typography variant=\"body2\" color=\"textSecondary\">\n Please try again or contact your administrator\n </Typography>\n </Box>\n );\n }\n\n const myRequests = ((requests || []) as APIKey[]).filter(\n (r) => !optimisticallyDeleted.has(r.metadata.name),\n );\n const plans = (apiProduct?.status?.discoveredPlans || []) as Plan[];\n\n const pendingRequests = myRequests.filter(\n (r) => !r.status?.phase || r.status.phase === \"Pending\",\n );\n const approvedRequests = myRequests.filter(\n (r) => r.status?.phase === \"Approved\",\n );\n const rejectedRequests = myRequests.filter(\n (r) => r.status?.phase === \"Rejected\",\n );\n\n const approvedColumns: TableColumn<APIKey>[] = [\n {\n title: \"Tier\",\n field: \"spec.planTier\",\n render: (row: APIKey) => (\n <Chip label={row.spec.planTier} color=\"primary\" size=\"small\" />\n ),\n },\n {\n title: \"Approved\",\n field: \"status.reviewedAt\",\n render: (row: APIKey) => (\n <Typography variant=\"body2\">\n {row.status?.reviewedAt\n ? new Date(row.status.reviewedAt).toLocaleDateString()\n : \"-\"}\n </Typography>\n ),\n },\n {\n title: \"API Key\",\n field: \"status.secretRef\",\n searchable: false,\n filtering: false,\n render: (row: APIKey) => {\n const key = `${row.metadata.namespace}/${row.metadata.name}`;\n const isVisible = visibleKeys.has(row.metadata.name);\n const isLoading = apiKeyLoading.has(key);\n const apiKeyValue = apiKeyValues.get(key);\n const hasSecretRef = row.status?.secretRef?.name;\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 // key has already been viewed and cannot be retrieved again\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 // hiding - clear the value from memory\n clearApiKeyValue(row.metadata.namespace, row.metadata.name);\n toggleVisibility(row.metadata.name);\n } else if (!isAlreadyRead) {\n // show warning dialog before first reveal\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\">\n <Typography\n variant=\"body2\"\n style={{\n fontFamily: \"monospace\",\n marginRight: 8,\n }}\n >\n {isLoading\n ? \"Loading...\"\n : isVisible && apiKeyValue\n ? apiKeyValue\n : \"••••••••••••••••\"}\n </Typography>\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 ? <VisibilityOffIcon /> : <VisibilityIcon />}\n </IconButton>\n </span>\n </Tooltip>\n </Box>\n );\n },\n },\n {\n title: \"\",\n field: \"actions\",\n searchable: false,\n filtering: false,\n render: (row: APIKey) => {\n const isDeleting = deleting === row.metadata.name;\n if (isDeleting) {\n return <CircularProgress size={20} />;\n }\n const ownerId = row.spec.requestedBy.userId;\n const canDelete = canDeleteResource(\n ownerId,\n userId,\n canDeleteOwnKey,\n canDeleteAllKeys,\n );\n if (!canDelete) return null;\n return (\n <IconButton\n size=\"small\"\n onClick={(e) => {\n e.stopPropagation();\n const rect = e.currentTarget.getBoundingClientRect();\n setMenuAnchor({ top: rect.bottom, left: rect.left });\n setMenuRequest(row);\n }}\n title=\"Actions\"\n aria-controls={menuAnchor ? \"actions-menu\" : undefined}\n aria-haspopup=\"true\"\n >\n <MoreVertIcon />\n </IconButton>\n );\n },\n },\n ];\n\n const requestColumns: TableColumn<APIKey>[] = [\n {\n title: \"Status\",\n field: \"status.phase\",\n render: (row: APIKey) => {\n const phase = row.status?.phase || \"Pending\";\n const isPending = phase === \"Pending\";\n return (\n <Chip\n label={phase}\n size=\"small\"\n icon={isPending ? <HourglassEmptyIcon /> : <CancelIcon />}\n color={isPending ? \"default\" : \"secondary\"}\n />\n );\n },\n },\n {\n title: \"Tier\",\n field: \"spec.planTier\",\n render: (row: APIKey) => (\n <Chip label={row.spec.planTier} color=\"primary\" size=\"small\" />\n ),\n },\n {\n title: \"Use Case\",\n field: \"spec.useCase\",\n render: (row: APIKey) => {\n if (!row.spec.useCase) {\n return <Typography variant=\"body2\">-</Typography>;\n }\n return (\n <Tooltip title={row.spec.useCase} placement=\"top\">\n <Typography\n variant=\"body2\"\n style={{\n maxWidth: \"200px\",\n overflow: \"hidden\",\n textOverflow: \"ellipsis\",\n whiteSpace: \"nowrap\",\n }}\n >\n {row.spec.useCase}\n </Typography>\n </Tooltip>\n );\n },\n },\n {\n title: \"Requested\",\n field: \"metadata.creationTimestamp\",\n render: (row: APIKey) => (\n <Typography variant=\"body2\">\n {row.metadata.creationTimestamp\n ? new Date(row.metadata.creationTimestamp).toLocaleDateString()\n : \"-\"}\n </Typography>\n ),\n },\n {\n title: \"Reviewed\",\n field: \"status.reviewedAt\",\n render: (row: APIKey) => {\n if (!row.status?.reviewedAt)\n return <Typography variant=\"body2\">-</Typography>;\n return (\n <Typography variant=\"body2\">\n {new Date(row.status.reviewedAt).toLocaleDateString()}\n </Typography>\n );\n },\n },\n {\n title: \"\",\n field: \"actions\",\n searchable: false,\n filtering: false,\n render: (row: APIKey) => {\n const isDeleting = deleting === row.metadata.name;\n if (isDeleting) {\n return <CircularProgress size={20} />;\n }\n const isPending = !row.status?.phase || row.status.phase === \"Pending\";\n const ownerId = row.spec.requestedBy.userId;\n const canDelete = canDeleteResource(\n ownerId,\n userId,\n canDeleteOwnKey,\n canDeleteAllKeys,\n );\n const canEdit = canUpdateRequest && ownerId === userId;\n if (!isPending || (!canEdit && !canDelete)) return null;\n return (\n <IconButton\n size=\"small\"\n onClick={(e) => {\n e.stopPropagation();\n const rect = e.currentTarget.getBoundingClientRect();\n setMenuAnchor({ top: rect.bottom, left: rect.left });\n setMenuRequest(row);\n }}\n title=\"Actions\"\n aria-controls={menuAnchor ? \"actions-menu\" : undefined}\n aria-haspopup=\"true\"\n >\n <MoreVertIcon />\n </IconButton>\n );\n },\n },\n ];\n\n // Filter columns for pending requests (no Reviewed or Reason)\n const pendingRequestColumns = requestColumns.filter(\n (col) => col.title !== \"Reviewed\" && col.title !== \"Reason\",\n );\n\n return (\n <Box p={2}>\n <Grid container spacing={3} direction=\"column\">\n {canCreateRequest && (\n <Grid item>\n <Box\n display=\"flex\"\n flexDirection=\"column\"\n alignItems=\"flex-end\"\n mb={2}\n >\n <Button\n variant=\"contained\"\n color=\"primary\"\n startIcon={<AddIcon />}\n onClick={() => setRequestDialogOpen(true)}\n disabled={plans.length === 0}\n data-testid=\"request-api-access-button\"\n data-plans-count={plans.length}\n >\n Request API Access\n </Button>\n {plans.length === 0 && (\n <Typography\n variant=\"caption\"\n color=\"textSecondary\"\n style={{ marginTop: 4 }}\n data-testid=\"no-plans-message\"\n >\n {!apiProduct\n ? \"API product not found\"\n : (() => {\n const readyCondition =\n apiProduct.status?.conditions?.find(\n (c: any) => c.type === \"Ready\",\n );\n const planCondition =\n apiProduct.status?.conditions?.find(\n (c: any) => c.type === \"PlanPolicyDiscovered\",\n );\n\n if (readyCondition?.status !== \"True\") {\n return `HTTPRoute not ready: ${readyCondition?.message || \"unknown\"}`;\n }\n if (planCondition?.status !== \"True\") {\n return `No plans discovered: ${planCondition?.message || \"no PlanPolicy found\"}`;\n }\n return \"No plans available\";\n })()}\n </Typography>\n )}\n </Box>\n </Grid>\n )}\n {pendingRequests.length === 0 &&\n rejectedRequests.length === 0 &&\n approvedRequests.length === 0 && (\n <Grid item>\n <Box p={3} textAlign=\"center\">\n <Typography variant=\"body1\" color=\"textSecondary\">\n No API keys yet. Request access to get started.\n </Typography>\n </Box>\n </Grid>\n )}\n {pendingRequests.length > 0 && (\n <Grid item>\n <Table\n title=\"Pending Requests\"\n options={{\n paging: pendingRequests.length > 5,\n pageSize: 20,\n search: true,\n filtering: true,\n debounceInterval: 300,\n toolbar: true,\n emptyRowsWhenPaging: false,\n }}\n columns={pendingRequestColumns}\n data={pendingRequests}\n />\n </Grid>\n )}\n {rejectedRequests.length > 0 && (\n <Grid item>\n <Table\n title=\"Rejected Requests\"\n options={{\n paging: rejectedRequests.length > 5,\n pageSize: 20,\n search: true,\n filtering: true,\n debounceInterval: 300,\n toolbar: true,\n emptyRowsWhenPaging: false,\n }}\n columns={requestColumns}\n data={rejectedRequests}\n />\n </Grid>\n )}\n {approvedRequests.length > 0 && (\n <Grid item>\n <Table\n key=\"api-keys-table\"\n title=\"API Keys\"\n options={{\n paging: approvedRequests.length > 5,\n pageSize: 20,\n search: true,\n filtering: true,\n debounceInterval: 300,\n toolbar: true,\n emptyRowsWhenPaging: false,\n }}\n columns={approvedColumns}\n data={approvedRequests}\n detailPanel={detailPanelConfig}\n />\n </Grid>\n )}\n </Grid>\n\n <RequestAccessDialog\n open={requestDialogOpen}\n onClose={() => setRequestDialogOpen(false)}\n onSuccess={() => {\n setRequestDialogOpen(false);\n setRefresh((r) => r + 1);\n }}\n apiProductName={apiProductName}\n namespace={namespace}\n userEmail={userEmail}\n plans={plans}\n />\n\n <Menu\n id=\"actions-menu\"\n open={Boolean(menuAnchor)}\n onClose={handleMenuClose}\n anchorReference=\"anchorPosition\"\n anchorPosition={menuAnchor || { top: 0, left: 0 }}\n >\n {menuRequest &&\n (() => {\n const isPending =\n !menuRequest.status?.phase ||\n menuRequest.status.phase === \"Pending\";\n const ownerId = menuRequest.spec.requestedBy.userId;\n const canEdit = canUpdateRequest && ownerId === userId && isPending;\n\n const items = [];\n if (canEdit) {\n items.push(\n <MenuItem key=\"edit\" onClick={handleMenuEdit}>\n Edit\n </MenuItem>,\n );\n }\n items.push(\n <MenuItem key=\"delete\" onClick={handleMenuDeleteClick}>\n Delete\n </MenuItem>,\n );\n return items;\n })()}\n </Menu>\n\n {requestToEdit && (\n <EditAPIKeyDialog\n open={editDialogOpen}\n onClose={() => {\n setEditDialogOpen(false);\n setRequestToEdit(null);\n }}\n onSuccess={handleEditSuccess}\n request={requestToEdit}\n availablePlans={plans}\n />\n )}\n\n <ConfirmDeleteDialog\n open={deleteDialogState.open}\n title=\"Delete Request\"\n description={`Are you sure you want to delete this ${deleteDialogState.request?.status?.phase === \"Approved\" ? \"API key\" : \"request\"}?`}\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 toggleVisibility(pendingKeyReveal.name);\n }\n setShowOnceWarningOpen(false);\n setPendingKeyReveal(null);\n }}\n >\n Reveal API Key\n </Button>\n </DialogActions>\n </Dialog>\n </Box>\n );\n};\n","import { jsx } from 'react/jsx-runtime';\nimport { stringifyEntityRef } from '@backstage/catalog-model';\nimport { AnalyticsContext } from '@backstage/core-plugin-api';\nimport { createVersionedContext, useVersionedContext, createVersionedValueMap } from '@backstage/version-bridge';\n\nconst NewEntityContext = createVersionedContext(\n \"entity-context\"\n);\nconst AsyncEntityProvider = (props) => {\n const { children, entity, loading, error, refresh } = props;\n const value = { entity, loading, error, refresh };\n return /* @__PURE__ */ jsx(NewEntityContext.Provider, { value: createVersionedValueMap({ 1: value }), children: /* @__PURE__ */ jsx(\n AnalyticsContext,\n {\n attributes: {\n ...entity ? { entityRef: stringifyEntityRef(entity) } : void 0\n },\n children\n }\n ) });\n};\nconst EntityProvider = (props) => /* @__PURE__ */ jsx(\n AsyncEntityProvider,\n {\n entity: props.entity,\n loading: !Boolean(props.entity),\n error: void 0,\n refresh: void 0,\n children: props.children\n }\n);\nfunction useEntity() {\n const versionedHolder = useVersionedContext(\n \"entity-context\"\n );\n if (!versionedHolder) {\n throw new Error(\"Entity context is not available\");\n }\n const value = versionedHolder.atVersion(1);\n if (!value) {\n throw new Error(\"EntityContext v1 not available\");\n }\n if (!value.entity) {\n throw new Error(\n \"useEntity hook is being called outside of an EntityLayout where the entity has not been loaded. If this is intentional, please use useAsyncEntity instead.\"\n );\n }\n return { entity: value.entity };\n}\nfunction useAsyncEntity() {\n const versionedHolder = useVersionedContext(\n \"entity-context\"\n );\n if (!versionedHolder) {\n throw new Error(\"Entity context is not available\");\n }\n const value = versionedHolder.atVersion(1);\n if (!value) {\n throw new Error(\"EntityContext v1 not available\");\n }\n const { entity, loading, error, refresh } = value;\n return { entity, loading, error, refresh };\n}\n\nexport { AsyncEntityProvider, EntityProvider, useAsyncEntity, useEntity };\n//# sourceMappingURL=useEntity.esm.js.map\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 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 { useEffect } from 'react';\nimport useAsyncFn from './useAsyncFn';\nexport default function useAsync(fn, deps) {\n if (deps === void 0) { deps = []; }\n var _a = useAsyncFn(fn, deps, {\n loading: true,\n }), state = _a[0], callback = _a[1];\n useEffect(function () {\n callback();\n }, [callback]);\n return state;\n}\n","import { __assign } from \"tslib\";\nimport { useCallback, useRef, useState } from 'react';\nimport useMountedState from './useMountedState';\nexport default function useAsyncFn(fn, deps, initialState) {\n if (deps === void 0) { deps = []; }\n if (initialState === void 0) { initialState = { loading: false }; }\n var lastCallId = useRef(0);\n var isMounted = useMountedState();\n var _a = useState(initialState), state = _a[0], set = _a[1];\n var callback = 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 (__assign(__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}\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 2C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10S17.52 2 12 2zm1 15h-2v-6h2v6zm0-8h-2V7h2v2z\"\n}), 'Info');\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 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","\"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 8c1.1 0 2-.9 2-2s-.9-2-2-2-2 .9-2 2 .9 2 2 2zm0 2c-1.1 0-2 .9-2 2s.9 2 2 2 2-.9 2-2-.9-2-2-2zm0 6c-1.1 0-2 .9-2 2s.9 2 2 2 2-.9 2-2-.9-2-2-2z\"\n}), 'MoreVert');\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: \"M1 21h22L12 2 1 21zm12-3h-2v-2h2v2zm0-4h-2v-4h2v4z\"\n}), 'Warning');\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: \"M6 2v6h.01L6 8.01 10 12l-4 4 .01.01H6V22h12v-5.99h-.01L18 16l-4-4 4-3.99-.01-.01H18V2H6zm10 14.5V20H8v-3.5l4-4 4 4zm-4-5l-4-4V4h8v3.5l-4 4z\"\n}), 'HourglassEmpty');\n\nexports.default = _default;"],"names":["EditAPIKeyDialog","open","onClose","onSuccess","request","availablePlans","kuadrantApi","useApi","kuadrantApiRef","planTier","setPlanTier","useState","useCase","setUseCase","saving","setSaving","error","setError","useEffect","spec","handleClose","Dialog","maxWidth","fullWidth","DialogTitle","DialogContent","Box","mb","p","bgcolor","color","borderRadius","Typography","variant","FormControl","margin","InputLabel","Select","value","onChange","e","target","disabled","map","plan","limitDesc","Object","entries","limits","key","val","join","MenuItem","tier","TextField","label","placeholder","multiline","rows","helperText","DialogActions","Button","onClick","async","patch","trim","updateRequest","metadata","name","namespace","err","console","Error","message","startIcon","CircularProgress","size","undefined","_interopRequireDefault","_interopRequireWildcard","exports","React","_default","default","createElement","d","generateAuthorizationHeaderSnippets","baseUrl","apiKey","prefix","prefixWithSpace","curl","nodejs","python","go","ApiKeyManagementTab","propNamespace","entity","apiProduct","deleteDialogState","useEntity","identityApi","identityApiRef","alertApi","alertApiRef","visibleKeys","setVisibleKeys","Set","refresh","setRefresh","userId","setUserId","userEmail","setUserEmail","requestDialogOpen","setRequestDialogOpen","editDialogOpen","setEditDialogOpen","requestToEdit","setRequestToEdit","menuAnchor","setMenuAnchor","menuRequest","setMenuRequest","deleting","setDeleting","optimisticallyDeleted","setOptimisticallyDeleted","setDeleteDialogState","apiKeyValues","setApiKeyValues","Map","apiKeyLoading","setApiKeyLoading","alreadyReadKeys","setAlreadyReadKeys","showOnceWarningOpen","setShowOnceWarningOpen","pendingKeyReveal","setPendingKeyReveal","apiProductName","annotations","useAsync","identity","getBackstageIdentity","profile","getProfileInfo","userEntityRef","email","requests","loading","requestsLoading","requestsError","getRequestsByNamespace","items","filter","r","apiProductRef","plansLoading","plansError","data","getApiProducts","find","resourceRef","allowed","canCreateRequest","createRequestPermissionLoading","createRequestPermissionError","useKuadrantPermission","kuadrantApiKeyCreatePermission","canDeleteOwnKey","deleteOwnPermissionLoading","deleteOwnPermissionError","kuadrantApiKeyDeleteOwnPermission","canDeleteAllKeys","deleteAllPermissionLoading","deleteAllPermissionError","kuadrantApiKeyDeleteAllPermission","canUpdateRequest","updateRequestPermissionLoading","updateRequestPermissionError","kuadrantApiKeyUpdateOwnPermission","handleMenuClose","handleMenuEdit","handleMenuDeleteClick","toggleVisibility","keyName","prev","newSet","has","delete","add","detailPanelConfig","useMemo","render","rowData","revealedKey","get","DetailPanelContent","apiName","revealedApiKey","api","selectedLanguage","setSelectedLanguage","hostname","status","apiHostname","displayApiKey","snippets","credentials","authorizationHeader","customHeader","headerName","generateCustomHeaderSnippets","queryString","paramName","generateQueryStringSnippets","cookie","cookieName","generateCookieSnippets","generateAuthCodeSnippets","authScheme","stopPropagation","gutterBottom","border","style","whiteSpace","wordBreak","overflowWrap","paragraph","Tabs","newValue","indicatorColor","Tab","mt","CodeSnippet","text","language","showCopyCodeButton","permissionError","Progress","ResponseErrorPanel","failedPermission","myRequests","plans","discoveredPlans","pendingRequests","phase","approvedRequests","rejectedRequests","approvedColumns","title","field","row","Chip","reviewedAt","Date","toLocaleDateString","searchable","filtering","isVisible","isLoading","apiKeyValue","hasSecretRef","secretRef","canReadSecret","isAlreadyRead","Tooltip","display","alignItems","fontFamily","marginRight","VisibilityOffIcon","fontSize","IconButton","navigator","clipboard","writeText","post","severity","FileCopyIcon","span","requestNamespace","requestName","next","clearApiKeyValue","VisibilityIcon","ownerId","requestedBy","canDeleteResource","rect","currentTarget","getBoundingClientRect","top","bottom","left","aria-controls","aria-haspopup","MoreVertIcon","requestColumns","isPending","icon","HourglassEmptyIcon","CancelIcon","placement","overflow","textOverflow","creationTimestamp","canDelete","pendingRequestColumns","col","Grid","container","spacing","direction","item","flexDirection","AddIcon","length","data-testid","data-plans-count","marginTop","readyCondition","conditions","c","type","planCondition","textAlign","Table","options","paging","pageSize","search","debounceInterval","toolbar","emptyRowsWhenPaging","columns","detailPanel","RequestAccessDialog","Menu","id","Boolean","anchorReference","anchorPosition","push","ConfirmDeleteDialog","description","onConfirm","deleteRequest","errorMessage","handleDeleteRequest","onCancel","WarningIcon","strong","getApiKeySecret","set","includes","fetchApiKeyFromSecret","versionedHolder","useVersionedContext","atVersion","createVersionedContext","fn","deps","_a","initialState","lastCallId","useRef","isMounted","useMountedState","state","callback","useCallback","args","_i","arguments","callId","current","prevState","__assign","apply","then","useAsyncFn","confirmText","inputValue","setInputValue","requiresTextConfirmation","canConfirm","gap","DialogContentText","autoFocus"],"sourceRoot":""}
@@ -1,2 +1,2 @@
1
- "use strict";(self.webpackChunkinternal_plugin_kuadrant=self.webpackChunkinternal_plugin_kuadrant||[]).push([[879],{72627:(e,n,t)=>{t.r(n),t.d(n,{ApiKeyApprovalPage:()=>u,ApiKeyDetailPage:()=>A,ApiProductDetailPage:()=>b,ApiProductsPage:()=>d,EntityKuadrantApiAccessCard:()=>m,EntityKuadrantApiApprovalTab:()=>y,EntityKuadrantApiKeyManagementTab:()=>c,EntityKuadrantApiKeysContent:()=>P,EntityKuadrantApiProductInfoContent:()=>s,EntityKuadrantApiProductOpenApiAlert:()=>K,KuadrantPage:()=>p,MyApiKeysPage:()=>l,kuadrantPlugin:()=>r});var a=t(22097);const o=(0,a.createRouteRef)({id:"kuadrant"}),i=(0,a.createSubRouteRef)({id:"kuadrant/resource",parent:o,path:"/:kind/:namespace/:name"}),r=(0,a.createPlugin)({id:"kuadrant",routes:{root:o,resource:i}}),p=r.provide((0,a.createRoutableExtension)({name:"KuadrantPage",component:()=>Promise.all([t.e(9400),t.e(9370),t.e(2628),t.e(4611),t.e(369),t.e(4382),t.e(7076),t.e(5235),t.e(5010),t.e(8563),t.e(6422),t.e(5478),t.e(9634),t.e(484),t.e(1942),t.e(5501),t.e(7976),t.e(8789),t.e(7005)]).then(t.bind(t,94251)).then(e=>e.ApiProductsPage),mountPoint:o})),d=r.provide((0,a.createRoutableExtension)({name:"ApiProductsPage",component:()=>Promise.all([t.e(9400),t.e(9370),t.e(2628),t.e(4611),t.e(369),t.e(4382),t.e(7076),t.e(5235),t.e(5010),t.e(8563),t.e(6422),t.e(5478),t.e(9634),t.e(484),t.e(1942),t.e(5501),t.e(7976),t.e(8789),t.e(7005)]).then(t.bind(t,94251)).then(e=>e.ApiProductsPage),mountPoint:o})),l=r.provide((0,a.createRoutableExtension)({name:"MyApiKeysPage",component:()=>Promise.all([t.e(9400),t.e(9370),t.e(2628),t.e(4611),t.e(369),t.e(7076),t.e(5235),t.e(5010),t.e(8563),t.e(5478),t.e(9634),t.e(484),t.e(1942),t.e(5501),t.e(7976),t.e(7270)]).then(t.bind(t,37270)).then(e=>e.MyApiKeysPage),mountPoint:o})),u=r.provide((0,a.createRoutableExtension)({name:"ApiKeyApprovalPage",component:()=>Promise.all([t.e(9400),t.e(9370),t.e(2628),t.e(4611),t.e(369),t.e(4382),t.e(7076),t.e(5235),t.e(5010),t.e(8563),t.e(5478),t.e(9634),t.e(484),t.e(1942),t.e(5501),t.e(7976),t.e(69),t.e(3976)]).then(t.bind(t,43976)).then(e=>e.ApiKeyApprovalPage),mountPoint:o})),m=r.provide((0,a.createComponentExtension)({name:"EntityKuadrantApiAccessCard",component:{lazy:()=>Promise.all([t.e(9400),t.e(9370),t.e(2628),t.e(4611),t.e(369),t.e(4382),t.e(2118),t.e(5478),t.e(9634),t.e(484),t.e(1942),t.e(5501),t.e(7976),t.e(5427),t.e(9051),t.e(5203)]).then(t.bind(t,60441)).then(e=>e.ApiAccessCard)}})),c=r.provide((0,a.createComponentExtension)({name:"EntityKuadrantApiKeyManagementTab",component:{lazy:()=>Promise.all([t.e(9400),t.e(9370),t.e(2628),t.e(4611),t.e(369),t.e(4382),t.e(7076),t.e(5235),t.e(5010),t.e(5478),t.e(484),t.e(1942),t.e(5501),t.e(7976),t.e(5427),t.e(9051),t.e(9838)]).then(t.bind(t,23650)).then(e=>e.ApiKeyManagementTab)}})),P=r.provide((0,a.createComponentExtension)({name:"EntityKuadrantApiKeysContent",component:{lazy:()=>Promise.all([t.e(9400),t.e(9370),t.e(2628),t.e(4611),t.e(369),t.e(4382),t.e(7076),t.e(5235),t.e(5010),t.e(5478),t.e(484),t.e(1942),t.e(5501),t.e(7976),t.e(5427),t.e(9051),t.e(9838)]).then(t.bind(t,23650)).then(e=>e.ApiKeyManagementTab)}})),s=r.provide((0,a.createComponentExtension)({name:"EntityKuadrantApiProductInfoContent",component:{lazy:()=>Promise.all([t.e(9400),t.e(9370),t.e(2628),t.e(4382),t.e(327),t.e(5478),t.e(9634),t.e(484),t.e(1942),t.e(5501),t.e(7976),t.e(5427),t.e(3947),t.e(4447)]).then(t.bind(t,75453)).then(e=>e.ApiProductInfoCard)}})),A=r.provide((0,a.createRoutableExtension)({name:"ApiKeyDetailPage",component:()=>Promise.all([t.e(9400),t.e(9370),t.e(2628),t.e(4611),t.e(7076),t.e(8563),t.e(5478),t.e(9634),t.e(484),t.e(1942),t.e(5501),t.e(7976),t.e(2967)]).then(t.bind(t,72967)).then(e=>e.ApiKeyDetailPage),mountPoint:o})),y=r.provide((0,a.createComponentExtension)({name:"EntityKuadrantApiApprovalTab",component:{lazy:()=>Promise.all([t.e(9400),t.e(9370),t.e(2628),t.e(4611),t.e(369),t.e(4382),t.e(7076),t.e(5235),t.e(5010),t.e(5478),t.e(484),t.e(1942),t.e(5501),t.e(7976),t.e(5427),t.e(69),t.e(6800)]).then(t.bind(t,46800)).then(e=>e.EntityApiApprovalTab)}})),K=r.provide((0,a.createComponentExtension)({name:"EntityKuadrantApiProductOpenApiAlert",component:{lazy:()=>Promise.all([t.e(9400),t.e(9370),t.e(5478),t.e(9634),t.e(484),t.e(1942),t.e(7976),t.e(5427),t.e(6371)]).then(t.bind(t,36371)).then(e=>e.ApiProductOpenApiAlert)}})),b=r.provide((0,a.createRoutableExtension)({name:"ApiProductDetailPage",component:()=>Promise.all([t.e(9400),t.e(9370),t.e(2628),t.e(4611),t.e(369),t.e(4382),t.e(7076),t.e(5235),t.e(8563),t.e(8804),t.e(5478),t.e(9634),t.e(484),t.e(1942),t.e(5501),t.e(7976),t.e(8789),t.e(3947),t.e(7791)]).then(t.bind(t,57791)).then(e=>e.ApiProductDetailPage),mountPoint:o}))}}]);
2
- //# sourceMappingURL=exposed-PluginRoot.8a8dd91f.chunk.js.map
1
+ "use strict";(self.webpackChunkinternal_plugin_kuadrant=self.webpackChunkinternal_plugin_kuadrant||[]).push([[879],{72627:(e,n,t)=>{t.r(n),t.d(n,{ApiKeyApprovalPage:()=>u,ApiKeyDetailPage:()=>A,ApiProductDetailPage:()=>b,ApiProductsPage:()=>d,EntityKuadrantApiAccessCard:()=>m,EntityKuadrantApiApprovalTab:()=>y,EntityKuadrantApiKeyManagementTab:()=>c,EntityKuadrantApiKeysContent:()=>P,EntityKuadrantApiProductInfoContent:()=>s,EntityKuadrantApiProductOpenApiAlert:()=>K,KuadrantPage:()=>p,MyApiKeysPage:()=>l,kuadrantPlugin:()=>r});var a=t(22097);const o=(0,a.createRouteRef)({id:"kuadrant"}),i=(0,a.createSubRouteRef)({id:"kuadrant/resource",parent:o,path:"/:kind/:namespace/:name"}),r=(0,a.createPlugin)({id:"kuadrant",routes:{root:o,resource:i}}),p=r.provide((0,a.createRoutableExtension)({name:"KuadrantPage",component:()=>Promise.all([t.e(9400),t.e(9370),t.e(2628),t.e(4611),t.e(369),t.e(4382),t.e(7076),t.e(5235),t.e(5010),t.e(8563),t.e(6422),t.e(5478),t.e(9634),t.e(484),t.e(1942),t.e(5501),t.e(7976),t.e(8789),t.e(7005)]).then(t.bind(t,94251)).then(e=>e.ApiProductsPage),mountPoint:o})),d=r.provide((0,a.createRoutableExtension)({name:"ApiProductsPage",component:()=>Promise.all([t.e(9400),t.e(9370),t.e(2628),t.e(4611),t.e(369),t.e(4382),t.e(7076),t.e(5235),t.e(5010),t.e(8563),t.e(6422),t.e(5478),t.e(9634),t.e(484),t.e(1942),t.e(5501),t.e(7976),t.e(8789),t.e(7005)]).then(t.bind(t,94251)).then(e=>e.ApiProductsPage),mountPoint:o})),l=r.provide((0,a.createRoutableExtension)({name:"MyApiKeysPage",component:()=>Promise.all([t.e(9400),t.e(9370),t.e(2628),t.e(4611),t.e(369),t.e(7076),t.e(5235),t.e(5010),t.e(8563),t.e(5478),t.e(9634),t.e(484),t.e(1942),t.e(5501),t.e(7976),t.e(7270)]).then(t.bind(t,37270)).then(e=>e.MyApiKeysPage),mountPoint:o})),u=r.provide((0,a.createRoutableExtension)({name:"ApiKeyApprovalPage",component:()=>Promise.all([t.e(9400),t.e(9370),t.e(2628),t.e(4611),t.e(369),t.e(4382),t.e(7076),t.e(5235),t.e(5010),t.e(8563),t.e(5478),t.e(9634),t.e(484),t.e(1942),t.e(5501),t.e(7976),t.e(69),t.e(3976)]).then(t.bind(t,43976)).then(e=>e.ApiKeyApprovalPage),mountPoint:o})),m=r.provide((0,a.createComponentExtension)({name:"EntityKuadrantApiAccessCard",component:{lazy:()=>Promise.all([t.e(9400),t.e(9370),t.e(2628),t.e(4611),t.e(369),t.e(4382),t.e(2118),t.e(5478),t.e(9634),t.e(484),t.e(1942),t.e(5501),t.e(7976),t.e(5427),t.e(9051),t.e(5203)]).then(t.bind(t,60441)).then(e=>e.ApiAccessCard)}})),c=r.provide((0,a.createComponentExtension)({name:"EntityKuadrantApiKeyManagementTab",component:{lazy:()=>Promise.all([t.e(9400),t.e(9370),t.e(2628),t.e(4611),t.e(369),t.e(4382),t.e(7076),t.e(5235),t.e(5010),t.e(5478),t.e(484),t.e(1942),t.e(5501),t.e(7976),t.e(5427),t.e(9051),t.e(9838)]).then(t.bind(t,23650)).then(e=>e.ApiKeyManagementTab)}})),P=r.provide((0,a.createComponentExtension)({name:"EntityKuadrantApiKeysContent",component:{lazy:()=>Promise.all([t.e(9400),t.e(9370),t.e(2628),t.e(4611),t.e(369),t.e(4382),t.e(7076),t.e(5235),t.e(5010),t.e(5478),t.e(484),t.e(1942),t.e(5501),t.e(7976),t.e(5427),t.e(9051),t.e(9838)]).then(t.bind(t,23650)).then(e=>e.ApiKeyManagementTab)}})),s=r.provide((0,a.createComponentExtension)({name:"EntityKuadrantApiProductInfoContent",component:{lazy:()=>Promise.all([t.e(9400),t.e(9370),t.e(2628),t.e(4382),t.e(327),t.e(5478),t.e(9634),t.e(484),t.e(1942),t.e(5501),t.e(7976),t.e(5427),t.e(3947),t.e(4447)]).then(t.bind(t,75453)).then(e=>e.ApiProductInfoCard)}})),A=r.provide((0,a.createRoutableExtension)({name:"ApiKeyDetailPage",component:()=>Promise.all([t.e(9400),t.e(9370),t.e(2628),t.e(4611),t.e(7076),t.e(8563),t.e(5478),t.e(9634),t.e(484),t.e(1942),t.e(5501),t.e(7976),t.e(2967)]).then(t.bind(t,72967)).then(e=>e.ApiKeyDetailPage),mountPoint:o})),y=r.provide((0,a.createComponentExtension)({name:"EntityKuadrantApiApprovalTab",component:{lazy:()=>Promise.all([t.e(9400),t.e(9370),t.e(2628),t.e(4611),t.e(369),t.e(4382),t.e(7076),t.e(5235),t.e(5010),t.e(5478),t.e(484),t.e(1942),t.e(5501),t.e(7976),t.e(5427),t.e(69),t.e(6800)]).then(t.bind(t,46800)).then(e=>e.EntityApiApprovalTab)}})),K=r.provide((0,a.createComponentExtension)({name:"EntityKuadrantApiProductOpenApiAlert",component:{lazy:()=>Promise.all([t.e(9400),t.e(9370),t.e(5478),t.e(9634),t.e(484),t.e(1942),t.e(7976),t.e(5427),t.e(6371)]).then(t.bind(t,36371)).then(e=>e.ApiProductOpenApiAlert)}})),b=r.provide((0,a.createRoutableExtension)({name:"ApiProductDetailPage",component:()=>Promise.all([t.e(9400),t.e(9370),t.e(2628),t.e(4611),t.e(369),t.e(4382),t.e(7076),t.e(5235),t.e(8563),t.e(6387),t.e(5478),t.e(9634),t.e(484),t.e(1942),t.e(5501),t.e(7976),t.e(8789),t.e(3947),t.e(7791)]).then(t.bind(t,57791)).then(e=>e.ApiProductDetailPage),mountPoint:o}))}}]);
2
+ //# sourceMappingURL=exposed-PluginRoot.3b24e5af.chunk.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"static/exposed-PluginRoot.8a8dd91f.chunk.js","mappings":"qiBAEO,MAAMA,GAAeC,EAAAA,EAAAA,gBAAe,CACzCC,GAAI,aAGOC,GAAmBC,EAAAA,EAAAA,mBAAkB,CAChDF,GAAI,oBACJG,OAAQL,EACRM,KAAM,4BCDKC,GAAiBC,EAAAA,EAAAA,cAAa,CACzCN,GAAI,WACJO,OAAQ,CACNC,KAAMV,EACNW,SAAUR,KAIDS,EAAeL,EAAeM,SACzCC,EAAAA,EAAAA,yBAAwB,CACtBC,KAAM,eACNC,UAAW,IACT,iOAAoCC,KAAKC,GAAKA,EAAEC,iBAClDC,WAAYpB,KAIHmB,EAAkBZ,EAAeM,SAC5CC,EAAAA,EAAAA,yBAAwB,CACtBC,KAAM,kBACNC,UAAW,IACT,iOAAoCC,KAAKC,GAAKA,EAAEC,iBAClDC,WAAYpB,KAIHqB,EAAgBd,EAAeM,SAC1CC,EAAAA,EAAAA,yBAAwB,CACtBC,KAAM,gBACNC,UAAW,IACT,mMAAqCC,KAAKC,GAAKA,EAAEG,eACnDD,WAAYpB,KAIHsB,EAAqBf,EAAeM,SAC/CC,EAAAA,EAAAA,yBAAwB,CACtBC,KAAM,qBACNC,UAAW,IACT,qNAA0CC,KAAKC,GAAKA,EAAEI,oBACxDF,WAAYpB,KAIHuB,EAA8BhB,EAAeM,SACxDW,EAAAA,EAAAA,0BAAyB,CACvBT,KAAM,8BACNC,UAAW,CACTS,KAAM,IACJ,mMAAqCR,KAAKC,GAAKA,EAAEQ,mBAK5CC,EAAoCpB,EAAeM,SAC9DW,EAAAA,EAAAA,0BAAyB,CACvBT,KAAM,oCACNC,UAAW,CACTS,KAAM,IACJ,6MAA2CR,KAAKC,GAAKA,EAAEU,yBAMlDC,EAA+BtB,EAAeM,SACzDW,EAAAA,EAAAA,0BAAyB,CACvBT,KAAM,+BACNC,UAAW,CACTS,KAAM,IACJ,6MAA2CR,KAAKC,GAAKA,EAAEU,yBAKlDE,EAAsCvB,EAAeM,SAChEW,EAAAA,EAAAA,0BAAyB,CACvBT,KAAM,sCACNC,UAAW,CACTS,KAAM,IACJ,+KAA0CR,KAAKC,GAAKA,EAAEa,wBAKjDC,EAAmBzB,EAAeM,SAC7CC,EAAAA,EAAAA,yBAAwB,CACtBC,KAAM,mBACNC,UAAW,IACT,sKAAwCC,KAAKC,GAAKA,EAAEc,kBACtDZ,WAAYpB,KAIHiC,EAA+B1B,EAAeM,SACzDW,EAAAA,EAAAA,0BAAyB,CACvBT,KAAM,+BACNC,UAAW,CACTS,KAAM,IACJ,2MAA4CR,KAAKC,GAAKA,EAAEgB,0BAKnDC,EAAuC5B,EAAeM,SACjEW,EAAAA,EAAAA,0BAAyB,CACvBT,KAAM,uCACNC,UAAW,CACTS,KAAM,IACJ,8HAA8CR,KAAKC,GAAKA,EAAEkB,4BAKrDC,EAAuB9B,EAAeM,SACjDC,EAAAA,EAAAA,yBAAwB,CACtBC,KAAM,uBACNC,UAAW,IACT,iOAA4CC,KAAKC,GAAKA,EAAEmB,sBAC1DjB,WAAYpB,I","sources":["webpack://internal.plugin-kuadrant/./src/routes.ts","webpack://internal.plugin-kuadrant/./src/plugin.ts"],"sourcesContent":["import { createRouteRef, createSubRouteRef } from '@backstage/core-plugin-api';\n\nexport const rootRouteRef = createRouteRef({\n id: 'kuadrant',\n});\n\nexport const resourceRouteRef = createSubRouteRef({\n id: 'kuadrant/resource',\n parent: rootRouteRef,\n path: '/:kind/:namespace/:name',\n});\n","import {\n createPlugin,\n createRoutableExtension,\n createComponentExtension,\n} from '@backstage/core-plugin-api';\n\nimport { rootRouteRef, resourceRouteRef } from './routes';\n\nexport const kuadrantPlugin = createPlugin({\n id: 'kuadrant',\n routes: {\n root: rootRouteRef,\n resource: resourceRouteRef,\n },\n});\n\nexport const KuadrantPage = kuadrantPlugin.provide(\n createRoutableExtension({\n name: 'KuadrantPage',\n component: () =>\n import('./components/KuadrantPage').then(m => m.ApiProductsPage),\n mountPoint: rootRouteRef,\n }),\n);\n\nexport const ApiProductsPage = kuadrantPlugin.provide(\n createRoutableExtension({\n name: 'ApiProductsPage',\n component: () =>\n import('./components/KuadrantPage').then(m => m.ApiProductsPage),\n mountPoint: rootRouteRef,\n }),\n);\n\nexport const MyApiKeysPage = kuadrantPlugin.provide(\n createRoutableExtension({\n name: 'MyApiKeysPage',\n component: () =>\n import('./components/MyApiKeysPage').then(m => m.MyApiKeysPage),\n mountPoint: rootRouteRef,\n }),\n);\n\nexport const ApiKeyApprovalPage = kuadrantPlugin.provide(\n createRoutableExtension({\n name: 'ApiKeyApprovalPage',\n component: () =>\n import('./components/ApiKeyApprovalPage').then(m => m.ApiKeyApprovalPage),\n mountPoint: rootRouteRef,\n }),\n);\n\nexport const EntityKuadrantApiAccessCard = kuadrantPlugin.provide(\n createComponentExtension({\n name: 'EntityKuadrantApiAccessCard',\n component: {\n lazy: () =>\n import('./components/ApiAccessCard').then(m => m.ApiAccessCard),\n },\n }),\n);\n\nexport const EntityKuadrantApiKeyManagementTab = kuadrantPlugin.provide(\n createComponentExtension({\n name: 'EntityKuadrantApiKeyManagementTab',\n component: {\n lazy: () =>\n import('./components/ApiKeyManagementTab').then(m => m.ApiKeyManagementTab),\n },\n }),\n);\n\n// entity content extension for api keys tab\nexport const EntityKuadrantApiKeysContent = kuadrantPlugin.provide(\n createComponentExtension({\n name: 'EntityKuadrantApiKeysContent',\n component: {\n lazy: () =>\n import('./components/ApiKeyManagementTab').then(m => m.ApiKeyManagementTab),\n },\n }),\n);\n\nexport const EntityKuadrantApiProductInfoContent = kuadrantPlugin.provide(\n createComponentExtension({\n name: 'EntityKuadrantApiProductInfoContent',\n component: {\n lazy: () =>\n import('./components/ApiProductInfoCard').then(m => m.ApiProductInfoCard),\n },\n }),\n);\n\nexport const ApiKeyDetailPage = kuadrantPlugin.provide(\n createRoutableExtension({\n name: 'ApiKeyDetailPage',\n component: () =>\n import('./components/ApiKeyDetailPage').then(m => m.ApiKeyDetailPage),\n mountPoint: rootRouteRef,\n }),\n);\n\nexport const EntityKuadrantApiApprovalTab = kuadrantPlugin.provide(\n createComponentExtension({\n name: 'EntityKuadrantApiApprovalTab',\n component: {\n lazy: () =>\n import('./components/EntityApiApprovalTab').then(m => m.EntityApiApprovalTab),\n },\n }),\n);\n\nexport const EntityKuadrantApiProductOpenApiAlert = kuadrantPlugin.provide(\n createComponentExtension({\n name: 'EntityKuadrantApiProductOpenApiAlert',\n component: {\n lazy: () =>\n import('./components/ApiProductOpenApiAlert').then(m => m.ApiProductOpenApiAlert),\n },\n }),\n);\n\nexport const ApiProductDetailPage = kuadrantPlugin.provide(\n createRoutableExtension({\n name: 'ApiProductDetailPage',\n component: () =>\n import('./components/ApiProductDetailPage').then(m => m.ApiProductDetailPage),\n mountPoint: rootRouteRef,\n }),\n);\n"],"names":["rootRouteRef","createRouteRef","id","resourceRouteRef","createSubRouteRef","parent","path","kuadrantPlugin","createPlugin","routes","root","resource","KuadrantPage","provide","createRoutableExtension","name","component","then","m","ApiProductsPage","mountPoint","MyApiKeysPage","ApiKeyApprovalPage","EntityKuadrantApiAccessCard","createComponentExtension","lazy","ApiAccessCard","EntityKuadrantApiKeyManagementTab","ApiKeyManagementTab","EntityKuadrantApiKeysContent","EntityKuadrantApiProductInfoContent","ApiProductInfoCard","ApiKeyDetailPage","EntityKuadrantApiApprovalTab","EntityApiApprovalTab","EntityKuadrantApiProductOpenApiAlert","ApiProductOpenApiAlert","ApiProductDetailPage"],"sourceRoot":""}
1
+ {"version":3,"file":"static/exposed-PluginRoot.3b24e5af.chunk.js","mappings":"qiBAEO,MAAMA,GAAeC,EAAAA,EAAAA,gBAAe,CACzCC,GAAI,aAGOC,GAAmBC,EAAAA,EAAAA,mBAAkB,CAChDF,GAAI,oBACJG,OAAQL,EACRM,KAAM,4BCDKC,GAAiBC,EAAAA,EAAAA,cAAa,CACzCN,GAAI,WACJO,OAAQ,CACNC,KAAMV,EACNW,SAAUR,KAIDS,EAAeL,EAAeM,SACzCC,EAAAA,EAAAA,yBAAwB,CACtBC,KAAM,eACNC,UAAW,IACT,iOAAoCC,KAAKC,GAAKA,EAAEC,iBAClDC,WAAYpB,KAIHmB,EAAkBZ,EAAeM,SAC5CC,EAAAA,EAAAA,yBAAwB,CACtBC,KAAM,kBACNC,UAAW,IACT,iOAAoCC,KAAKC,GAAKA,EAAEC,iBAClDC,WAAYpB,KAIHqB,EAAgBd,EAAeM,SAC1CC,EAAAA,EAAAA,yBAAwB,CACtBC,KAAM,gBACNC,UAAW,IACT,mMAAqCC,KAAKC,GAAKA,EAAEG,eACnDD,WAAYpB,KAIHsB,EAAqBf,EAAeM,SAC/CC,EAAAA,EAAAA,yBAAwB,CACtBC,KAAM,qBACNC,UAAW,IACT,qNAA0CC,KAAKC,GAAKA,EAAEI,oBACxDF,WAAYpB,KAIHuB,EAA8BhB,EAAeM,SACxDW,EAAAA,EAAAA,0BAAyB,CACvBT,KAAM,8BACNC,UAAW,CACTS,KAAM,IACJ,mMAAqCR,KAAKC,GAAKA,EAAEQ,mBAK5CC,EAAoCpB,EAAeM,SAC9DW,EAAAA,EAAAA,0BAAyB,CACvBT,KAAM,oCACNC,UAAW,CACTS,KAAM,IACJ,6MAA2CR,KAAKC,GAAKA,EAAEU,yBAMlDC,EAA+BtB,EAAeM,SACzDW,EAAAA,EAAAA,0BAAyB,CACvBT,KAAM,+BACNC,UAAW,CACTS,KAAM,IACJ,6MAA2CR,KAAKC,GAAKA,EAAEU,yBAKlDE,EAAsCvB,EAAeM,SAChEW,EAAAA,EAAAA,0BAAyB,CACvBT,KAAM,sCACNC,UAAW,CACTS,KAAM,IACJ,+KAA0CR,KAAKC,GAAKA,EAAEa,wBAKjDC,EAAmBzB,EAAeM,SAC7CC,EAAAA,EAAAA,yBAAwB,CACtBC,KAAM,mBACNC,UAAW,IACT,sKAAwCC,KAAKC,GAAKA,EAAEc,kBACtDZ,WAAYpB,KAIHiC,EAA+B1B,EAAeM,SACzDW,EAAAA,EAAAA,0BAAyB,CACvBT,KAAM,+BACNC,UAAW,CACTS,KAAM,IACJ,2MAA4CR,KAAKC,GAAKA,EAAEgB,0BAKnDC,EAAuC5B,EAAeM,SACjEW,EAAAA,EAAAA,0BAAyB,CACvBT,KAAM,uCACNC,UAAW,CACTS,KAAM,IACJ,8HAA8CR,KAAKC,GAAKA,EAAEkB,4BAKrDC,EAAuB9B,EAAeM,SACjDC,EAAAA,EAAAA,yBAAwB,CACtBC,KAAM,uBACNC,UAAW,IACT,iOAA4CC,KAAKC,GAAKA,EAAEmB,sBAC1DjB,WAAYpB,I","sources":["webpack://internal.plugin-kuadrant/./src/routes.ts","webpack://internal.plugin-kuadrant/./src/plugin.ts"],"sourcesContent":["import { createRouteRef, createSubRouteRef } from '@backstage/core-plugin-api';\n\nexport const rootRouteRef = createRouteRef({\n id: 'kuadrant',\n});\n\nexport const resourceRouteRef = createSubRouteRef({\n id: 'kuadrant/resource',\n parent: rootRouteRef,\n path: '/:kind/:namespace/:name',\n});\n","import {\n createPlugin,\n createRoutableExtension,\n createComponentExtension,\n} from '@backstage/core-plugin-api';\n\nimport { rootRouteRef, resourceRouteRef } from './routes';\n\nexport const kuadrantPlugin = createPlugin({\n id: 'kuadrant',\n routes: {\n root: rootRouteRef,\n resource: resourceRouteRef,\n },\n});\n\nexport const KuadrantPage = kuadrantPlugin.provide(\n createRoutableExtension({\n name: 'KuadrantPage',\n component: () =>\n import('./components/KuadrantPage').then(m => m.ApiProductsPage),\n mountPoint: rootRouteRef,\n }),\n);\n\nexport const ApiProductsPage = kuadrantPlugin.provide(\n createRoutableExtension({\n name: 'ApiProductsPage',\n component: () =>\n import('./components/KuadrantPage').then(m => m.ApiProductsPage),\n mountPoint: rootRouteRef,\n }),\n);\n\nexport const MyApiKeysPage = kuadrantPlugin.provide(\n createRoutableExtension({\n name: 'MyApiKeysPage',\n component: () =>\n import('./components/MyApiKeysPage').then(m => m.MyApiKeysPage),\n mountPoint: rootRouteRef,\n }),\n);\n\nexport const ApiKeyApprovalPage = kuadrantPlugin.provide(\n createRoutableExtension({\n name: 'ApiKeyApprovalPage',\n component: () =>\n import('./components/ApiKeyApprovalPage').then(m => m.ApiKeyApprovalPage),\n mountPoint: rootRouteRef,\n }),\n);\n\nexport const EntityKuadrantApiAccessCard = kuadrantPlugin.provide(\n createComponentExtension({\n name: 'EntityKuadrantApiAccessCard',\n component: {\n lazy: () =>\n import('./components/ApiAccessCard').then(m => m.ApiAccessCard),\n },\n }),\n);\n\nexport const EntityKuadrantApiKeyManagementTab = kuadrantPlugin.provide(\n createComponentExtension({\n name: 'EntityKuadrantApiKeyManagementTab',\n component: {\n lazy: () =>\n import('./components/ApiKeyManagementTab').then(m => m.ApiKeyManagementTab),\n },\n }),\n);\n\n// entity content extension for api keys tab\nexport const EntityKuadrantApiKeysContent = kuadrantPlugin.provide(\n createComponentExtension({\n name: 'EntityKuadrantApiKeysContent',\n component: {\n lazy: () =>\n import('./components/ApiKeyManagementTab').then(m => m.ApiKeyManagementTab),\n },\n }),\n);\n\nexport const EntityKuadrantApiProductInfoContent = kuadrantPlugin.provide(\n createComponentExtension({\n name: 'EntityKuadrantApiProductInfoContent',\n component: {\n lazy: () =>\n import('./components/ApiProductInfoCard').then(m => m.ApiProductInfoCard),\n },\n }),\n);\n\nexport const ApiKeyDetailPage = kuadrantPlugin.provide(\n createRoutableExtension({\n name: 'ApiKeyDetailPage',\n component: () =>\n import('./components/ApiKeyDetailPage').then(m => m.ApiKeyDetailPage),\n mountPoint: rootRouteRef,\n }),\n);\n\nexport const EntityKuadrantApiApprovalTab = kuadrantPlugin.provide(\n createComponentExtension({\n name: 'EntityKuadrantApiApprovalTab',\n component: {\n lazy: () =>\n import('./components/EntityApiApprovalTab').then(m => m.EntityApiApprovalTab),\n },\n }),\n);\n\nexport const EntityKuadrantApiProductOpenApiAlert = kuadrantPlugin.provide(\n createComponentExtension({\n name: 'EntityKuadrantApiProductOpenApiAlert',\n component: {\n lazy: () =>\n import('./components/ApiProductOpenApiAlert').then(m => m.ApiProductOpenApiAlert),\n },\n }),\n);\n\nexport const ApiProductDetailPage = kuadrantPlugin.provide(\n createRoutableExtension({\n name: 'ApiProductDetailPage',\n component: () =>\n import('./components/ApiProductDetailPage').then(m => m.ApiProductDetailPage),\n mountPoint: rootRouteRef,\n }),\n);\n"],"names":["rootRouteRef","createRouteRef","id","resourceRouteRef","createSubRouteRef","parent","path","kuadrantPlugin","createPlugin","routes","root","resource","KuadrantPage","provide","createRoutableExtension","name","component","then","m","ApiProductsPage","mountPoint","MyApiKeysPage","ApiKeyApprovalPage","EntityKuadrantApiAccessCard","createComponentExtension","lazy","ApiAccessCard","EntityKuadrantApiKeyManagementTab","ApiKeyManagementTab","EntityKuadrantApiKeysContent","EntityKuadrantApiProductInfoContent","ApiProductInfoCard","ApiKeyDetailPage","EntityKuadrantApiApprovalTab","EntityApiApprovalTab","EntityKuadrantApiProductOpenApiAlert","ApiProductOpenApiAlert","ApiProductDetailPage"],"sourceRoot":""}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@kuadrant/kuadrant-backstage-plugin-frontend",
3
- "version": "0.0.2-dev-9be93a3",
3
+ "version": "0.0.2-dev-8189c1c",
4
4
  "license": "Apache-2.0",
5
5
  "repository": {
6
6
  "type": "git",
@@ -1,48 +0,0 @@
1
- import React from 'react';
2
- import { useTheme, Box, Typography, Chip } from '@material-ui/core';
3
- import { Alert } from '@material-ui/lab';
4
-
5
- const PlanPolicyDetails = ({
6
- discoveredPlans,
7
- alertSeverity = "warning",
8
- alertMessage = "No PlanPolicy found for this HTTPRoute. API keys and rate limiting may not be available.",
9
- includeTopMargin = true
10
- }) => {
11
- const theme = useTheme();
12
- return /* @__PURE__ */ React.createElement(
13
- Box,
14
- {
15
- mt: includeTopMargin ? 1 : 0,
16
- p: 2,
17
- bgcolor: theme.palette.background.default,
18
- borderRadius: 1,
19
- border: `1px solid ${theme.palette.divider}`
20
- },
21
- discoveredPlans ? /* @__PURE__ */ React.createElement(React.Fragment, null, discoveredPlans.length > 0 ? /* @__PURE__ */ React.createElement(React.Fragment, null, /* @__PURE__ */ React.createElement(
22
- Typography,
23
- {
24
- variant: "caption",
25
- display: "block",
26
- gutterBottom: true,
27
- color: "textSecondary",
28
- style: { marginTop: 8 }
29
- },
30
- "Available PlanPolicy Tiers:"
31
- ), /* @__PURE__ */ React.createElement(Box, { display: "flex", flexWrap: "wrap", mt: 1, style: { gap: 8 } }, discoveredPlans.map((plan, idx) => {
32
- const limitText = plan.limits?.daily ? `${plan.limits.daily}/day` : plan.limits?.monthly ? `${plan.limits.monthly}/month` : plan.limits?.yearly ? `${plan.limits.yearly}/year` : "No limit";
33
- return /* @__PURE__ */ React.createElement(
34
- Chip,
35
- {
36
- key: idx,
37
- label: `${plan.tier}: ${limitText}`,
38
- size: "small",
39
- variant: "outlined",
40
- color: "primary"
41
- }
42
- );
43
- }))) : /* @__PURE__ */ React.createElement(Typography, { variant: "caption", color: "textSecondary" }, "No plans defined in this PlanPolicy")) : /* @__PURE__ */ React.createElement(Alert, { severity: alertSeverity }, alertMessage)
44
- );
45
- };
46
-
47
- export { PlanPolicyDetails };
48
- //# sourceMappingURL=PlanPolicyDetails.esm.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"PlanPolicyDetails.esm.js","sources":["../../../src/components/PlanPolicyDetailsCard/PlanPolicyDetails.tsx"],"sourcesContent":["import React from 'react';\nimport { Box, Typography, Chip, useTheme } from '@material-ui/core';\nimport { Alert } from '@material-ui/lab';\n\ninterface PlanPolicyDetailsProps {\n discoveredPlans?: Array<{\n tier: string;\n limits?: {\n daily?: number;\n monthly?: number;\n yearly?: number;\n };\n }> | null;\n alertSeverity?: 'warning' | 'info';\n alertMessage?: string;\n includeTopMargin?: boolean;\n}\n\nexport const PlanPolicyDetails: React.FC<PlanPolicyDetailsProps> = ({\n discoveredPlans,\n alertSeverity = 'warning',\n alertMessage = 'No PlanPolicy found for this HTTPRoute. API keys and rate limiting may not be available.',\n includeTopMargin = true,\n}) => {\n const theme = useTheme();\n return (\n <Box\n mt={includeTopMargin ? 1 : 0}\n p={2}\n bgcolor={theme.palette.background.default}\n borderRadius={1}\n border={`1px solid ${theme.palette.divider}`}\n >\n {discoveredPlans ? (\n <>\n {discoveredPlans.length > 0 ? (\n <>\n <Typography\n variant=\"caption\"\n display=\"block\"\n gutterBottom\n color=\"textSecondary\"\n style={{ marginTop: 8 }}\n >\n Available PlanPolicy Tiers:\n </Typography>\n <Box display=\"flex\" flexWrap=\"wrap\" mt={1} style={{ gap: 8 }}>\n {discoveredPlans.map((plan: any, idx: number) => {\n const limitText = plan.limits?.daily\n ? `${plan.limits.daily}/day`\n : plan.limits?.monthly\n ? `${plan.limits.monthly}/month`\n : plan.limits?.yearly\n ? `${plan.limits.yearly}/year`\n : 'No limit';\n\n return (\n <Chip\n key={idx}\n label={`${plan.tier}: ${limitText}`}\n size=\"small\"\n variant=\"outlined\"\n color=\"primary\"\n />\n );\n })}\n </Box>\n </>\n ) : (\n <Typography variant=\"caption\" color=\"textSecondary\">\n No plans defined in this PlanPolicy\n </Typography>\n )}\n </>\n ) : (\n <Alert severity={alertSeverity}>{alertMessage}</Alert>\n )}\n </Box>\n );\n};\n"],"names":[],"mappings":";;;;AAkBO,MAAM,oBAAsD,CAAC;AAAA,EAClE,eAAA;AAAA,EACA,aAAgB,GAAA,SAAA;AAAA,EAChB,YAAe,GAAA,0FAAA;AAAA,EACf,gBAAmB,GAAA;AACrB,CAAM,KAAA;AACJ,EAAA,MAAM,QAAQ,QAAS,EAAA;AACvB,EACE,uBAAA,KAAA,CAAA,aAAA;AAAA,IAAC,GAAA;AAAA,IAAA;AAAA,MACC,EAAA,EAAI,mBAAmB,CAAI,GAAA,CAAA;AAAA,MAC3B,CAAG,EAAA,CAAA;AAAA,MACH,OAAA,EAAS,KAAM,CAAA,OAAA,CAAQ,UAAW,CAAA,OAAA;AAAA,MAClC,YAAc,EAAA,CAAA;AAAA,MACd,MAAQ,EAAA,CAAA,UAAA,EAAa,KAAM,CAAA,OAAA,CAAQ,OAAO,CAAA;AAAA,KAAA;AAAA,IAEzC,eACC,mBAAA,KAAA,CAAA,aAAA,CAAA,KAAA,CAAA,QAAA,EAAA,IAAA,EACG,eAAgB,CAAA,MAAA,GAAS,oBAEtB,KAAA,CAAA,aAAA,CAAA,KAAA,CAAA,QAAA,EAAA,IAAA,kBAAA,KAAA,CAAA,aAAA;AAAA,MAAC,UAAA;AAAA,MAAA;AAAA,QACC,OAAQ,EAAA,SAAA;AAAA,QACR,OAAQ,EAAA,OAAA;AAAA,QACR,YAAY,EAAA,IAAA;AAAA,QACZ,KAAM,EAAA,eAAA;AAAA,QACN,KAAA,EAAO,EAAE,SAAA,EAAW,CAAE;AAAA,OAAA;AAAA,MACvB;AAAA,uBAGA,KAAA,CAAA,aAAA,CAAA,GAAA,EAAA,EAAI,SAAQ,MAAO,EAAA,QAAA,EAAS,QAAO,EAAI,EAAA,CAAA,EAAG,KAAO,EAAA,EAAE,KAAK,CAAE,EAAA,EAAA,EACxD,gBAAgB,GAAI,CAAA,CAAC,MAAW,GAAgB,KAAA;AAC/C,MAAM,MAAA,SAAA,GAAY,IAAK,CAAA,MAAA,EAAQ,KAC3B,GAAA,CAAA,EAAG,KAAK,MAAO,CAAA,KAAK,CACpB,IAAA,CAAA,GAAA,IAAA,CAAK,MAAQ,EAAA,OAAA,GACX,GAAG,IAAK,CAAA,MAAA,CAAO,OAAO,CAAA,MAAA,CAAA,GACtB,IAAK,CAAA,MAAA,EAAQ,SACX,CAAG,EAAA,IAAA,CAAK,MAAO,CAAA,MAAM,CACrB,KAAA,CAAA,GAAA,UAAA;AAER,MACE,uBAAA,KAAA,CAAA,aAAA;AAAA,QAAC,IAAA;AAAA,QAAA;AAAA,UACC,GAAK,EAAA,GAAA;AAAA,UACL,KAAO,EAAA,CAAA,EAAG,IAAK,CAAA,IAAI,KAAK,SAAS,CAAA,CAAA;AAAA,UACjC,IAAK,EAAA,OAAA;AAAA,UACL,OAAQ,EAAA,UAAA;AAAA,UACR,KAAM,EAAA;AAAA;AAAA,OACR;AAAA,KAEH,CACH,CACF,CAEA,mBAAA,KAAA,CAAA,aAAA,CAAC,cAAW,OAAQ,EAAA,SAAA,EAAU,KAAM,EAAA,eAAA,EAAA,EAAgB,qCAEpD,CAEJ,CAAA,uCAEC,KAAM,EAAA,EAAA,QAAA,EAAU,iBAAgB,YAAa;AAAA,GAElD;AAEJ;;;;"}