@kuadrant/kuadrant-backstage-plugin-frontend 0.0.2-dev-cd184fc → 0.0.2-dev-d2ba42b

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 (119) hide show
  1. package/dist/components/ApiAccessCard/ApiAccessCard.esm.js +2 -1
  2. package/dist/components/ApiAccessCard/ApiAccessCard.esm.js.map +1 -1
  3. package/dist/components/ApiKeyApprovalPage/ApiKeyApprovalPage.esm.js +10 -0
  4. package/dist/components/ApiKeyApprovalPage/ApiKeyApprovalPage.esm.js.map +1 -0
  5. package/dist/components/ApiKeyApprovalPage/index.esm.js +2 -0
  6. package/dist/components/ApiKeyDetailPage/ApiKeyDetailPage.esm.js +4 -4
  7. package/dist/components/ApiKeyDetailPage/ApiKeyDetailPage.esm.js.map +1 -1
  8. package/dist/components/ApiProductDetailPage/ApiProductDetailPage.esm.js +28 -2
  9. package/dist/components/ApiProductDetailPage/ApiProductDetailPage.esm.js.map +1 -1
  10. package/dist/components/ApiProductDetails/ApiProductDetails.esm.js +13 -3
  11. package/dist/components/ApiProductDetails/ApiProductDetails.esm.js.map +1 -1
  12. package/dist/components/ApprovalQueueTable/ApprovalQueueTable.esm.js +2 -2
  13. package/dist/components/ApprovalQueueTable/ApprovalQueueTable.esm.js.map +1 -1
  14. package/dist/components/CreateAPIProductDialog/CreateAPIProductDialog.esm.js +117 -10
  15. package/dist/components/CreateAPIProductDialog/CreateAPIProductDialog.esm.js.map +1 -1
  16. package/dist/components/EditAPIProductDialog/EditAPIProductDialog.esm.js +48 -24
  17. package/dist/components/EditAPIProductDialog/EditAPIProductDialog.esm.js.map +1 -1
  18. package/dist/components/EntityApiApprovalTab/EntityApiApprovalTab.esm.js +2 -2
  19. package/dist/components/EntityApiApprovalTab/EntityApiApprovalTab.esm.js.map +1 -1
  20. package/dist/components/KuadrantPage/ApiProductsPage.esm.js +53 -3
  21. package/dist/components/KuadrantPage/ApiProductsPage.esm.js.map +1 -1
  22. package/dist/components/MyApiKeysPage/MyApiKeysPage.esm.js +10 -0
  23. package/dist/components/MyApiKeysPage/MyApiKeysPage.esm.js.map +1 -0
  24. package/dist/components/MyApiKeysPage/index.esm.js +2 -0
  25. package/dist/components/MyApiKeysPage/index.esm.js.map +1 -0
  26. package/dist/components/MyApiKeysTable/MyApiKeysTable.esm.js +26 -5
  27. package/dist/components/MyApiKeysTable/MyApiKeysTable.esm.js.map +1 -1
  28. package/dist/components/PlanPolicyDetailsCard/PlanPolicyDetails.esm.js +4 -4
  29. package/dist/components/PlanPolicyDetailsCard/PlanPolicyDetails.esm.js.map +1 -1
  30. package/dist/components/SimpleRequestAccessDialog/SimpleRequestAccessDialog.esm.js +206 -0
  31. package/dist/components/SimpleRequestAccessDialog/SimpleRequestAccessDialog.esm.js.map +1 -0
  32. package/dist/index.d.ts +3 -2
  33. package/dist/index.esm.js +1 -1
  34. package/dist/plugin.esm.js +11 -4
  35. package/dist/plugin.esm.js.map +1 -1
  36. package/dist/utils/styles.esm.js +44 -5
  37. package/dist/utils/styles.esm.js.map +1 -1
  38. package/dist-scalprum/internal.plugin-kuadrant.eacc31ca4d6c1340f8a8.js +2 -0
  39. package/dist-scalprum/{internal.plugin-kuadrant.c830fc2d56f9e96a055e.js.map → internal.plugin-kuadrant.eacc31ca4d6c1340f8a8.js.map} +1 -1
  40. package/dist-scalprum/plugin-manifest.json +3 -3
  41. package/dist-scalprum/static/2118.82433765.chunk.js +2 -0
  42. package/dist-scalprum/static/2118.82433765.chunk.js.map +1 -0
  43. package/dist-scalprum/static/2967.5bade048.chunk.js +2 -0
  44. package/dist-scalprum/static/2967.5bade048.chunk.js.map +1 -0
  45. package/dist-scalprum/static/3650.89dfc64c.chunk.js +2 -0
  46. package/dist-scalprum/static/3650.89dfc64c.chunk.js.map +1 -0
  47. package/dist-scalprum/static/369.2374941c.chunk.js +2 -0
  48. package/dist-scalprum/static/369.2374941c.chunk.js.map +1 -0
  49. package/dist-scalprum/static/3947.ff1c25cf.chunk.js +2 -0
  50. package/dist-scalprum/static/3947.ff1c25cf.chunk.js.map +1 -0
  51. package/dist-scalprum/static/3976.4cf18515.chunk.js +2 -0
  52. package/dist-scalprum/static/3976.4cf18515.chunk.js.map +1 -0
  53. package/dist-scalprum/static/4447.adbf663f.chunk.js +2 -0
  54. package/dist-scalprum/static/4447.adbf663f.chunk.js.map +1 -0
  55. package/dist-scalprum/static/5010.4ef96c87.chunk.js +3 -0
  56. package/dist-scalprum/static/5010.4ef96c87.chunk.js.map +1 -0
  57. package/dist-scalprum/static/5203.fce2a28f.chunk.js +2 -0
  58. package/dist-scalprum/static/5203.fce2a28f.chunk.js.map +1 -0
  59. package/dist-scalprum/static/6422.969d9b8c.chunk.js +2 -0
  60. package/dist-scalprum/static/6422.969d9b8c.chunk.js.map +1 -0
  61. package/dist-scalprum/static/6800.cd5c7bcb.chunk.js +2 -0
  62. package/dist-scalprum/static/6800.cd5c7bcb.chunk.js.map +1 -0
  63. package/dist-scalprum/static/7005.72759857.chunk.js +2 -0
  64. package/dist-scalprum/static/7005.72759857.chunk.js.map +1 -0
  65. package/dist-scalprum/static/7270.9473c969.chunk.js +2 -0
  66. package/dist-scalprum/static/7270.9473c969.chunk.js.map +1 -0
  67. package/dist-scalprum/static/7791.01371352.chunk.js +2 -0
  68. package/dist-scalprum/static/7791.01371352.chunk.js.map +1 -0
  69. package/dist-scalprum/static/{6763.d6cd937f.chunk.js → 8563.46f1a3e1.chunk.js} +3 -3
  70. package/dist-scalprum/static/8563.46f1a3e1.chunk.js.map +1 -0
  71. package/dist-scalprum/static/8789.30227526.chunk.js +2 -0
  72. package/dist-scalprum/static/8789.30227526.chunk.js.map +1 -0
  73. package/dist-scalprum/static/8804.63919453.chunk.js +2 -0
  74. package/dist-scalprum/static/8804.63919453.chunk.js.map +1 -0
  75. package/dist-scalprum/static/exposed-PluginRoot.8d8f0b09.chunk.js +2 -0
  76. package/dist-scalprum/static/exposed-PluginRoot.8d8f0b09.chunk.js.map +1 -0
  77. package/package.json +1 -1
  78. package/dist/components/ApiKeysPage/ApiKeysPage.esm.js +0 -43
  79. package/dist/components/ApiKeysPage/ApiKeysPage.esm.js.map +0 -1
  80. package/dist/components/ApiKeysPage/index.esm.js +0 -2
  81. package/dist-scalprum/internal.plugin-kuadrant.c830fc2d56f9e96a055e.js +0 -2
  82. package/dist-scalprum/static/2821.972ae33b.chunk.js +0 -2
  83. package/dist-scalprum/static/2821.972ae33b.chunk.js.map +0 -1
  84. package/dist-scalprum/static/2967.65c51af8.chunk.js +0 -2
  85. package/dist-scalprum/static/2967.65c51af8.chunk.js.map +0 -1
  86. package/dist-scalprum/static/3091.9a74ea5f.chunk.js +0 -2
  87. package/dist-scalprum/static/3091.9a74ea5f.chunk.js.map +0 -1
  88. package/dist-scalprum/static/3650.ee76eba9.chunk.js +0 -2
  89. package/dist-scalprum/static/3650.ee76eba9.chunk.js.map +0 -1
  90. package/dist-scalprum/static/3947.7458971d.chunk.js +0 -2
  91. package/dist-scalprum/static/3947.7458971d.chunk.js.map +0 -1
  92. package/dist-scalprum/static/3984.647ef00c.chunk.js +0 -2
  93. package/dist-scalprum/static/3984.647ef00c.chunk.js.map +0 -1
  94. package/dist-scalprum/static/4651.c85cecc4.chunk.js +0 -2
  95. package/dist-scalprum/static/4651.c85cecc4.chunk.js.map +0 -1
  96. package/dist-scalprum/static/4682.75f17114.chunk.js +0 -2
  97. package/dist-scalprum/static/4682.75f17114.chunk.js.map +0 -1
  98. package/dist-scalprum/static/5010.0cd6c959.chunk.js +0 -3
  99. package/dist-scalprum/static/5010.0cd6c959.chunk.js.map +0 -1
  100. package/dist-scalprum/static/5203.b654e8e0.chunk.js +0 -2
  101. package/dist-scalprum/static/5203.b654e8e0.chunk.js.map +0 -1
  102. package/dist-scalprum/static/5453.ecdee66d.chunk.js +0 -2
  103. package/dist-scalprum/static/5453.ecdee66d.chunk.js.map +0 -1
  104. package/dist-scalprum/static/6763.d6cd937f.chunk.js.map +0 -1
  105. package/dist-scalprum/static/6800.20f46859.chunk.js +0 -2
  106. package/dist-scalprum/static/6800.20f46859.chunk.js.map +0 -1
  107. package/dist-scalprum/static/6840.1653e6b0.chunk.js +0 -2
  108. package/dist-scalprum/static/6840.1653e6b0.chunk.js.map +0 -1
  109. package/dist-scalprum/static/7791.ac1ac509.chunk.js +0 -2
  110. package/dist-scalprum/static/7791.ac1ac509.chunk.js.map +0 -1
  111. package/dist-scalprum/static/8172.cbe1f2c4.chunk.js +0 -2
  112. package/dist-scalprum/static/8172.cbe1f2c4.chunk.js.map +0 -1
  113. package/dist-scalprum/static/8799.52bbeef1.chunk.js +0 -2
  114. package/dist-scalprum/static/8799.52bbeef1.chunk.js.map +0 -1
  115. package/dist-scalprum/static/exposed-PluginRoot.55777e36.chunk.js +0 -2
  116. package/dist-scalprum/static/exposed-PluginRoot.55777e36.chunk.js.map +0 -1
  117. /package/dist/components/{ApiKeysPage → ApiKeyApprovalPage}/index.esm.js.map +0 -0
  118. /package/dist-scalprum/static/{5010.0cd6c959.chunk.js.LICENSE.txt → 5010.4ef96c87.chunk.js.LICENSE.txt} +0 -0
  119. /package/dist-scalprum/static/{6763.d6cd937f.chunk.js.LICENSE.txt → 8563.46f1a3e1.chunk.js.LICENSE.txt} +0 -0
@@ -0,0 +1,2 @@
1
+ "use strict";(self.webpackChunkinternal_plugin_kuadrant=self.webpackChunkinternal_plugin_kuadrant||[]).push([[3947],{71255:(e,a,i)=>{i.d(a,{n:()=>d});var s=i(31085),n=(i(95478),i(82300)),t=i(37725),l=i(96040),c=i(10394),r=i(42899),o=i(72501);const d=({issuerUrl:e,tokenEndpoint:a})=>{const i=a||e;return(0,s.jsx)(l.n,{title:"OIDC Provider Discovery",children:(0,s.jsx)(c.A,{p:2,children:(0,s.jsxs)(r.A,{container:!0,spacing:2,children:[(0,s.jsx)(r.A,{item:!0,xs:12,children:(0,s.jsx)(o.A,{variant:"body2",children:"This API uses OIDC authentication. Obtain a token from the identity provider below."})}),(0,s.jsx)(r.A,{item:!0,xs:12,children:(0,s.jsxs)(o.A,{variant:"body2",children:[(0,s.jsx)("strong",{children:"Identity Provider: "}),(0,s.jsx)(t.N_,{to:e,target:"_blank",children:e})]})}),a&&(0,s.jsx)(r.A,{item:!0,xs:12,children:(0,s.jsxs)(o.A,{variant:"body2",children:[(0,s.jsx)("strong",{children:"Token Endpoint: "}),(0,s.jsx)(t.N_,{to:a,target:"_blank",children:a})]})}),(0,s.jsx)(r.A,{item:!0,xs:12,children:(0,s.jsx)(n.z,{text:`# Example (Client Credentials Flow):\ncurl -X POST \\\n -d "grant_type=client_credentials" \\\n -d "client_id=YOUR_CLIENT_ID" \\\n -d "client_secret=YOUR_CLIENT_SECRET" \\\n ${i}`,language:"bash",showCopyCodeButton:!0})})]})})})}},89509:(e,a,i)=>{i.d(a,{O:()=>v});var s=i(31085),n=(i(95478),i(10394)),t=i(72501),l=i(67720),c=i(61009),r=i(47625),o=i(9719),d=i(54801),m=i(13677),h=i(42899),p=i(58837),x=i(37725),j=i(46299);const A=(0,p.A)(e=>({label:{fontWeight:600,color:e.palette.text.secondary,marginBottom:e.spacing(.5),fontSize:"0.75rem",textTransform:"uppercase"},tierChip:{marginRight:e.spacing(.5),marginBottom:e.spacing(.5)},statusChipPublished:{backgroundColor:e.palette.primary.main,color:e.palette.primary.contrastText},statusChipDraft:{backgroundColor:e.palette.grey[600],color:e.palette.common.white},infoGrid:{display:"grid",gridTemplateColumns:"repeat(auto-fill, minmax(180px, 1fr))",gap:e.spacing(3),marginBottom:e.spacing(3)},infoItem:{minWidth:0},apiLink:{color:e.palette.primary.main,textDecoration:"none","&:hover":{textDecoration:"underline"}}})),v=({product:e,showStatus:a=!0,showCatalogLink:i=!0,httpRouteHostnames:p})=>{var v,u,b,g,N,y,f,I,C,_,k,P,R,D,L,S,T,w,O;const U=A(),E=(null===(v=e.spec)||void 0===v?void 0:v.publishStatus)||"Draft",z="Published"===E,B=(null===(u=e.status)||void 0===u?void 0:u.discoveredPlans)||[],G=(null===(g=e.status)||void 0===g||null===(b=g.discoveredAuthScheme)||void 0===b?void 0:b.authentication)||{},H=Object.values(G).some(e=>e.hasOwnProperty("apiKey"));return(0,s.jsxs)(s.Fragment,{children:[(null===(N=e.spec)||void 0===N?void 0:N.description)&&(0,s.jsxs)(n.A,{mb:3,children:[(0,s.jsx)(t.A,{variant:"caption",className:U.label,children:"Description"}),(0,s.jsx)(t.A,{variant:"body1",children:e.spec.description})]}),(0,s.jsxs)(n.A,{className:U.infoGrid,children:[a&&(0,s.jsxs)(n.A,{className:U.infoItem,children:[(0,s.jsx)(t.A,{variant:"caption",className:U.label,children:"Publish Status"}),(0,s.jsx)(n.A,{children:(0,s.jsx)(l.A,{label:E,size:"small",className:z?U.statusChipPublished:U.statusChipDraft,"data-testid":"publish-status-chip"})})]}),(null===(y=e.metadata.labels)||void 0===y?void 0:y.lifecycle)&&(0,s.jsxs)(n.A,{className:U.infoItem,children:[(0,s.jsx)(t.A,{variant:"caption",className:U.label,children:"Lifecycle"}),(0,s.jsx)(n.A,{children:(0,s.jsx)(l.A,{label:e.metadata.labels.lifecycle,size:"small",style:(0,j.ee)(e.metadata.labels.lifecycle),"data-testid":"lifecycle-chip"})})]}),(0,s.jsxs)(n.A,{className:U.infoItem,children:[(0,s.jsx)(t.A,{variant:"caption",className:U.label,children:"Version"}),(0,s.jsx)(t.A,{variant:"body2",children:(null===(f=e.spec)||void 0===f?void 0:f.version)||"v1"})]}),(0,s.jsxs)(n.A,{className:U.infoItem,children:[(0,s.jsx)(t.A,{variant:"caption",className:U.label,children:"Namespace"}),(0,s.jsx)(t.A,{variant:"body2",children:e.metadata.namespace})]}),H&&(0,s.jsxs)(n.A,{className:U.infoItem,children:[(0,s.jsx)(t.A,{variant:"caption",className:U.label,children:"API Key Approval"}),(0,s.jsx)(t.A,{variant:"body2",children:"automatic"===(null===(I=e.spec)||void 0===I?void 0:I.approvalMode)?"Automatic":"Need manual approval"})]}),(null===(C=e.spec)||void 0===C?void 0:C.tags)&&e.spec.tags.length>0&&(0,s.jsxs)(n.A,{className:U.infoItem,children:[(0,s.jsx)(t.A,{variant:"caption",className:U.label,children:"Tags"}),(0,s.jsx)(n.A,{children:e.spec.tags.map(e=>(0,s.jsx)(l.A,{label:e,size:"small",variant:"outlined",className:U.tierChip},e))})]})]}),(0,s.jsxs)(n.A,{className:U.infoGrid,children:[(0,s.jsxs)(n.A,{className:U.infoItem,children:[(0,s.jsx)(t.A,{variant:"caption",className:U.label,children:"API"}),(0,s.jsx)("br",{}),i?(0,s.jsx)(x.N_,{to:`/catalog/default/api/${e.metadata.name}`,className:U.apiLink,children:e.metadata.name}):(0,s.jsx)(t.A,{variant:"body2",children:e.metadata.name})]}),(0,s.jsxs)(n.A,{className:U.infoItem,children:[(0,s.jsx)(t.A,{variant:"caption",className:U.label,children:"Route"}),(0,s.jsx)(t.A,{variant:"body2",children:(null===(k=e.spec)||void 0===k||null===(_=k.targetRef)||void 0===_?void 0:_.name)||"-"})]}),p&&p.length>0&&(0,s.jsxs)(n.A,{className:U.infoItem,children:[(0,s.jsx)(t.A,{variant:"caption",className:U.label,children:p.length>1?"Hostnames":"Hostname"}),p.map((e,a)=>(0,s.jsx)(t.A,{variant:"body2",children:e},a))]})]}),B.length>0&&(0,s.jsxs)(n.A,{mb:3,children:[(0,s.jsx)(t.A,{variant:"caption",className:U.label,children:"Available Tiers"}),(0,s.jsxs)(c.A,{size:"small",children:[(0,s.jsx)(r.A,{children:(0,s.jsxs)(o.A,{children:[(0,s.jsx)(d.A,{children:"Tier"}),(0,s.jsx)(d.A,{children:"Rate Limits"})]})}),(0,s.jsx)(m.A,{children:B.map(e=>(0,s.jsxs)(o.A,{children:[(0,s.jsx)(d.A,{children:(0,s.jsx)(l.A,{label:e.tier,size:"small"})}),(0,s.jsx)(d.A,{children:e.limits&&Object.entries(e.limits).map(([e,a])=>(0,s.jsxs)(t.A,{variant:"body2",children:[String(a)," per ",e]},e))})]},e.tier))})]})]}),(0,s.jsxs)(h.A,{container:!0,spacing:3,children:[((null===(R=e.spec)||void 0===R||null===(P=R.contact)||void 0===P?void 0:P.email)||(null===(L=e.spec)||void 0===L||null===(D=L.contact)||void 0===D?void 0:D.team))&&(0,s.jsxs)(h.A,{item:!0,xs:12,md:6,children:[(0,s.jsx)(t.A,{variant:"caption",className:U.label,children:"Contact Information"}),(0,s.jsxs)(n.A,{mt:1,children:[e.spec.contact.team&&(0,s.jsxs)(t.A,{variant:"body2",children:[(0,s.jsx)("strong",{children:"Team:"})," ",e.spec.contact.team]}),e.spec.contact.email&&(0,s.jsxs)(t.A,{variant:"body2",children:[(0,s.jsx)("strong",{children:"Email:"})," ",(0,s.jsx)(x.N_,{to:`mailto:${e.spec.contact.email}`,children:e.spec.contact.email})]})]})]}),((null===(T=e.spec)||void 0===T||null===(S=T.documentation)||void 0===S?void 0:S.docsURL)||(null===(O=e.spec)||void 0===O||null===(w=O.documentation)||void 0===w?void 0:w.openAPISpecURL))&&(0,s.jsxs)(h.A,{item:!0,xs:12,md:6,children:[(0,s.jsx)(t.A,{variant:"caption",className:U.label,children:"Documentation"}),(0,s.jsxs)(n.A,{mt:1,children:[e.spec.documentation.docsURL&&(0,s.jsxs)(t.A,{variant:"body2",children:[(0,s.jsx)("strong",{children:"Docs:"})," ",(0,s.jsx)(x.N_,{to:e.spec.documentation.docsURL,target:"_blank",children:"View Documentation"})]}),e.spec.documentation.openAPISpecURL&&(0,s.jsxs)(t.A,{variant:"body2",children:[(0,s.jsx)("strong",{children:"OpenAPI Spec:"})," ",(0,s.jsx)(x.N_,{to:e.spec.documentation.openAPISpecURL,target:"_blank",children:"View Spec"})]})]})]})]})]})}}}]);
2
+ //# sourceMappingURL=3947.ff1c25cf.chunk.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"static/3947.ff1c25cf.chunk.js","mappings":"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","sources":["webpack://internal.plugin-kuadrant/./src/components/OidcProviderCard/OidcProviderCard.tsx","webpack://internal.plugin-kuadrant/./src/components/ApiProductDetails/ApiProductDetails.tsx"],"sourcesContent":["import React from 'react';\nimport { InfoCard, CodeSnippet, Link } from '@backstage/core-components';\nimport { Typography, Box, Grid } from '@material-ui/core';\n\ninterface OidcProviderCardProps {\n issuerUrl: string;\n tokenEndpoint?: string;\n}\n\nexport const OidcProviderCard = ({ issuerUrl, tokenEndpoint }: OidcProviderCardProps) => {\n const effectiveTokenEndpoint = tokenEndpoint || issuerUrl;\n\n return (\n <InfoCard title=\"OIDC Provider Discovery\">\n <Box p={2}>\n <Grid container spacing={2}>\n <Grid item xs={12}>\n <Typography variant=\"body2\">\n This API uses OIDC authentication. Obtain a token from the identity provider below.\n </Typography>\n </Grid>\n <Grid item xs={12}>\n <Typography variant=\"body2\">\n <strong>Identity Provider: </strong>\n <Link to={issuerUrl} target=\"_blank\">\n {issuerUrl}\n </Link>\n </Typography>\n </Grid>\n {tokenEndpoint && (\n <Grid item xs={12}>\n <Typography variant=\"body2\">\n <strong>Token Endpoint: </strong>\n <Link to={tokenEndpoint} target=\"_blank\">\n {tokenEndpoint}\n </Link>\n </Typography>\n </Grid>\n )}\n <Grid item xs={12}>\n <CodeSnippet\n text={`# Example (Client Credentials Flow):\ncurl -X POST \\\\\n -d \"grant_type=client_credentials\" \\\\\n -d \"client_id=YOUR_CLIENT_ID\" \\\\\n -d \"client_secret=YOUR_CLIENT_SECRET\" \\\\\n ${effectiveTokenEndpoint}`}\n language=\"bash\"\n showCopyCodeButton\n />\n </Grid>\n </Grid>\n </Box>\n </InfoCard>\n );\n};\n","import React from \"react\";\nimport {\n Box,\n Typography,\n Chip,\n makeStyles,\n Table,\n TableBody,\n TableCell,\n TableHead,\n TableRow,\n Grid,\n} from \"@material-ui/core\";\nimport { Link } from \"@backstage/core-components\";\nimport { APIProduct, Plan } from \"../../types/api-management\";\nimport { getLifecycleChipStyle } from \"../../utils/styles\";\n\nconst useStyles = makeStyles((theme) => ({\n label: {\n fontWeight: 600,\n color: theme.palette.text.secondary,\n marginBottom: theme.spacing(0.5),\n fontSize: \"0.75rem\",\n textTransform: \"uppercase\",\n },\n tierChip: {\n marginRight: theme.spacing(0.5),\n marginBottom: theme.spacing(0.5),\n },\n statusChipPublished: {\n backgroundColor: theme.palette.primary.main,\n color: theme.palette.primary.contrastText,\n },\n statusChipDraft: {\n backgroundColor: theme.palette.grey[600],\n color: theme.palette.common.white,\n },\n infoGrid: {\n display: \"grid\",\n gridTemplateColumns: \"repeat(auto-fill, minmax(180px, 1fr))\",\n gap: theme.spacing(3),\n marginBottom: theme.spacing(3),\n },\n infoItem: {\n minWidth: 0,\n },\n apiLink: {\n color: theme.palette.primary.main,\n textDecoration: \"none\",\n \"&:hover\": {\n textDecoration: \"underline\",\n },\n },\n}));\n\ninterface ApiProductDetailsProps {\n product: APIProduct;\n showStatus?: boolean;\n showCatalogLink?: boolean;\n httpRouteHostnames?: string[] | null;\n}\n\nexport const ApiProductDetails = ({\n product,\n showStatus = true,\n showCatalogLink = true,\n httpRouteHostnames,\n}: ApiProductDetailsProps) => {\n const classes = useStyles();\n\n const publishStatus = product.spec?.publishStatus || \"Draft\";\n const isPublished = publishStatus === \"Published\";\n const tiers = product.status?.discoveredPlans || [];\n\n // check if product has API key auth\n const authSchemes = product.status?.discoveredAuthScheme?.authentication || {};\n const schemeObjects = Object.values(authSchemes);\n const hasApiKey = schemeObjects.some(\n (scheme: any) => scheme.hasOwnProperty(\"apiKey\"),\n );\n\n return (\n <>\n {product.spec?.description && (\n <Box mb={3}>\n <Typography variant=\"caption\" className={classes.label}>\n Description\n </Typography>\n <Typography variant=\"body1\">{product.spec.description}</Typography>\n </Box>\n )}\n\n <Box className={classes.infoGrid}>\n {showStatus && (\n <Box className={classes.infoItem}>\n <Typography variant=\"caption\" className={classes.label}>\n Publish Status\n </Typography>\n <Box>\n <Chip\n label={publishStatus}\n size=\"small\"\n className={\n isPublished\n ? classes.statusChipPublished\n : classes.statusChipDraft\n }\n data-testid=\"publish-status-chip\"\n />\n </Box>\n </Box>\n )}\n {product.metadata.labels?.lifecycle && (\n <Box className={classes.infoItem}>\n <Typography variant=\"caption\" className={classes.label}>\n Lifecycle\n </Typography>\n <Box>\n <Chip\n label={product.metadata.labels.lifecycle}\n size=\"small\"\n style={getLifecycleChipStyle(product.metadata.labels.lifecycle)}\n data-testid=\"lifecycle-chip\"\n />\n </Box>\n </Box>\n )}\n <Box className={classes.infoItem}>\n <Typography variant=\"caption\" className={classes.label}>\n Version\n </Typography>\n <Typography variant=\"body2\">\n {product.spec?.version || \"v1\"}\n </Typography>\n </Box>\n <Box className={classes.infoItem}>\n <Typography variant=\"caption\" className={classes.label}>\n Namespace\n </Typography>\n <Typography variant=\"body2\">{product.metadata.namespace}</Typography>\n </Box>\n {hasApiKey && (\n <Box className={classes.infoItem}>\n <Typography variant=\"caption\" className={classes.label}>\n API Key Approval\n </Typography>\n <Typography variant=\"body2\">\n {product.spec?.approvalMode === \"automatic\"\n ? \"Automatic\"\n : \"Need manual approval\"}\n </Typography>\n </Box>\n )}\n {product.spec?.tags && product.spec.tags.length > 0 && (\n <Box className={classes.infoItem}>\n <Typography variant=\"caption\" className={classes.label}>\n Tags\n </Typography>\n <Box>\n {product.spec.tags.map((tag) => (\n <Chip\n key={tag}\n label={tag}\n size=\"small\"\n variant=\"outlined\"\n className={classes.tierChip}\n />\n ))}\n </Box>\n </Box>\n )}\n </Box>\n\n <Box className={classes.infoGrid}>\n <Box className={classes.infoItem}>\n <Typography variant=\"caption\" className={classes.label}>\n API\n </Typography>\n <br />\n {showCatalogLink ? (\n <Link\n to={`/catalog/default/api/${product.metadata.name}`}\n className={classes.apiLink}\n >\n {product.metadata.name}\n </Link>\n ) : (\n <Typography variant=\"body2\">{product.metadata.name}</Typography>\n )}\n </Box>\n <Box className={classes.infoItem}>\n <Typography variant=\"caption\" className={classes.label}>\n Route\n </Typography>\n <Typography variant=\"body2\">\n {product.spec?.targetRef?.name || \"-\"}\n </Typography>\n </Box>\n {httpRouteHostnames && httpRouteHostnames.length > 0 && (\n <Box className={classes.infoItem}>\n <Typography variant=\"caption\" className={classes.label}>\n {httpRouteHostnames.length > 1 ? \"Hostnames\" : \"Hostname\"}\n </Typography>\n {httpRouteHostnames.map((hostname, index) => (\n <Typography key={index} variant=\"body2\">\n {hostname}\n </Typography>\n ))}\n </Box>\n )}\n </Box>\n\n {tiers.length > 0 && (\n <Box mb={3}>\n <Typography variant=\"caption\" className={classes.label}>\n Available Tiers\n </Typography>\n <Table size=\"small\">\n <TableHead>\n <TableRow>\n <TableCell>Tier</TableCell>\n <TableCell>Rate Limits</TableCell>\n </TableRow>\n </TableHead>\n <TableBody>\n {tiers.map((plan: Plan) => (\n <TableRow key={plan.tier}>\n <TableCell>\n <Chip label={plan.tier} size=\"small\" />\n </TableCell>\n <TableCell>\n {plan.limits &&\n Object.entries(plan.limits).map(([key, value]) => (\n <Typography key={key} variant=\"body2\">\n {String(value)} per {key}\n </Typography>\n ))}\n </TableCell>\n </TableRow>\n ))}\n </TableBody>\n </Table>\n </Box>\n )}\n\n <Grid container spacing={3}>\n {(product.spec?.contact?.email || product.spec?.contact?.team) && (\n <Grid item xs={12} md={6}>\n <Typography variant=\"caption\" className={classes.label}>\n Contact Information\n </Typography>\n <Box mt={1}>\n {product.spec.contact.team && (\n <Typography variant=\"body2\">\n <strong>Team:</strong> {product.spec.contact.team}\n </Typography>\n )}\n {product.spec.contact.email && (\n <Typography variant=\"body2\">\n <strong>Email:</strong>{\" \"}\n <Link to={`mailto:${product.spec.contact.email}`}>\n {product.spec.contact.email}\n </Link>\n </Typography>\n )}\n </Box>\n </Grid>\n )}\n\n {(product.spec?.documentation?.docsURL ||\n product.spec?.documentation?.openAPISpecURL) && (\n <Grid item xs={12} md={6}>\n <Typography variant=\"caption\" className={classes.label}>\n Documentation\n </Typography>\n <Box mt={1}>\n {product.spec.documentation.docsURL && (\n <Typography variant=\"body2\">\n <strong>Docs:</strong>{\" \"}\n <Link to={product.spec.documentation.docsURL} target=\"_blank\">\n View Documentation\n </Link>\n </Typography>\n )}\n {product.spec.documentation.openAPISpecURL && (\n <Typography variant=\"body2\">\n <strong>OpenAPI Spec:</strong>{\" \"}\n <Link\n to={product.spec.documentation.openAPISpecURL}\n target=\"_blank\"\n >\n View Spec\n </Link>\n </Typography>\n )}\n </Box>\n </Grid>\n )}\n </Grid>\n </>\n );\n};\n"],"names":["OidcProviderCard","issuerUrl","tokenEndpoint","effectiveTokenEndpoint","InfoCard","title","Box","p","Grid","container","spacing","item","xs","Typography","variant","strong","Link","to","target","CodeSnippet","text","language","showCopyCodeButton","useStyles","makeStyles","theme","label","fontWeight","color","palette","secondary","marginBottom","fontSize","textTransform","tierChip","marginRight","statusChipPublished","backgroundColor","primary","main","contrastText","statusChipDraft","grey","common","white","infoGrid","display","gridTemplateColumns","gap","infoItem","minWidth","apiLink","textDecoration","ApiProductDetails","product","showStatus","showCatalogLink","httpRouteHostnames","classes","publishStatus","spec","isPublished","tiers","status","discoveredPlans","authSchemes","discoveredAuthScheme","authentication","hasApiKey","Object","values","some","scheme","hasOwnProperty","description","mb","className","Chip","size","data-testid","metadata","labels","lifecycle","style","getLifecycleChipStyle","version","namespace","approvalMode","tags","length","map","tag","br","name","targetRef","hostname","index","Table","TableHead","TableRow","TableCell","TableBody","plan","tier","limits","entries","key","value","String","contact","email","team","md","mt","documentation","docsURL","openAPISpecURL"],"sourceRoot":""}
@@ -0,0 +1,2 @@
1
+ "use strict";(self.webpackChunkinternal_plugin_kuadrant=self.webpackChunkinternal_plugin_kuadrant||[]).push([[3976],{12229:(e,t,a)=>{a.d(t,{Z:()=>x});var r=a(31085),n=a(95478),s=a.n(n),o=a(10394),i=a(72501),l=a(64947),c=a(37197),d=a(69621),u=a(12981),p=a(86901),m=a(69076),g=a(58837),h=a(6924),v=a(23164);const f=(0,g.A)(e=>({root:{width:240,minWidth:240,padding:e.spacing(2),borderRight:`1px solid ${e.palette.divider}`,backgroundColor:e.palette.background.paper,height:"100%",overflowY:"auto"},sectionTitle:{fontWeight:600,fontSize:"0.75rem",textTransform:"uppercase",letterSpacing:"0.05em",color:e.palette.text.secondary,marginBottom:e.spacing(1),display:"flex",alignItems:"center",justifyContent:"space-between",cursor:"pointer",userSelect:"none"},filterSection:{marginBottom:e.spacing(2)},checkbox:{padding:e.spacing(.5)},checkboxLabel:{fontSize:"0.875rem"},clearButton:{marginTop:e.spacing(2)},count:{fontSize:"0.75rem",color:e.palette.text.secondary,marginLeft:e.spacing(1)}})),x=({sections:e,filters:t,onChange:a,onClear:n})=>{const g=f(),[x,A]=s().useState(new Set(e.filter(e=>e.collapsed).map(e=>e.id))),j=Object.values(t).some(e=>e.length>0);return(0,r.jsxs)(o.A,{className:g.root,children:[(0,r.jsxs)(o.A,{display:"flex",justifyContent:"space-between",alignItems:"center",mb:2,children:[(0,r.jsx)(i.A,{variant:"subtitle2",children:"Filters"}),j&&(0,r.jsx)(l.A,{size:"small",color:"primary",onClick:()=>{const t={};e.forEach(e=>{t[e.id]=[]}),a(t),null==n||n()},children:"Clear all"})]}),(0,r.jsx)(c.A,{}),e.map(e=>{const n=x.has(e.id),s=(t[e.id]||[]).length;return(0,r.jsxs)(o.A,{className:g.filterSection,mt:2,children:[(0,r.jsxs)(o.A,{className:g.sectionTitle,onClick:()=>{return t=e.id,void A(e=>{const a=new Set(e);return a.has(t)?a.delete(t):a.add(t),a});var t},children:[(0,r.jsxs)(o.A,{display:"flex",alignItems:"center",children:[(0,r.jsx)("span",{children:e.title}),s>0&&(0,r.jsxs)("span",{className:g.count,children:["(",s,")"]})]}),n?(0,r.jsx)(h.A,{fontSize:"small"}):(0,r.jsx)(v.A,{fontSize:"small"})]}),(0,r.jsx)(d.A,{in:!n,children:(0,r.jsx)(u.A,{children:e.options.map(n=>(0,r.jsx)(p.A,{control:(0,r.jsx)(m.A,{checked:(t[e.id]||[]).includes(n.value),onChange:()=>((e,r)=>{const n=t[e]||[],s=n.includes(r)?n.filter(e=>e!==r):[...n,r];a({...t,[e]:s})})(e.id,n.value),size:"small",className:g.checkbox,color:"primary"}),label:(0,r.jsxs)(o.A,{display:"flex",alignItems:"center",children:[(0,r.jsx)("span",{className:g.checkboxLabel,children:n.label}),void 0!==n.count&&(0,r.jsxs)("span",{className:g.count,children:["(",n.count,")"]})]})},n.value))})})]},e.id)})]})}},23164:(e,t,a)=>{var r=a(4293),n=a(78920);t.A=void 0;var s=n(a(95478)),o=(0,r(a(74044)).default)(s.createElement("path",{d:"M12 8l-6 6 1.41 1.41L12 10.83l4.59 4.58L18 14z"}),"ExpandLess");t.A=o},27799:(e,t,a)=>{var r=a(4293),n=a(78920);t.A=void 0;var s=n(a(95478)),o=(0,r(a(74044)).default)(s.createElement("path",{d:"M21 5v6.59l-3-3.01-4 4.01-4-4-4 4-3-3.01V5c0-1.1.9-2 2-2h14c1.1 0 2 .9 2 2zm-3 6.42l3 3.01V19c0 1.1-.9 2-2 2H5c-1.1 0-2-.9-2-2v-6.58l3 2.99 4-4 4 4 4-3.99z"}),"BrokenImage");t.A=o},34955:(e,t,a)=>{a.d(t,{Al:()=>o,EM:()=>c,FL:()=>s,J:()=>n,KV:()=>f,R_:()=>d,U3:()=>i,dp:()=>p,jH:()=>v,q0:()=>m,uL:()=>h,v_:()=>l,vs:()=>u,z4:()=>g});var r=a(83572);(0,r.i)({name:"kuadrant.planpolicy.create",attributes:{action:"create"}}),(0,r.i)({name:"kuadrant.planpolicy.read",attributes:{action:"read"}}),(0,r.i)({name:"kuadrant.planpolicy.update",attributes:{action:"update"}}),(0,r.i)({name:"kuadrant.planpolicy.delete",attributes:{action:"delete"}});const n=(0,r.i)({name:"kuadrant.planpolicy.list",attributes:{action:"read"}}),s=(0,r.i)({name:"kuadrant.apiproduct.create",attributes:{action:"create"}}),o=((0,r.i)({name:"kuadrant.apiproduct.read.own",attributes:{action:"read"}}),(0,r.i)({name:"kuadrant.apiproduct.read.all",attributes:{action:"read"}})),i=(0,r.i)({name:"kuadrant.apiproduct.update.own",attributes:{action:"update"}}),l=(0,r.i)({name:"kuadrant.apiproduct.update.all",attributes:{action:"update"}}),c=(0,r.i)({name:"kuadrant.apiproduct.delete.own",attributes:{action:"delete"}}),d=(0,r.i)({name:"kuadrant.apiproduct.delete.all",attributes:{action:"delete"}}),u=(0,r.i)({name:"kuadrant.apiproduct.list",attributes:{action:"read"}}),p=(0,r.i)({name:"kuadrant.apikey.create",attributes:{action:"create"},resourceType:"apiproduct"}),m=((0,r.i)({name:"kuadrant.apikey.read.own",attributes:{action:"read"}}),(0,r.i)({name:"kuadrant.apikey.read.all",attributes:{action:"read"}}),(0,r.i)({name:"kuadrant.apikey.update.own",attributes:{action:"update"}})),g=(0,r.i)({name:"kuadrant.apikey.update.all",attributes:{action:"update"}}),h=(0,r.i)({name:"kuadrant.apikey.delete.own",attributes:{action:"delete"}}),v=(0,r.i)({name:"kuadrant.apikey.delete.all",attributes:{action:"delete"}}),f=(0,r.i)({name:"kuadrant.apikey.approve",attributes:{action:"update"}})},35015:(e,t,a)=>{a.d(t,{A:()=>o});var r=a(95478),n=a(85608),s=a(71581);function o(e,t){void 0===t&&(t=[]);var a=function(e,t,a){void 0===t&&(t=[]),void 0===a&&(a={loading:!1});var o=(0,r.useRef)(0),i=(0,s.A)(),l=(0,r.useState)(a),c=l[0],d=l[1],u=(0,r.useCallback)(function(){for(var t=[],a=0;a<arguments.length;a++)t[a]=arguments[a];var r=++o.current;return c.loading||d(function(e){return(0,n.__assign)((0,n.__assign)({},e),{loading:!0})}),e.apply(void 0,t).then(function(e){return i()&&r===o.current&&d({value:e,loading:!1}),e},function(e){return i()&&r===o.current&&d({error:e,loading:!1}),e})},t);return[c,u]}(e,t,{loading:!0}),o=a[0],i=a[1];return(0,r.useEffect)(function(){i()},[i]),o}},43976:(e,t,a)=>{a.r(t),a.d(t,{ApiKeyApprovalPage:()=>M});var r=a(31085),n=a(95478),s=a.n(n),o=a(289),i=a(15831),l=a(45210),c=a(46681),d=a(22097),u=a(35015),p=a(37725),m=a(86687),g=a(42367),h=a(25010),v=a(34955),f=a(46205),x=a(58837),A=a(76891),j=a(61477),y=a(10394),b=a(46805),k=a(72501),w=a(16249),C=a(93453),S=a(64947),R=a(78467),q=a(67720),P=a(55429),I=a(92399),B=a(12229),$=a(46299),T=a(77318);const N=(0,x.A)(e=>({container:{display:"flex",height:"100%",minHeight:400},tableContainer:{flex:1,overflow:"auto",padding:10},useCasePanel:{padding:e.spacing(2),backgroundColor:e.palette.background.default},useCaseLabel:{fontWeight:600,marginBottom:e.spacing(1),color:e.palette.text.secondary,textTransform:"uppercase",fontSize:"0.75rem"},bulkActions:{padding:e.spacing(2),backgroundColor:e.palette.background.default,borderBottom:`1px solid ${e.palette.divider}`,display:"flex",alignItems:"center",justifyContent:"space-between"}})),z=({open:e,request:t,action:a,processing:n,onClose:o,onConfirm:i})=>{var l,c;const[d,u]=s().useState(""),p="approve"===a?"Approve":"Reject",m="approve"===a?"Approving...":"Rejecting...",g="reject"===a,h=(null==t||null===(l=t.spec.requestedBy)||void 0===l?void 0:l.userId)||"",v=!g||d===h;return s().useEffect(()=>{e||u("")},[e]),(0,r.jsxs)(A.A,{open:e,onClose:n?void 0:o,maxWidth:"sm",fullWidth:!0,children:[(0,r.jsx)(j.A,{children:g?(0,r.jsxs)(y.A,{display:"flex",alignItems:"center",style:{gap:8},children:[(0,r.jsx)(I.A,{color:"error"}),(0,r.jsxs)("span",{children:[p," API Key"]})]}):(0,r.jsxs)("span",{children:[p," API Key"]})}),(0,r.jsx)(b.A,{children:t&&(0,r.jsxs)(r.Fragment,{children:[(0,r.jsxs)("p",{children:[(0,r.jsx)("strong",{children:"User:"})," ",t.spec.requestedBy.userId]}),(0,r.jsxs)("p",{children:[(0,r.jsx)("strong",{children:"API:"})," ",(null===(c=t.spec.apiProductRef)||void 0===c?void 0:c.name)||"unknown"]}),(0,r.jsxs)("p",{children:[(0,r.jsx)("strong",{children:"Tier:"})," ",t.spec.planTier]}),(0,r.jsxs)(y.A,{mb:2,children:[(0,r.jsx)(k.A,{variant:"body2",component:"span",style:{fontWeight:"bold"},children:"Use Case:"})," ",(0,r.jsx)(k.A,{variant:"body2",component:"span",style:{whiteSpace:"pre-wrap"},children:t.spec.useCase||"-"})]}),g&&(0,r.jsxs)(y.A,{mt:2,children:[(0,r.jsxs)(k.A,{variant:"body2",color:"textSecondary",gutterBottom:!0,children:["Type ",(0,r.jsx)("strong",{children:h})," to confirm rejection:"]}),(0,r.jsx)(w.A,{fullWidth:!0,variant:"outlined",size:"small",value:d,onChange:e=>u(e.target.value),disabled:n,autoFocus:!0,placeholder:h})]})]})}),(0,r.jsxs)(C.A,{children:[(0,r.jsx)(S.A,{onClick:o,disabled:n,children:"Cancel"}),(0,r.jsx)(S.A,{onClick:i,color:"approve"===a?"primary":"secondary",variant:"contained",disabled:n||!v,startIcon:n?(0,r.jsx)(R.A,{size:16,color:"inherit"}):void 0,children:n?m:p})]})]})},L=({open:e,requests:t,action:a,processing:n,onClose:s,onConfirm:o})=>{const i="approve"===a,l=i?"Approve All":"Reject All",c=i?"Approving...":"Rejecting...";return(0,r.jsxs)(A.A,{open:e,onClose:n?void 0:s,maxWidth:"md",fullWidth:!0,children:[(0,r.jsxs)(j.A,{children:[i?"Approve":"Reject"," ",t.length," API Keys"]}),(0,r.jsxs)(b.A,{children:[(0,r.jsxs)(k.A,{variant:"body2",paragraph:!0,children:["You are about to ",i?"approve":"reject"," the following API keys:"]}),(0,r.jsx)(y.A,{mb:2,maxHeight:200,overflow:"auto",children:t.map(e=>{var t;return(0,r.jsx)(y.A,{mb:1,p:1,bgcolor:"background.default",children:(0,r.jsxs)(k.A,{variant:"body2",children:[(0,r.jsx)("strong",{children:e.spec.requestedBy.userId})," -"," ",(null===(t=e.spec.apiProductRef)||void 0===t?void 0:t.name)||"unknown"," (",e.spec.planTier,")"]})},`${e.metadata.namespace}/${e.metadata.name}`)})})]}),(0,r.jsxs)(C.A,{children:[(0,r.jsx)(S.A,{onClick:s,disabled:n,children:"Cancel"}),(0,r.jsx)(S.A,{onClick:o,color:i?"primary":"secondary",variant:"contained",disabled:n,startIcon:n?(0,r.jsx)(R.A,{size:16,color:"inherit"}):void 0,children:n?c:l})]})]})},E=({request:e})=>{const t=N();return(0,r.jsxs)(y.A,{className:t.useCasePanel,onClick:e=>e.stopPropagation(),children:[(0,r.jsx)(k.A,{className:t.useCaseLabel,children:"Use Case"}),(0,r.jsx)(k.A,{variant:"body2",children:e.spec.useCase||"No use case provided"})]})},W=(0,x.A)(e=>({successSection:{padding:e.spacing(2),backgroundColor:e.palette.success.main+"14",borderRadius:e.shape.borderRadius,marginBottom:e.spacing(2),border:`1px solid ${e.palette.success.main}40`},failureSection:{padding:e.spacing(2),backgroundColor:e.palette.error.main+"14",borderRadius:e.shape.borderRadius,marginBottom:e.spacing(2),border:`1px solid ${e.palette.error.main}40`},warningSection:{padding:e.spacing(2),backgroundColor:e.palette.warning.main+"14",borderRadius:e.shape.borderRadius,marginBottom:e.spacing(2),border:`1px solid ${e.palette.warning.main}40`},statRow:{display:"flex",alignItems:"center",gap:e.spacing(1)},errorList:{marginTop:e.spacing(2),padding:0,listStyle:"none"},errorItem:{padding:e.spacing(1.5),marginBottom:e.spacing(1),backgroundColor:e.palette.background.paper,borderRadius:e.shape.borderRadius,border:`1px solid ${e.palette.divider}`},errorName:{fontWeight:600,fontSize:"0.875rem",marginBottom:e.spacing(.5)},errorMessage:{fontSize:"0.813rem",color:e.palette.error.main,fontFamily:"monospace",wordBreak:"break-word"},expandButton:{marginTop:e.spacing(1),padding:e.spacing(.5)}})),F=({open:e,results:t,isApprove:a,onClose:s})=>{const o=W(),[i,l]=(0,n.useState)(!0),c=t.filter(e=>e.success),d=t.filter(e=>!e.success),u=c.length>0;return(0,r.jsxs)(A.A,{open:e,onClose:s,maxWidth:"md",fullWidth:!0,children:[(0,r.jsx)(j.A,{children:(0,r.jsx)(y.A,{display:"flex",alignItems:"center",style:{gap:8},children:u?(0,r.jsxs)(r.Fragment,{children:[(0,r.jsx)(I.A,{style:{color:"#ff9800"}}),(0,r.jsxs)("span",{children:["Bulk ",a?"Approve":"Reject"," Completed with Issues"]})]}):(0,r.jsxs)(r.Fragment,{children:[(0,r.jsx)(I.A,{color:"error"}),(0,r.jsxs)("span",{children:["Bulk ",a?"Approve":"Reject"," Failed"]})]})})}),(0,r.jsxs)(b.A,{children:[u&&(0,r.jsx)(y.A,{className:o.successSection,children:(0,r.jsxs)(y.A,{className:o.statRow,children:[(0,r.jsx)(P.A,{style:{color:"#4caf50"}}),(0,r.jsxs)(k.A,{variant:"body1",children:[(0,r.jsx)("strong",{children:c.length})," API key",1!==c.length?"s":""," ",a?"approved":"rejected"," successfully"]})]})}),(0,r.jsxs)(y.A,{className:u?o.warningSection:o.failureSection,children:[(0,r.jsxs)(y.A,{className:o.statRow,children:[(0,r.jsx)(I.A,{color:"error"}),(0,r.jsxs)(k.A,{variant:"body1",children:[(0,r.jsx)("strong",{children:d.length})," request",1!==d.length?"s":""," failed"]})]}),(0,r.jsxs)(S.A,{size:"small",className:o.expandButton,onClick:()=>l(!i),children:[i?"Hide":"Show"," Error Details"]}),i&&(0,r.jsx)("ul",{className:o.errorList,children:d.map((e,t)=>(0,r.jsxs)("li",{className:o.errorItem,children:[(0,r.jsxs)("div",{className:o.errorName,children:[e.namespace,"/",e.name]}),(0,r.jsx)("div",{className:o.errorMessage,children:e.error||"Unknown error"})]},e.name||t))})]})]}),(0,r.jsx)(C.A,{children:(0,r.jsx)(S.A,{onClick:s,color:"primary",variant:"contained",children:"Close"})})]})},O=()=>{var e;const t=N(),a=(0,d.useApi)(d.configApiRef),s=(0,d.useApi)(d.fetchApiRef),o=(0,d.useApi)(d.identityApiRef),i=(0,d.useApi)(d.alertApiRef),l=a.getString("backend.baseUrl"),[c,x]=(0,n.useState)(0),[A,j]=(0,n.useState)([]),[b,w]=(0,n.useState)({open:!1,request:null,action:"approve",processing:!1}),[C,R]=(0,n.useState)({open:!1,requests:[],action:"approve",processing:!1}),[W,O]=(0,n.useState)({open:!1,results:[],isApprove:!0}),[M,_]=(0,n.useState)({status:[],apiProduct:[],tier:[]}),{allowed:U,loading:H,error:K}=(0,f.l)(v.z4),{allowed:Y,loading:D,error:J}=(0,f.l)(v.q0),V=H||D,Z=K||J,{value:G,loading:X,error:Q}=(0,u.A)(async()=>{const e=(await o.getBackstageIdentity()).userEntityRef,[t,a]=await Promise.all([s.fetch(`${l}/api/kuadrant/requests`),s.fetch(`${l}/api/kuadrant/apiproducts`)]);if(!t.ok)return{allRequests:[],reviewedBy:e,ownedApiProducts:new Set};const r=t.headers.get("content-type");if(!(null==r?void 0:r.includes("application/json")))return i.post({message:"Unexpected content-type from the server response.",display:"transient",severity:"warning"}),{allRequests:[],reviewedBy:e,ownedApiProducts:new Set};const n=(await t.json()).items||[],c=new Set;if(a.ok){const t=await a.json();for(const a of t.items||[]){var d,u;(null===(u=a.metadata)||void 0===u||null===(d=u.annotations)||void 0===d?void 0:d["backstage.io/owner"])===e&&c.add(`${a.metadata.namespace}/${a.metadata.name}`)}}return{allRequests:n,reviewedBy:e,ownedApiProducts:c}},[l,s,o,c]),ee=(0,n.useMemo)(()=>{if(!(null==G?void 0:G.allRequests))return[];const e={Approved:0,Pending:0,Rejected:0},t=new Map,a=new Map;return G.allRequests.forEach(r=>{var n,s;const o=(null===(n=r.status)||void 0===n?void 0:n.phase)||"Pending";e[o]++;const i=(null===(s=r.spec.apiProductRef)||void 0===s?void 0:s.name)||"unknown";t.set(i,(t.get(i)||0)+1);const l=r.spec.planTier||"unknown";a.set(l,(a.get(l)||0)+1)}),[{id:"status",title:"Status",options:[{value:"Pending",label:"Pending",count:e.Pending},{value:"Approved",label:"Approved",count:e.Approved},{value:"Rejected",label:"Rejected",count:e.Rejected}]},{id:"apiProduct",title:"API Product",options:Array.from(t.entries()).map(([e,t])=>({value:e,label:e,count:t})),collapsed:t.size>5},{id:"tier",title:"Tier",options:Array.from(a.entries()).map(([e,t])=>({value:e,label:e.charAt(0).toUpperCase()+e.slice(1),count:t}))}]},[null==G?void 0:G.allRequests]),te=(0,n.useMemo)(()=>(null==G?void 0:G.allRequests)?G.allRequests.filter(e=>{if(M.status.length>0){var t;const a=(null===(t=e.status)||void 0===t?void 0:t.phase)||"Pending";if(!M.status.includes(a))return!1}if(M.apiProduct.length>0){var a;const t=(null===(a=e.spec.apiProductRef)||void 0===a?void 0:a.name)||"unknown";if(!M.apiProduct.includes(t))return!1}if(M.tier.length>0){const t=e.spec.planTier||"unknown";if(!M.tier.includes(t))return!1}return!0}):[],[null==G?void 0:G.allRequests,M]),ae=e=>new Date(e).toLocaleDateString("en-GB",{year:"numeric",month:"short",day:"numeric"}),re=[{title:"Requester",field:"spec.requestedBy.userId",render:e=>(0,r.jsx)(k.A,{variant:"body2",children:e.spec.requestedBy.userId})},{title:"API Product",field:"spec.apiProductRef.name",render:e=>{var t;const a=(null===(t=e.spec.apiProductRef)||void 0===t?void 0:t.name)||"unknown";return(0,r.jsx)(p.N_,{to:`/catalog/default/api/${a}`,children:(0,r.jsx)("strong",{children:a})})}},{title:"Status",field:"status.phase",render:e=>{var t;const a=(null===(t=e.status)||void 0===t?void 0:t.phase)||"Pending";return(0,r.jsx)(q.A,{label:a,size:"small",style:(0,$.uU)(a)})}},{title:"Tier",field:"spec.planTier",render:e=>(0,r.jsx)(q.A,{label:e.spec.planTier,size:"small",variant:"outlined"})},{title:"Requested",field:"metadata.creationTimestamp",render:e=>(0,r.jsx)(k.A,{variant:"body2",children:e.metadata.creationTimestamp?ae(e.metadata.creationTimestamp):"-"})},{title:"Reviewed By",field:"status.reviewedBy",render:e=>{var t;if(!(null===(t=e.status)||void 0===t?void 0:t.reviewedBy))return(0,r.jsx)(k.A,{variant:"body2",color:"textSecondary",children:"-"});const a=e.status.reviewedBy.replace(/^user:default\//,""),n="system"===a;return(0,r.jsxs)(y.A,{children:[(0,r.jsx)(k.A,{variant:"body2",children:n?"Automatic":a}),e.status.reviewedAt&&(0,r.jsx)(k.A,{variant:"caption",color:"textSecondary",children:ae(e.status.reviewedAt)})]})}},{title:"Actions",filtering:!1,render:e=>{var t,a,n;if("Pending"!==((null===(t=e.status)||void 0===t?void 0:t.phase)||"Pending"))return null;const s=`${e.metadata.namespace}/${(null===(a=e.spec.apiProductRef)||void 0===a?void 0:a.name)||"unknown"}`;var o;const i=null!==(o=null==G||null===(n=G.ownedApiProducts)||void 0===n?void 0:n.has(s))&&void 0!==o&&o;return U||Y&&i?(0,r.jsxs)(y.A,{display:"flex",style:{gap:8},children:[(0,r.jsx)(S.A,{size:"small",startIcon:(0,r.jsx)(P.A,{}),onClick:()=>{w({open:!0,request:e,action:"approve",processing:!1})},color:"primary",variant:"outlined",children:"Approve"}),(0,r.jsx)(S.A,{size:"small",startIcon:(0,r.jsx)(I.A,{}),onClick:()=>{w({open:!0,request:e,action:"reject",processing:!1})},color:"secondary",variant:"outlined",children:"Reject"})]}):null}}],ne=(0,n.useMemo)(()=>[{render:e=>{var t;const a=e.rowData;return(null==a||null===(t=a.metadata)||void 0===t?void 0:t.name)?(0,r.jsx)(E,{request:a}):(0,r.jsx)(y.A,{})}}],[]);if(X||V)return(0,r.jsx)(m.k,{});if(Q)return(0,r.jsx)(g._,{error:Q});if(Z)return(0,r.jsx)(y.A,{p:2,children:(0,r.jsxs)(k.A,{color:"error",children:["Unable to check permissions: ",Z.message]})});const se=U||Y;return(0,r.jsxs)(r.Fragment,{children:[(0,r.jsxs)(y.A,{className:t.container,children:[(0,r.jsx)(B.Z,{sections:ee,filters:M,onChange:_}),(0,r.jsxs)(y.A,{className:t.tableContainer,children:[A.length>0&&(0,r.jsxs)(y.A,{className:t.bulkActions,children:[(0,r.jsxs)(k.A,{variant:"body2",children:[A.length," request",1!==A.length?"s":""," selected"]}),(0,r.jsxs)(y.A,{display:"flex",style:{gap:8},children:[(0,r.jsx)(S.A,{size:"small",variant:"contained",color:"primary",startIcon:(0,r.jsx)(P.A,{}),onClick:()=>{0!==A.length&&R({open:!0,requests:A,action:"approve",processing:!1})},children:"Approve Selected"}),(0,r.jsx)(S.A,{size:"small",variant:"contained",color:"secondary",startIcon:(0,r.jsx)(I.A,{}),onClick:()=>{0!==A.length&&R({open:!0,requests:A,action:"reject",processing:!1})},children:"Reject Selected"})]})]}),0===te.length?(0,r.jsx)(y.A,{p:4,textAlign:"center",children:(0,r.jsx)(k.A,{variant:"body1",color:"textSecondary",children:0===(null==G||null===(e=G.allRequests)||void 0===e?void 0:e.length)?"No API keys found.":"No API keys match the selected filters."})}):(0,r.jsx)(h.X,{options:{selection:se,showSelectAllCheckbox:te.some(e=>{var t;return!(null===(t=e.status)||void 0===t?void 0:t.phase)||"Pending"===e.status.phase}),selectionProps:e=>{var t,a;return{disabled:"Pending"!==(null===(t=e.status)||void 0===t?void 0:t.phase)&&void 0!==(null===(a=e.status)||void 0===a?void 0:a.phase)}},paging:te.length>10,pageSize:20,search:!0,filtering:!1,debounceInterval:300,showTextRowsSelected:!1,toolbar:!0,emptyRowsWhenPaging:!1},columns:re,data:te.map(e=>{const t=A.some(t=>t.metadata.name===e.metadata.name&&t.metadata.namespace===e.metadata.namespace);return{...e,id:e.metadata.name,tableData:{checked:t}}}),onSelectionChange:e=>{const t=e.filter(e=>{var t;return!(null===(t=e.status)||void 0===t?void 0:t.phase)||"Pending"===e.status.phase});j(t)},detailPanel:ne})]})]}),(0,r.jsx)(z,{open:b.open,request:b.request,action:b.action,processing:b.processing,onClose:()=>w({open:!1,request:null,action:"approve",processing:!1}),onConfirm:async()=>{if(!b.request||!G)return;w(e=>({...e,processing:!0}));const e="approve"===b.action?`${l}/api/kuadrant/requests/${b.request.metadata.namespace}/${b.request.metadata.name}/approve`:`${l}/api/kuadrant/requests/${b.request.metadata.namespace}/${b.request.metadata.name}/reject`;try{const t=await s.fetch(e,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({reviewedBy:G.reviewedBy})});if(!t.ok){const e=await(0,T.T)(t);throw new Error(e)}w({open:!1,request:null,action:"approve",processing:!1}),j(e=>e.filter(e=>{var t,a;return e.metadata.name!==(null===(t=b.request)||void 0===t?void 0:t.metadata.name)||e.metadata.namespace!==(null===(a=b.request)||void 0===a?void 0:a.metadata.namespace)})),x(e=>e+1);const a="approve"===b.action?"approved":"rejected";i.post({message:`API key ${a}`,severity:"success",display:"transient"})}catch(e){console.error(`error ${b.action}ing request:`,e),w(e=>({...e,processing:!1}));const t=e instanceof Error?e.message:"unknown error occurred";i.post({message:`Failed to ${b.action} API key. ${t}`,severity:"error",display:"transient"})}}}),(0,r.jsx)(L,{open:C.open,requests:C.requests,action:C.action,processing:C.processing,onClose:()=>R({open:!1,requests:[],action:"approve",processing:!1}),onConfirm:async()=>{if(!G||0===C.requests.length)return;R(e=>({...e,processing:!0})),O({open:!1,results:[],isApprove:!0});const e="approve"===C.action,t=e?`${l}/api/kuadrant/requests/bulk-approve`:`${l}/api/kuadrant/requests/bulk-reject`;try{const a=await s.fetch(t,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({requests:C.requests.map(e=>({namespace:e.metadata.namespace,name:e.metadata.name})),reviewedBy:G.reviewedBy})});if(!a.ok){const e=await(0,T.T)(a);throw new Error(e)}const r=(await a.json()).results||[],n=r.filter(e=>e.success),o=0===r.filter(e=>!e.success).length,l=e?"approved":"rejected";R({open:!1,requests:[],action:"approve",processing:!1}),O({open:!o,results:r,isApprove:e}),j([]),x(e=>e+1),o&&i.post({message:`${n.length} API keys ${l}`,severity:"success",display:"transient"})}catch(e){console.error(`error bulk ${C.action}ing requests:`,e),R(e=>({...e,processing:!1}));const t=e instanceof Error?e.message:"unknown error occurred";i.post({message:`Failed to bulk ${C.action} API keys. ${t}`,severity:"error",display:"transient"})}}}),(0,r.jsx)(F,{open:W.open,results:W.results,isApprove:W.isApprove,onClose:()=>O({open:!1,results:[],isApprove:!0})})]})},M=()=>(0,r.jsxs)(o.Y,{themeId:"tool",children:[(0,r.jsx)(i.Y,{title:"API Key Approval",subtitle:"Review and approve API key requests",children:(0,r.jsx)(l.Y,{children:"Approve or reject API key access requests"})}),(0,r.jsx)(c.U,{children:(0,r.jsx)(O,{})})]})},45210:(e,t,a)=>{a.d(t,{Y:()=>P});var r=a(31085),n=a(22097),s=a(10394),o=a(64947),i=a(93453),l=a(29365),c=a(46423),d=a(5951),u=a(26343),p=a(28233),m=a(55197),g=a(37976),h=a(72501),v=a(5893),f=a(95478),x=(a(45250),a(49634),a(21006)),A=a(75501);const j=()=>{const{t:e}=(0,A.useTranslationRef)(x.O);return{url:"https://github.com/backstage/backstage/issues",items:[{title:e("supportConfig.default.title"),icon:"warning",links:[{title:e("supportConfig.default.linkTitle"),url:"https://github.com/backstage/backstage/blob/master/app-config.yaml"}]}]}};var y=a(27799);function b(e){const{id:t,Fallback:a=y.A,...s}=e,o=(0,n.useApp)().getSystemIcon(t)??a;return(0,r.jsx)(o,{...s})}function k(e){return(0,r.jsx)(b,{id:"help",...e})}var w=a(37725);const C=(0,g.makeStyles)({popoverList:{minWidth:260,maxWidth:400},menuItem:{whiteSpace:"normal"}},{name:"BackstageSupportButton"}),S=({icon:e})=>{const t=(0,n.useApp)(),a=e?t.getSystemIcon(e)??k:k;return(0,r.jsx)(a,{})},R=({link:e})=>(0,r.jsx)(w.N_,{to:e.url,children:e.title??e.url}),q=({item:e})=>(0,r.jsxs)(u.A,{button:!1,children:[(0,r.jsx)(c.A,{children:(0,r.jsx)(S,{icon:e.icon})}),(0,r.jsx)(d.A,{primary:e.title,secondary:e.links?.reduce((e,t,a)=>[...e,a>0&&(0,r.jsx)("br",{},a),(0,r.jsx)(R,{link:t},t.url)],[])})]});function P(e){const{t}=(0,A.useTranslationRef)(x.O),{title:a,items:c,children:d}=e,{items:g}=function(){const e=(0,n.useApiHolder)().get(n.configApiRef),t=e?.getOptionalConfig("app.support"),a=j();return t?{url:t.getString("url"),items:t.getConfigArray("items").flatMap(e=>({title:e.getString("title"),icon:e.getOptionalString("icon"),links:(e.getOptionalConfigArray("links")??[]).flatMap(e=>({url:e.getString("url"),title:e.getOptionalString("title")??""}))}))}:a}(),[y,b]=(0,f.useState)(!1),[w,S]=(0,f.useState)(null),R=C(),P=(0,n.useApi)(n.configApiRef).getOptionalConfig("app.support"),I=(0,v.A)(e=>e.breakpoints.down("sm")),B=e=>{S(e.currentTarget),b(!0)},$=()=>{b(!1)};return P?(0,r.jsxs)(r.Fragment,{children:[(0,r.jsx)(s.A,{display:"flex",ml:1,children:I?(0,r.jsx)(l.A,{color:"primary",size:"small",onClick:B,"data-testid":"support-button","aria-label":"Support",children:(0,r.jsx)(k,{})}):(0,r.jsx)(o.A,{"data-testid":"support-button","aria-label":"Support",color:"primary",onClick:B,startIcon:(0,r.jsx)(k,{}),children:t("supportButton.title")})}),(0,r.jsxs)(m.Ay,{"data-testid":"support-button-popover",open:y,anchorEl:w,anchorOrigin:{vertical:"bottom",horizontal:"right"},transformOrigin:{vertical:"top",horizontal:"right"},onClose:$,children:[(0,r.jsxs)(p.A,{className:R.popoverList,autoFocusItem:Boolean(w),children:[a&&(0,r.jsx)(u.A,{button:!1,alignItems:"flex-start",className:R.menuItem,children:(0,r.jsx)(h.A,{variant:"subtitle1",children:a})}),f.Children.map(d,(e,t)=>(0,r.jsx)(u.A,{button:!1,alignItems:"flex-start",className:R.menuItem,children:e},`child-${t}`)),(c??g).map((e,t)=>(0,r.jsx)(q,{item:e},`item-${t}`))]}),(0,r.jsx)(i.A,{children:(0,r.jsx)(o.A,{color:"primary",onClick:$,"aria-label":"Close",children:t("supportButton.close")})})]})]}):null}},46205:(e,t,a)=>{a.d(t,{W:()=>s,l:()=>n});var r=a(24217);function n(e,t){const a="resourceType"in e?{permission:e,resourceRef:t}:{permission:e},n=(0,r.J)(a);return{allowed:n.allowed,loading:n.loading,error:n.error}}function s(e,t,a,r){return!!r||!(!a||e!==t)}},46299:(e,t,a)=>{a.d(t,{ee:()=>s,g9:()=>r,uU:()=>n});const r=e=>{const t={border:"none"};switch(e){case"Approved":return{...t,backgroundColor:"#1976d2",color:"#fff"};case"Rejected":return{...t,backgroundColor:"#d32f2f",color:"#fff"};default:return{...t,backgroundColor:"#9c27b0",color:"#fff"}}},n=e=>{const t={border:"none"};switch(e){case"Approved":return{...t,backgroundColor:"#2e7d32",color:"#fff"};case"Rejected":return{...t,backgroundColor:"#d32f2f",color:"#fff"};default:return{...t,backgroundColor:"#ed6c02",color:"#fff"}}},s=e=>{switch(e){case"production":return{backgroundColor:"#1976d2",color:"#fff"};case"experimental":return{backgroundColor:"#9c27b0",color:"#fff"};case"deprecated":return{backgroundColor:"#ff9800",color:"#fff"};case"retired":return{backgroundColor:"#d32f2f",color:"#fff"};default:return{}}}},55429:(e,t,a)=>{var r=a(4293),n=a(78920);t.A=void 0;var s=n(a(95478)),o=(0,r(a(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 2zm-2 15l-5-5 1.41-1.41L10 14.17l7.59-7.59L19 8l-9 9z"}),"CheckCircle");t.A=o},77318:(e,t,a)=>{async function r(e){const t=await e.json().catch(()=>({}));switch(e.status){case 400:return t.error||"Invalid request. Please check your input.";case 403:return"Permission denied. Contact your administrator.";case 404:return"Resource not found. It may have been deleted.";case 409:return"Resource already exists or conflicts with existing data.";case 500:return"Server error. Please try again or contact support.";default:return t.error||`Request failed (${e.status})`}}a.d(t,{T:()=>r})}}]);
2
+ //# sourceMappingURL=3976.4cf18515.chunk.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"static/3976.4cf18515.chunk.js","mappings":"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","sources":["webpack://internal.plugin-kuadrant/./src/components/FilterPanel/FilterPanel.tsx","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/icons/ExpandLess.js","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/icons/BrokenImage.js","webpack://internal.plugin-kuadrant/./src/permissions.ts","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/./src/components/ApprovalQueueTable/ApprovalQueueTable.tsx","webpack://internal.plugin-kuadrant/./src/components/ApiKeyApprovalPage/ApiKeyApprovalPage.tsx","webpack://internal.plugin-kuadrant/./node_modules/@backstage/core-components/dist/hooks/useSupportConfig.esm.js","webpack://internal.plugin-kuadrant/./node_modules/@backstage/core-components/dist/icons/icons.esm.js","webpack://internal.plugin-kuadrant/./node_modules/@backstage/core-components/dist/components/SupportButton/SupportButton.esm.js","webpack://internal.plugin-kuadrant/./src/utils/permissions.ts","webpack://internal.plugin-kuadrant/./src/utils/styles.ts","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/icons/CheckCircle.js","webpack://internal.plugin-kuadrant/./src/utils/errors.ts"],"sourcesContent":["import React from 'react';\nimport {\n Box,\n Typography,\n Checkbox,\n FormControlLabel,\n FormGroup,\n Divider,\n Button,\n Collapse,\n makeStyles,\n} from '@material-ui/core';\nimport ExpandMoreIcon from '@material-ui/icons/ExpandMore';\nimport ExpandLessIcon from '@material-ui/icons/ExpandLess';\n\nconst useStyles = makeStyles(theme => ({\n root: {\n width: 240,\n minWidth: 240,\n padding: theme.spacing(2),\n borderRight: `1px solid ${theme.palette.divider}`,\n backgroundColor: theme.palette.background.paper,\n height: '100%',\n overflowY: 'auto',\n },\n sectionTitle: {\n fontWeight: 600,\n fontSize: '0.75rem',\n textTransform: 'uppercase',\n letterSpacing: '0.05em',\n color: theme.palette.text.secondary,\n marginBottom: theme.spacing(1),\n display: 'flex',\n alignItems: 'center',\n justifyContent: 'space-between',\n cursor: 'pointer',\n userSelect: 'none',\n },\n filterSection: {\n marginBottom: theme.spacing(2),\n },\n checkbox: {\n padding: theme.spacing(0.5),\n },\n checkboxLabel: {\n fontSize: '0.875rem',\n },\n clearButton: {\n marginTop: theme.spacing(2),\n },\n count: {\n fontSize: '0.75rem',\n color: theme.palette.text.secondary,\n marginLeft: theme.spacing(1),\n },\n}));\n\nexport interface FilterOption {\n value: string;\n label: string;\n count?: number;\n}\n\nexport interface FilterSection {\n id: string;\n title: string;\n options: FilterOption[];\n collapsed?: boolean;\n}\n\nexport interface FilterState {\n [sectionId: string]: string[];\n}\n\ninterface FilterPanelProps {\n sections: FilterSection[];\n filters: FilterState;\n onChange: (filters: FilterState) => void;\n onClear?: () => void;\n}\n\nexport const FilterPanel = ({\n sections,\n filters,\n onChange,\n onClear,\n}: FilterPanelProps) => {\n const classes = useStyles();\n const [collapsedSections, setCollapsedSections] = React.useState<Set<string>>(\n new Set(sections.filter(s => s.collapsed).map(s => s.id)),\n );\n\n const toggleSection = (sectionId: string) => {\n setCollapsedSections(prev => {\n const next = new Set(prev);\n if (next.has(sectionId)) {\n next.delete(sectionId);\n } else {\n next.add(sectionId);\n }\n return next;\n });\n };\n\n const handleCheckboxChange = (sectionId: string, value: string) => {\n const currentValues = filters[sectionId] || [];\n const newValues = currentValues.includes(value)\n ? currentValues.filter(v => v !== value)\n : [...currentValues, value];\n\n onChange({\n ...filters,\n [sectionId]: newValues,\n });\n };\n\n const hasActiveFilters = Object.values(filters).some(\n values => values.length > 0,\n );\n\n const handleClear = () => {\n const clearedFilters: FilterState = {};\n sections.forEach(section => {\n clearedFilters[section.id] = [];\n });\n onChange(clearedFilters);\n onClear?.();\n };\n\n return (\n <Box className={classes.root}>\n <Box display=\"flex\" justifyContent=\"space-between\" alignItems=\"center\" mb={2}>\n <Typography variant=\"subtitle2\">Filters</Typography>\n {hasActiveFilters && (\n <Button\n size=\"small\"\n color=\"primary\"\n onClick={handleClear}\n >\n Clear all\n </Button>\n )}\n </Box>\n\n <Divider />\n\n {sections.map(section => {\n const isCollapsed = collapsedSections.has(section.id);\n const selectedCount = (filters[section.id] || []).length;\n\n return (\n <Box key={section.id} className={classes.filterSection} mt={2}>\n <Box\n className={classes.sectionTitle}\n onClick={() => toggleSection(section.id)}\n >\n <Box display=\"flex\" alignItems=\"center\">\n <span>{section.title}</span>\n {selectedCount > 0 && (\n <span className={classes.count}>({selectedCount})</span>\n )}\n </Box>\n {isCollapsed ? (\n <ExpandMoreIcon fontSize=\"small\" />\n ) : (\n <ExpandLessIcon fontSize=\"small\" />\n )}\n </Box>\n\n <Collapse in={!isCollapsed}>\n <FormGroup>\n {section.options.map(option => (\n <FormControlLabel\n key={option.value}\n control={\n <Checkbox\n checked={(filters[section.id] || []).includes(option.value)}\n onChange={() =>\n handleCheckboxChange(section.id, option.value)\n }\n size=\"small\"\n className={classes.checkbox}\n color=\"primary\"\n />\n }\n label={\n <Box display=\"flex\" alignItems=\"center\">\n <span className={classes.checkboxLabel}>\n {option.label}\n </span>\n {option.count !== undefined && (\n <span className={classes.count}>({option.count})</span>\n )}\n </Box>\n }\n />\n ))}\n </FormGroup>\n </Collapse>\n </Box>\n );\n })}\n </Box>\n );\n};\n","\"use strict\";\n\nvar _interopRequireDefault = require(\"@babel/runtime/helpers/interopRequireDefault\");\n\nvar _interopRequireWildcard = require(\"@babel/runtime/helpers/interopRequireWildcard\");\n\nObject.defineProperty(exports, \"__esModule\", {\n value: true\n});\nexports.default = void 0;\n\nvar React = _interopRequireWildcard(require(\"react\"));\n\nvar _createSvgIcon = _interopRequireDefault(require(\"./utils/createSvgIcon\"));\n\nvar _default = (0, _createSvgIcon.default)( /*#__PURE__*/React.createElement(\"path\", {\n d: \"M12 8l-6 6 1.41 1.41L12 10.83l4.59 4.58L18 14z\"\n}), 'ExpandLess');\n\nexports.default = _default;","\"use strict\";\n\nvar _interopRequireDefault = require(\"@babel/runtime/helpers/interopRequireDefault\");\n\nvar _interopRequireWildcard = require(\"@babel/runtime/helpers/interopRequireWildcard\");\n\nObject.defineProperty(exports, \"__esModule\", {\n value: true\n});\nexports.default = void 0;\n\nvar React = _interopRequireWildcard(require(\"react\"));\n\nvar _createSvgIcon = _interopRequireDefault(require(\"./utils/createSvgIcon\"));\n\nvar _default = (0, _createSvgIcon.default)( /*#__PURE__*/React.createElement(\"path\", {\n d: \"M21 5v6.59l-3-3.01-4 4.01-4-4-4 4-3-3.01V5c0-1.1.9-2 2-2h14c1.1 0 2 .9 2 2zm-3 6.42l3 3.01V19c0 1.1-.9 2-2 2H5c-1.1 0-2-.9-2-2v-6.58l3 2.99 4-4 4 4 4-3.99z\"\n}), 'BrokenImage');\n\nexports.default = _default;","import { createPermission } from '@backstage/plugin-permission-common';\n\n/**\n * permission definitions for the kuadrant plugin\n *\n * these permissions control access to kuadrant resources and operations.\n * they must match the permissions defined in the backend plugin.\n *\n * permission types:\n * - BasicPermission: standard permission that applies globally\n * - ResourcePermission: permission scoped to specific resource types (e.g., apiproduct)\n *\n * permission patterns:\n * - `.create` - create new resources\n * - `.read` - read resource details\n * - `.read.own` - read only resources owned by the user\n * - `.read.all` - read all resources regardless of ownership\n * - `.update` - modify existing resources\n * - `.delete` - delete resources\n * - `.delete.own` - delete only resources owned by the user\n * - `.delete.all` - delete any resource regardless of ownership\n * - `.list` - list/view collections of resources\n */\n\n// planpolicy permissions\nexport const kuadrantPlanPolicyCreatePermission = createPermission({\n name: 'kuadrant.planpolicy.create',\n attributes: { action: 'create' },\n});\n\nexport const kuadrantPlanPolicyReadPermission = createPermission({\n name: 'kuadrant.planpolicy.read',\n attributes: { action: 'read' },\n});\n\nexport const kuadrantPlanPolicyUpdatePermission = createPermission({\n name: 'kuadrant.planpolicy.update',\n attributes: { action: 'update' },\n});\n\nexport const kuadrantPlanPolicyDeletePermission = createPermission({\n name: 'kuadrant.planpolicy.delete',\n attributes: { action: 'delete' },\n});\n\nexport const kuadrantPlanPolicyListPermission = createPermission({\n name: 'kuadrant.planpolicy.list',\n attributes: { action: 'read' },\n});\n\n// apiproduct permissions\n\n/**\n * permission to create new API products\n * granted to api owners and admins\n */\nexport const kuadrantApiProductCreatePermission = createPermission({\n name: 'kuadrant.apiproduct.create',\n attributes: { action: 'create' },\n});\n\n/**\n * permission to read API products owned by the current user\n * for api owners to view their own products\n */\nexport const kuadrantApiProductReadOwnPermission = createPermission({\n name: 'kuadrant.apiproduct.read.own',\n attributes: { action: 'read' },\n});\n\n/**\n * permission to read all API products regardless of ownership\n * for platform engineers/admins who need to view all products\n */\nexport const kuadrantApiProductReadAllPermission = createPermission({\n name: 'kuadrant.apiproduct.read.all',\n attributes: { action: 'read' },\n});\n\n/**\n * permission to update API products owned by the current user\n * for api owners to modify their own products\n */\nexport const kuadrantApiProductUpdateOwnPermission = createPermission({\n name: 'kuadrant.apiproduct.update.own',\n attributes: { action: 'update' },\n});\n\n/**\n * permission to update any API product regardless of ownership\n * for platform engineers/admins\n */\nexport const kuadrantApiProductUpdateAllPermission = createPermission({\n name: 'kuadrant.apiproduct.update.all',\n attributes: { action: 'update' },\n});\n\n/**\n * permission to delete API products owned by the current user\n * for api owners to remove their own products\n */\nexport const kuadrantApiProductDeleteOwnPermission = createPermission({\n name: 'kuadrant.apiproduct.delete.own',\n attributes: { action: 'delete' },\n});\n\n/**\n * permission to delete any API product regardless of ownership\n * for platform engineers/admins\n */\nexport const kuadrantApiProductDeleteAllPermission = createPermission({\n name: 'kuadrant.apiproduct.delete.all',\n attributes: { action: 'delete' },\n});\n\n/**\n * permission to list API products\n * backend filters results based on .own vs .all read permissions\n */\nexport const kuadrantApiProductListPermission = createPermission({\n name: 'kuadrant.apiproduct.list',\n attributes: { action: 'read' },\n});\n\n// apikey permissions\n\n/**\n * permission to create API keys (request API access)\n *\n * this is a ResourcePermission scoped to 'apiproduct', allowing\n * fine-grained control over which API products users can request access to.\n *\n * use in frontend: useKuadrantPermission(kuadrantApiKeyCreatePermission)\n * use in backend with resource: { permission, resourceRef: 'apiproduct:namespace/name' }\n */\nexport const kuadrantApiKeyCreatePermission = createPermission({\n name: 'kuadrant.apikey.create',\n attributes: { action: 'create' },\n resourceType: 'apiproduct',\n});\n\n/**\n * permission to read API keys owned by the current user\n * allows users to view their own API keys and request history\n */\nexport const kuadrantApiKeyReadOwnPermission = createPermission({\n name: 'kuadrant.apikey.read.own',\n attributes: { action: 'read' },\n});\n\n/**\n * permission to read all API keys regardless of ownership\n * for platform engineers/admins who need to view the approval queue and audit keys\n */\nexport const kuadrantApiKeyReadAllPermission = createPermission({\n name: 'kuadrant.apikey.read.all',\n attributes: { action: 'read' },\n});\n\n/**\n * permission to update API keys owned by the current user\n * allows users to edit their own pending requests (change plan tier, use case)\n */\nexport const kuadrantApiKeyUpdateOwnPermission = createPermission({\n name: 'kuadrant.apikey.update.own',\n attributes: { action: 'update' },\n});\n\n/**\n * permission to update any API key regardless of ownership\n * typically granted to API owners and platform engineers for approving/rejecting requests\n */\nexport const kuadrantApiKeyUpdateAllPermission = createPermission({\n name: 'kuadrant.apikey.update.all',\n attributes: { action: 'update' },\n});\n\n/**\n * permission to delete API keys owned by the current user\n * allows users to cancel their own requests or revoke their own access\n */\nexport const kuadrantApiKeyDeleteOwnPermission = createPermission({\n name: 'kuadrant.apikey.delete.own',\n attributes: { action: 'delete' },\n});\n\n/**\n * permission to delete any API key regardless of ownership\n * for platform engineers/admins who need to revoke access\n */\nexport const kuadrantApiKeyDeleteAllPermission = createPermission({\n name: 'kuadrant.apikey.delete.all',\n attributes: { action: 'delete' },\n});\n\n/**\n * permission to approve/reject API key requests\n * grants access to the approval queue - for API owners and admins only\n * separate from update.own which consumers use to edit their pending requests\n */\nexport const kuadrantApiKeyApprovePermission = createPermission({\n name: 'kuadrant.apikey.approve',\n attributes: { action: 'update' },\n});\n\nexport const kuadrantPermissions = [\n kuadrantPlanPolicyCreatePermission,\n kuadrantPlanPolicyReadPermission,\n kuadrantPlanPolicyUpdatePermission,\n kuadrantPlanPolicyDeletePermission,\n kuadrantPlanPolicyListPermission,\n kuadrantApiProductCreatePermission,\n kuadrantApiProductReadOwnPermission,\n kuadrantApiProductReadAllPermission,\n kuadrantApiProductUpdateOwnPermission,\n kuadrantApiProductUpdateAllPermission,\n kuadrantApiProductDeleteOwnPermission,\n kuadrantApiProductDeleteAllPermission,\n kuadrantApiProductListPermission,\n kuadrantApiKeyCreatePermission,\n kuadrantApiKeyReadOwnPermission,\n kuadrantApiKeyReadAllPermission,\n kuadrantApiKeyUpdateOwnPermission,\n kuadrantApiKeyUpdateAllPermission,\n kuadrantApiKeyDeleteOwnPermission,\n kuadrantApiKeyDeleteAllPermission,\n kuadrantApiKeyApprovePermission,\n];\n","import { 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","import React, { useState, useMemo } from \"react\";\nimport {\n useApi,\n fetchApiRef,\n identityApiRef,\n configApiRef,\n alertApiRef,\n} from \"@backstage/core-plugin-api\";\nimport { useAsync } from \"react-use\";\nimport {\n Table,\n TableColumn,\n Progress,\n ResponseErrorPanel,\n Link,\n} from \"@backstage/core-components\";\nimport {\n kuadrantApiKeyUpdateAllPermission,\n kuadrantApiKeyUpdateOwnPermission,\n} from \"../../permissions\";\nimport { useKuadrantPermission } from \"../../utils/permissions\";\nimport {\n Button,\n Dialog,\n DialogTitle,\n DialogContent,\n DialogActions,\n Chip,\n Typography,\n Box,\n CircularProgress,\n TextField,\n makeStyles,\n} from \"@material-ui/core\";\nimport CheckCircleIcon from \"@material-ui/icons/CheckCircle\";\nimport CancelIcon from \"@material-ui/icons/Cancel\";\nimport { FilterPanel, FilterSection, FilterState } from \"../FilterPanel\";\nimport { APIKey } from \"../../types/api-management\";\nimport { getApprovalQueueStatusChipStyle } from \"../../utils/styles\";\nimport { handleFetchError } from \"../../utils/errors.ts\";\n\nconst useStyles = makeStyles((theme) => ({\n container: {\n display: \"flex\",\n height: \"100%\",\n minHeight: 400,\n },\n tableContainer: {\n flex: 1,\n overflow: \"auto\",\n padding: 10,\n },\n useCasePanel: {\n padding: theme.spacing(2),\n backgroundColor: theme.palette.background.default,\n },\n useCaseLabel: {\n fontWeight: 600,\n marginBottom: theme.spacing(1),\n color: theme.palette.text.secondary,\n textTransform: \"uppercase\",\n fontSize: \"0.75rem\",\n },\n bulkActions: {\n padding: theme.spacing(2),\n backgroundColor: theme.palette.background.default,\n borderBottom: `1px solid ${theme.palette.divider}`,\n display: \"flex\",\n alignItems: \"center\",\n justifyContent: \"space-between\",\n },\n}));\n\ninterface ApprovalDialogProps {\n open: boolean;\n request: APIKey | null;\n action: \"approve\" | \"reject\";\n processing: boolean;\n onClose: () => void;\n onConfirm: () => void;\n}\n\nconst ApprovalDialog = ({\n open,\n request,\n action,\n processing,\n onClose,\n onConfirm,\n}: ApprovalDialogProps) => {\n const [confirmInput, setConfirmInput] = React.useState(\"\");\n const actionLabel = action === \"approve\" ? \"Approve\" : \"Reject\";\n const processingLabel =\n action === \"approve\" ? \"Approving...\" : \"Rejecting...\";\n\n const isReject = action === \"reject\";\n const confirmText = request?.spec.requestedBy?.userId || \"\";\n const canConfirm = isReject ? confirmInput === confirmText : true;\n\n // reset input when dialog closes\n React.useEffect(() => {\n if (!open) {\n setConfirmInput(\"\");\n }\n }, [open]);\n\n return (\n <Dialog\n open={open}\n onClose={processing ? undefined : onClose}\n maxWidth=\"sm\"\n fullWidth\n >\n <DialogTitle>\n {isReject ? (\n <Box display=\"flex\" alignItems=\"center\" style={{ gap: 8 }}>\n <CancelIcon color=\"error\" />\n <span>{actionLabel} API Key</span>\n </Box>\n ) : (\n <span>{actionLabel} API Key</span>\n )}\n </DialogTitle>\n <DialogContent>\n {request && (\n <>\n <p>\n <strong>User:</strong> {request.spec.requestedBy.userId}\n </p>\n <p>\n <strong>API:</strong>{\" \"}\n {request.spec.apiProductRef?.name || \"unknown\"}\n </p>\n <p>\n <strong>Tier:</strong> {request.spec.planTier}\n </p>\n <Box mb={2}>\n <Typography\n variant=\"body2\"\n component=\"span\"\n style={{ fontWeight: \"bold\" }}\n >\n Use Case:\n </Typography>{\" \"}\n <Typography\n variant=\"body2\"\n component=\"span\"\n style={{ whiteSpace: \"pre-wrap\" }}\n >\n {request.spec.useCase || \"-\"}\n </Typography>\n </Box>\n {isReject && (\n <Box mt={2}>\n <Typography variant=\"body2\" color=\"textSecondary\" gutterBottom>\n Type <strong>{confirmText}</strong> to confirm rejection:\n </Typography>\n <TextField\n fullWidth\n variant=\"outlined\"\n size=\"small\"\n value={confirmInput}\n onChange={(e) => setConfirmInput(e.target.value)}\n disabled={processing}\n autoFocus\n placeholder={confirmText}\n />\n </Box>\n )}\n </>\n )}\n </DialogContent>\n <DialogActions>\n <Button onClick={onClose} disabled={processing}>\n Cancel\n </Button>\n <Button\n onClick={onConfirm}\n color={action === \"approve\" ? \"primary\" : \"secondary\"}\n variant=\"contained\"\n disabled={processing || !canConfirm}\n startIcon={\n processing ? (\n <CircularProgress size={16} color=\"inherit\" />\n ) : undefined\n }\n >\n {processing ? processingLabel : actionLabel}\n </Button>\n </DialogActions>\n </Dialog>\n );\n};\n\ninterface BulkActionDialogProps {\n open: boolean;\n requests: APIKey[];\n action: \"approve\" | \"reject\";\n processing: boolean;\n onClose: () => void;\n onConfirm: () => void;\n}\n\nconst BulkActionDialog = ({\n open,\n requests,\n action,\n processing,\n onClose,\n onConfirm,\n}: BulkActionDialogProps) => {\n const isApprove = action === \"approve\";\n const actionLabel = isApprove ? \"Approve All\" : \"Reject All\";\n const processingLabel = isApprove ? \"Approving...\" : \"Rejecting...\";\n\n return (\n <Dialog\n open={open}\n onClose={processing ? undefined : onClose}\n maxWidth=\"md\"\n fullWidth\n >\n <DialogTitle>\n {isApprove ? \"Approve\" : \"Reject\"} {requests.length} API Keys\n </DialogTitle>\n <DialogContent>\n <Typography variant=\"body2\" paragraph>\n You are about to {isApprove ? \"approve\" : \"reject\"} the following API\n keys:\n </Typography>\n <Box mb={2} maxHeight={200} overflow=\"auto\">\n {requests.map((request) => (\n <Box\n key={`${request.metadata.namespace}/${request.metadata.name}`}\n mb={1}\n p={1}\n bgcolor=\"background.default\"\n >\n <Typography variant=\"body2\">\n <strong>{request.spec.requestedBy.userId}</strong> -{\" \"}\n {request.spec.apiProductRef?.name || \"unknown\"} (\n {request.spec.planTier})\n </Typography>\n </Box>\n ))}\n </Box>\n </DialogContent>\n <DialogActions>\n <Button onClick={onClose} disabled={processing}>\n Cancel\n </Button>\n <Button\n onClick={onConfirm}\n color={isApprove ? \"primary\" : \"secondary\"}\n variant=\"contained\"\n disabled={processing}\n startIcon={\n processing ? (\n <CircularProgress size={16} color=\"inherit\" />\n ) : undefined\n }\n >\n {processing ? processingLabel : actionLabel}\n </Button>\n </DialogActions>\n </Dialog>\n );\n};\n\ninterface ExpandedRowProps {\n request: APIKey;\n}\n\nconst ExpandedRowContent = ({ request }: ExpandedRowProps) => {\n const classes = useStyles();\n\n return (\n <Box className={classes.useCasePanel} onClick={(e) => e.stopPropagation()}>\n <Typography className={classes.useCaseLabel}>Use Case</Typography>\n <Typography variant=\"body2\">\n {request.spec.useCase || \"No use case provided\"}\n </Typography>\n </Box>\n );\n};\n\ninterface BulkRequestResult {\n name: string;\n namespace: string;\n success: boolean;\n error: string | null;\n}\n\ninterface AlertDialogProps {\n open: boolean;\n results: BulkRequestResult[];\n isApprove: boolean;\n onClose: () => void;\n}\n\nconst useBulkAlertStyles = makeStyles((theme) => ({\n successSection: {\n padding: theme.spacing(2),\n backgroundColor: theme.palette.success.main + '14',\n borderRadius: theme.shape.borderRadius,\n marginBottom: theme.spacing(2),\n border: `1px solid ${theme.palette.success.main}40`,\n },\n failureSection: {\n padding: theme.spacing(2),\n backgroundColor: theme.palette.error.main + '14',\n borderRadius: theme.shape.borderRadius,\n marginBottom: theme.spacing(2),\n border: `1px solid ${theme.palette.error.main}40`,\n },\n warningSection: {\n padding: theme.spacing(2),\n backgroundColor: theme.palette.warning.main + '14',\n borderRadius: theme.shape.borderRadius,\n marginBottom: theme.spacing(2),\n border: `1px solid ${theme.palette.warning.main}40`,\n },\n statRow: {\n display: 'flex',\n alignItems: 'center',\n gap: theme.spacing(1),\n },\n errorList: {\n marginTop: theme.spacing(2),\n padding: 0,\n listStyle: 'none',\n },\n errorItem: {\n padding: theme.spacing(1.5),\n marginBottom: theme.spacing(1),\n backgroundColor: theme.palette.background.paper,\n borderRadius: theme.shape.borderRadius,\n border: `1px solid ${theme.palette.divider}`,\n },\n errorName: {\n fontWeight: 600,\n fontSize: '0.875rem',\n marginBottom: theme.spacing(0.5),\n },\n errorMessage: {\n fontSize: '0.813rem',\n color: theme.palette.error.main,\n fontFamily: 'monospace',\n wordBreak: 'break-word',\n },\n expandButton: {\n marginTop: theme.spacing(1),\n padding: theme.spacing(0.5),\n },\n}));\n\nconst BulkAlertDialog = ({\n open,\n results,\n isApprove,\n onClose,\n}: AlertDialogProps) => {\n const classes = useBulkAlertStyles();\n const [showErrors, setShowErrors] = useState(true);\n const successResults = results.filter((res: any) => res.success);\n const failedResults = results.filter((res: any) => !res.success);\n const hasPartialSuccess = successResults.length > 0;\n\n return (\n <Dialog\n open={open}\n onClose={onClose}\n maxWidth=\"md\"\n fullWidth\n >\n <DialogTitle>\n <Box display=\"flex\" alignItems=\"center\" style={{ gap: 8 }}>\n {hasPartialSuccess ? (\n <>\n <CancelIcon style={{ color: '#ff9800' }} />\n <span>Bulk {isApprove ? \"Approve\" : \"Reject\"} Completed with Issues</span>\n </>\n ) : (\n <>\n <CancelIcon color=\"error\" />\n <span>Bulk {isApprove ? \"Approve\" : \"Reject\"} Failed</span>\n </>\n )}\n </Box>\n </DialogTitle>\n <DialogContent>\n {hasPartialSuccess && (\n <Box className={classes.successSection}>\n <Box className={classes.statRow}>\n <CheckCircleIcon style={{ color: '#4caf50' }} />\n <Typography variant=\"body1\">\n <strong>{successResults.length}</strong> API key{successResults.length !== 1 ? 's' : ''} {isApprove ? 'approved' : 'rejected'} successfully\n </Typography>\n </Box>\n </Box>\n )}\n\n <Box className={hasPartialSuccess ? classes.warningSection : classes.failureSection}>\n <Box className={classes.statRow}>\n <CancelIcon color=\"error\" />\n <Typography variant=\"body1\">\n <strong>{failedResults.length}</strong> request{failedResults.length !== 1 ? 's' : ''} failed\n </Typography>\n </Box>\n\n <Button\n size=\"small\"\n className={classes.expandButton}\n onClick={() => setShowErrors(!showErrors)}\n >\n {showErrors ? 'Hide' : 'Show'} Error Details\n </Button>\n\n {showErrors && (\n <ul className={classes.errorList}>\n {failedResults.map((result, index) => (\n <li key={result.name || index} className={classes.errorItem}>\n <div className={classes.errorName}>\n {result.namespace}/{result.name}\n </div>\n <div className={classes.errorMessage}>\n {result.error || 'Unknown error'}\n </div>\n </li>\n ))}\n </ul>\n )}\n </Box>\n </DialogContent>\n <DialogActions>\n <Button onClick={onClose} color=\"primary\" variant=\"contained\">\n Close\n </Button>\n </DialogActions>\n </Dialog>\n );\n};\n\nexport const ApprovalQueueTable = () => {\n const classes = useStyles();\n const config = useApi(configApiRef);\n const fetchApi = useApi(fetchApiRef);\n const identityApi = useApi(identityApiRef);\n const alertApi = useApi(alertApiRef);\n const backendUrl = config.getString(\"backend.baseUrl\");\n const [refresh, setRefresh] = useState(0);\n const [selectedRequests, setSelectedRequests] = useState<APIKey[]>([]);\n const [dialogState, setDialogState] = useState<{\n open: boolean;\n request: APIKey | null;\n action: \"approve\" | \"reject\";\n processing: boolean;\n }>({\n open: false,\n request: null,\n action: \"approve\",\n processing: false,\n });\n const [bulkDialogState, setBulkDialogState] = useState<{\n open: boolean;\n requests: APIKey[];\n action: \"approve\" | \"reject\";\n processing: boolean;\n }>({\n open: false,\n requests: [],\n action: \"approve\",\n processing: false,\n });\n const [bulkAlertDialogState, setBulkAlertDialogState] = useState<{\n open: boolean;\n results: BulkRequestResult[];\n isApprove: boolean;\n }>({\n open: false,\n results: [],\n isApprove: true,\n });\n const [filters, setFilters] = useState<FilterState>({\n status: [],\n apiProduct: [],\n tier: [],\n });\n\n const {\n allowed: canUpdateAllRequests,\n loading: updateAllPermissionLoading,\n error: updateAllPermissionError,\n } = useKuadrantPermission(kuadrantApiKeyUpdateAllPermission);\n\n const {\n allowed: canUpdateOwnRequests,\n loading: updateOwnPermissionLoading,\n error: updateOwnPermissionError,\n } = useKuadrantPermission(kuadrantApiKeyUpdateOwnPermission);\n\n const updatePermissionLoading =\n updateAllPermissionLoading || updateOwnPermissionLoading;\n const updatePermissionError =\n updateAllPermissionError || updateOwnPermissionError;\n\n const { value, loading, error } = useAsync(async () => {\n const identity = await identityApi.getBackstageIdentity();\n const reviewedBy = identity.userEntityRef;\n\n const [requestsResponse, apiProductsResponse] = await Promise.all([\n fetchApi.fetch(`${backendUrl}/api/kuadrant/requests`),\n fetchApi.fetch(`${backendUrl}/api/kuadrant/apiproducts`),\n ]);\n\n if (!requestsResponse.ok) {\n return {\n allRequests: [] as APIKey[],\n reviewedBy,\n ownedApiProducts: new Set<string>(),\n };\n }\n\n const contentType = requestsResponse.headers.get(\"content-type\");\n if (!contentType?.includes(\"application/json\")) {\n alertApi.post({\n message: \"Unexpected content-type from the server response.\",\n display: \"transient\",\n severity: \"warning\",\n });\n return {\n allRequests: [] as APIKey[],\n reviewedBy,\n ownedApiProducts: new Set<string>(),\n };\n }\n\n const data = await requestsResponse.json();\n const allRequests = data.items || [];\n\n const ownedApiProducts = new Set<string>();\n if (apiProductsResponse.ok) {\n const apiProductsData = await apiProductsResponse.json();\n for (const product of apiProductsData.items || []) {\n const owner = product.metadata?.annotations?.[\"backstage.io/owner\"];\n if (owner === reviewedBy) {\n ownedApiProducts.add(\n `${product.metadata.namespace}/${product.metadata.name}`,\n );\n }\n }\n }\n\n return { allRequests, reviewedBy, ownedApiProducts };\n }, [backendUrl, fetchApi, identityApi, refresh]);\n\n const filterSections: FilterSection[] = useMemo(() => {\n if (!value?.allRequests) return [];\n\n const statusCounts = { Approved: 0, Pending: 0, Rejected: 0 };\n const apiProductCounts = new Map<string, number>();\n const tierCounts = new Map<string, number>();\n\n value.allRequests.forEach((r: APIKey) => {\n const status = r.status?.phase || \"Pending\";\n statusCounts[status as keyof typeof statusCounts]++;\n\n const apiProduct = r.spec.apiProductRef?.name || \"unknown\";\n apiProductCounts.set(\n apiProduct,\n (apiProductCounts.get(apiProduct) || 0) + 1,\n );\n\n const tier = r.spec.planTier || \"unknown\";\n tierCounts.set(tier, (tierCounts.get(tier) || 0) + 1);\n });\n\n return [\n {\n id: \"status\",\n title: \"Status\",\n options: [\n { value: \"Pending\", label: \"Pending\", count: statusCounts.Pending },\n {\n value: \"Approved\",\n label: \"Approved\",\n count: statusCounts.Approved,\n },\n {\n value: \"Rejected\",\n label: \"Rejected\",\n count: statusCounts.Rejected,\n },\n ],\n },\n {\n id: \"apiProduct\",\n title: \"API Product\",\n options: Array.from(apiProductCounts.entries()).map(\n ([name, count]) => ({\n value: name,\n label: name,\n count,\n }),\n ),\n collapsed: apiProductCounts.size > 5,\n },\n {\n id: \"tier\",\n title: \"Tier\",\n options: Array.from(tierCounts.entries()).map(([tier, count]) => ({\n value: tier,\n label: tier.charAt(0).toUpperCase() + tier.slice(1),\n count,\n })),\n },\n ];\n }, [value?.allRequests]);\n\n const filteredRequests = useMemo(() => {\n if (!value?.allRequests) return [];\n\n return value.allRequests.filter((r: APIKey) => {\n if (filters.status.length > 0) {\n const status = r.status?.phase || \"Pending\";\n if (!filters.status.includes(status)) return false;\n }\n if (filters.apiProduct.length > 0) {\n const apiProduct = r.spec.apiProductRef?.name || \"unknown\";\n if (!filters.apiProduct.includes(apiProduct)) return false;\n }\n if (filters.tier.length > 0) {\n const tier = r.spec.planTier || \"unknown\";\n if (!filters.tier.includes(tier)) return false;\n }\n return true;\n });\n }, [value?.allRequests, filters]);\n\n const handleApprove = (request: APIKey) => {\n setDialogState({\n open: true,\n request,\n action: \"approve\",\n processing: false,\n });\n };\n\n const handleReject = (request: APIKey) => {\n setDialogState({\n open: true,\n request,\n action: \"reject\",\n processing: false,\n });\n };\n\n const handleConfirm = async () => {\n if (!dialogState.request || !value) return;\n\n setDialogState((prev) => ({ ...prev, processing: true }));\n\n const endpoint =\n dialogState.action === \"approve\"\n ? `${backendUrl}/api/kuadrant/requests/${dialogState.request.metadata.namespace}/${dialogState.request.metadata.name}/approve`\n : `${backendUrl}/api/kuadrant/requests/${dialogState.request.metadata.namespace}/${dialogState.request.metadata.name}/reject`;\n\n try {\n const response = await fetchApi.fetch(endpoint, {\n method: \"POST\",\n headers: { \"Content-Type\": \"application/json\" },\n body: JSON.stringify({ reviewedBy: value.reviewedBy }),\n });\n\n if (!response.ok) {\n const err = await handleFetchError(response);\n throw new Error(err);\n }\n\n setDialogState({\n open: false,\n request: null,\n action: \"approve\",\n processing: false,\n });\n // remove the processed request from selection\n setSelectedRequests((prev) =>\n prev.filter(\n (r) =>\n r.metadata.name !== dialogState.request?.metadata.name ||\n r.metadata.namespace !== dialogState.request?.metadata.namespace,\n ),\n );\n setRefresh((r) => r + 1);\n const action = dialogState.action === \"approve\" ? \"approved\" : \"rejected\";\n alertApi.post({\n message: `API key ${action}`,\n severity: \"success\",\n display: \"transient\",\n });\n } catch (err) {\n console.error(`error ${dialogState.action}ing request:`, err);\n setDialogState((prev) => ({ ...prev, processing: false }));\n const errorMessage = err instanceof Error ? err.message : \"unknown error occurred\";\n alertApi.post({\n message: `Failed to ${dialogState.action} API key. ${errorMessage}`,\n severity: \"error\",\n display: \"transient\",\n });\n }\n };\n\n const handleBulkApprove = () => {\n if (selectedRequests.length === 0) return;\n setBulkDialogState({\n open: true,\n requests: selectedRequests,\n action: \"approve\",\n processing: false,\n });\n };\n\n const handleBulkReject = () => {\n if (selectedRequests.length === 0) return;\n setBulkDialogState({\n open: true,\n requests: selectedRequests,\n action: \"reject\",\n processing: false,\n });\n };\n\n const handleBulkConfirm = async () => {\n if (!value || bulkDialogState.requests.length === 0) return;\n\n setBulkDialogState((prev) => ({ ...prev, processing: true }));\n setBulkAlertDialogState({ open: false, results: [], isApprove: true, });\n\n const isApprove = bulkDialogState.action === \"approve\";\n const endpoint = isApprove\n ? `${backendUrl}/api/kuadrant/requests/bulk-approve`\n : `${backendUrl}/api/kuadrant/requests/bulk-reject`;\n\n try {\n const response = await fetchApi.fetch(endpoint, {\n method: \"POST\",\n headers: { \"Content-Type\": \"application/json\" },\n body: JSON.stringify({\n requests: bulkDialogState.requests.map((r) => ({\n namespace: r.metadata.namespace,\n name: r.metadata.name,\n })),\n reviewedBy: value.reviewedBy,\n }),\n });\n\n if (!response.ok) {\n const err = await handleFetchError(response);\n throw new Error(err);\n }\n const bulkResponseRaw = await response.json();\n const bulkResponse = bulkResponseRaw.results || [];\n\n const successfulItems = bulkResponse.filter((res: any) => res.success);\n const failedItems = bulkResponse.filter((res: any) => !res.success);\n const totalSuccess = failedItems.length === 0;\n\n const action = isApprove ? \"approved\" : \"rejected\";\n\n setBulkDialogState({\n open: false,\n requests: [],\n action: \"approve\",\n processing: false,\n });\n setBulkAlertDialogState({\n open: !totalSuccess,\n results: bulkResponse,\n isApprove,\n });\n setSelectedRequests([]);\n setRefresh((r) => r + 1);\n\n if (totalSuccess) {\n alertApi.post({\n message: `${successfulItems.length} API keys ${action}`,\n severity: \"success\",\n display: \"transient\",\n });\n }\n } catch (err) {\n console.error(`error bulk ${bulkDialogState.action}ing requests:`, err);\n setBulkDialogState((prev) => ({ ...prev, processing: false }));\n const errorMessage = err instanceof Error ? err.message : \"unknown error occurred\";\n alertApi.post({\n message: `Failed to bulk ${bulkDialogState.action} API keys. ${errorMessage}`,\n severity: \"error\",\n display: \"transient\",\n });\n }\n };\n\n const formatDate = (dateString: string) => {\n const date = new Date(dateString);\n return date.toLocaleDateString(\"en-GB\", {\n year: \"numeric\",\n month: \"short\",\n day: \"numeric\",\n });\n };\n\n const columns: TableColumn<APIKey>[] = [\n {\n title: \"Requester\",\n field: \"spec.requestedBy.userId\",\n render: (row) => (\n <Typography variant=\"body2\">{row.spec.requestedBy.userId}</Typography>\n ),\n },\n {\n title: \"API Product\",\n field: \"spec.apiProductRef.name\",\n render: (row) => {\n const name = row.spec.apiProductRef?.name || \"unknown\";\n return (\n <Link to={`/catalog/default/api/${name}`}>\n <strong>{name}</strong>\n </Link>\n );\n },\n },\n {\n title: \"Status\",\n field: \"status.phase\",\n render: (row) => {\n const phase = row.status?.phase || \"Pending\";\n return (\n <Chip label={phase} size=\"small\" style={getApprovalQueueStatusChipStyle(phase)} />\n );\n },\n },\n {\n title: \"Tier\",\n field: \"spec.planTier\",\n render: (row) => (\n <Chip label={row.spec.planTier} size=\"small\" variant=\"outlined\" />\n ),\n },\n {\n title: \"Requested\",\n field: \"metadata.creationTimestamp\",\n render: (row) => (\n <Typography variant=\"body2\">\n {row.metadata.creationTimestamp\n ? formatDate(row.metadata.creationTimestamp)\n : \"-\"}\n </Typography>\n ),\n },\n {\n title: \"Reviewed By\",\n field: \"status.reviewedBy\",\n render: (row) => {\n if (!row.status?.reviewedBy)\n return (\n <Typography variant=\"body2\" color=\"textSecondary\">\n -\n </Typography>\n );\n const reviewer = row.status.reviewedBy.replace(/^user:default\\//, \"\");\n const isAutomatic = reviewer === \"system\";\n return (\n <Box>\n <Typography variant=\"body2\">\n {isAutomatic ? \"Automatic\" : reviewer}\n </Typography>\n {row.status.reviewedAt && (\n <Typography variant=\"caption\" color=\"textSecondary\">\n {formatDate(row.status.reviewedAt)}\n </Typography>\n )}\n </Box>\n );\n },\n },\n {\n title: \"Actions\",\n filtering: false,\n render: (row) => {\n const phase = row.status?.phase || \"Pending\";\n if (phase !== \"Pending\") return null;\n\n const apiProductKey = `${row.metadata.namespace}/${row.spec.apiProductRef?.name || \"unknown\"}`;\n const ownsApiProduct =\n value?.ownedApiProducts?.has(apiProductKey) ?? false;\n const canUpdate =\n canUpdateAllRequests || (canUpdateOwnRequests && ownsApiProduct);\n if (!canUpdate) return null;\n\n return (\n <Box display=\"flex\" style={{ gap: 8 }}>\n <Button\n size=\"small\"\n startIcon={<CheckCircleIcon />}\n onClick={() => handleApprove(row)}\n color=\"primary\"\n variant=\"outlined\"\n >\n Approve\n </Button>\n <Button\n size=\"small\"\n startIcon={<CancelIcon />}\n onClick={() => handleReject(row)}\n color=\"secondary\"\n variant=\"outlined\"\n >\n Reject\n </Button>\n </Box>\n );\n },\n },\n ];\n\n const detailPanelConfig = useMemo(\n () => [\n {\n render: (data: any) => {\n const request = data.rowData as APIKey;\n if (!request?.metadata?.name) {\n return <Box />;\n }\n return <ExpandedRowContent request={request} />;\n },\n },\n ],\n [],\n );\n\n if (loading || updatePermissionLoading) {\n return <Progress />;\n }\n\n if (error) {\n return <ResponseErrorPanel error={error} />;\n }\n\n if (updatePermissionError) {\n return (\n <Box p={2}>\n <Typography color=\"error\">\n Unable to check permissions: {updatePermissionError.message}\n </Typography>\n </Box>\n );\n }\n\n const canSelectRows = canUpdateAllRequests || canUpdateOwnRequests;\n\n return (\n <>\n <Box className={classes.container}>\n <FilterPanel\n sections={filterSections}\n filters={filters}\n onChange={setFilters}\n />\n <Box className={classes.tableContainer}>\n {selectedRequests.length > 0 && (\n <Box className={classes.bulkActions}>\n <Typography variant=\"body2\">\n {selectedRequests.length} request\n {selectedRequests.length !== 1 ? \"s\" : \"\"} selected\n </Typography>\n <Box display=\"flex\" style={{ gap: 8 }}>\n <Button\n size=\"small\"\n variant=\"contained\"\n color=\"primary\"\n startIcon={<CheckCircleIcon />}\n onClick={handleBulkApprove}\n >\n Approve Selected\n </Button>\n <Button\n size=\"small\"\n variant=\"contained\"\n color=\"secondary\"\n startIcon={<CancelIcon />}\n onClick={handleBulkReject}\n >\n Reject Selected\n </Button>\n </Box>\n </Box>\n )}\n\n {filteredRequests.length === 0 ? (\n <Box p={4} textAlign=\"center\">\n <Typography variant=\"body1\" color=\"textSecondary\">\n {value?.allRequests?.length === 0\n ? \"No API keys found.\"\n : \"No API keys match the selected filters.\"}\n </Typography>\n </Box>\n ) : (\n <Table\n options={{\n selection: canSelectRows,\n showSelectAllCheckbox: filteredRequests.some(\n (r: APIKey) =>\n !r.status?.phase || r.status.phase === \"Pending\",\n ),\n selectionProps: (row: APIKey) => ({\n disabled:\n row.status?.phase !== \"Pending\" &&\n row.status?.phase !== undefined,\n }),\n paging: filteredRequests.length > 10,\n pageSize: 20,\n search: true,\n filtering: false,\n debounceInterval: 300,\n showTextRowsSelected: false,\n toolbar: true,\n emptyRowsWhenPaging: false,\n }}\n columns={columns}\n data={filteredRequests.map((item: APIKey) => {\n const isSelected = selectedRequests.some(\n (selected) =>\n selected.metadata.name === item.metadata.name &&\n selected.metadata.namespace === item.metadata.namespace,\n );\n return {\n ...item,\n id: item.metadata.name,\n tableData: { checked: isSelected },\n };\n })}\n onSelectionChange={(rows) => {\n // only allow selecting pending requests\n const pendingOnly = (rows as APIKey[]).filter(\n (r) => !r.status?.phase || r.status.phase === \"Pending\",\n );\n setSelectedRequests(pendingOnly);\n }}\n detailPanel={detailPanelConfig}\n />\n )}\n </Box>\n </Box>\n\n <ApprovalDialog\n open={dialogState.open}\n request={dialogState.request}\n action={dialogState.action}\n processing={dialogState.processing}\n onClose={() =>\n setDialogState({\n open: false,\n request: null,\n action: \"approve\",\n processing: false,\n })\n }\n onConfirm={handleConfirm}\n />\n <BulkActionDialog\n open={bulkDialogState.open}\n requests={bulkDialogState.requests}\n action={bulkDialogState.action}\n processing={bulkDialogState.processing}\n onClose={() =>\n setBulkDialogState({\n open: false,\n requests: [],\n action: \"approve\",\n processing: false,\n })\n }\n onConfirm={handleBulkConfirm}\n />\n <BulkAlertDialog\n open={bulkAlertDialogState.open}\n results={bulkAlertDialogState.results}\n isApprove={bulkAlertDialogState.isApprove}\n onClose={() =>\n setBulkAlertDialogState({\n open: false,\n results: [],\n isApprove: true,\n })\n }\n />\n </>\n );\n};\n","import React from 'react';\nimport {\n Header,\n Page,\n Content,\n SupportButton,\n} from '@backstage/core-components';\nimport { ApprovalQueueTable } from '../ApprovalQueueTable';\n\nexport const ApiKeyApprovalPage = () => {\n return (\n <Page themeId=\"tool\">\n <Header title=\"API Key Approval\" subtitle=\"Review and approve API key requests\">\n <SupportButton>Approve or reject API key access requests</SupportButton>\n </Header>\n <Content>\n <ApprovalQueueTable />\n </Content>\n </Page>\n );\n};\n","import { useApiHolder, configApiRef } from '@backstage/core-plugin-api';\nimport { coreComponentsTranslationRef } from '../translation.esm.js';\nimport { useTranslationRef } from '@backstage/core-plugin-api/alpha';\n\nconst useDefaultSupportConfig = () => {\n const { t } = useTranslationRef(coreComponentsTranslationRef);\n return {\n url: \"https://github.com/backstage/backstage/issues\",\n items: [\n {\n title: t(\"supportConfig.default.title\"),\n icon: \"warning\",\n links: [\n {\n // TODO: Update to dedicated support page on backstage.io/docs\n title: t(\"supportConfig.default.linkTitle\"),\n url: \"https://github.com/backstage/backstage/blob/master/app-config.yaml\"\n }\n ]\n }\n ]\n };\n};\nfunction useSupportConfig() {\n const apiHolder = useApiHolder();\n const config = apiHolder.get(configApiRef);\n const supportConfig = config?.getOptionalConfig(\"app.support\");\n const defaultSupportConfig = useDefaultSupportConfig();\n if (!supportConfig) {\n return defaultSupportConfig;\n }\n return {\n url: supportConfig.getString(\"url\"),\n items: supportConfig.getConfigArray(\"items\").flatMap((itemConf) => ({\n title: itemConf.getString(\"title\"),\n icon: itemConf.getOptionalString(\"icon\"),\n links: (itemConf.getOptionalConfigArray(\"links\") ?? []).flatMap(\n (linkConf) => ({\n url: linkConf.getString(\"url\"),\n title: linkConf.getOptionalString(\"title\") ?? \"\"\n })\n )\n }))\n };\n}\n\nexport { useSupportConfig };\n//# sourceMappingURL=useSupportConfig.esm.js.map\n","import { jsx } from 'react/jsx-runtime';\nimport { useApp } from '@backstage/core-plugin-api';\nimport MuiBrokenImageIcon from '@material-ui/icons/BrokenImage';\n\nfunction AppIcon(props) {\n const { id: key, Fallback = MuiBrokenImageIcon, ...rest } = props;\n const app = useApp();\n const Icon = app.getSystemIcon(key) ?? Fallback;\n return /* @__PURE__ */ jsx(Icon, { ...rest });\n}\nfunction BrokenImageIcon(props) {\n return /* @__PURE__ */ jsx(AppIcon, { id: \"brokenImage\", ...props });\n}\nfunction CatalogIcon(props) {\n return /* @__PURE__ */ jsx(AppIcon, { id: \"catalog\", ...props });\n}\nfunction ChatIcon(props) {\n return /* @__PURE__ */ jsx(AppIcon, { id: \"chat\", ...props });\n}\nfunction DashboardIcon(props) {\n return /* @__PURE__ */ jsx(AppIcon, { id: \"dashboard\", ...props });\n}\nfunction DocsIcon(props) {\n return /* @__PURE__ */ jsx(AppIcon, { id: \"docs\", ...props });\n}\nfunction EmailIcon(props) {\n return /* @__PURE__ */ jsx(AppIcon, { id: \"email\", ...props });\n}\nfunction GitHubIcon(props) {\n return /* @__PURE__ */ jsx(AppIcon, { id: \"github\", ...props });\n}\nfunction GroupIcon(props) {\n return /* @__PURE__ */ jsx(AppIcon, { id: \"group\", ...props });\n}\nfunction HelpIcon(props) {\n return /* @__PURE__ */ jsx(AppIcon, { id: \"help\", ...props });\n}\nfunction UserIcon(props) {\n return /* @__PURE__ */ jsx(AppIcon, { id: \"user\", ...props });\n}\nfunction WarningIcon(props) {\n return /* @__PURE__ */ jsx(AppIcon, { id: \"warning\", ...props });\n}\nfunction StarIcon(props) {\n return /* @__PURE__ */ jsx(AppIcon, { id: \"star\", ...props });\n}\nfunction UnstarredIcon(props) {\n return /* @__PURE__ */ jsx(AppIcon, { id: \"unstarred\", ...props });\n}\n\nexport { AppIcon, BrokenImageIcon, CatalogIcon, ChatIcon, DashboardIcon, DocsIcon, EmailIcon, GitHubIcon, GroupIcon, HelpIcon, StarIcon, UnstarredIcon, UserIcon, WarningIcon };\n//# sourceMappingURL=icons.esm.js.map\n","import { jsxs, Fragment, jsx } from 'react/jsx-runtime';\nimport { useApi, configApiRef, useApp } from '@backstage/core-plugin-api';\nimport Box from '@material-ui/core/Box';\nimport Button from '@material-ui/core/Button';\nimport DialogActions from '@material-ui/core/DialogActions';\nimport IconButton from '@material-ui/core/IconButton';\nimport ListItemIcon from '@material-ui/core/ListItemIcon';\nimport ListItemText from '@material-ui/core/ListItemText';\nimport MenuItem from '@material-ui/core/MenuItem';\nimport MenuList from '@material-ui/core/MenuList';\nimport Popover from '@material-ui/core/Popover';\nimport { makeStyles } from '@material-ui/core/styles';\nimport Typography from '@material-ui/core/Typography';\nimport useMediaQuery from '@material-ui/core/useMediaQuery';\nimport { useState, Children } from 'react';\nimport 'lodash';\nimport 'qs';\nimport 'react-router-dom';\nimport '@react-hookz/web';\nimport { useSupportConfig } from '../../hooks/useSupportConfig.esm.js';\nimport { HelpIcon } from '../../icons/icons.esm.js';\nimport { Link } from '../Link/Link.esm.js';\nimport { coreComponentsTranslationRef } from '../../translation.esm.js';\nimport { useTranslationRef } from '@backstage/core-plugin-api/alpha';\n\nconst useStyles = makeStyles(\n {\n popoverList: {\n minWidth: 260,\n maxWidth: 400\n },\n menuItem: {\n whiteSpace: \"normal\"\n }\n },\n { name: \"BackstageSupportButton\" }\n);\nconst SupportIcon = ({ icon }) => {\n const app = useApp();\n const Icon = icon ? app.getSystemIcon(icon) ?? HelpIcon : HelpIcon;\n return /* @__PURE__ */ jsx(Icon, {});\n};\nconst SupportLink = ({ link }) => /* @__PURE__ */ jsx(Link, { to: link.url, children: link.title ?? link.url });\nconst SupportListItem = ({ item }) => {\n return /* @__PURE__ */ jsxs(MenuItem, { button: false, children: [\n /* @__PURE__ */ jsx(ListItemIcon, { children: /* @__PURE__ */ jsx(SupportIcon, { icon: item.icon }) }),\n /* @__PURE__ */ jsx(\n ListItemText,\n {\n primary: item.title,\n secondary: item.links?.reduce(\n (prev, link, idx) => [\n ...prev,\n idx > 0 && /* @__PURE__ */ jsx(\"br\", {}, idx),\n /* @__PURE__ */ jsx(SupportLink, { link }, link.url)\n ],\n []\n )\n }\n )\n ] });\n};\nfunction SupportButton(props) {\n const { t } = useTranslationRef(coreComponentsTranslationRef);\n const { title, items, children } = props;\n const { items: configItems } = useSupportConfig();\n const [popoverOpen, setPopoverOpen] = useState(false);\n const [anchorEl, setAnchorEl] = useState(null);\n const classes = useStyles();\n const supportConfig = useApi(configApiRef).getOptionalConfig(\"app.support\");\n const isSmallScreen = useMediaQuery(\n (theme) => theme.breakpoints.down(\"sm\")\n );\n const onClickHandler = (event) => {\n setAnchorEl(event.currentTarget);\n setPopoverOpen(true);\n };\n const popoverCloseHandler = () => {\n setPopoverOpen(false);\n };\n if (!supportConfig) {\n return null;\n }\n return /* @__PURE__ */ jsxs(Fragment, { children: [\n /* @__PURE__ */ jsx(Box, { display: \"flex\", ml: 1, children: isSmallScreen ? /* @__PURE__ */ jsx(\n IconButton,\n {\n color: \"primary\",\n size: \"small\",\n onClick: onClickHandler,\n \"data-testid\": \"support-button\",\n \"aria-label\": \"Support\",\n children: /* @__PURE__ */ jsx(HelpIcon, {})\n }\n ) : /* @__PURE__ */ jsx(\n Button,\n {\n \"data-testid\": \"support-button\",\n \"aria-label\": \"Support\",\n color: \"primary\",\n onClick: onClickHandler,\n startIcon: /* @__PURE__ */ jsx(HelpIcon, {}),\n children: t(\"supportButton.title\")\n }\n ) }),\n /* @__PURE__ */ jsxs(\n Popover,\n {\n \"data-testid\": \"support-button-popover\",\n open: popoverOpen,\n anchorEl,\n anchorOrigin: {\n vertical: \"bottom\",\n horizontal: \"right\"\n },\n transformOrigin: {\n vertical: \"top\",\n horizontal: \"right\"\n },\n onClose: popoverCloseHandler,\n children: [\n /* @__PURE__ */ jsxs(\n MenuList,\n {\n className: classes.popoverList,\n autoFocusItem: Boolean(anchorEl),\n children: [\n title && /* @__PURE__ */ jsx(\n MenuItem,\n {\n button: false,\n alignItems: \"flex-start\",\n className: classes.menuItem,\n children: /* @__PURE__ */ jsx(Typography, { variant: \"subtitle1\", children: title })\n }\n ),\n Children.map(children, (child, i) => /* @__PURE__ */ jsx(\n MenuItem,\n {\n button: false,\n alignItems: \"flex-start\",\n className: classes.menuItem,\n children: child\n },\n `child-${i}`\n )),\n (items ?? configItems).map((item, i) => /* @__PURE__ */ jsx(SupportListItem, { item }, `item-${i}`))\n ]\n }\n ),\n /* @__PURE__ */ jsx(DialogActions, { children: /* @__PURE__ */ jsx(\n Button,\n {\n color: \"primary\",\n onClick: popoverCloseHandler,\n \"aria-label\": \"Close\",\n children: t(\"supportButton.close\")\n }\n ) })\n ]\n }\n )\n ] });\n}\n\nexport { SupportButton };\n//# sourceMappingURL=SupportButton.esm.js.map\n","import { 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 { CSSProperties } from \"react\";\n\n/**\n * Returns inline styles for API key status chips on the My API Keys page.\n */\nexport const getMyApiKeysStatusChipStyle = (phase: string): CSSProperties => {\n const base = { border: \"none\" };\n switch (phase) {\n case \"Approved\":\n return { ...base, backgroundColor: \"#1976d2\", color: \"#fff\" }; // Blue\n case \"Rejected\":\n return { ...base, backgroundColor: \"#d32f2f\", color: \"#fff\" }; // Red\n case \"Pending\":\n return { ...base, backgroundColor: \"#9c27b0\", color: \"#fff\" }; // Purple\n default:\n return { ...base, backgroundColor: \"#9c27b0\", color: \"#fff\" }; // Purple (fallback for Pending)\n }\n};\n\n/**\n * Returns inline styles for API key status chips on the Approval Queue page.\n * Uses inline styles to ensure proper specificity with Material-UI Chip.\n */\nexport const getApprovalQueueStatusChipStyle = (phase: string): CSSProperties => {\n const base = { border: \"none\" };\n switch (phase) {\n case \"Approved\":\n return { ...base, backgroundColor: \"#2e7d32\", color: \"#fff\" }; // Green\n case \"Rejected\":\n return { ...base, backgroundColor: \"#d32f2f\", color: \"#fff\" }; // Red\n case \"Pending\":\n return { ...base, backgroundColor: \"#ed6c02\", color: \"#fff\" }; // Orange\n default:\n return { ...base, backgroundColor: \"#ed6c02\", color: \"#fff\" }; // Orange (fallback for Pending)\n }\n};\n\n/**\n * Returns inline styles for lifecycle chips.\n * Uses inline styles to ensure proper specificity with Material-UI Chip.\n */\nexport const getLifecycleChipStyle = (lifecycle: string): CSSProperties => {\n switch (lifecycle) {\n case \"production\":\n return { backgroundColor: \"#1976d2\", color: \"#fff\" }; // Blue\n case \"experimental\":\n return { backgroundColor: \"#9c27b0\", color: \"#fff\" }; // Purple\n case \"deprecated\":\n return { backgroundColor: \"#ff9800\", color: \"#fff\" }; // Orange\n case \"retired\":\n return { backgroundColor: \"#d32f2f\", color: \"#fff\" }; // Red\n default:\n return {};\n }\n};\n","\"use strict\";\n\nvar _interopRequireDefault = require(\"@babel/runtime/helpers/interopRequireDefault\");\n\nvar _interopRequireWildcard = require(\"@babel/runtime/helpers/interopRequireWildcard\");\n\nObject.defineProperty(exports, \"__esModule\", {\n value: true\n});\nexports.default = void 0;\n\nvar React = _interopRequireWildcard(require(\"react\"));\n\nvar _createSvgIcon = _interopRequireDefault(require(\"./utils/createSvgIcon\"));\n\nvar _default = (0, _createSvgIcon.default)( /*#__PURE__*/React.createElement(\"path\", {\n d: \"M12 2C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10S17.52 2 12 2zm-2 15l-5-5 1.41-1.41L10 14.17l7.59-7.59L19 8l-9 9z\"\n}), 'CheckCircle');\n\nexports.default = _default;","export async function handleFetchError(response: Response): Promise<string> {\n const errorData = await response.json().catch(() => ({}));\n\n switch (response.status) {\n case 400:\n return errorData.error || 'Invalid request. Please check your input.';\n case 403:\n return 'Permission denied. Contact your administrator.';\n case 404:\n return 'Resource not found. It may have been deleted.';\n case 409:\n return 'Resource already exists or conflicts with existing data.';\n case 500:\n return 'Server error. Please try again or contact support.';\n default:\n return errorData.error || `Request failed (${response.status})`;\n }\n}\n"],"names":["useStyles","makeStyles","theme","root","width","minWidth","padding","spacing","borderRight","palette","divider","backgroundColor","background","paper","height","overflowY","sectionTitle","fontWeight","fontSize","textTransform","letterSpacing","color","text","secondary","marginBottom","display","alignItems","justifyContent","cursor","userSelect","filterSection","checkbox","checkboxLabel","clearButton","marginTop","count","marginLeft","FilterPanel","sections","filters","onChange","onClear","classes","collapsedSections","setCollapsedSections","React","Set","filter","s","collapsed","map","id","hasActiveFilters","Object","values","some","length","Box","className","mb","Typography","variant","Button","size","onClick","clearedFilters","forEach","section","Divider","isCollapsed","has","selectedCount","mt","toggleSection","sectionId","prev","next","delete","add","span","title","ExpandMoreIcon","ExpandLessIcon","Collapse","in","FormGroup","options","option","FormControlLabel","control","Checkbox","checked","includes","value","currentValues","newValues","v","handleCheckboxChange","label","undefined","_interopRequireDefault","_interopRequireWildcard","exports","_default","default","createElement","d","createPermission","name","attributes","action","kuadrantPlanPolicyListPermission","kuadrantApiProductCreatePermission","kuadrantApiProductReadAllPermission","kuadrantApiProductUpdateOwnPermission","kuadrantApiProductUpdateAllPermission","kuadrantApiProductDeleteOwnPermission","kuadrantApiProductDeleteAllPermission","kuadrantApiProductListPermission","kuadrantApiKeyCreatePermission","resourceType","kuadrantApiKeyUpdateOwnPermission","kuadrantApiKeyUpdateAllPermission","kuadrantApiKeyDeleteOwnPermission","kuadrantApiKeyDeleteAllPermission","kuadrantApiKeyApprovePermission","useAsync","fn","deps","_a","initialState","loading","lastCallId","useRef","isMounted","useMountedState","useState","state","set","callback","useCallback","args","_i","arguments","callId","current","prevState","__assign","apply","then","error","useAsyncFn","useEffect","container","minHeight","tableContainer","flex","overflow","useCasePanel","useCaseLabel","bulkActions","borderBottom","ApprovalDialog","open","request","processing","onClose","onConfirm","confirmInput","setConfirmInput","actionLabel","processingLabel","isReject","confirmText","spec","requestedBy","userId","canConfirm","Dialog","maxWidth","fullWidth","DialogTitle","style","gap","CancelIcon","DialogContent","p","strong","apiProductRef","planTier","component","whiteSpace","useCase","gutterBottom","TextField","e","target","disabled","autoFocus","placeholder","DialogActions","startIcon","CircularProgress","BulkActionDialog","requests","isApprove","paragraph","maxHeight","bgcolor","metadata","namespace","ExpandedRowContent","stopPropagation","useBulkAlertStyles","successSection","success","main","borderRadius","shape","border","failureSection","warningSection","warning","statRow","errorList","listStyle","errorItem","errorName","errorMessage","fontFamily","wordBreak","expandButton","BulkAlertDialog","results","showErrors","setShowErrors","successResults","res","failedResults","hasPartialSuccess","CheckCircleIcon","ul","result","index","li","div","ApprovalQueueTable","config","useApi","configApiRef","fetchApi","fetchApiRef","identityApi","identityApiRef","alertApi","alertApiRef","backendUrl","getString","refresh","setRefresh","selectedRequests","setSelectedRequests","dialogState","setDialogState","bulkDialogState","setBulkDialogState","bulkAlertDialogState","setBulkAlertDialogState","setFilters","status","apiProduct","tier","allowed","canUpdateAllRequests","updateAllPermissionLoading","updateAllPermissionError","useKuadrantPermission","canUpdateOwnRequests","updateOwnPermissionLoading","updateOwnPermissionError","updatePermissionLoading","updatePermissionError","async","reviewedBy","getBackstageIdentity","userEntityRef","requestsResponse","apiProductsResponse","Promise","all","fetch","ok","allRequests","ownedApiProducts","contentType","headers","get","post","message","severity","json","items","apiProductsData","product","annotations","filterSections","useMemo","statusCounts","Approved","Pending","Rejected","apiProductCounts","Map","tierCounts","r","phase","Array","from","entries","charAt","toUpperCase","slice","filteredRequests","formatDate","dateString","Date","toLocaleDateString","year","month","day","columns","field","render","row","Link","to","Chip","getApprovalQueueStatusChipStyle","creationTimestamp","reviewer","replace","isAutomatic","reviewedAt","filtering","apiProductKey","ownsApiProduct","detailPanelConfig","data","rowData","Progress","ResponseErrorPanel","canSelectRows","textAlign","Table","selection","showSelectAllCheckbox","selectionProps","paging","pageSize","search","debounceInterval","showTextRowsSelected","toolbar","emptyRowsWhenPaging","item","isSelected","selected","tableData","onSelectionChange","rows","pendingOnly","detailPanel","endpoint","response","method","body","JSON","stringify","err","handleFetchError","Error","console","bulkResponse","successfulItems","totalSuccess","ApiKeyApprovalPage","Page","themeId","Header","subtitle","SupportButton","Content","useDefaultSupportConfig","t","useTranslationRef","url","icon","links","AppIcon","props","key","Fallback","rest","Icon","useApp","getSystemIcon","jsx","HelpIcon","popoverList","menuItem","SupportIcon","app","SupportLink","link","children","SupportListItem","jsxs","MenuItem","button","ListItemIcon","ListItemText","primary","reduce","idx","configItems","useApiHolder","supportConfig","getOptionalConfig","defaultSupportConfig","getConfigArray","flatMap","itemConf","getOptionalString","getOptionalConfigArray","linkConf","useSupportConfig","popoverOpen","setPopoverOpen","anchorEl","setAnchorEl","isSmallScreen","useMediaQuery","breakpoints","down","onClickHandler","event","currentTarget","popoverCloseHandler","Fragment","ml","IconButton","Popover","anchorOrigin","vertical","horizontal","transformOrigin","MenuList","autoFocusItem","Boolean","Children","child","i","permission","resourceRef","permissionRequest","usePermission","canDeleteResource","ownerId","currentUserId","canDeleteOwn","canDeleteAll","getMyApiKeysStatusChipStyle","base","getLifecycleChipStyle","lifecycle","errorData","catch"],"sourceRoot":""}
@@ -0,0 +1,2 @@
1
+ "use strict";(self.webpackChunkinternal_plugin_kuadrant=self.webpackChunkinternal_plugin_kuadrant||[]).push([[4447],{34955:(t,e,a)=>{a.d(e,{Al:()=>i,EM:()=>c,FL:()=>o,J:()=>n,KV:()=>b,R_:()=>l,U3:()=>d,dp:()=>p,jH:()=>h,q0:()=>f,uL:()=>m,v_:()=>u,vs:()=>s,z4:()=>k});var r=a(83572);(0,r.i)({name:"kuadrant.planpolicy.create",attributes:{action:"create"}}),(0,r.i)({name:"kuadrant.planpolicy.read",attributes:{action:"read"}}),(0,r.i)({name:"kuadrant.planpolicy.update",attributes:{action:"update"}}),(0,r.i)({name:"kuadrant.planpolicy.delete",attributes:{action:"delete"}});const n=(0,r.i)({name:"kuadrant.planpolicy.list",attributes:{action:"read"}}),o=(0,r.i)({name:"kuadrant.apiproduct.create",attributes:{action:"create"}}),i=((0,r.i)({name:"kuadrant.apiproduct.read.own",attributes:{action:"read"}}),(0,r.i)({name:"kuadrant.apiproduct.read.all",attributes:{action:"read"}})),d=(0,r.i)({name:"kuadrant.apiproduct.update.own",attributes:{action:"update"}}),u=(0,r.i)({name:"kuadrant.apiproduct.update.all",attributes:{action:"update"}}),c=(0,r.i)({name:"kuadrant.apiproduct.delete.own",attributes:{action:"delete"}}),l=(0,r.i)({name:"kuadrant.apiproduct.delete.all",attributes:{action:"delete"}}),s=(0,r.i)({name:"kuadrant.apiproduct.list",attributes:{action:"read"}}),p=(0,r.i)({name:"kuadrant.apikey.create",attributes:{action:"create"},resourceType:"apiproduct"}),f=((0,r.i)({name:"kuadrant.apikey.read.own",attributes:{action:"read"}}),(0,r.i)({name:"kuadrant.apikey.read.all",attributes:{action:"read"}}),(0,r.i)({name:"kuadrant.apikey.update.own",attributes:{action:"update"}})),k=(0,r.i)({name:"kuadrant.apikey.update.all",attributes:{action:"update"}}),m=(0,r.i)({name:"kuadrant.apikey.delete.own",attributes:{action:"delete"}}),h=(0,r.i)({name:"kuadrant.apikey.delete.all",attributes:{action:"delete"}}),b=(0,r.i)({name:"kuadrant.apikey.approve",attributes:{action:"update"}})},46205:(t,e,a)=>{a.d(e,{W:()=>o,l:()=>n});var r=a(24217);function n(t,e){const a="resourceType"in t?{permission:t,resourceRef:e}:{permission:t},n=(0,r.J)(a);return{allowed:n.allowed,loading:n.loading,error:n.error}}function o(t,e,a,r){return!!r||!(!a||t!==e)}},46299:(t,e,a)=>{a.d(e,{ee:()=>o,g9:()=>r,uU:()=>n});const r=t=>{const e={border:"none"};switch(t){case"Approved":return{...e,backgroundColor:"#1976d2",color:"#fff"};case"Rejected":return{...e,backgroundColor:"#d32f2f",color:"#fff"};default:return{...e,backgroundColor:"#9c27b0",color:"#fff"}}},n=t=>{const e={border:"none"};switch(t){case"Approved":return{...e,backgroundColor:"#2e7d32",color:"#fff"};case"Rejected":return{...e,backgroundColor:"#d32f2f",color:"#fff"};default:return{...e,backgroundColor:"#ed6c02",color:"#fff"}}},o=t=>{switch(t){case"production":return{backgroundColor:"#1976d2",color:"#fff"};case"experimental":return{backgroundColor:"#9c27b0",color:"#fff"};case"deprecated":return{backgroundColor:"#ff9800",color:"#fff"};case"retired":return{backgroundColor:"#d32f2f",color:"#fff"};default:return{}}}},75453:(t,e,a)=>{a.r(e),a.d(e,{ApiProductInfoCard:()=>v});var r=a(31085),n=(a(95478),a(25467)),o=a(22097),i=a(96040),d=a(86687),u=a(42367),c=a(58837),l=a(72501),s=a(10394),p=a(91638),f=a(46205),k=a(34955),m=a(89509),h=a(71255);const b=(0,c.A)(t=>({label:{fontWeight:600,color:t.palette.text.secondary,marginBottom:t.spacing(.5),fontSize:"0.75rem",textTransform:"uppercase"}})),v=()=>{var t,e,a,c,v,A,y,w,g,j;const x=b(),{entity:P}=(0,n.tN)(),I=(0,o.useApi)(o.configApiRef),C=(0,o.useApi)(o.fetchApiRef),R=(0,o.useApi)(o.identityApiRef),_=I.getString("backend.baseUrl"),{allowed:S,loading:E}=(0,f.l)(k.Al),N=null===(t=P.metadata.annotations)||void 0===t?void 0:t["kuadrant.io/namespace"],O=null===(e=P.metadata.annotations)||void 0===e?void 0:e["kuadrant.io/apiproduct"],{value:U}=(0,p.A)(async()=>(await R.getBackstageIdentity()).userEntityRef.split("/")[1]||"guest",[R]),{value:$,loading:B,error:F}=(0,p.A)(async()=>{if(!N||!O)return null;const t=await C.fetch(`${_}/api/kuadrant/apiproducts/${N}/${O}`);if(!t.ok){const e=await t.json();throw new Error(e.error||`Failed to fetch API product: ${t.status}`)}return t.json()},[_,C,N,O]),L=null==$||null===(c=$.metadata)||void 0===c||null===(a=c.annotations)||void 0===a?void 0:a["backstage.io/owner"],T=null==L?void 0:L.split("/")[1],z=S||U&&T===U;if(!N||!O)return(0,r.jsx)(i.n,{title:"API Product Information",children:(0,r.jsx)(l.A,{children:"No APIProduct linked to this API entity"})});if(B||E)return(0,r.jsx)(i.n,{title:"API Product Information",children:(0,r.jsx)(d.k,{})});if($&&!z)return(0,r.jsx)(i.n,{title:"API Product Information",children:(0,r.jsx)(s.A,{p:2,children:(0,r.jsx)(l.A,{variant:"body2",color:"textSecondary",children:"You don't have permission to view this API product's details. Only the API owner or users with admin permissions can view this information."})})});if(F&&F.message.includes("you can only read your own"))return(0,r.jsx)(i.n,{title:"API Product Information",children:(0,r.jsx)(s.A,{p:2,children:(0,r.jsx)(l.A,{variant:"body2",color:"textSecondary",children:"You don't have permission to view this API product's details. Only the API owner or users with admin permissions can view this information."})})});if(F)return(0,r.jsx)(i.n,{title:"API Product Information",children:(0,r.jsx)(u._,{error:F})});if(!$)return(0,r.jsx)(i.n,{title:"API Product Information",children:(0,r.jsx)(l.A,{children:"APIProduct not found"})});const D=(null===(A=$.status)||void 0===A||null===(v=A.discoveredAuthScheme)||void 0===v?void 0:v.authentication)||{},J=Object.values(D).find(t=>t.hasOwnProperty("jwt")),W=Boolean(J),Y=null==J||null===(y=J.jwt)||void 0===y?void 0:y.issuerUrl,q=null===(g=$.status)||void 0===g||null===(w=g.oidcDiscovery)||void 0===w?void 0:w.tokenEndpoint;return(0,r.jsxs)(r.Fragment,{children:[(0,r.jsxs)(i.n,{title:"API Product Details",children:[(0,r.jsxs)(s.A,{mb:2,children:[(0,r.jsx)(l.A,{variant:"caption",className:x.label,children:"Product Name"}),(0,r.jsx)(l.A,{variant:"h6",children:(null===(j=$.spec)||void 0===j?void 0:j.displayName)||O})]}),(0,r.jsx)(m.O,{product:$,showStatus:!1,showCatalogLink:!1})]}),W&&Y&&(0,r.jsx)(s.A,{mt:2,children:(0,r.jsx)(h.n,{issuerUrl:Y,tokenEndpoint:q})})]})}}}]);
2
+ //# sourceMappingURL=4447.adbf663f.chunk.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"static/4447.adbf663f.chunk.js","mappings":"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","sources":["webpack://internal.plugin-kuadrant/./src/permissions.ts","webpack://internal.plugin-kuadrant/./src/utils/permissions.ts","webpack://internal.plugin-kuadrant/./src/utils/styles.ts","webpack://internal.plugin-kuadrant/./src/components/ApiProductInfoCard/ApiProductInfoCard.tsx"],"sourcesContent":["import { createPermission } from '@backstage/plugin-permission-common';\n\n/**\n * permission definitions for the kuadrant plugin\n *\n * these permissions control access to kuadrant resources and operations.\n * they must match the permissions defined in the backend plugin.\n *\n * permission types:\n * - BasicPermission: standard permission that applies globally\n * - ResourcePermission: permission scoped to specific resource types (e.g., apiproduct)\n *\n * permission patterns:\n * - `.create` - create new resources\n * - `.read` - read resource details\n * - `.read.own` - read only resources owned by the user\n * - `.read.all` - read all resources regardless of ownership\n * - `.update` - modify existing resources\n * - `.delete` - delete resources\n * - `.delete.own` - delete only resources owned by the user\n * - `.delete.all` - delete any resource regardless of ownership\n * - `.list` - list/view collections of resources\n */\n\n// planpolicy permissions\nexport const kuadrantPlanPolicyCreatePermission = createPermission({\n name: 'kuadrant.planpolicy.create',\n attributes: { action: 'create' },\n});\n\nexport const kuadrantPlanPolicyReadPermission = createPermission({\n name: 'kuadrant.planpolicy.read',\n attributes: { action: 'read' },\n});\n\nexport const kuadrantPlanPolicyUpdatePermission = createPermission({\n name: 'kuadrant.planpolicy.update',\n attributes: { action: 'update' },\n});\n\nexport const kuadrantPlanPolicyDeletePermission = createPermission({\n name: 'kuadrant.planpolicy.delete',\n attributes: { action: 'delete' },\n});\n\nexport const kuadrantPlanPolicyListPermission = createPermission({\n name: 'kuadrant.planpolicy.list',\n attributes: { action: 'read' },\n});\n\n// apiproduct permissions\n\n/**\n * permission to create new API products\n * granted to api owners and admins\n */\nexport const kuadrantApiProductCreatePermission = createPermission({\n name: 'kuadrant.apiproduct.create',\n attributes: { action: 'create' },\n});\n\n/**\n * permission to read API products owned by the current user\n * for api owners to view their own products\n */\nexport const kuadrantApiProductReadOwnPermission = createPermission({\n name: 'kuadrant.apiproduct.read.own',\n attributes: { action: 'read' },\n});\n\n/**\n * permission to read all API products regardless of ownership\n * for platform engineers/admins who need to view all products\n */\nexport const kuadrantApiProductReadAllPermission = createPermission({\n name: 'kuadrant.apiproduct.read.all',\n attributes: { action: 'read' },\n});\n\n/**\n * permission to update API products owned by the current user\n * for api owners to modify their own products\n */\nexport const kuadrantApiProductUpdateOwnPermission = createPermission({\n name: 'kuadrant.apiproduct.update.own',\n attributes: { action: 'update' },\n});\n\n/**\n * permission to update any API product regardless of ownership\n * for platform engineers/admins\n */\nexport const kuadrantApiProductUpdateAllPermission = createPermission({\n name: 'kuadrant.apiproduct.update.all',\n attributes: { action: 'update' },\n});\n\n/**\n * permission to delete API products owned by the current user\n * for api owners to remove their own products\n */\nexport const kuadrantApiProductDeleteOwnPermission = createPermission({\n name: 'kuadrant.apiproduct.delete.own',\n attributes: { action: 'delete' },\n});\n\n/**\n * permission to delete any API product regardless of ownership\n * for platform engineers/admins\n */\nexport const kuadrantApiProductDeleteAllPermission = createPermission({\n name: 'kuadrant.apiproduct.delete.all',\n attributes: { action: 'delete' },\n});\n\n/**\n * permission to list API products\n * backend filters results based on .own vs .all read permissions\n */\nexport const kuadrantApiProductListPermission = createPermission({\n name: 'kuadrant.apiproduct.list',\n attributes: { action: 'read' },\n});\n\n// apikey permissions\n\n/**\n * permission to create API keys (request API access)\n *\n * this is a ResourcePermission scoped to 'apiproduct', allowing\n * fine-grained control over which API products users can request access to.\n *\n * use in frontend: useKuadrantPermission(kuadrantApiKeyCreatePermission)\n * use in backend with resource: { permission, resourceRef: 'apiproduct:namespace/name' }\n */\nexport const kuadrantApiKeyCreatePermission = createPermission({\n name: 'kuadrant.apikey.create',\n attributes: { action: 'create' },\n resourceType: 'apiproduct',\n});\n\n/**\n * permission to read API keys owned by the current user\n * allows users to view their own API keys and request history\n */\nexport const kuadrantApiKeyReadOwnPermission = createPermission({\n name: 'kuadrant.apikey.read.own',\n attributes: { action: 'read' },\n});\n\n/**\n * permission to read all API keys regardless of ownership\n * for platform engineers/admins who need to view the approval queue and audit keys\n */\nexport const kuadrantApiKeyReadAllPermission = createPermission({\n name: 'kuadrant.apikey.read.all',\n attributes: { action: 'read' },\n});\n\n/**\n * permission to update API keys owned by the current user\n * allows users to edit their own pending requests (change plan tier, use case)\n */\nexport const kuadrantApiKeyUpdateOwnPermission = createPermission({\n name: 'kuadrant.apikey.update.own',\n attributes: { action: 'update' },\n});\n\n/**\n * permission to update any API key regardless of ownership\n * typically granted to API owners and platform engineers for approving/rejecting requests\n */\nexport const kuadrantApiKeyUpdateAllPermission = createPermission({\n name: 'kuadrant.apikey.update.all',\n attributes: { action: 'update' },\n});\n\n/**\n * permission to delete API keys owned by the current user\n * allows users to cancel their own requests or revoke their own access\n */\nexport const kuadrantApiKeyDeleteOwnPermission = createPermission({\n name: 'kuadrant.apikey.delete.own',\n attributes: { action: 'delete' },\n});\n\n/**\n * permission to delete any API key regardless of ownership\n * for platform engineers/admins who need to revoke access\n */\nexport const kuadrantApiKeyDeleteAllPermission = createPermission({\n name: 'kuadrant.apikey.delete.all',\n attributes: { action: 'delete' },\n});\n\n/**\n * permission to approve/reject API key requests\n * grants access to the approval queue - for API owners and admins only\n * separate from update.own which consumers use to edit their pending requests\n */\nexport const kuadrantApiKeyApprovePermission = createPermission({\n name: 'kuadrant.apikey.approve',\n attributes: { action: 'update' },\n});\n\nexport const kuadrantPermissions = [\n kuadrantPlanPolicyCreatePermission,\n kuadrantPlanPolicyReadPermission,\n kuadrantPlanPolicyUpdatePermission,\n kuadrantPlanPolicyDeletePermission,\n kuadrantPlanPolicyListPermission,\n kuadrantApiProductCreatePermission,\n kuadrantApiProductReadOwnPermission,\n kuadrantApiProductReadAllPermission,\n kuadrantApiProductUpdateOwnPermission,\n kuadrantApiProductUpdateAllPermission,\n kuadrantApiProductDeleteOwnPermission,\n kuadrantApiProductDeleteAllPermission,\n kuadrantApiProductListPermission,\n kuadrantApiKeyCreatePermission,\n kuadrantApiKeyReadOwnPermission,\n kuadrantApiKeyReadAllPermission,\n kuadrantApiKeyUpdateOwnPermission,\n kuadrantApiKeyUpdateAllPermission,\n kuadrantApiKeyDeleteOwnPermission,\n kuadrantApiKeyDeleteAllPermission,\n kuadrantApiKeyApprovePermission,\n];\n","import { usePermission } from '@backstage/plugin-permission-react';\nimport { Permission, ResourcePermission } from '@backstage/plugin-permission-common';\n\n/**\n * result of a permission check including error state\n */\nexport interface PermissionCheckResult {\n allowed: boolean;\n loading: boolean;\n error?: Error;\n}\n\n/**\n * custom hook for checking kuadrant permissions that handles both\n * BasicPermission and ResourcePermission types without type bypasses\n *\n * @param permission - the permission to check\n * @param resourceRef - optional resource reference for ResourcePermissions\n * @returns permission check result with error handling\n *\n * @example\n * // basic permission\n * const { allowed, loading, error } = useKuadrantPermission(\n * kuadrantApiProductListPermission\n * );\n *\n * @example\n * // resource permission\n * const { allowed, loading, error } = useKuadrantPermission(\n * kuadrantApiKeyCreatePermission,\n * 'apiproduct:namespace/name'\n * );\n */\nexport function useKuadrantPermission(\n permission: Permission,\n resourceRef?: string,\n): PermissionCheckResult {\n // construct the permission request based on whether it's a ResourcePermission\n const permissionRequest = 'resourceType' in permission\n ? { permission: permission as ResourcePermission, resourceRef }\n : { permission };\n\n const result = usePermission(permissionRequest as any);\n\n return {\n allowed: result.allowed,\n loading: result.loading,\n error: result.error,\n };\n}\n\n/**\n * helper to determine if a user can delete a specific API key or request\n *\n * @param ownerId - the user id who owns the key/request\n * @param currentUserId - the current user's id\n * @param canDeleteOwn - whether user has permission to delete their own keys\n * @param canDeleteAll - whether user has permission to delete all keys\n * @returns true if user can delete this specific key/request\n */\nexport function canDeleteResource(\n ownerId: string,\n currentUserId: string,\n canDeleteOwn: boolean,\n canDeleteAll: boolean,\n): boolean {\n if (canDeleteAll) return true;\n if (canDeleteOwn && ownerId === currentUserId) return true;\n return false;\n}\n","import { CSSProperties } from \"react\";\n\n/**\n * Returns inline styles for API key status chips on the My API Keys page.\n */\nexport const getMyApiKeysStatusChipStyle = (phase: string): CSSProperties => {\n const base = { border: \"none\" };\n switch (phase) {\n case \"Approved\":\n return { ...base, backgroundColor: \"#1976d2\", color: \"#fff\" }; // Blue\n case \"Rejected\":\n return { ...base, backgroundColor: \"#d32f2f\", color: \"#fff\" }; // Red\n case \"Pending\":\n return { ...base, backgroundColor: \"#9c27b0\", color: \"#fff\" }; // Purple\n default:\n return { ...base, backgroundColor: \"#9c27b0\", color: \"#fff\" }; // Purple (fallback for Pending)\n }\n};\n\n/**\n * Returns inline styles for API key status chips on the Approval Queue page.\n * Uses inline styles to ensure proper specificity with Material-UI Chip.\n */\nexport const getApprovalQueueStatusChipStyle = (phase: string): CSSProperties => {\n const base = { border: \"none\" };\n switch (phase) {\n case \"Approved\":\n return { ...base, backgroundColor: \"#2e7d32\", color: \"#fff\" }; // Green\n case \"Rejected\":\n return { ...base, backgroundColor: \"#d32f2f\", color: \"#fff\" }; // Red\n case \"Pending\":\n return { ...base, backgroundColor: \"#ed6c02\", color: \"#fff\" }; // Orange\n default:\n return { ...base, backgroundColor: \"#ed6c02\", color: \"#fff\" }; // Orange (fallback for Pending)\n }\n};\n\n/**\n * Returns inline styles for lifecycle chips.\n * Uses inline styles to ensure proper specificity with Material-UI Chip.\n */\nexport const getLifecycleChipStyle = (lifecycle: string): CSSProperties => {\n switch (lifecycle) {\n case \"production\":\n return { backgroundColor: \"#1976d2\", color: \"#fff\" }; // Blue\n case \"experimental\":\n return { backgroundColor: \"#9c27b0\", color: \"#fff\" }; // Purple\n case \"deprecated\":\n return { backgroundColor: \"#ff9800\", color: \"#fff\" }; // Orange\n case \"retired\":\n return { backgroundColor: \"#d32f2f\", color: \"#fff\" }; // Red\n default:\n return {};\n }\n};\n","import React from 'react';\nimport { useEntity } from '@backstage/plugin-catalog-react';\nimport { useApi, configApiRef, fetchApiRef, identityApiRef } from '@backstage/core-plugin-api';\nimport { InfoCard, Progress, ResponseErrorPanel } from '@backstage/core-components';\nimport { Typography, Box, makeStyles } from '@material-ui/core';\nimport useAsync from 'react-use/lib/useAsync';\nimport { useKuadrantPermission } from '../../utils/permissions';\nimport { kuadrantApiProductReadAllPermission } from '../../permissions';\nimport { ApiProductDetails } from '../ApiProductDetails';\nimport { OidcProviderCard } from '../OidcProviderCard';\n\nconst useStyles = makeStyles((theme) => ({\n label: {\n fontWeight: 600,\n color: theme.palette.text.secondary,\n marginBottom: theme.spacing(0.5),\n fontSize: '0.75rem',\n textTransform: 'uppercase',\n },\n}));\n\nexport const ApiProductInfoCard = () => {\n const classes = useStyles();\n const { entity } = useEntity();\n const config = useApi(configApiRef);\n const fetchApi = useApi(fetchApiRef);\n const identityApi = useApi(identityApiRef);\n const backendUrl = config.getString('backend.baseUrl');\n\n const { allowed: canReadAll, loading: permLoading } = useKuadrantPermission(\n kuadrantApiProductReadAllPermission\n );\n\n const namespace = entity.metadata.annotations?.['kuadrant.io/namespace'];\n const apiProductName = entity.metadata.annotations?.['kuadrant.io/apiproduct'];\n\n const { value: currentUserId } = useAsync(async () => {\n const identity = await identityApi.getBackstageIdentity();\n return identity.userEntityRef.split('/')[1] || 'guest';\n }, [identityApi]);\n\n const { value: apiProduct, loading, error } = useAsync(async () => {\n if (!namespace || !apiProductName) {\n return null;\n }\n\n const productResponse = await fetchApi.fetch(\n `${backendUrl}/api/kuadrant/apiproducts/${namespace}/${apiProductName}`\n );\n\n if (!productResponse.ok) {\n const errorData = await productResponse.json();\n throw new Error(errorData.error || `Failed to fetch API product: ${productResponse.status}`);\n }\n\n return productResponse.json();\n }, [backendUrl, fetchApi, namespace, apiProductName]);\n\n // check if user has permission to view this api product\n const owner = apiProduct?.metadata?.annotations?.['backstage.io/owner'];\n const ownerUserId = owner?.split('/')[1];\n const canView = canReadAll || (currentUserId && ownerUserId === currentUserId);\n\n if (!namespace || !apiProductName) {\n return (\n <InfoCard title=\"API Product Information\">\n <Typography>No APIProduct linked to this API entity</Typography>\n </InfoCard>\n );\n }\n\n if (loading || permLoading) {\n return (\n <InfoCard title=\"API Product Information\">\n <Progress />\n </InfoCard>\n );\n }\n\n // show permission message if user doesn't have permission\n if (apiProduct && !canView) {\n return (\n <InfoCard title=\"API Product Information\">\n <Box p={2}>\n <Typography variant=\"body2\" color=\"textSecondary\">\n You don't have permission to view this API product's details. Only the API owner or users with admin permissions can view this information.\n </Typography>\n </Box>\n </InfoCard>\n );\n }\n\n // also show permission message if we got a permission error from the backend\n if (error && error.message.includes('you can only read your own')) {\n return (\n <InfoCard title=\"API Product Information\">\n <Box p={2}>\n <Typography variant=\"body2\" color=\"textSecondary\">\n You don't have permission to view this API product's details. Only the API owner or users with admin permissions can view this information.\n </Typography>\n </Box>\n </InfoCard>\n );\n }\n\n if (error) {\n return (\n <InfoCard title=\"API Product Information\">\n <ResponseErrorPanel error={error} />\n </InfoCard>\n );\n }\n\n if (!apiProduct) {\n return (\n <InfoCard title=\"API Product Information\">\n <Typography>APIProduct not found</Typography>\n </InfoCard>\n );\n }\n\n // check for OIDC auth scheme\n const authSchemes = apiProduct.status?.discoveredAuthScheme?.authentication || {};\n const schemeObjects = Object.values(authSchemes);\n const jwtScheme = schemeObjects.find((scheme: any) => scheme.hasOwnProperty('jwt'));\n const hasOidc = Boolean(jwtScheme);\n const jwtIssuer = (jwtScheme as any)?.jwt?.issuerUrl;\n const jwtTokenEndpoint = apiProduct.status?.oidcDiscovery?.tokenEndpoint;\n\n return (\n <>\n <InfoCard title=\"API Product Details\">\n <Box mb={2}>\n <Typography variant=\"caption\" className={classes.label}>\n Product Name\n </Typography>\n <Typography variant=\"h6\">\n {apiProduct.spec?.displayName || apiProductName}\n </Typography>\n </Box>\n <ApiProductDetails\n product={apiProduct}\n showStatus={false}\n showCatalogLink={false}\n />\n </InfoCard>\n {hasOidc && jwtIssuer && (\n <Box mt={2}>\n <OidcProviderCard\n issuerUrl={jwtIssuer}\n tokenEndpoint={jwtTokenEndpoint}\n />\n </Box>\n )}\n </>\n );\n};\n"],"names":["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","getMyApiKeysStatusChipStyle","phase","base","border","backgroundColor","color","getApprovalQueueStatusChipStyle","getLifecycleChipStyle","lifecycle","useStyles","makeStyles","theme","label","fontWeight","palette","text","secondary","marginBottom","spacing","fontSize","textTransform","ApiProductInfoCard","entity","apiProduct","classes","useEntity","config","useApi","configApiRef","fetchApi","fetchApiRef","identityApi","identityApiRef","backendUrl","getString","canReadAll","permLoading","namespace","metadata","annotations","apiProductName","value","useAsync","async","getBackstageIdentity","userEntityRef","split","productResponse","fetch","ok","errorData","json","Error","status","owner","ownerUserId","canView","InfoCard","title","Typography","Progress","Box","p","variant","message","includes","ResponseErrorPanel","authSchemes","discoveredAuthScheme","authentication","jwtScheme","Object","values","find","scheme","hasOwnProperty","hasOidc","Boolean","jwtIssuer","jwt","issuerUrl","jwtTokenEndpoint","oidcDiscovery","tokenEndpoint","mb","className","spec","displayName","ApiProductDetails","product","showStatus","showCatalogLink","mt","OidcProviderCard"],"sourceRoot":""}