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

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 (96) hide show
  1. package/dist/api.esm.js +220 -0
  2. package/dist/api.esm.js.map +1 -0
  3. package/dist/apis.esm.js +15 -0
  4. package/dist/apis.esm.js.map +1 -0
  5. package/dist/components/ApiAccessCard/ApiAccessCard.esm.js +25 -36
  6. package/dist/components/ApiAccessCard/ApiAccessCard.esm.js.map +1 -1
  7. package/dist/components/ApiKeyDetailPage/ApiKeyDetailPage.esm.js +24 -36
  8. package/dist/components/ApiKeyDetailPage/ApiKeyDetailPage.esm.js.map +1 -1
  9. package/dist/components/ApiKeyManagementTab/ApiKeyManagementTab.esm.js +20 -46
  10. package/dist/components/ApiKeyManagementTab/ApiKeyManagementTab.esm.js.map +1 -1
  11. package/dist/components/ApiProductDetailPage/ApiProductDetailPage.esm.js +9 -43
  12. package/dist/components/ApiProductDetailPage/ApiProductDetailPage.esm.js.map +1 -1
  13. package/dist/components/ApiProductInfoCard/ApiProductInfoCard.esm.js +5 -13
  14. package/dist/components/ApiProductInfoCard/ApiProductInfoCard.esm.js.map +1 -1
  15. package/dist/components/ApiProductOpenApiAlert/ApiProductOpenApiAlert.esm.js +7 -10
  16. package/dist/components/ApiProductOpenApiAlert/ApiProductOpenApiAlert.esm.js.map +1 -1
  17. package/dist/components/ApprovalQueueTable/ApprovalQueueTable.esm.js +32 -64
  18. package/dist/components/ApprovalQueueTable/ApprovalQueueTable.esm.js.map +1 -1
  19. package/dist/components/CreateAPIProductDialog/CreateAPIProductDialog.esm.js +29 -51
  20. package/dist/components/CreateAPIProductDialog/CreateAPIProductDialog.esm.js.map +1 -1
  21. package/dist/components/EditAPIKeyDialog/EditAPIKeyDialog.esm.js +8 -18
  22. package/dist/components/EditAPIKeyDialog/EditAPIKeyDialog.esm.js.map +1 -1
  23. package/dist/components/EditAPIProductDialog/EditAPIProductDialog.esm.js +8 -28
  24. package/dist/components/EditAPIProductDialog/EditAPIProductDialog.esm.js.map +1 -1
  25. package/dist/components/EntityApiApprovalTab/EntityApiApprovalTab.esm.js +9 -25
  26. package/dist/components/EntityApiApprovalTab/EntityApiApprovalTab.esm.js.map +1 -1
  27. package/dist/components/KuadrantPage/ApiProductsPage.esm.js +19 -53
  28. package/dist/components/KuadrantPage/ApiProductsPage.esm.js.map +1 -1
  29. package/dist/components/MyApiKeysTable/MyApiKeysTable.esm.js +25 -52
  30. package/dist/components/MyApiKeysTable/MyApiKeysTable.esm.js.map +1 -1
  31. package/dist/components/RequestAccessDialog/RequestAccessDialog.esm.js +10 -25
  32. package/dist/components/RequestAccessDialog/RequestAccessDialog.esm.js.map +1 -1
  33. package/dist/index.d.ts +459 -1
  34. package/dist/index.esm.js +2 -0
  35. package/dist/index.esm.js.map +1 -1
  36. package/dist-scalprum/internal.plugin-kuadrant.8bf02d5482a5a3ed6337.js +2 -0
  37. package/dist-scalprum/internal.plugin-kuadrant.8bf02d5482a5a3ed6337.js.map +1 -0
  38. package/dist-scalprum/plugin-manifest.json +3 -3
  39. package/dist-scalprum/static/2967.28e7c7f8.chunk.js +2 -0
  40. package/dist-scalprum/static/2967.28e7c7f8.chunk.js.map +1 -0
  41. package/dist-scalprum/static/3976.cf3ec7be.chunk.js +2 -0
  42. package/dist-scalprum/static/3976.cf3ec7be.chunk.js.map +1 -0
  43. package/dist-scalprum/static/4447.222bb58d.chunk.js +2 -0
  44. package/dist-scalprum/static/4447.222bb58d.chunk.js.map +1 -0
  45. package/dist-scalprum/static/5203.e92a3353.chunk.js +2 -0
  46. package/dist-scalprum/static/5203.e92a3353.chunk.js.map +1 -0
  47. package/dist-scalprum/static/6371.6ee1f11d.chunk.js +2 -0
  48. package/dist-scalprum/static/6371.6ee1f11d.chunk.js.map +1 -0
  49. package/dist-scalprum/static/6800.a7645f4c.chunk.js +2 -0
  50. package/dist-scalprum/static/6800.a7645f4c.chunk.js.map +1 -0
  51. package/dist-scalprum/static/69.1decc5e8.chunk.js +2 -0
  52. package/dist-scalprum/static/69.1decc5e8.chunk.js.map +1 -0
  53. package/dist-scalprum/static/7005.a9f73153.chunk.js +2 -0
  54. package/dist-scalprum/static/7005.a9f73153.chunk.js.map +1 -0
  55. package/dist-scalprum/static/7270.b0e185f1.chunk.js +2 -0
  56. package/dist-scalprum/static/7270.b0e185f1.chunk.js.map +1 -0
  57. package/dist-scalprum/static/7791.5c1eea8a.chunk.js +2 -0
  58. package/dist-scalprum/static/7791.5c1eea8a.chunk.js.map +1 -0
  59. package/dist-scalprum/static/8789.84963cbc.chunk.js +2 -0
  60. package/dist-scalprum/static/8789.84963cbc.chunk.js.map +1 -0
  61. package/dist-scalprum/static/9051.db875198.chunk.js +2 -0
  62. package/dist-scalprum/static/9051.db875198.chunk.js.map +1 -0
  63. package/dist-scalprum/static/{2946.a35243f1.chunk.js → 9370.2e9fe34b.chunk.js} +3 -3
  64. package/dist-scalprum/static/9370.2e9fe34b.chunk.js.map +1 -0
  65. package/dist-scalprum/static/9838.5df9b1de.chunk.js +2 -0
  66. package/dist-scalprum/static/9838.5df9b1de.chunk.js.map +1 -0
  67. package/dist-scalprum/static/exposed-PluginRoot.8a8dd91f.chunk.js +2 -0
  68. package/dist-scalprum/static/{exposed-PluginRoot.8d8f0b09.chunk.js.map → exposed-PluginRoot.8a8dd91f.chunk.js.map} +1 -1
  69. package/package.json +3 -1
  70. package/dist-scalprum/internal.plugin-kuadrant.eacc31ca4d6c1340f8a8.js +0 -2
  71. package/dist-scalprum/internal.plugin-kuadrant.eacc31ca4d6c1340f8a8.js.map +0 -1
  72. package/dist-scalprum/static/2946.a35243f1.chunk.js.map +0 -1
  73. package/dist-scalprum/static/2967.5bade048.chunk.js +0 -2
  74. package/dist-scalprum/static/2967.5bade048.chunk.js.map +0 -1
  75. package/dist-scalprum/static/3650.89dfc64c.chunk.js +0 -2
  76. package/dist-scalprum/static/3650.89dfc64c.chunk.js.map +0 -1
  77. package/dist-scalprum/static/3976.4cf18515.chunk.js +0 -2
  78. package/dist-scalprum/static/3976.4cf18515.chunk.js.map +0 -1
  79. package/dist-scalprum/static/4447.adbf663f.chunk.js +0 -2
  80. package/dist-scalprum/static/4447.adbf663f.chunk.js.map +0 -1
  81. package/dist-scalprum/static/5203.fce2a28f.chunk.js +0 -2
  82. package/dist-scalprum/static/5203.fce2a28f.chunk.js.map +0 -1
  83. package/dist-scalprum/static/6371.d45f37cc.chunk.js +0 -2
  84. package/dist-scalprum/static/6371.d45f37cc.chunk.js.map +0 -1
  85. package/dist-scalprum/static/6800.cd5c7bcb.chunk.js +0 -2
  86. package/dist-scalprum/static/6800.cd5c7bcb.chunk.js.map +0 -1
  87. package/dist-scalprum/static/7005.72759857.chunk.js +0 -2
  88. package/dist-scalprum/static/7005.72759857.chunk.js.map +0 -1
  89. package/dist-scalprum/static/7270.9473c969.chunk.js +0 -2
  90. package/dist-scalprum/static/7270.9473c969.chunk.js.map +0 -1
  91. package/dist-scalprum/static/7791.01371352.chunk.js +0 -2
  92. package/dist-scalprum/static/7791.01371352.chunk.js.map +0 -1
  93. package/dist-scalprum/static/8789.30227526.chunk.js +0 -2
  94. package/dist-scalprum/static/8789.30227526.chunk.js.map +0 -1
  95. package/dist-scalprum/static/exposed-PluginRoot.8d8f0b09.chunk.js +0 -2
  96. /package/dist-scalprum/static/{2946.a35243f1.chunk.js.LICENSE.txt → 9370.2e9fe34b.chunk.js.LICENSE.txt} +0 -0
@@ -1,2 +0,0 @@
1
- "use strict";(self.webpackChunkinternal_plugin_kuadrant=self.webpackChunkinternal_plugin_kuadrant||[]).push([[6371],{5030:(e,t,n)=>{Object.defineProperty(t,"__esModule",{value:!0});var r=n(85608),a=n(95478),o=r.__importDefault(n(10009));t.default=function(e,t,n){void 0===t&&(t=[]),void 0===n&&(n={loading:!1});var i=a.useRef(0),l=o.default(),c=a.useState(n),s=c[0],u=c[1],d=a.useCallback(function(){for(var t=[],n=0;n<arguments.length;n++)t[n]=arguments[n];var a=++i.current;return s.loading||u(function(e){return r.__assign(r.__assign({},e),{loading:!0})}),e.apply(void 0,t).then(function(e){return l()&&a===i.current&&u({value:e,loading:!1}),e},function(e){return l()&&a===i.current&&u({error:e,loading:!1}),e})},t);return[s,d]}},10009:(e,t,n)=>{Object.defineProperty(t,"__esModule",{value:!0});var r=n(95478);t.default=function(){var e=r.useRef(!1),t=r.useCallback(function(){return e.current},[]);return r.useEffect(function(){return e.current=!0,function(){e.current=!1}},[]),t}},25467:(e,t,n)=>{n.d(t,{tN:()=>a}),n(31085),n(22097);var r=n(15427);function a(){const e=(0,r.useVersionedContext)("entity-context");if(!e)throw new Error("Entity context is not available");const t=e.atVersion(1);if(!t)throw new Error("EntityContext v1 not available");if(!t.entity)throw new Error("useEntity hook is being called outside of an EntityLayout where the entity has not been loaded. If this is intentional, please use useAsyncEntity instead.");return{entity:t.entity}}(0,r.createVersionedContext)("entity-context")},36371:(e,t,n)=>{n.r(t),n.d(t,{ApiProductOpenApiAlert:()=>d});var r=n(31085),a=(n(95478),n(25467)),o=n(22097),i=n(37725),l=n(10394),c=n(72501),s=n(84441),u=n(91638);const d=()=>{var e,t,n,d;const{entity:p}=(0,a.tN)(),f=(0,o.useApi)(o.configApiRef),m=(0,o.useApi)(o.fetchApiRef),h=f.getString("backend.baseUrl"),g=null===(e=p.metadata.annotations)||void 0===e?void 0:e["kuadrant.io/namespace"],v=null===(t=p.metadata.annotations)||void 0===t?void 0:t["kuadrant.io/apiproduct"],{value:y,loading:A,error:b}=(0,u.A)(async()=>{if(!g||!v)return null;const e=await m.fetch(`${h}/api/kuadrant/apiproducts/${g}/${v}`);return e.ok?await e.json():null},[h,m,g,v]);if(!g||!v||A||b||!y)return null;const{spec:w,status:k}=y,x=null==k||null===(n=k.conditions)||void 0===n?void 0:n.find(e=>"OpenAPISpecReady"===e.type&&"False"===e.status);return x?(0,r.jsx)(l.A,{mb:2,children:(0,r.jsxs)(s.A,{severity:"warning",children:[(0,r.jsx)(c.A,{variant:"body2",gutterBottom:!0,children:(0,r.jsx)("strong",{children:"OpenAPI Spec Issue"})}),(0,r.jsx)(c.A,{variant:"body2",gutterBottom:!0,children:x.message}),(null===(d=w.documentation)||void 0===d?void 0:d.openAPISpecURL)&&(0,r.jsxs)(c.A,{variant:"body2",children:["Spec URL:"," ",(0,r.jsx)(i.N_,{to:w.documentation.openAPISpecURL,target:"_blank",children:w.documentation.openAPISpecURL})]})]})}):null}},37725:(e,t,n)=>{n.d(t,{N_:()=>b});var r=n(31085),a=n(22097),o=n(49203),i=n(37976),l=n(72501),c=n(53373),s=n.n(c),u=n(45250),d=n(95478),p=n(49634),f=n(39330);const m=(0,i.makeStyles)(e=>({visuallyHidden:{clip:"rect(0 0 0 0)",clipPath:"inset(50%)",overflow:"hidden",position:"absolute",userSelect:"none",whiteSpace:"nowrap",height:1,width:1},externalLink:{position:"relative"},externalLinkIcon:{verticalAlign:"bottom",marginLeft:e.spacing(.5)}}),{name:"Link"}),h=()=>{const e=(0,a.useApp)().getSystemIcon("externalLink")||f.A,t=m();return(0,r.jsx)(e,{className:t.externalLinkIcon})},g=e=>/^([a-z+.-]+):/.test(e),v=/^[\u0000-\u001F ]*j[\r\n\t]*a[\r\n\t]*v[\r\n\t]*a[\r\n\t]*s[\r\n\t]*c[\r\n\t]*r[\r\n\t]*i[\r\n\t]*p[\r\n\t]*t[\r\n\t]*\:/i,y=window.open;if(y&&!y.__backstage){const e=function(...e){const t=String(e[0]);if(v.test(t))throw new Error("Rejected window.open() with a javascript: URL as a security precaution");return y.apply(this,e)};e.__backstage=!0,window.open=e}const A=e=>e instanceof Array?e.map(A).join(" ").trim():"object"==typeof e&&e?A(e?.props?.children):["string","number"].includes(typeof e)?String(e):"",b=(0,d.forwardRef)(({onClick:e,noTrack:t,externalLinkIcon:n,...i},c)=>{const d=m(),f=(0,a.useAnalytics)(),y=function(){const[e]=(0,p.createRoutesFromChildren)((0,r.jsx)(p.Route,{index:!0,element:(0,r.jsx)("div",{})}));return!e.index}()?(e=>{let t=String(e);const n=(()=>{const e=(()=>{try{return(0,a.useApi)(a.configApiRef).getOptionalString("app.baseUrl")}catch{return}})()??"/",{pathname:t}=new URL(e,"http://sample.dev");return(0,u.trimEnd)(t,"/")})(),r=g(t),o=t.startsWith(n);return r||o||(t=n.concat(t)),t})(i.to):i.to,b=A(i.children)||y,w=g(y),k=w&&!!/^https?:/.exec(y);if(v.test(y))throw new Error("Link component rejected javascript: URL as a security precaution");const x=n=>{e?.(n),t||f.captureEvent("click",b,{attributes:{to:y}})};return w?(0,r.jsxs)(o.A,{...k?{target:"_blank",rel:"noopener"}:{},...i,...i["aria-label"]?{"aria-label":`${i["aria-label"]}, Opens in a new window`}:{},ref:c,href:y,onClick:x,className:s()(d.externalLink,i.className),children:[i.children,n&&(0,r.jsx)(h,{}),(0,r.jsx)(l.A,{component:"span",className:d.visuallyHidden,children:", Opens in a new window"})]}):(0,r.jsx)(o.A,{...i,ref:c,component:p.Link,to:y,onClick:x})})},39330:(e,t,n)=>{var r=n(4293),a=n(78920);t.A=void 0;var o=a(n(95478)),i=(0,r(n(74044)).default)(o.createElement("path",{d:"M19 19H5V5h7V3H5c-1.11 0-2 .9-2 2v14c0 1.1.89 2 2 2h14c1.1 0 2-.9 2-2v-7h-2v7zM14 3v2h3.59l-9.83 9.83 1.41 1.41L19 6.41V10h2V3h-7z"}),"OpenInNew");t.A=i},84441:(e,t,n)=>{n.d(t,{A:()=>w});var r=n(39850),a=n(89575),o=n(95478);function i(e){var t,n,r="";if("string"==typeof e||"number"==typeof e)r+=e;else if("object"==typeof e)if(Array.isArray(e))for(t=0;t<e.length;t++)e[t]&&(n=i(e[t]))&&(r&&(r+=" "),r+=n);else for(t in e)e[t]&&(r&&(r+=" "),r+=t);return r}const l=function(){for(var e,t,n=0,r="";n<arguments.length;)(e=arguments[n++])&&(t=i(e))&&(r&&(r+=" "),r+=t);return r};var c=n(37976),s=n(4321),u=n(38483);const d=(0,u.A)(o.createElement("path",{d:"M20,12A8,8 0 0,1 12,20A8,8 0 0,1 4,12A8,8 0 0,1 12,4C12.76,4 13.5,4.11 14.2, 4.31L15.77,2.74C14.61,2.26 13.34,2 12,2A10,10 0 0,0 2,12A10,10 0 0,0 12,22A10,10 0 0, 0 22,12M7.91,10.08L6.5,11.5L11,16L21,6L19.59,4.58L11,13.17L7.91,10.08Z"}),"SuccessOutlined"),p=(0,u.A)(o.createElement("path",{d:"M12 5.99L19.53 19H4.47L12 5.99M12 2L1 21h22L12 2zm1 14h-2v2h2v-2zm0-6h-2v4h2v-4z"}),"ReportProblemOutlined"),f=(0,u.A)(o.createElement("path",{d:"M11 15h2v2h-2zm0-8h2v6h-2zm.99-5C6.47 2 2 6.48 2 12s4.47 10 9.99 10C17.52 22 22 17.52 22 12S17.52 2 11.99 2zM12 20c-4.42 0-8-3.58-8-8s3.58-8 8-8 8 3.58 8 8-3.58 8-8 8z"}),"ErrorOutline"),m=(0,u.A)(o.createElement("path",{d:"M11,9H13V7H11M12,20C7.59,20 4,16.41 4,12C4,7.59 7.59,4 12,4C16.41,4 20,7.59 20, 12C20,16.41 16.41,20 12,20M12,2A10,10 0 0,0 2,12A10,10 0 0,0 12,22A10,10 0 0,0 22,12A10, 10 0 0,0 12,2M11,17H13V11H11V17Z"}),"InfoOutlined"),h=(0,u.A)(o.createElement("path",{d:"M19 6.41L17.59 5 12 10.59 6.41 5 5 6.41 10.59 12 5 17.59 6.41 19 12 13.41 17.59 19 19 17.59 13.41 12z"}),"Close");var g=n(29365),v=n(11192),y={success:o.createElement(d,{fontSize:"inherit"}),warning:o.createElement(p,{fontSize:"inherit"}),error:o.createElement(f,{fontSize:"inherit"}),info:o.createElement(m,{fontSize:"inherit"})},A=o.createElement(h,{fontSize:"small"}),b=o.forwardRef(function(e,t){var n=e.action,i=e.children,c=e.classes,u=e.className,d=e.closeText,p=void 0===d?"Close":d,f=e.color,m=e.icon,h=e.iconMapping,b=void 0===h?y:h,w=e.onClose,k=e.role,x=void 0===k?"alert":k,E=e.severity,C=void 0===E?"success":E,L=e.variant,S=void 0===L?"standard":L,j=(0,r.A)(e,["action","children","classes","className","closeText","color","icon","iconMapping","onClose","role","severity","variant"]);return o.createElement(s.A,(0,a.A)({role:x,square:!0,elevation:0,className:l(c.root,c["".concat(S).concat((0,v.A)(f||C))],u),ref:t},j),!1!==m?o.createElement("div",{className:c.icon},m||b[C]||y[C]):null,o.createElement("div",{className:c.message},i),null!=n?o.createElement("div",{className:c.action},n):null,null==n&&w?o.createElement("div",{className:c.action},o.createElement(g.A,{size:"small","aria-label":p,title:p,color:"inherit",onClick:w},A)):null)});const w=(0,c.withStyles)(function(e){var t="light"===e.palette.type?c.darken:c.lighten,n="light"===e.palette.type?c.lighten:c.darken;return{root:(0,a.A)({},e.typography.body2,{borderRadius:e.shape.borderRadius,backgroundColor:"transparent",display:"flex",padding:"6px 16px"}),standardSuccess:{color:t(e.palette.success.main,.6),backgroundColor:n(e.palette.success.main,.9),"& $icon":{color:e.palette.success.main}},standardInfo:{color:t(e.palette.info.main,.6),backgroundColor:n(e.palette.info.main,.9),"& $icon":{color:e.palette.info.main}},standardWarning:{color:t(e.palette.warning.main,.6),backgroundColor:n(e.palette.warning.main,.9),"& $icon":{color:e.palette.warning.main}},standardError:{color:t(e.palette.error.main,.6),backgroundColor:n(e.palette.error.main,.9),"& $icon":{color:e.palette.error.main}},outlinedSuccess:{color:t(e.palette.success.main,.6),border:"1px solid ".concat(e.palette.success.main),"& $icon":{color:e.palette.success.main}},outlinedInfo:{color:t(e.palette.info.main,.6),border:"1px solid ".concat(e.palette.info.main),"& $icon":{color:e.palette.info.main}},outlinedWarning:{color:t(e.palette.warning.main,.6),border:"1px solid ".concat(e.palette.warning.main),"& $icon":{color:e.palette.warning.main}},outlinedError:{color:t(e.palette.error.main,.6),border:"1px solid ".concat(e.palette.error.main),"& $icon":{color:e.palette.error.main}},filledSuccess:{color:"#fff",fontWeight:e.typography.fontWeightMedium,backgroundColor:e.palette.success.main},filledInfo:{color:"#fff",fontWeight:e.typography.fontWeightMedium,backgroundColor:e.palette.info.main},filledWarning:{color:"#fff",fontWeight:e.typography.fontWeightMedium,backgroundColor:e.palette.warning.main},filledError:{color:"#fff",fontWeight:e.typography.fontWeightMedium,backgroundColor:e.palette.error.main},icon:{marginRight:12,padding:"7px 0",display:"flex",fontSize:22,opacity:.9},message:{padding:"8px 0"},action:{display:"flex",alignItems:"center",marginLeft:"auto",paddingLeft:16,marginRight:-8}}},{name:"MuiAlert"})(b)},91638:(e,t,n)=>{var r=n(85608),a=n(95478),o=r.__importDefault(n(5030));t.A=function(e,t){void 0===t&&(t=[]);var n=o.default(e,t,{loading:!0}),r=n[0],i=n[1];return a.useEffect(function(){i()},[i]),r}}}]);
2
- //# sourceMappingURL=6371.d45f37cc.chunk.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"static/6371.d45f37cc.chunk.js","mappings":"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","sources":["webpack://internal.plugin-kuadrant/../../node_modules/react-use/lib/useAsyncFn.js","webpack://internal.plugin-kuadrant/../../node_modules/react-use/lib/useMountedState.js","webpack://internal.plugin-kuadrant/./node_modules/@backstage/plugin-catalog-react/dist/hooks/useEntity.esm.js","webpack://internal.plugin-kuadrant/./src/components/ApiProductOpenApiAlert/ApiProductOpenApiAlert.tsx","webpack://internal.plugin-kuadrant/./node_modules/@backstage/core-components/dist/components/Link/Link.esm.js","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/icons/OpenInNew.js","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/lab/node_modules/clsx/dist/clsx.m.js","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/lab/esm/internal/svg-icons/SuccessOutlined.js","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/lab/esm/internal/svg-icons/ReportProblemOutlined.js","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/lab/esm/internal/svg-icons/ErrorOutline.js","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/lab/esm/internal/svg-icons/InfoOutlined.js","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/lab/esm/internal/svg-icons/Close.js","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/lab/esm/Alert/Alert.js","webpack://internal.plugin-kuadrant/../../node_modules/react-use/lib/useAsync.js"],"sourcesContent":["\"use strict\";\nObject.defineProperty(exports, \"__esModule\", { value: true });\nvar tslib_1 = require(\"tslib\");\nvar react_1 = require(\"react\");\nvar useMountedState_1 = tslib_1.__importDefault(require(\"./useMountedState\"));\nfunction useAsyncFn(fn, deps, initialState) {\n if (deps === void 0) { deps = []; }\n if (initialState === void 0) { initialState = { loading: false }; }\n var lastCallId = react_1.useRef(0);\n var isMounted = useMountedState_1.default();\n var _a = react_1.useState(initialState), state = _a[0], set = _a[1];\n var callback = react_1.useCallback(function () {\n var args = [];\n for (var _i = 0; _i < arguments.length; _i++) {\n args[_i] = arguments[_i];\n }\n var callId = ++lastCallId.current;\n if (!state.loading) {\n set(function (prevState) { return (tslib_1.__assign(tslib_1.__assign({}, prevState), { loading: true })); });\n }\n return fn.apply(void 0, args).then(function (value) {\n isMounted() && callId === lastCallId.current && set({ value: value, loading: false });\n return value;\n }, function (error) {\n isMounted() && callId === lastCallId.current && set({ error: error, loading: false });\n return error;\n });\n }, deps);\n return [state, callback];\n}\nexports.default = useAsyncFn;\n","\"use strict\";\nObject.defineProperty(exports, \"__esModule\", { value: true });\nvar react_1 = require(\"react\");\nfunction useMountedState() {\n var mountedRef = react_1.useRef(false);\n var get = react_1.useCallback(function () { return mountedRef.current; }, []);\n react_1.useEffect(function () {\n mountedRef.current = true;\n return function () {\n mountedRef.current = false;\n };\n }, []);\n return get;\n}\nexports.default = useMountedState;\n","import { jsx } from 'react/jsx-runtime';\nimport { stringifyEntityRef } from '@backstage/catalog-model';\nimport { AnalyticsContext } from '@backstage/core-plugin-api';\nimport { createVersionedContext, useVersionedContext, createVersionedValueMap } from '@backstage/version-bridge';\n\nconst NewEntityContext = createVersionedContext(\n \"entity-context\"\n);\nconst AsyncEntityProvider = (props) => {\n const { children, entity, loading, error, refresh } = props;\n const value = { entity, loading, error, refresh };\n return /* @__PURE__ */ jsx(NewEntityContext.Provider, { value: createVersionedValueMap({ 1: value }), children: /* @__PURE__ */ jsx(\n AnalyticsContext,\n {\n attributes: {\n ...entity ? { entityRef: stringifyEntityRef(entity) } : void 0\n },\n children\n }\n ) });\n};\nconst EntityProvider = (props) => /* @__PURE__ */ jsx(\n AsyncEntityProvider,\n {\n entity: props.entity,\n loading: !Boolean(props.entity),\n error: void 0,\n refresh: void 0,\n children: props.children\n }\n);\nfunction useEntity() {\n const versionedHolder = useVersionedContext(\n \"entity-context\"\n );\n if (!versionedHolder) {\n throw new Error(\"Entity context is not available\");\n }\n const value = versionedHolder.atVersion(1);\n if (!value) {\n throw new Error(\"EntityContext v1 not available\");\n }\n if (!value.entity) {\n throw new Error(\n \"useEntity hook is being called outside of an EntityLayout where the entity has not been loaded. If this is intentional, please use useAsyncEntity instead.\"\n );\n }\n return { entity: value.entity };\n}\nfunction useAsyncEntity() {\n const versionedHolder = useVersionedContext(\n \"entity-context\"\n );\n if (!versionedHolder) {\n throw new Error(\"Entity context is not available\");\n }\n const value = versionedHolder.atVersion(1);\n if (!value) {\n throw new Error(\"EntityContext v1 not available\");\n }\n const { entity, loading, error, refresh } = value;\n return { entity, loading, error, refresh };\n}\n\nexport { AsyncEntityProvider, EntityProvider, useAsyncEntity, useEntity };\n//# sourceMappingURL=useEntity.esm.js.map\n","import React from \"react\";\nimport { useEntity } from \"@backstage/plugin-catalog-react\";\nimport { useApi, configApiRef, fetchApiRef } from \"@backstage/core-plugin-api\";\nimport { Link } from \"@backstage/core-components\";\nimport { Box, Typography } from \"@material-ui/core\";\nimport { Alert } from \"@material-ui/lab\";\nimport useAsync from \"react-use/lib/useAsync\";\n\n// Displays alerts for OpenAPI spec issues.\nexport const ApiProductOpenApiAlert = () => {\n const { entity } = useEntity();\n const config = useApi(configApiRef);\n const fetchApi = useApi(fetchApiRef);\n const backendUrl = config.getString(\"backend.baseUrl\");\n\n // Get APIProduct reference from entity annotations\n const namespace = entity.metadata.annotations?.[\"kuadrant.io/namespace\"];\n const apiProductName =\n entity.metadata.annotations?.[\"kuadrant.io/apiproduct\"];\n\n // Fetch the full APIProduct resource to check status conditions\n const {value: apiProduct,loading,error,} = useAsync(async () => {\n if (!namespace || !apiProductName) {\n return null;\n }\n\n const response = await fetchApi.fetch(\n `${backendUrl}/api/kuadrant/apiproducts/${namespace}/${apiProductName}`,\n );\n\n if (!response.ok) {\n return null;\n }\n\n return await response.json();\n }, [backendUrl, fetchApi, namespace, apiProductName]);\n\n // Don't render anything if data is missing or still loading\n if (!namespace || !apiProductName || loading || error || !apiProduct) {\n return null;\n }\n\n const { spec, status } = apiProduct;\n\n const openAPICondition = status?.conditions?.find(\n (c: any) => c.type === \"OpenAPISpecReady\" && c.status === \"False\",\n );\n\n if (!openAPICondition) {\n return null;\n }\n\n return (\n <Box mb={2}>\n <Alert severity=\"warning\">\n <Typography variant=\"body2\" gutterBottom>\n <strong>OpenAPI Spec Issue</strong>\n </Typography>\n <Typography variant=\"body2\" gutterBottom>\n {openAPICondition.message}\n </Typography>\n {spec.documentation?.openAPISpecURL && (\n <Typography variant=\"body2\">\n Spec URL:{\" \"}\n <Link to={spec.documentation.openAPISpecURL} target=\"_blank\">\n {spec.documentation.openAPISpecURL}\n </Link>\n </Typography>\n )}\n </Alert>\n </Box>\n );\n};\n","import { jsxs, jsx } from 'react/jsx-runtime';\nimport { useAnalytics, useApp, useApi, configApiRef } from '@backstage/core-plugin-api';\nimport MaterialLink from '@material-ui/core/Link';\nimport { makeStyles } from '@material-ui/core/styles';\nimport Typography from '@material-ui/core/Typography';\nimport classNames from 'classnames';\nimport { trimEnd } from 'lodash';\nimport { forwardRef } from 'react';\nimport { Link as Link$1, createRoutesFromChildren, Route } from 'react-router-dom';\nimport OpenInNew from '@material-ui/icons/OpenInNew';\n\nfunction isReactRouterBeta() {\n const [obj] = createRoutesFromChildren(/* @__PURE__ */ jsx(Route, { index: true, element: /* @__PURE__ */ jsx(\"div\", {}) }));\n return !obj.index;\n}\nconst useStyles = makeStyles(\n (theme) => ({\n visuallyHidden: {\n clip: \"rect(0 0 0 0)\",\n clipPath: \"inset(50%)\",\n overflow: \"hidden\",\n position: \"absolute\",\n userSelect: \"none\",\n whiteSpace: \"nowrap\",\n height: 1,\n width: 1\n },\n externalLink: {\n position: \"relative\"\n },\n externalLinkIcon: {\n verticalAlign: \"bottom\",\n marginLeft: theme.spacing(0.5)\n }\n }),\n { name: \"Link\" }\n);\nconst ExternalLinkIcon = () => {\n const app = useApp();\n const Icon = app.getSystemIcon(\"externalLink\") || OpenInNew;\n const classes = useStyles();\n return /* @__PURE__ */ jsx(Icon, { className: classes.externalLinkIcon });\n};\nconst isExternalUri = (uri) => /^([a-z+.-]+):/.test(uri);\nconst scriptProtocolPattern = (\n // eslint-disable-next-line no-control-regex\n /^[\\u0000-\\u001F ]*j[\\r\\n\\t]*a[\\r\\n\\t]*v[\\r\\n\\t]*a[\\r\\n\\t]*s[\\r\\n\\t]*c[\\r\\n\\t]*r[\\r\\n\\t]*i[\\r\\n\\t]*p[\\r\\n\\t]*t[\\r\\n\\t]*\\:/i\n);\nconst originalWindowOpen = window.open;\nif (originalWindowOpen && !originalWindowOpen.__backstage) {\n const newOpen = function open(...args) {\n const url = String(args[0]);\n if (scriptProtocolPattern.test(url)) {\n throw new Error(\n \"Rejected window.open() with a javascript: URL as a security precaution\"\n );\n }\n return originalWindowOpen.apply(this, args);\n };\n newOpen.__backstage = true;\n window.open = newOpen;\n}\nconst useBaseUrl = () => {\n try {\n const config = useApi(configApiRef);\n return config.getOptionalString(\"app.baseUrl\");\n } catch {\n return void 0;\n }\n};\nconst useBasePath = () => {\n const base = \"http://sample.dev\";\n const url = useBaseUrl() ?? \"/\";\n const { pathname } = new URL(url, base);\n return trimEnd(pathname, \"/\");\n};\nconst useResolvedPath = (uri) => {\n let resolvedPath = String(uri);\n const basePath = useBasePath();\n const external = isExternalUri(resolvedPath);\n const startsWithBasePath = resolvedPath.startsWith(basePath);\n if (!external && !startsWithBasePath) {\n resolvedPath = basePath.concat(resolvedPath);\n }\n return resolvedPath;\n};\nconst getNodeText = (node) => {\n if (node instanceof Array) {\n return node.map(getNodeText).join(\" \").trim();\n }\n if (typeof node === \"object\" && node) {\n return getNodeText(node?.props?.children);\n }\n if ([\"string\", \"number\"].includes(typeof node)) {\n return String(node);\n }\n return \"\";\n};\nconst Link = forwardRef(\n ({ onClick, noTrack, externalLinkIcon, ...props }, ref) => {\n const classes = useStyles();\n const analytics = useAnalytics();\n const to = isReactRouterBeta() ? useResolvedPath(props.to) : props.to;\n const linkText = getNodeText(props.children) || to;\n const external = isExternalUri(to);\n const newWindow = external && !!/^https?:/.exec(to);\n if (scriptProtocolPattern.test(to)) {\n throw new Error(\n \"Link component rejected javascript: URL as a security precaution\"\n );\n }\n const handleClick = (event) => {\n onClick?.(event);\n if (!noTrack) {\n analytics.captureEvent(\"click\", linkText, { attributes: { to } });\n }\n };\n return external ? (\n // External links\n /* @__PURE__ */ jsxs(\n MaterialLink,\n {\n ...newWindow ? { target: \"_blank\", rel: \"noopener\" } : {},\n ...props,\n ...props[\"aria-label\"] ? { \"aria-label\": `${props[\"aria-label\"]}, Opens in a new window` } : {},\n ref,\n href: to,\n onClick: handleClick,\n className: classNames(classes.externalLink, props.className),\n children: [\n props.children,\n externalLinkIcon && /* @__PURE__ */ jsx(ExternalLinkIcon, {}),\n /* @__PURE__ */ jsx(Typography, { component: \"span\", className: classes.visuallyHidden, children: \", Opens in a new window\" })\n ]\n }\n )\n ) : (\n // Interact with React Router for internal links\n /* @__PURE__ */ jsx(\n MaterialLink,\n {\n ...props,\n ref,\n component: Link$1,\n to,\n onClick: handleClick\n }\n )\n );\n }\n);\n\nexport { Link, isExternalUri, isReactRouterBeta, useResolvedPath };\n//# sourceMappingURL=Link.esm.js.map\n","\"use strict\";\n\nvar _interopRequireDefault = require(\"@babel/runtime/helpers/interopRequireDefault\");\n\nvar _interopRequireWildcard = require(\"@babel/runtime/helpers/interopRequireWildcard\");\n\nObject.defineProperty(exports, \"__esModule\", {\n value: true\n});\nexports.default = void 0;\n\nvar React = _interopRequireWildcard(require(\"react\"));\n\nvar _createSvgIcon = _interopRequireDefault(require(\"./utils/createSvgIcon\"));\n\nvar _default = (0, _createSvgIcon.default)( /*#__PURE__*/React.createElement(\"path\", {\n d: \"M19 19H5V5h7V3H5c-1.11 0-2 .9-2 2v14c0 1.1.89 2 2 2h14c1.1 0 2-.9 2-2v-7h-2v7zM14 3v2h3.59l-9.83 9.83 1.41 1.41L19 6.41V10h2V3h-7z\"\n}), 'OpenInNew');\n\nexports.default = _default;","function r(e){var t,f,n=\"\";if(\"string\"==typeof e||\"number\"==typeof e)n+=e;else if(\"object\"==typeof e)if(Array.isArray(e))for(t=0;t<e.length;t++)e[t]&&(f=r(e[t]))&&(n&&(n+=\" \"),n+=f);else for(t in e)e[t]&&(n&&(n+=\" \"),n+=t);return n}export function clsx(){for(var e,t,f=0,n=\"\";f<arguments.length;)(e=arguments[f++])&&(t=r(e))&&(n&&(n+=\" \"),n+=t);return n}export default clsx;","import * as React from 'react';\nimport { createSvgIcon } from '@material-ui/core/utils';\n/**\n * @ignore - internal component.\n */\n\nexport default createSvgIcon( /*#__PURE__*/React.createElement(\"path\", {\n d: \"M20,12A8,8 0 0,1 12,20A8,8 0 0,1 4,12A8,8 0 0,1 12,4C12.76,4 13.5,4.11 14.2, 4.31L15.77,2.74C14.61,2.26 13.34,2 12,2A10,10 0 0,0 2,12A10,10 0 0,0 12,22A10,10 0 0, 0 22,12M7.91,10.08L6.5,11.5L11,16L21,6L19.59,4.58L11,13.17L7.91,10.08Z\"\n}), 'SuccessOutlined');","import * as React from 'react';\nimport { createSvgIcon } from '@material-ui/core/utils';\n/**\n * @ignore - internal component.\n */\n\nexport default createSvgIcon( /*#__PURE__*/React.createElement(\"path\", {\n d: \"M12 5.99L19.53 19H4.47L12 5.99M12 2L1 21h22L12 2zm1 14h-2v2h2v-2zm0-6h-2v4h2v-4z\"\n}), 'ReportProblemOutlined');","import * as React from 'react';\nimport { createSvgIcon } from '@material-ui/core/utils';\n/**\n * @ignore - internal component.\n */\n\nexport default createSvgIcon( /*#__PURE__*/React.createElement(\"path\", {\n d: \"M11 15h2v2h-2zm0-8h2v6h-2zm.99-5C6.47 2 2 6.48 2 12s4.47 10 9.99 10C17.52 22 22 17.52 22 12S17.52 2 11.99 2zM12 20c-4.42 0-8-3.58-8-8s3.58-8 8-8 8 3.58 8 8-3.58 8-8 8z\"\n}), 'ErrorOutline');","import * as React from 'react';\nimport { createSvgIcon } from '@material-ui/core/utils';\n/**\n * @ignore - internal component.\n */\n\nexport default createSvgIcon( /*#__PURE__*/React.createElement(\"path\", {\n d: \"M11,9H13V7H11M12,20C7.59,20 4,16.41 4,12C4,7.59 7.59,4 12,4C16.41,4 20,7.59 20, 12C20,16.41 16.41,20 12,20M12,2A10,10 0 0,0 2,12A10,10 0 0,0 12,22A10,10 0 0,0 22,12A10, 10 0 0,0 12,2M11,17H13V11H11V17Z\"\n}), 'InfoOutlined');","import * as React from 'react';\nimport { createSvgIcon } from '@material-ui/core/utils';\n/**\n * @ignore - internal component.\n */\n\nexport default createSvgIcon( /*#__PURE__*/React.createElement(\"path\", {\n d: \"M19 6.41L17.59 5 12 10.59 6.41 5 5 6.41 10.59 12 5 17.59 6.41 19 12 13.41 17.59 19 19 17.59 13.41 12z\"\n}), 'Close');","import _objectWithoutProperties from \"@babel/runtime/helpers/esm/objectWithoutProperties\";\nimport _extends from \"@babel/runtime/helpers/esm/extends\";\nimport * as React from 'react';\nimport PropTypes from 'prop-types';\nimport clsx from 'clsx';\nimport { withStyles, lighten, darken } from '@material-ui/core/styles';\nimport Paper from '@material-ui/core/Paper';\nimport SuccessOutlinedIcon from '../internal/svg-icons/SuccessOutlined';\nimport ReportProblemOutlinedIcon from '../internal/svg-icons/ReportProblemOutlined';\nimport ErrorOutlineIcon from '../internal/svg-icons/ErrorOutline';\nimport InfoOutlinedIcon from '../internal/svg-icons/InfoOutlined';\nimport CloseIcon from '../internal/svg-icons/Close';\nimport IconButton from '@material-ui/core/IconButton';\nimport { capitalize } from '@material-ui/core/utils';\nexport var styles = function styles(theme) {\n var getColor = theme.palette.type === 'light' ? darken : lighten;\n var getBackgroundColor = theme.palette.type === 'light' ? lighten : darken;\n return {\n /* Styles applied to the root element. */\n root: _extends({}, theme.typography.body2, {\n borderRadius: theme.shape.borderRadius,\n backgroundColor: 'transparent',\n display: 'flex',\n padding: '6px 16px'\n }),\n\n /* Styles applied to the root element if `variant=\"standard\"` and `color=\"success\"`. */\n standardSuccess: {\n color: getColor(theme.palette.success.main, 0.6),\n backgroundColor: getBackgroundColor(theme.palette.success.main, 0.9),\n '& $icon': {\n color: theme.palette.success.main\n }\n },\n\n /* Styles applied to the root element if `variant=\"standard\"` and `color=\"info\"`. */\n standardInfo: {\n color: getColor(theme.palette.info.main, 0.6),\n backgroundColor: getBackgroundColor(theme.palette.info.main, 0.9),\n '& $icon': {\n color: theme.palette.info.main\n }\n },\n\n /* Styles applied to the root element if `variant=\"standard\"` and `color=\"warning\"`. */\n standardWarning: {\n color: getColor(theme.palette.warning.main, 0.6),\n backgroundColor: getBackgroundColor(theme.palette.warning.main, 0.9),\n '& $icon': {\n color: theme.palette.warning.main\n }\n },\n\n /* Styles applied to the root element if `variant=\"standard\"` and `color=\"error\"`. */\n standardError: {\n color: getColor(theme.palette.error.main, 0.6),\n backgroundColor: getBackgroundColor(theme.palette.error.main, 0.9),\n '& $icon': {\n color: theme.palette.error.main\n }\n },\n\n /* Styles applied to the root element if `variant=\"outlined\"` and `color=\"success\"`. */\n outlinedSuccess: {\n color: getColor(theme.palette.success.main, 0.6),\n border: \"1px solid \".concat(theme.palette.success.main),\n '& $icon': {\n color: theme.palette.success.main\n }\n },\n\n /* Styles applied to the root element if `variant=\"outlined\"` and `color=\"info\"`. */\n outlinedInfo: {\n color: getColor(theme.palette.info.main, 0.6),\n border: \"1px solid \".concat(theme.palette.info.main),\n '& $icon': {\n color: theme.palette.info.main\n }\n },\n\n /* Styles applied to the root element if `variant=\"outlined\"` and `color=\"warning\"`. */\n outlinedWarning: {\n color: getColor(theme.palette.warning.main, 0.6),\n border: \"1px solid \".concat(theme.palette.warning.main),\n '& $icon': {\n color: theme.palette.warning.main\n }\n },\n\n /* Styles applied to the root element if `variant=\"outlined\"` and `color=\"error\"`. */\n outlinedError: {\n color: getColor(theme.palette.error.main, 0.6),\n border: \"1px solid \".concat(theme.palette.error.main),\n '& $icon': {\n color: theme.palette.error.main\n }\n },\n\n /* Styles applied to the root element if `variant=\"filled\"` and `color=\"success\"`. */\n filledSuccess: {\n color: '#fff',\n fontWeight: theme.typography.fontWeightMedium,\n backgroundColor: theme.palette.success.main\n },\n\n /* Styles applied to the root element if `variant=\"filled\"` and `color=\"info\"`. */\n filledInfo: {\n color: '#fff',\n fontWeight: theme.typography.fontWeightMedium,\n backgroundColor: theme.palette.info.main\n },\n\n /* Styles applied to the root element if `variant=\"filled\"` and `color=\"warning\"`. */\n filledWarning: {\n color: '#fff',\n fontWeight: theme.typography.fontWeightMedium,\n backgroundColor: theme.palette.warning.main\n },\n\n /* Styles applied to the root element if `variant=\"filled\"` and `color=\"error\"`. */\n filledError: {\n color: '#fff',\n fontWeight: theme.typography.fontWeightMedium,\n backgroundColor: theme.palette.error.main\n },\n\n /* Styles applied to the icon wrapper element. */\n icon: {\n marginRight: 12,\n padding: '7px 0',\n display: 'flex',\n fontSize: 22,\n opacity: 0.9\n },\n\n /* Styles applied to the message wrapper element. */\n message: {\n padding: '8px 0'\n },\n\n /* Styles applied to the action wrapper element if `action` is provided. */\n action: {\n display: 'flex',\n alignItems: 'center',\n marginLeft: 'auto',\n paddingLeft: 16,\n marginRight: -8\n }\n };\n};\nvar defaultIconMapping = {\n success: /*#__PURE__*/React.createElement(SuccessOutlinedIcon, {\n fontSize: \"inherit\"\n }),\n warning: /*#__PURE__*/React.createElement(ReportProblemOutlinedIcon, {\n fontSize: \"inherit\"\n }),\n error: /*#__PURE__*/React.createElement(ErrorOutlineIcon, {\n fontSize: \"inherit\"\n }),\n info: /*#__PURE__*/React.createElement(InfoOutlinedIcon, {\n fontSize: \"inherit\"\n })\n};\n\nvar _ref = /*#__PURE__*/React.createElement(CloseIcon, {\n fontSize: \"small\"\n});\n\nvar Alert = /*#__PURE__*/React.forwardRef(function Alert(props, ref) {\n var action = props.action,\n children = props.children,\n classes = props.classes,\n className = props.className,\n _props$closeText = props.closeText,\n closeText = _props$closeText === void 0 ? 'Close' : _props$closeText,\n color = props.color,\n icon = props.icon,\n _props$iconMapping = props.iconMapping,\n iconMapping = _props$iconMapping === void 0 ? defaultIconMapping : _props$iconMapping,\n onClose = props.onClose,\n _props$role = props.role,\n role = _props$role === void 0 ? 'alert' : _props$role,\n _props$severity = props.severity,\n severity = _props$severity === void 0 ? 'success' : _props$severity,\n _props$variant = props.variant,\n variant = _props$variant === void 0 ? 'standard' : _props$variant,\n other = _objectWithoutProperties(props, [\"action\", \"children\", \"classes\", \"className\", \"closeText\", \"color\", \"icon\", \"iconMapping\", \"onClose\", \"role\", \"severity\", \"variant\"]);\n\n return /*#__PURE__*/React.createElement(Paper, _extends({\n role: role,\n square: true,\n elevation: 0,\n className: clsx(classes.root, classes[\"\".concat(variant).concat(capitalize(color || severity))], className),\n ref: ref\n }, other), icon !== false ? /*#__PURE__*/React.createElement(\"div\", {\n className: classes.icon\n }, icon || iconMapping[severity] || defaultIconMapping[severity]) : null, /*#__PURE__*/React.createElement(\"div\", {\n className: classes.message\n }, children), action != null ? /*#__PURE__*/React.createElement(\"div\", {\n className: classes.action\n }, action) : null, action == null && onClose ? /*#__PURE__*/React.createElement(\"div\", {\n className: classes.action\n }, /*#__PURE__*/React.createElement(IconButton, {\n size: \"small\",\n \"aria-label\": closeText,\n title: closeText,\n color: \"inherit\",\n onClick: onClose\n }, _ref)) : null);\n});\nprocess.env.NODE_ENV !== \"production\" ? Alert.propTypes = {\n // ----------------------------- Warning --------------------------------\n // | These PropTypes are generated from the TypeScript type definitions |\n // | To update them edit the d.ts file and run \"yarn proptypes\" |\n // ----------------------------------------------------------------------\n\n /**\n * The action to display. It renders after the message, at the end of the alert.\n */\n action: PropTypes.node,\n\n /**\n * The content of the component.\n */\n children: PropTypes.node,\n\n /**\n * Override or extend the styles applied to the component.\n * See [CSS API](#css) below for more details.\n */\n classes: PropTypes.object,\n\n /**\n * @ignore\n */\n className: PropTypes.string,\n\n /**\n * Override the default label for the *close popup* icon button.\n *\n * For localization purposes, you can use the provided [translations](/guides/localization/).\n */\n closeText: PropTypes.string,\n\n /**\n * The main color for the alert. Unless provided, the value is taken from the `severity` prop.\n */\n color: PropTypes.oneOf(['error', 'info', 'success', 'warning']),\n\n /**\n * Override the icon displayed before the children.\n * Unless provided, the icon is mapped to the value of the `severity` prop.\n */\n icon: PropTypes.node,\n\n /**\n * The component maps the `severity` prop to a range of different icons,\n * for instance success to `<SuccessOutlined>`.\n * If you wish to change this mapping, you can provide your own.\n * Alternatively, you can use the `icon` prop to override the icon displayed.\n */\n iconMapping: PropTypes.shape({\n error: PropTypes.node,\n info: PropTypes.node,\n success: PropTypes.node,\n warning: PropTypes.node\n }),\n\n /**\n * Callback fired when the component requests to be closed.\n * When provided and no `action` prop is set, a close icon button is displayed that triggers the callback when clicked.\n *\n * @param {object} event The event source of the callback.\n */\n onClose: PropTypes.func,\n\n /**\n * The ARIA role attribute of the element.\n */\n role: PropTypes.string,\n\n /**\n * The severity of the alert. This defines the color and icon used.\n */\n severity: PropTypes.oneOf(['error', 'info', 'success', 'warning']),\n\n /**\n * The variant to use.\n */\n variant: PropTypes.oneOf(['filled', 'outlined', 'standard'])\n} : void 0;\nexport default withStyles(styles, {\n name: 'MuiAlert'\n})(Alert);","\"use strict\";\nObject.defineProperty(exports, \"__esModule\", { value: true });\nvar tslib_1 = require(\"tslib\");\nvar react_1 = require(\"react\");\nvar useAsyncFn_1 = tslib_1.__importDefault(require(\"./useAsyncFn\"));\nfunction useAsync(fn, deps) {\n if (deps === void 0) { deps = []; }\n var _a = useAsyncFn_1.default(fn, deps, {\n loading: true,\n }), state = _a[0], callback = _a[1];\n react_1.useEffect(function () {\n callback();\n }, [callback]);\n return state;\n}\nexports.default = useAsync;\n"],"names":["Object","defineProperty","exports","value","tslib_1","react_1","useMountedState_1","__importDefault","fn","deps","initialState","loading","lastCallId","useRef","isMounted","default","_a","useState","state","set","callback","useCallback","args","_i","arguments","length","callId","current","prevState","__assign","apply","then","error","mountedRef","get","useEffect","useEntity","versionedHolder","useVersionedContext","Error","atVersion","entity","createVersionedContext","ApiProductOpenApiAlert","status","spec","config","useApi","configApiRef","fetchApi","fetchApiRef","backendUrl","getString","namespace","metadata","annotations","apiProductName","apiProduct","useAsync","async","response","fetch","ok","json","openAPICondition","conditions","find","c","type","Box","mb","Alert","severity","Typography","variant","gutterBottom","strong","message","documentation","openAPISpecURL","Link","to","target","useStyles","makeStyles","theme","visuallyHidden","clip","clipPath","overflow","position","userSelect","whiteSpace","height","width","externalLink","externalLinkIcon","verticalAlign","marginLeft","spacing","name","ExternalLinkIcon","Icon","useApp","getSystemIcon","classes","jsx","className","isExternalUri","uri","test","scriptProtocolPattern","originalWindowOpen","window","open","__backstage","newOpen","url","String","this","getNodeText","node","Array","map","join","trim","props","children","includes","forwardRef","onClick","noTrack","ref","analytics","useAnalytics","obj","createRoutesFromChildren","Route","index","element","isReactRouterBeta","resolvedPath","basePath","getOptionalString","useBaseUrl","pathname","URL","trimEnd","useBasePath","external","startsWithBasePath","startsWith","concat","useResolvedPath","linkText","newWindow","exec","handleClick","event","captureEvent","attributes","jsxs","rel","href","component","_interopRequireDefault","_interopRequireWildcard","React","_default","createElement","d","r","e","t","f","n","isArray","createSvgIcon","defaultIconMapping","success","SuccessOutlined","fontSize","warning","ReportProblemOutlined","ErrorOutline","info","InfoOutlined","_ref","Close","action","_props$closeText","closeText","color","icon","_props$iconMapping","iconMapping","onClose","_props$role","role","_props$severity","_props$variant","other","Paper","square","elevation","root","capitalize","IconButton","size","title","withStyles","getColor","palette","darken","lighten","getBackgroundColor","typography","body2","borderRadius","shape","backgroundColor","display","padding","standardSuccess","main","standardInfo","standardWarning","standardError","outlinedSuccess","border","outlinedInfo","outlinedWarning","outlinedError","filledSuccess","fontWeight","fontWeightMedium","filledInfo","filledWarning","filledError","marginRight","opacity","alignItems","paddingLeft","useAsyncFn_1"],"sourceRoot":""}
@@ -1,2 +0,0 @@
1
- "use strict";(self.webpackChunkinternal_plugin_kuadrant=self.webpackChunkinternal_plugin_kuadrant||[]).push([[6800],{25467:(e,t,a)=>{a.d(t,{tN:()=>n}),a(31085),a(22097);var r=a(15427);function n(){const e=(0,r.useVersionedContext)("entity-context");if(!e)throw new Error("Entity context is not available");const t=e.atVersion(1);if(!t)throw new Error("EntityContext v1 not available");if(!t.entity)throw new Error("useEntity hook is being called outside of an EntityLayout where the entity has not been loaded. If this is intentional, please use useAsyncEntity instead.");return{entity:t.entity}}(0,r.createVersionedContext)("entity-context")},34955:(e,t,a)=>{a.d(t,{Al:()=>o,EM:()=>c,FL:()=>i,J:()=>n,KV:()=>g,R_:()=>l,U3:()=>s,dp:()=>p,jH:()=>m,q0:()=>f,uL:()=>h,v_:()=>d,vs:()=>u,z4:()=>v});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"}}),i=(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"}})),s=(0,r.i)({name:"kuadrant.apiproduct.update.own",attributes:{action:"update"}}),d=(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"}}),u=(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"}})),v=(0,r.i)({name:"kuadrant.apikey.update.all",attributes:{action:"update"}}),h=(0,r.i)({name:"kuadrant.apikey.delete.own",attributes:{action:"delete"}}),m=(0,r.i)({name:"kuadrant.apikey.delete.all",attributes:{action:"delete"}}),g=(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),i=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),s=(0,i.A)(),d=(0,r.useState)(a),c=d[0],l=d[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||l(function(e){return(0,n.__assign)((0,n.__assign)({},e),{loading:!0})}),e.apply(void 0,t).then(function(e){return s()&&r===o.current&&l({value:e,loading:!1}),e},function(e){return s()&&r===o.current&&l({error:e,loading:!1}),e})},t);return[c,u]}(e,t,{loading:!0}),o=a[0],s=a[1];return(0,r.useEffect)(function(){s()},[s]),o}},46205:(e,t,a)=>{a.d(t,{W:()=>i,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 i(e,t,a,r){return!!r||!(!a||e!==t)}},46299:(e,t,a)=>{a.d(t,{ee:()=>i,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"}}},i=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{}}}},46800:(e,t,a)=>{a.r(t),a.d(t,{EntityApiApprovalTab:()=>$});var r=a(31085),n=a(95478),i=a(25467),o=a(22097),s=a(35015),d=a(86687),c=a(42367),l=a(25010),u=a(34955),p=a(46205),f=a(58837),v=a(10394),h=a(72501),m=a(76891),g=a(61477),y=a(46805),A=a(16249),b=a(93453),k=a(64947),x=a(78467),j=a(67720),w=a(71677),C=a(29365),q=a(55429),R=a(92399),P=a(46299),T=a(77318);const E=(0,f.A)(e=>({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"}})),I=({request:e})=>{const t=E();return(0,r.jsxs)(v.A,{className:t.useCasePanel,onClick:e=>e.stopPropagation(),children:[(0,r.jsx)(h.A,{className:t.useCaseLabel,children:"Use Case"}),(0,r.jsx)(h.A,{variant:"body2",children:e.spec.useCase||"No use case provided"})]})},S=({open:e,request:t,action:a,processing:i,onClose:o,onConfirm:s})=>{const[d,c]=(0,n.useState)(""),l=()=>{c(""),o()},u="reject"===a,p=(null==t?void 0:t.metadata.name)||"",f=d===p;return(0,r.jsxs)(m.A,{open:e,onClose:i?void 0:l,maxWidth:"sm",fullWidth:!0,children:[(0,r.jsx)(g.A,{children:u?"Reject API key":"Approve API key"}),(0,r.jsx)(y.A,{children:t&&(0,r.jsxs)(r.Fragment,{children:[(0,r.jsxs)(h.A,{variant:"body2",paragraph:!0,children:[(0,r.jsx)("strong",{children:"Requester:"})," ",t.spec.requestedBy.userId]}),(0,r.jsxs)(h.A,{variant:"body2",paragraph:!0,children:[(0,r.jsx)("strong",{children:"Tier:"})," ",t.spec.planTier]}),t.spec.useCase&&(0,r.jsxs)(h.A,{variant:"body2",paragraph:!0,children:[(0,r.jsx)("strong",{children:"Use Case:"})," ",t.spec.useCase]}),u&&(0,r.jsx)(v.A,{mt:2,p:2,bgcolor:"error.light",borderRadius:1,children:(0,r.jsx)(h.A,{variant:"body2",children:"This action will permanently deny access. The user will need to submit a new request."})}),(0,r.jsx)(v.A,{mt:2,children:(0,r.jsx)(A.A,{fullWidth:!0,label:`Type "${p}" to confirm`,value:d,onChange:e=>c(e.target.value),disabled:i,autoFocus:!0})})]})}),(0,r.jsxs)(b.A,{children:[(0,r.jsx)(k.A,{onClick:l,disabled:i,children:"Cancel"}),(0,r.jsx)(k.A,{onClick:()=>{s(d),c("")},color:u?"secondary":"primary",variant:"contained",disabled:!f||i,startIcon:i?(0,r.jsx)(x.A,{size:16,color:"inherit"}):void 0,children:i?u?"Rejecting...":"Approving...":u?"Reject":"Approve"})]})]})},$=()=>{var e,t;const{entity:a}=(0,i.tN)(),f=(0,o.useApi)(o.configApiRef),m=(0,o.useApi)(o.fetchApiRef),g=(0,o.useApi)(o.identityApiRef),y=(0,o.useApi)(o.alertApiRef),A=f.getString("backend.baseUrl"),b=(null===(e=a.metadata.annotations)||void 0===e?void 0:e["kuadrant.io/apiproduct"])||a.metadata.name,k=(null===(t=a.metadata.annotations)||void 0===t?void 0:t["kuadrant.io/namespace"])||"default",[x,E]=(0,n.useState)(0),[$,_]=(0,n.useState)({open:!1,request:null,action:"approve",processing:!1}),{allowed:z,loading:B,error:L}=(0,p.l)(u.KV),{value:N,loading:U,error:W}=(0,s.A)(async()=>{const e=(await g.getBackstageIdentity()).userEntityRef,t=await m.fetch(`${A}/api/kuadrant/requests`);if(!t.ok){const e=await(0,T.T)(t);throw new Error(`Failed to fetch requests. ${e}`)}return{requests:((await t.json()).items||[]).filter(e=>{var t;return(null===(t=e.spec.apiProductRef)||void 0===t?void 0:t.name)===b&&e.metadata.namespace===k}),reviewedBy:e}},[A,m,g,b,k,x]);if(U||B)return(0,r.jsx)(d.k,{});if(W)return(0,r.jsx)(c._,{error:W});if(L)return(0,r.jsx)(v.A,{p:2,children:(0,r.jsxs)(h.A,{color:"error",children:["Unable to check permissions: ",L.message]})});if(!z)return(0,r.jsx)(v.A,{p:3,textAlign:"center",children:(0,r.jsx)(h.A,{variant:"body1",color:"textSecondary",children:"You don't have permission to view the approval queue for this API."})});const F=(null==N?void 0:N.requests)||[],V=F.filter(e=>{var t;return!(null===(t=e.status)||void 0===t?void 0:t.phase)||"Pending"===e.status.phase}),D=F.filter(e=>{var t;return"Approved"===(null===(t=e.status)||void 0===t?void 0:t.phase)}),J=F.filter(e=>{var t;return"Rejected"===(null===(t=e.status)||void 0===t?void 0:t.phase)}),K=[{title:"Requester",field:"spec.requestedBy.userId",render:e=>(0,r.jsx)(h.A,{variant:"body2",children:e.spec.requestedBy.userId})},{title:"Status",field:"status.phase",render:e=>{var t;const a=(null===(t=e.status)||void 0===t?void 0:t.phase)||"Pending",n="Approved"===a?"Active":a;return(0,r.jsx)(j.A,{label:n,size:"small",style:(0,P.uU)(a)})}},{title:"Tier",field:"spec.planTier",render:e=>(0,r.jsx)(j.A,{label:e.spec.planTier,size:"small",variant:"outlined"})},{title:"Requested",field:"metadata.creationTimestamp",render:e=>e.metadata.creationTimestamp?(0,r.jsx)(h.A,{variant:"body2",children:new Date(e.metadata.creationTimestamp).toLocaleDateString()}):(0,r.jsx)(h.A,{variant:"body2",children:"-"})},{title:"Actions",filtering:!1,render:e=>{var t;return"Pending"!==((null===(t=e.status)||void 0===t?void 0:t.phase)||"Pending")?null:(0,r.jsxs)(v.A,{display:"flex",style:{gap:4},children:[(0,r.jsx)(w.Ay,{title:"Approve",children:(0,r.jsx)(C.A,{size:"small",onClick:()=>{_({open:!0,request:e,action:"approve",processing:!1})},children:(0,r.jsx)(q.A,{color:"primary"})})}),(0,r.jsx)(w.Ay,{title:"Reject",children:(0,r.jsx)(C.A,{size:"small",onClick:()=>{_({open:!0,request:e,action:"reject",processing:!1})},children:(0,r.jsx)(R.A,{color:"error"})})})]})}}],M=[{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)(I,{request:a}):(0,r.jsx)(v.A,{})}}];return(0,r.jsxs)(v.A,{p:2,children:[(0,r.jsx)(v.A,{mb:2,children:(0,r.jsxs)(h.A,{variant:"body2",color:"textSecondary",children:[V.length," pending, ",D.length," approved,"," ",J.length," rejected"]})}),0===F.length?(0,r.jsx)(v.A,{p:3,textAlign:"center",children:(0,r.jsx)(h.A,{variant:"body1",color:"textSecondary",children:"No API keys for this API."})}):(0,r.jsx)(l.X,{options:{paging:F.length>10,pageSize:20,search:!0,filtering:!1,debounceInterval:300,toolbar:!0,emptyRowsWhenPaging:!1},columns:K,data:F.map(e=>({...e,id:e.metadata.name})),detailPanel:M}),(0,r.jsx)(S,{open:$.open,request:$.request,action:$.action,processing:$.processing,onClose:()=>_({open:!1,request:null,action:"approve",processing:!1}),onConfirm:async()=>{if(!$.request||!N)return;_(e=>({...e,processing:!0}));const e="approve"===$.action?`${A}/api/kuadrant/requests/${$.request.metadata.namespace}/${$.request.metadata.name}/approve`:`${A}/api/kuadrant/requests/${$.request.metadata.namespace}/${$.request.metadata.name}/reject`;try{const t=await m.fetch(e,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({reviewedBy:N.reviewedBy})});if(!t.ok){const e=await(0,T.T)(t);throw new Error(e)}_({open:!1,request:null,action:"approve",processing:!1}),E(e=>e+1),y.post({message:"API key "+("approve"===$.action?"approved":"rejected"),severity:"success",display:"transient"})}catch(e){console.error(`Error ${$.action}ing request:`,e),_(e=>({...e,processing:!1}));const t=e instanceof Error?e.message:"unknown error occurred";y.post({message:`Failed to ${$.action} APIKey. ${t}`,severity:"error",display:"transient"})}}})]})}},55429:(e,t,a)=>{var r=a(4293),n=a(78920);t.A=void 0;var i=n(a(95478)),o=(0,r(a(74044)).default)(i.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=6800.cd5c7bcb.chunk.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"static/6800.cd5c7bcb.chunk.js","mappings":"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","sources":["webpack://internal.plugin-kuadrant/./node_modules/@backstage/plugin-catalog-react/dist/hooks/useEntity.esm.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/utils/permissions.ts","webpack://internal.plugin-kuadrant/./src/utils/styles.ts","webpack://internal.plugin-kuadrant/./src/components/EntityApiApprovalTab/EntityApiApprovalTab.tsx","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/icons/CheckCircle.js","webpack://internal.plugin-kuadrant/./src/utils/errors.ts"],"sourcesContent":["import { jsx } from 'react/jsx-runtime';\nimport { stringifyEntityRef } from '@backstage/catalog-model';\nimport { AnalyticsContext } from '@backstage/core-plugin-api';\nimport { createVersionedContext, useVersionedContext, createVersionedValueMap } from '@backstage/version-bridge';\n\nconst NewEntityContext = createVersionedContext(\n \"entity-context\"\n);\nconst AsyncEntityProvider = (props) => {\n const { children, entity, loading, error, refresh } = props;\n const value = { entity, loading, error, refresh };\n return /* @__PURE__ */ jsx(NewEntityContext.Provider, { value: createVersionedValueMap({ 1: value }), children: /* @__PURE__ */ jsx(\n AnalyticsContext,\n {\n attributes: {\n ...entity ? { entityRef: stringifyEntityRef(entity) } : void 0\n },\n children\n }\n ) });\n};\nconst EntityProvider = (props) => /* @__PURE__ */ jsx(\n AsyncEntityProvider,\n {\n entity: props.entity,\n loading: !Boolean(props.entity),\n error: void 0,\n refresh: void 0,\n children: props.children\n }\n);\nfunction useEntity() {\n const versionedHolder = useVersionedContext(\n \"entity-context\"\n );\n if (!versionedHolder) {\n throw new Error(\"Entity context is not available\");\n }\n const value = versionedHolder.atVersion(1);\n if (!value) {\n throw new Error(\"EntityContext v1 not available\");\n }\n if (!value.entity) {\n throw new Error(\n \"useEntity hook is being called outside of an EntityLayout where the entity has not been loaded. If this is intentional, please use useAsyncEntity instead.\"\n );\n }\n return { entity: value.entity };\n}\nfunction useAsyncEntity() {\n const versionedHolder = useVersionedContext(\n \"entity-context\"\n );\n if (!versionedHolder) {\n throw new Error(\"Entity context is not available\");\n }\n const value = versionedHolder.atVersion(1);\n if (!value) {\n throw new Error(\"EntityContext v1 not available\");\n }\n const { entity, loading, error, refresh } = value;\n return { entity, loading, error, refresh };\n}\n\nexport { AsyncEntityProvider, EntityProvider, useAsyncEntity, useEntity };\n//# sourceMappingURL=useEntity.esm.js.map\n","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 { 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, { useState } from \"react\";\nimport { useEntity } from \"@backstage/plugin-catalog-react\";\nimport {\n useApi,\n configApiRef,\n fetchApiRef,\n identityApiRef,\n alertApiRef,\n} from \"@backstage/core-plugin-api\";\nimport { useAsync } from \"react-use\";\nimport {\n Table,\n TableColumn,\n Progress,\n ResponseErrorPanel,\n} from \"@backstage/core-components\";\nimport { kuadrantApiKeyApprovePermission } from \"../../permissions\";\nimport { useKuadrantPermission } from \"../../utils/permissions\";\nimport {\n Box,\n Typography,\n Chip,\n IconButton,\n Tooltip,\n Button,\n Dialog,\n DialogTitle,\n DialogContent,\n DialogActions,\n TextField,\n CircularProgress,\n makeStyles,\n} from \"@material-ui/core\";\nimport CheckCircleIcon from \"@material-ui/icons/CheckCircle\";\nimport CancelIcon from \"@material-ui/icons/Cancel\";\nimport { APIKey } from \"../../types/api-management\";\nimport { getApprovalQueueStatusChipStyle } from \"../../utils/styles\";\nimport {handleFetchError} from \"../../utils/errors.ts\";\n\nconst useStyles = makeStyles((theme) => ({\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}));\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 ApprovalDialogProps {\n open: boolean;\n request: APIKey | null;\n action: \"approve\" | \"reject\";\n processing: boolean;\n onClose: () => void;\n onConfirm: (confirmText: string) => void;\n}\n\nconst ApprovalDialog = ({\n open,\n request,\n action,\n processing,\n onClose,\n onConfirm,\n}: ApprovalDialogProps) => {\n const [confirmText, setConfirmText] = useState(\"\");\n\n const handleClose = () => {\n setConfirmText(\"\");\n onClose();\n };\n\n const handleConfirm = () => {\n onConfirm(confirmText);\n setConfirmText(\"\");\n };\n\n const isReject = action === \"reject\";\n const expectedText = request?.metadata.name || \"\";\n const isConfirmValid = confirmText === expectedText;\n\n return (\n <Dialog\n open={open}\n onClose={processing ? undefined : handleClose}\n maxWidth=\"sm\"\n fullWidth\n >\n <DialogTitle>\n {isReject ? \"Reject API key\" : \"Approve API key\"}\n </DialogTitle>\n <DialogContent>\n {request && (\n <>\n <Typography variant=\"body2\" paragraph>\n <strong>Requester:</strong> {request.spec.requestedBy.userId}\n </Typography>\n <Typography variant=\"body2\" paragraph>\n <strong>Tier:</strong> {request.spec.planTier}\n </Typography>\n {request.spec.useCase && (\n <Typography variant=\"body2\" paragraph>\n <strong>Use Case:</strong> {request.spec.useCase}\n </Typography>\n )}\n {isReject && (\n <Box mt={2} p={2} bgcolor=\"error.light\" borderRadius={1}>\n <Typography variant=\"body2\">\n This action will permanently deny access. The user will need\n to submit a new request.\n </Typography>\n </Box>\n )}\n <Box mt={2}>\n <TextField\n fullWidth\n label={`Type \"${expectedText}\" to confirm`}\n value={confirmText}\n onChange={(e) => setConfirmText(e.target.value)}\n disabled={processing}\n autoFocus\n />\n </Box>\n </>\n )}\n </DialogContent>\n <DialogActions>\n <Button onClick={handleClose} disabled={processing}>\n Cancel\n </Button>\n <Button\n onClick={handleConfirm}\n color={isReject ? \"secondary\" : \"primary\"}\n variant=\"contained\"\n disabled={!isConfirmValid || processing}\n startIcon={\n processing ? (\n <CircularProgress size={16} color=\"inherit\" />\n ) : undefined\n }\n >\n {processing\n ? isReject\n ? \"Rejecting...\"\n : \"Approving...\"\n : isReject\n ? \"Reject\"\n : \"Approve\"}\n </Button>\n </DialogActions>\n </Dialog>\n );\n};\n\nexport const EntityApiApprovalTab = () => {\n const { entity } = useEntity();\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\n const apiProductName =\n entity.metadata.annotations?.[\"kuadrant.io/apiproduct\"] ||\n entity.metadata.name;\n const namespace =\n entity.metadata.annotations?.[\"kuadrant.io/namespace\"] || \"default\";\n\n const [refresh, setRefresh] = useState(0);\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\n const {\n allowed: canApprove,\n loading: permissionLoading,\n error: permissionError,\n } = useKuadrantPermission(kuadrantApiKeyApprovePermission);\n\n const { value, loading, error } = useAsync(async () => {\n const identity = await identityApi.getBackstageIdentity();\n const reviewedBy = identity.userEntityRef;\n\n const response = await fetchApi.fetch(\n `${backendUrl}/api/kuadrant/requests`,\n );\n if (!response.ok) {\n const err = await handleFetchError(response);\n throw new Error(`Failed to fetch requests. ${err}`);\n }\n\n const data = await response.json();\n const allRequests: APIKey[] = data.items || [];\n\n // filter to this API product only\n const filtered = allRequests.filter(\n (r) =>\n r.spec.apiProductRef?.name === apiProductName &&\n r.metadata.namespace === namespace,\n );\n\n return { requests: filtered, reviewedBy };\n }, [backendUrl, fetchApi, identityApi, apiProductName, namespace, refresh]);\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 setRefresh((r) => r + 1);\n alertApi.post({\n message: `API key ${dialogState.action === \"approve\" ? \"approved\" : \"rejected\"}`,\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} APIKey. ${errorMessage}`,\n severity: 'error',\n display: 'transient',\n });\n }\n };\n\n if (loading || permissionLoading) {\n return <Progress />;\n }\n\n if (error) {\n return <ResponseErrorPanel error={error} />;\n }\n\n if (permissionError) {\n return (\n <Box p={2}>\n <Typography color=\"error\">\n Unable to check permissions: {permissionError.message}\n </Typography>\n </Box>\n );\n }\n\n if (!canApprove) {\n return (\n <Box p={3} textAlign=\"center\">\n <Typography variant=\"body1\" color=\"textSecondary\">\n You don't have permission to view the approval queue for this API.\n </Typography>\n </Box>\n );\n }\n\n const requests = value?.requests || [];\n const pendingRequests = requests.filter(\n (r) => !r.status?.phase || r.status.phase === \"Pending\",\n );\n const approvedRequests = requests.filter(\n (r) => r.status?.phase === \"Approved\",\n );\n const rejectedRequests = requests.filter(\n (r) => r.status?.phase === \"Rejected\",\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: \"Status\",\n field: \"status.phase\",\n render: (row) => {\n const phase = row.status?.phase || \"Pending\";\n const label = phase === \"Approved\" ? \"Active\" : phase;\n return (\n <Chip label={label} 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 if (!row.metadata.creationTimestamp)\n return <Typography variant=\"body2\">-</Typography>;\n return (\n <Typography variant=\"body2\">\n {new Date(row.metadata.creationTimestamp).toLocaleDateString()}\n </Typography>\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 return (\n <Box display=\"flex\" style={{ gap: 4 }}>\n <Tooltip title=\"Approve\">\n <IconButton size=\"small\" onClick={() => handleApprove(row)}>\n <CheckCircleIcon color=\"primary\" />\n </IconButton>\n </Tooltip>\n <Tooltip title=\"Reject\">\n <IconButton size=\"small\" onClick={() => handleReject(row)}>\n <CancelIcon color=\"error\" />\n </IconButton>\n </Tooltip>\n </Box>\n );\n },\n },\n ];\n\n const detailPanelConfig = [\n {\n render: (data: any) => {\n const request = data.rowData as APIKey;\n if (!request?.metadata?.name) return <Box />;\n return <ExpandedRowContent request={request} />;\n },\n },\n ];\n\n return (\n <Box p={2}>\n <Box mb={2}>\n <Typography variant=\"body2\" color=\"textSecondary\">\n {pendingRequests.length} pending, {approvedRequests.length} approved,{\" \"}\n {rejectedRequests.length} rejected\n </Typography>\n </Box>\n\n {requests.length === 0 ? (\n <Box p={3} textAlign=\"center\">\n <Typography variant=\"body1\" color=\"textSecondary\">\n No API keys for this API.\n </Typography>\n </Box>\n ) : (\n <Table\n options={{\n paging: requests.length > 10,\n pageSize: 20,\n search: true,\n filtering: false,\n debounceInterval: 300,\n toolbar: true,\n emptyRowsWhenPaging: false,\n }}\n columns={columns}\n data={requests.map((item) => ({ ...item, id: item.metadata.name }))}\n detailPanel={detailPanelConfig}\n />\n )}\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 </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 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":["useEntity","versionedHolder","useVersionedContext","Error","value","atVersion","entity","createVersionedContext","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","length","callId","current","prevState","__assign","apply","then","error","useAsyncFn","useEffect","useKuadrantPermission","permission","resourceRef","permissionRequest","result","usePermission","allowed","canDeleteResource","ownerId","currentUserId","canDeleteOwn","canDeleteAll","getMyApiKeysStatusChipStyle","phase","base","border","backgroundColor","color","getApprovalQueueStatusChipStyle","getLifecycleChipStyle","lifecycle","useStyles","makeStyles","theme","useCasePanel","padding","spacing","palette","background","default","useCaseLabel","fontWeight","marginBottom","text","secondary","textTransform","fontSize","ExpandedRowContent","request","classes","Box","className","onClick","e","stopPropagation","Typography","variant","spec","useCase","ApprovalDialog","open","processing","onClose","onConfirm","confirmText","setConfirmText","handleClose","isReject","expectedText","metadata","isConfirmValid","Dialog","undefined","maxWidth","fullWidth","DialogTitle","DialogContent","paragraph","strong","requestedBy","userId","planTier","mt","p","bgcolor","borderRadius","TextField","label","onChange","target","disabled","autoFocus","DialogActions","Button","startIcon","CircularProgress","size","EntityApiApprovalTab","config","useApi","configApiRef","fetchApi","fetchApiRef","identityApi","identityApiRef","alertApi","alertApiRef","backendUrl","getString","apiProductName","annotations","namespace","refresh","setRefresh","dialogState","setDialogState","canApprove","permissionLoading","permissionError","async","reviewedBy","getBackstageIdentity","userEntityRef","response","fetch","ok","err","handleFetchError","requests","json","items","filter","r","apiProductRef","Progress","ResponseErrorPanel","message","textAlign","pendingRequests","status","approvedRequests","rejectedRequests","columns","title","field","render","row","Chip","style","creationTimestamp","Date","toLocaleDateString","filtering","display","gap","Tooltip","IconButton","CheckCircleIcon","CancelIcon","detailPanelConfig","data","rowData","mb","Table","options","paging","pageSize","search","debounceInterval","toolbar","emptyRowsWhenPaging","map","item","id","detailPanel","prev","endpoint","method","headers","body","JSON","stringify","post","severity","console","errorMessage","_interopRequireDefault","_interopRequireWildcard","exports","React","_default","createElement","d","errorData","catch"],"sourceRoot":""}
@@ -1,2 +0,0 @@
1
- "use strict";(self.webpackChunkinternal_plugin_kuadrant=self.webpackChunkinternal_plugin_kuadrant||[]).push([[7005],{12229:(e,t,a)=>{a.d(t,{Z:()=>A});var s=a(31085),n=a(95478),l=a.n(n),i=a(10394),r=a(72501),o=a(64947),c=a(37197),d=a(69621),u=a(12981),p=a(86901),m=a(69076),h=a(58837),g=a(6924),x=a(23164);const v=(0,h.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)}})),A=({sections:e,filters:t,onChange:a,onClear:n})=>{const h=v(),[A,f]=l().useState(new Set(e.filter(e=>e.collapsed).map(e=>e.id))),y=Object.values(t).some(e=>e.length>0);return(0,s.jsxs)(i.A,{className:h.root,children:[(0,s.jsxs)(i.A,{display:"flex",justifyContent:"space-between",alignItems:"center",mb:2,children:[(0,s.jsx)(r.A,{variant:"subtitle2",children:"Filters"}),y&&(0,s.jsx)(o.A,{size:"small",color:"primary",onClick:()=>{const t={};e.forEach(e=>{t[e.id]=[]}),a(t),null==n||n()},children:"Clear all"})]}),(0,s.jsx)(c.A,{}),e.map(e=>{const n=A.has(e.id),l=(t[e.id]||[]).length;return(0,s.jsxs)(i.A,{className:h.filterSection,mt:2,children:[(0,s.jsxs)(i.A,{className:h.sectionTitle,onClick:()=>{return t=e.id,void f(e=>{const a=new Set(e);return a.has(t)?a.delete(t):a.add(t),a});var t},children:[(0,s.jsxs)(i.A,{display:"flex",alignItems:"center",children:[(0,s.jsx)("span",{children:e.title}),l>0&&(0,s.jsxs)("span",{className:h.count,children:["(",l,")"]})]}),n?(0,s.jsx)(g.A,{fontSize:"small"}):(0,s.jsx)(x.A,{fontSize:"small"})]}),(0,s.jsx)(d.A,{in:!n,children:(0,s.jsx)(u.A,{children:e.options.map(n=>(0,s.jsx)(p.A,{control:(0,s.jsx)(m.A,{checked:(t[e.id]||[]).includes(n.value),onChange:()=>((e,s)=>{const n=t[e]||[],l=n.includes(s)?n.filter(e=>e!==s):[...n,s];a({...t,[e]:l})})(e.id,n.value),size:"small",className:h.checkbox,color:"primary"}),label:(0,s.jsxs)(i.A,{display:"flex",alignItems:"center",children:[(0,s.jsx)("span",{className:h.checkboxLabel,children:n.label}),void 0!==n.count&&(0,s.jsxs)("span",{className:h.count,children:["(",n.count,")"]})]})},n.value))})})]},e.id)})]})}},94251:(e,t,a)=>{a.r(t),a.d(t,{ApiProductsPage:()=>se});var s=a(31085),n=a(95478),l=a(58837),i=a(10394),r=a(67720),o=a(72501),c=a(64947),d=a(29365),u=a(78467),p=a(18466),m=a(39590),h=a(75625),g=a(21702),x=a(85142),v=a(12229),A=a(37725),f=a(289),y=a(15831),j=a(45210),b=a(46681),P=a(42367),w=a(25010),S=a(91638),k=a(22097),I=a(86687),C=a(46205);const T=({children:e,permission:t,fallback:a,errorMessage:n})=>{const{allowed:l,loading:r,error:c}=(0,C.l)(t);return r?(0,s.jsx)(I.k,{}):c?(0,s.jsxs)(i.A,{p:4,children:[(0,s.jsxs)(o.A,{color:"error",children:["Unable to check permissions: ",c.message]}),(0,s.jsx)(o.A,{variant:"body2",color:"textSecondary",children:"Please try again or contact your administrator"})]}):l?(0,s.jsx)(s.Fragment,{children:e}):a?(0,s.jsx)(s.Fragment,{children:a}):(0,s.jsxs)(i.A,{p:4,children:[(0,s.jsx)(o.A,{color:"textSecondary",children:n||"You don't have permission to view this page"}),(0,s.jsx)(i.A,{mt:1,children:(0,s.jsxs)(o.A,{variant:"caption",color:"textSecondary",children:["Required permission: ",t.name]})})]})};var N=a(76891),$=a(61477),R=a(46805),z=a(42899),D=a(16249),E=a(34839),L=a(71677),H=a(26343),M=a(95061),W=a(29635),O=a(86901),q=a(30285),K=a(93453),U=a(89031),B=a(84441),F=a(65867),_=a(24170),Y=a(77318);const V=(0,l.A)(e=>({asterisk:{color:"#f44336"},sectionHeader:{display:"flex",alignItems:"center",gap:e.spacing(.5),marginTop:e.spacing(2),marginBottom:e.spacing(1)},infoIcon:{fontSize:18,color:e.palette.text.secondary},tagChip:{marginRight:e.spacing(.5),marginBottom:e.spacing(.5)}})),J=({open:e,onClose:t,onSuccess:a})=>{const l=V(),m=(0,k.useApi)(k.configApiRef),h=(0,k.useApi)(k.fetchApiRef),g=m.getString("backend.baseUrl"),[x,v]=(0,n.useState)(""),[A,f]=(0,n.useState)(""),[y,j]=(0,n.useState)(""),[b,P]=(0,n.useState)("v1"),[w,I]=(0,n.useState)("manual"),[C,T]=(0,n.useState)("Published"),[J,Z]=(0,n.useState)("production"),[G,X]=(0,n.useState)([]),[Q,ee]=(0,n.useState)(""),[te,ae]=(0,n.useState)(""),[se,ne]=(0,n.useState)(""),[le,ie]=(0,n.useState)(""),[re,oe]=(0,n.useState)(""),[ce,de]=(0,n.useState)(""),[ue,pe]=(0,n.useState)(""),[me,he]=(0,n.useState)(!1),[ge,xe]=(0,n.useState)(0),[ve,Ae]=(0,n.useState)(null),[fe,ye]=(0,n.useState)(null),[je,be]=(0,n.useState)(""),[Pe,we]=(0,n.useState)("name"),{value:Se,loading:ke,error:Ie}=(0,S.A)(async()=>{const e=await h.fetch(`${g}/api/kuadrant/httproutes`);if(!e.ok){const t=await(0,Y.T)(e);throw new Error(`failed to fetch routes. ${t}`)}return(await e.json()).items||[]},[g,h,e,ge]),{value:Ce,error:Te}=(0,S.A)(async()=>{const e=await h.fetch(`${g}/api/kuadrant/planpolicies`);if(!e.ok){const t=await(0,Y.T)(e);throw new Error(`failed to fetch PlanPolicies. ${t}`)}return await e.json()},[g,h,e]),Ne=(e,t)=>(null==Ce?void 0:Ce.items)?Ce.items.find(a=>{const s=a.targetRef;return"HTTPRoute"===(null==s?void 0:s.kind)&&(null==s?void 0:s.name)===t&&(!(null==s?void 0:s.namespace)||(null==s?void 0:s.namespace)===e)}):null,$e=te?te.split("/"):null,Re=$e?Ne($e[0],$e[1]):null,ze=(e,t)=>{const a=Ne(e,t);return a?`${a.metadata.name}${(e=>{var t;if(!(null==e||null===(t=e.spec)||void 0===t?void 0:t.plans))return"";const a=Object.entries(e.spec.plans).map(([e,t])=>{var a;const s=null==t||null===(a=t.limits)||void 0===a?void 0:a.requests;return s?`${e}: ${s.count}/${s.period}`:e}).join("; ");return a?` (${a})`:""})(a)}`:"N/A"};(0,n.useEffect)(()=>{e&&(Ae(null),ye(null))},[e]);const De=()=>{Q.trim()&&!G.includes(Q.trim())&&(X([...G,Q.trim()]),ee(""))},Ee=()=>{v(""),f(""),j(""),P("v1"),I("manual"),T("Published"),Z("production"),X([]),ee(""),ae(""),ne(""),ie(""),oe(""),de(""),pe(""),Ae(null),ye(null),t()},Le=!!ve||!!fe;return(0,s.jsxs)(N.A,{open:e,onClose:Ee,maxWidth:"md",fullWidth:!0,children:[(0,s.jsx)($.A,{children:"Create API Product"}),(0,s.jsxs)(R.A,{children:[ue&&(0,s.jsx)(B.A,{severity:"error",style:{marginBottom:16},children:ue}),Ie&&(0,s.jsxs)(B.A,{severity:"error",style:{marginBottom:16},children:[(0,s.jsx)("strong",{children:"Failed to load HTTPRoutes:"})," ",Ie.message,(0,s.jsx)(i.A,{mt:1,children:(0,s.jsx)(c.A,{size:"small",variant:"outlined",onClick:()=>xe(e=>e+1),children:"Retry"})})]}),Te&&(0,s.jsxs)(B.A,{severity:"warning",style:{marginBottom:16},children:[(0,s.jsx)("strong",{children:"Failed to load PlanPolicies:"})," ",Te.message,(0,s.jsx)(o.A,{variant:"body2",style:{marginTop:8},children:"You can still create the API Product, but plan information may be incomplete."})]}),(0,s.jsx)(i.A,{className:l.sectionHeader,children:(0,s.jsx)(o.A,{variant:"subtitle1",children:(0,s.jsx)("strong",{children:"API product info"})})}),(0,s.jsxs)(z.A,{container:!0,spacing:2,children:[(0,s.jsx)(z.A,{item:!0,xs:6,children:(0,s.jsx)(D.A,{fullWidth:!0,label:"API product name",value:A,onChange:e=>(e=>{if(f(e),!x||x.match(/-[a-f0-9]{6}$/)){const t=`${e.toLowerCase().replace(/[^a-z0-9-]/g,"-").replace(/-+/g,"-").replace(/^-|-$/g,"")}-${Math.floor(16777215*Math.random()).toString(16).padStart(6,"0")}`;v(t),Ae((0,_.o)(t))}})(e.target.value),placeholder:"My API",helperText:"Display name for your API product (shown to users)",margin:"normal",required:!0,disabled:me,InputLabelProps:{classes:{asterisk:l.asterisk}}})}),(0,s.jsx)(z.A,{item:!0,xs:6,children:(0,s.jsx)(D.A,{fullWidth:!0,label:"Kubernetes resource name",value:x,onChange:e=>{return t=e.target.value,v(t),void Ae((0,_.o)(t));var t},placeholder:"my-api",helperText:ve||"Auto-generated from product name. Only lowercase, numbers, and hyphens allowed.",error:!!ve,margin:"normal",required:!0,disabled:me,InputLabelProps:{classes:{asterisk:l.asterisk}}})}),(0,s.jsx)(z.A,{item:!0,xs:6,children:(0,s.jsx)(D.A,{fullWidth:!0,label:"Version",value:b,onChange:e=>P(e.target.value),placeholder:"v1",helperText:"Give a version to your API product",margin:"normal",required:!0,disabled:me,InputLabelProps:{classes:{asterisk:l.asterisk}}})}),(0,s.jsx)(z.A,{item:!0,xs:6,children:(0,s.jsx)(D.A,{fullWidth:!0,label:"Tag",value:Q,onChange:e=>ee(e.target.value),onKeyPress:e=>{"Enter"===e.key&&(e.preventDefault(),De())},placeholder:"Add tag",helperText:"Add a tag to your API product",margin:"normal",disabled:me,InputProps:{endAdornment:Q?(0,s.jsx)(E.A,{position:"end",children:(0,s.jsx)(d.A,{size:"small",onClick:De,disabled:me,children:(0,s.jsx)(p.A,{fontSize:"small"})})}):void 0}})}),G.length>0&&(0,s.jsx)(z.A,{item:!0,xs:12,children:(0,s.jsx)(i.A,{display:"flex",flexWrap:"wrap",children:G.map(e=>(0,s.jsx)(r.A,{label:e,onDelete:me?void 0:()=>{return t=e,void X(G.filter(e=>e!==t));var t},size:"small",className:l.tagChip,disabled:me},e))})}),(0,s.jsx)(z.A,{item:!0,xs:12,children:(0,s.jsx)(D.A,{fullWidth:!0,label:"Description",value:y,onChange:e=>j(e.target.value),placeholder:"API description",margin:"normal",multiline:!0,rows:2,required:!0,disabled:me,InputLabelProps:{classes:{asterisk:l.asterisk}}})})]}),(0,s.jsxs)(i.A,{className:l.sectionHeader,children:[(0,s.jsx)(o.A,{variant:"subtitle1",children:(0,s.jsx)("strong",{children:"Add API and Associate route"})}),(0,s.jsx)(L.Ay,{title:"Register an existing API and associate HTTPRoute for your API product",children:(0,s.jsx)(U.A,{className:l.infoIcon})})]}),(0,s.jsxs)(z.A,{container:!0,spacing:2,children:[(0,s.jsx)(z.A,{item:!0,xs:12,children:(0,s.jsx)(D.A,{fullWidth:!0,label:"OpenAPI Spec URL",value:ce,onChange:e=>{return t=e.target.value,de(t),void ye((0,_.q)(t));var t},placeholder:"https://api.example.com/openapi.json",helperText:fe||"Enter the full path to your API spec file",error:!!fe,margin:"normal",required:!0,disabled:me,InputLabelProps:{classes:{asterisk:l.asterisk}}})}),(0,s.jsx)(z.A,{item:!0,xs:12,children:(0,s.jsx)(D.A,{fullWidth:!0,label:"Documentation URL",value:re,onChange:e=>oe(e.target.value),placeholder:"https://docs.example.com/api",helperText:"Link to external documentation for this API",margin:"normal",disabled:me})}),(0,s.jsx)(z.A,{item:!0,xs:12,children:(0,s.jsxs)(D.A,{fullWidth:!0,select:!0,label:"HTTPRoute",value:te,onChange:e=>ae(e.target.value),margin:"normal",required:!0,helperText:Ie?"Unable to load HTTPRoutes. Please retry.":"Select an HTTPRoute. APIProduct will be created in the same namespace.",error:!!Ie,disabled:ke||me||!!Ie,InputLabelProps:{classes:{asterisk:l.asterisk}},SelectProps:{"data-testid":"httproute-select",MenuProps:{PaperProps:{style:{maxHeight:400}},anchorOrigin:{vertical:"bottom",horizontal:"left"},transformOrigin:{vertical:"top",horizontal:"left"},getContentAnchorEl:null}},children:[(0,s.jsx)(i.A,{px:2,pt:1,pb:1,style:{position:"sticky",top:0,zIndex:1},children:(0,s.jsx)(D.A,{fullWidth:!0,size:"small",placeholder:"Search...",value:je,onChange:e=>be(e.target.value),onKeyDown:e=>e.stopPropagation(),onClick:e=>e.stopPropagation(),InputProps:{endAdornment:(0,s.jsx)(E.A,{position:"end",children:(0,s.jsxs)(D.A,{select:!0,size:"small",value:Pe,onChange:e=>we(e.target.value),onKeyDown:e=>e.stopPropagation(),onClick:e=>e.stopPropagation(),style:{minWidth:120},variant:"standard",children:[(0,s.jsx)(H.A,{value:"name",children:"Name"}),(0,s.jsx)(H.A,{value:"namespace",children:"Namespace"}),(0,s.jsx)(H.A,{value:"planpolicy",children:"PlanPolicy"})]})})}})}),ke&&(0,s.jsx)(H.A,{value:"",children:"Loading..."}),Ie&&(0,s.jsx)(H.A,{value:"",children:"Error loading routes"}),!ke&&!Ie&&Se&&0===Se.length&&(0,s.jsx)(H.A,{value:"",children:"No HTTPRoutes available"}),!ke&&!Ie&&Se&&Se.filter(e=>{if(!je)return!0;const t=e.metadata.namespace,a=e.metadata.name,s=ze(t,a),n=je.toLowerCase();switch(Pe){case"name":return a.toLowerCase().includes(n);case"namespace":return t.toLowerCase().includes(n);case"planpolicy":return s.toLowerCase().includes(n);default:return!0}}).map(e=>{const t=e.metadata.namespace,a=e.metadata.name,n=ze(t,a);return(0,s.jsx)(H.A,{value:`${t}/${a}`,children:(0,s.jsxs)(i.A,{children:[(0,s.jsx)(o.A,{variant:"body1",children:a}),(0,s.jsxs)(o.A,{variant:"caption",color:"textSecondary",children:["Associated PlanPolicy: ",n]})]})},`${t}/${a}`)})]})})]}),te&&Re&&(0,s.jsxs)(s.Fragment,{children:[(0,s.jsxs)(i.A,{className:l.sectionHeader,children:[(0,s.jsx)(o.A,{variant:"subtitle1",children:(0,s.jsx)("strong",{children:"HTTPRoute policies"})}),(0,s.jsx)(L.Ay,{title:"Shows the associated policies and rate limit tiers for the selected HTTPRoute",children:(0,s.jsx)(U.A,{className:l.infoIcon})})]}),(0,s.jsx)(F.g,{discoveredPlans:Re.plans,alertSeverity:"warning",alertMessage:"No PlanPolicy found for this HTTPRoute. API keys and rate limiting may not be available.",includeTopMargin:!1})]}),(0,s.jsxs)(i.A,{className:l.sectionHeader,children:[(0,s.jsx)(o.A,{variant:"subtitle1",children:(0,s.jsx)("strong",{children:"Lifecycle and Visibility"})}),(0,s.jsx)(L.Ay,{title:"Control the lifecycle state and catalog visibility of this API product",children:(0,s.jsx)(U.A,{className:l.infoIcon})})]}),(0,s.jsxs)(z.A,{container:!0,spacing:2,children:[(0,s.jsx)(z.A,{item:!0,xs:6,children:(0,s.jsxs)(D.A,{fullWidth:!0,select:!0,label:"Lifecycle",value:J,onChange:e=>Z(e.target.value),margin:"normal",helperText:"API lifecycle state",disabled:me,children:[(0,s.jsx)(H.A,{value:"experimental",children:"Experimental"}),(0,s.jsx)(H.A,{value:"production",children:"Production"}),(0,s.jsx)(H.A,{value:"deprecated",children:"Deprecated"}),(0,s.jsx)(H.A,{value:"retired",children:"Retired"})]})}),(0,s.jsx)(z.A,{item:!0,xs:6,children:(0,s.jsxs)(D.A,{fullWidth:!0,select:!0,label:"Publish Status",value:C,onChange:e=>T(e.target.value),margin:"normal",helperText:"Controls catalog visibility (Draft = hidden from consumers)",disabled:me,children:[(0,s.jsx)(H.A,{value:"Draft",children:"Draft"}),(0,s.jsx)(H.A,{value:"Published",children:"Published"})]})})]}),(0,s.jsxs)(i.A,{className:l.sectionHeader,children:[(0,s.jsx)(o.A,{variant:"subtitle1",children:(0,s.jsx)("strong",{children:"API Key approval"})}),(0,s.jsx)(L.Ay,{title:"Choose how API key requests are handled for this product",children:(0,s.jsx)(U.A,{className:l.infoIcon})})]}),(0,s.jsx)(M.A,{component:"fieldset",disabled:me,children:(0,s.jsxs)(W.A,{row:!0,value:w,onChange:e=>I(e.target.value),children:[(0,s.jsx)(O.A,{value:"manual",control:(0,s.jsx)(q.A,{color:"primary"}),label:(0,s.jsxs)(i.A,{children:[(0,s.jsx)(o.A,{variant:"body2",children:"Need manual approval"}),(0,s.jsx)(o.A,{variant:"caption",color:"textSecondary",children:"Requires approval for requesting this API"})]})}),(0,s.jsx)(O.A,{value:"automatic",control:(0,s.jsx)(q.A,{color:"primary"}),label:(0,s.jsxs)(i.A,{children:[(0,s.jsx)(o.A,{variant:"body2",children:"Automatic"}),(0,s.jsx)(o.A,{variant:"caption",color:"textSecondary",children:"Keys are created without need to be approved"})]})})]})})]}),(0,s.jsxs)(K.A,{children:[(0,s.jsx)(c.A,{onClick:Ee,disabled:me,children:"Cancel"}),(0,s.jsx)(c.A,{onClick:async()=>{pe(""),he(!0);try{if(!te)throw new Error("Please select an HTTPRoute");const[e,t]=te.split("/"),s=e,n={apiVersion:"devportal.kuadrant.io/v1alpha1",kind:"APIProduct",metadata:{name:x,namespace:s,labels:{lifecycle:J}},spec:{displayName:A,description:y,version:b,approvalMode:w,publishStatus:C,tags:G,targetRef:{group:"gateway.networking.k8s.io",kind:"HTTPRoute",name:t,namespace:e},...se||le?{contact:{...se&&{email:se},...le&&{team:le}}}:{},...re||ce?{documentation:{...re&&{docsURL:re},...ce&&{openAPISpecURL:ce}}}:{}}},l=await h.fetch(`${g}/api/kuadrant/apiproducts`,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify(n)});if(!l.ok){const e=await(0,Y.T)(l);throw new Error(`failed to create APIProduct. ${e}`)}a({namespace:s,name:x,displayName:A}),Ee()}catch(e){pe(e instanceof Error?e.message:String(e))}finally{he(!1)}},color:"primary",variant:"contained",disabled:me||!x||!A||!y||!te||Le,startIcon:me?(0,s.jsx)(u.A,{size:16,color:"inherit"}):void 0,children:me?"Creating...":"Create"})]})]})};var Z=a(34955),G=a(26997),X=a(63221);const Q=a.p+"static/empty-state-illustration.7e3ad5a9..png";var ee=a(46299);const te=(0,l.A)(e=>({container:{display:"flex",height:"100%",minHeight:400},tableContainer:{flex:1,overflow:"auto",padding:10},emptyState:{display:"flex",alignItems:"center",justifyContent:"center",padding:e.spacing(6),minHeight:400},emptyStateContent:{display:"flex",alignItems:"center",gap:e.spacing(6),maxWidth:900},emptyStateText:{flex:1},emptyStateTitle:{marginBottom:e.spacing(2)},emptyStateDescription:{marginBottom:e.spacing(3),color:e.palette.text.secondary},emptyStateImage:{maxWidth:400,height:"auto"}})),ae=()=>{const e=te(),t=(0,k.useApi)(k.configApiRef),a=(0,k.useApi)(k.fetchApiRef),l=(0,k.useApi)(k.alertApiRef),I=(0,k.useApi)(k.identityApiRef),T=t.getString("backend.baseUrl"),[N,$]=(0,n.useState)(""),[R,z]=(0,n.useState)(!1),[D,E]=(0,n.useState)(!1),[L,H]=(0,n.useState)(0),[M,W]=(0,n.useState)(!1),[O,q]=(0,n.useState)(null),[K,U]=(0,n.useState)(null),[B,F]=(0,n.useState)(!1),[_,V]=(0,n.useState)(null),[ae,se]=(0,n.useState)({status:[],lifecycle:[],policy:[],route:[],namespace:[],tags:[],authentication:[]}),{allowed:ne,loading:le,error:ie}=(0,C.l)(Z.FL),{allowed:re,loading:oe}=(0,C.l)(Z.EM),{allowed:ce,loading:de,error:ue}=(0,C.l)(Z.R_),{allowed:pe}=(0,C.l)(Z.U3),{allowed:me}=(0,C.l)(Z.v_),he=oe||de,{allowed:ge,loading:xe,error:ve}=(0,C.l)(Z.J);(0,S.A)(async()=>{const e=await I.getBackstageIdentity();$(e.userEntityRef)},[I]);const{value:Ae,loading:fe,error:ye}=(0,S.A)(async()=>{const e=await a.fetch(`${T}/api/kuadrant/apiproducts`);if(!e.ok){const t=await(0,Y.T)(e);throw new Error(`failed to fetch APIProducts. ${t}`)}return await e.json()},[T,a,L]),{value:je,loading:be,error:Pe}=(0,S.A)(async()=>{if(!ge)return{items:[]};const e=await a.fetch(`${T}/api/kuadrant/planpolicies`);if(!e.ok){const t=await(0,Y.T)(e);throw new Error(`failed to fetch PlanPolicies: ${t}`)}return await e.json()},[T,a,L,ge]),we=(0,n.useCallback)(e=>{var t;if(!(null==je?void 0:je.items))return null;const a=null===(t=e.spec)||void 0===t?void 0:t.targetRef;if(!a)return null;const s=je.items.find(t=>{const s=t.targetRef;return"HTTPRoute"===(null==s?void 0:s.kind)&&(null==s?void 0:s.name)===a.name&&(!(null==s?void 0:s.namespace)||(null==s?void 0:s.namespace)===(a.namespace||e.metadata.namespace))});return(null==s?void 0:s.metadata.name)||null},[je]),Se=(0,n.useCallback)(e=>{var t,a;const s=(null===(a=e.status)||void 0===a||null===(t=a.discoveredAuthScheme)||void 0===t?void 0:t.authentication)||{},n=Object.values(s),l=[];return n.some(e=>e.hasOwnProperty("apiKey"))&&l.push("API Key"),n.some(e=>e.hasOwnProperty("jwt"))&&l.push("OIDC"),0===l.length&&l.push("Unknown"),l},[]),ke=fe||be||le||he||xe,Ie=ye||Pe,Ce=ie||ue||ve,Te=(0,n.useMemo)(()=>{const e=(null==Ae?void 0:Ae.items)||[];return ne||pe||me?e:e.filter(e=>{var t;return"Published"===((null===(t=e.spec)||void 0===t?void 0:t.publishStatus)||"Draft")})},[Ae,ne,pe,me]),Ne=(0,n.useMemo)(()=>{const e={Draft:0,Published:0},t=new Map,a=new Map,s=new Map,n=new Map,l=new Map,i=new Map;Te.forEach(r=>{var o,c,d,u,p;const m=(null===(o=r.spec)||void 0===o?void 0:o.publishStatus)||"Draft";e[m]++;const h=(null===(c=r.metadata.labels)||void 0===c?void 0:c.lifecycle)||"production";t.set(h,(t.get(h)||0)+1);const g=we(r)||"N/A";a.set(g,(a.get(g)||0)+1);const x=(null===(u=r.spec)||void 0===u||null===(d=u.targetRef)||void 0===d?void 0:d.name)||"unknown";s.set(x,(s.get(x)||0)+1);const v=r.metadata.namespace;n.set(v,(n.get(v)||0)+1),((null===(p=r.spec)||void 0===p?void 0:p.tags)||[]).forEach(e=>{l.set(e,(l.get(e)||0)+1)}),Se(r).forEach(e=>{i.set(e,(i.get(e)||0)+1)})});const r=[{id:"status",title:"Publish Status",options:[{value:"Draft",label:"Draft",count:e.Draft},{value:"Published",label:"Published",count:e.Published}]},{id:"lifecycle",title:"Lifecycle",options:Array.from(t.entries()).map(([e,t])=>({value:e,label:e.charAt(0).toUpperCase()+e.slice(1),count:t}))},{id:"authentication",title:"Authentication",options:Array.from(i.entries()).map(([e,t])=>({value:e,label:e,count:t}))},{id:"route",title:"Route",options:Array.from(s.entries()).map(([e,t])=>({value:e,label:e,count:t})),collapsed:s.size>5},{id:"namespace",title:"Namespace",options:Array.from(n.entries()).map(([e,t])=>({value:e,label:e,count:t})),collapsed:n.size>5},{id:"tags",title:"Tags",options:Array.from(l.entries()).map(([e,t])=>({value:e,label:e,count:t})),collapsed:l.size>5}];return ge&&r.splice(2,0,{id:"policy",title:"Policy",options:Array.from(a.entries()).map(([e,t])=>({value:e,label:e,count:t})),collapsed:a.size>5}),r},[Te,we,Se,ge]),$e=(0,n.useMemo)(()=>Te.filter(e=>{if(ae.status.length>0){var t;const a=(null===(t=e.spec)||void 0===t?void 0:t.publishStatus)||"Draft";if(!ae.status.includes(a))return!1}if(ae.lifecycle&&ae.lifecycle.length>0){var a;const t=(null===(a=e.metadata.labels)||void 0===a?void 0:a.lifecycle)||"production";if(!ae.lifecycle.includes(t))return!1}if(ae.authentication.length>0){const t=Se(e);if(!ae.authentication.some(e=>t.includes(e)))return!1}if(ae.policy.length>0){const t=we(e)||"N/A";if(!ae.policy.includes(t))return!1}if(ae.route.length>0){var s,n;const t=(null===(n=e.spec)||void 0===n||null===(s=n.targetRef)||void 0===s?void 0:s.name)||"unknown";if(!ae.route.includes(t))return!1}if(ae.namespace.length>0&&!ae.namespace.includes(e.metadata.namespace))return!1;if(ae.tags.length>0){var l;const t=(null===(l=e.spec)||void 0===l?void 0:l.tags)||[];if(!ae.tags.some(e=>t.includes(e)))return!1}return!0}),[Te,ae,we,Se]),Re=[{title:"Name",field:"spec.displayName",render:e=>{var t,a;const n=null!==(a=null===(t=e.spec)||void 0===t?void 0:t.displayName)&&void 0!==a?a:e.metadata.name;return(0,s.jsx)(A.N_,{to:`/kuadrant/api-products/${e.metadata.namespace}/${e.metadata.name}`,children:(0,s.jsx)("strong",{children:n})})},customFilterAndSearch:(e,t)=>{var a;return((null===(a=t.spec)||void 0===a?void 0:a.displayName)||t.metadata.name||"").toLowerCase().includes(e.toLowerCase())}},{title:"Version",field:"spec.version",render:e=>{var t;return(null===(t=e.spec)||void 0===t?void 0:t.version)||"-"}},{title:"Route",field:"spec.targetRef.name",render:e=>{var t,a;return(null===(a=e.spec)||void 0===a||null===(t=a.targetRef)||void 0===t?void 0:t.name)||"-"}},...ge?[{title:"Policy",field:"policy",render:e=>we(e)||"N/A"}]:[],{title:"Tags",field:"spec.tags",render:e=>{var t;const a=(null===(t=e.spec)||void 0===t?void 0:t.tags)||[];return 0===a.length?"-":(0,s.jsx)(i.A,{display:"flex",style:{gap:4,flexWrap:"wrap"},children:a.map(e=>(0,s.jsx)(r.A,{label:e,size:"small",variant:"outlined"},e))})}},{title:"Status",field:"spec.publishStatus",render:e=>{var t;const a=(null===(t=e.spec)||void 0===t?void 0:t.publishStatus)||"Draft";return(0,s.jsx)(r.A,{label:a,size:"small",color:"Published"===a?"primary":"default"})}},{title:"Lifecycle",field:"metadata.labels.lifecycle",render:e=>{var t;const a=null===(t=e.metadata.labels)||void 0===t?void 0:t.lifecycle;return a?(0,s.jsx)(r.A,{label:a.charAt(0).toUpperCase()+a.slice(1),size:"small",style:(0,ee.ee)(a)}):"-"}},{title:"Authentication",field:"status.discoveredAuthScheme",render:e=>{var t,a;const n=(null===(a=e.status)||void 0===a||null===(t=a.discoveredAuthScheme)||void 0===t?void 0:t.authentication)||{},l=Object.values(n),c=l.some(e=>e.hasOwnProperty("apiKey")),d=l.some(e=>e.hasOwnProperty("jwt"));return c||d?(0,s.jsxs)(i.A,{display:"flex",style:{gap:4},children:[c&&(0,s.jsx)(r.A,{icon:(0,s.jsx)(g.A,{}),label:"API Key",size:"small",color:"primary"}),d&&(0,s.jsx)(r.A,{icon:(0,s.jsx)(x.A,{}),label:"OIDC",size:"small",color:"secondary"})]}):(0,s.jsx)(o.A,{variant:"body2",style:{fontStyle:"italic"},children:"unknown"})}},{title:"Namespace",field:"metadata.namespace"},{title:"Actions",field:"actions",filtering:!1,render:e=>{var t,n,r;const o=(null===(n=e.metadata)||void 0===n||null===(t=n.annotations)||void 0===t?void 0:t["backstage.io/owner"])===N,u=me||pe&&o,p=ce||re&&o,g="Published"===(null===(r=e.spec)||void 0===r?void 0:r.publishStatus);return(0,s.jsxs)(i.A,{display:"flex",alignItems:"center",style:{gap:4},children:[u&&(0,s.jsx)(c.A,{size:"small",color:"primary",onClick:()=>(async e=>{var t,s,n;const i=e.metadata.namespace,r=e.metadata.name,o=(null===(t=e.spec)||void 0===t?void 0:t.displayName)||r,c="Published"===((null===(s=e.spec)||void 0===s?void 0:s.publishStatus)||"Draft")?"Draft":"Published",d=null===(n=e.metadata.labels)||void 0===n?void 0:n.lifecycle;if("Published"!==c||"retired"!==d)try{const e=await a.fetch(`${T}/api/kuadrant/apiproducts/${i}/${r}`,{method:"PATCH",headers:{"Content-Type":"application/json"},body:JSON.stringify({spec:{publishStatus:c}})});if(!e.ok){const t=await(0,Y.T)(e);throw new Error(t)}H(e=>e+1),l.post({message:`"${o}" ${"Published"===c?"published":"unpublished"} successfully`,severity:"success",display:"transient"})}catch(e){const t=e instanceof Error?e.message:"unknown error occurred";l.post({message:`Failed to update publish status: ${t}`,severity:"error",display:"transient"})}else l.post({message:"Cannot publish a retired API product. Please change the lifecycle status first.",severity:"error",display:"transient"})})(e),style:{marginRight:4,textTransform:"none"},children:g?"Unpublish":"Publish"}),u&&(0,s.jsx)(d.A,{size:"small",onClick:()=>{return t=e.metadata.namespace,a=e.metadata.name,U({namespace:t,name:a}),void E(!0);var t,a},title:"Edit API Product",children:(0,s.jsx)(h.A,{fontSize:"small"})}),p&&(0,s.jsx)(d.A,{size:"small",onClick:()=>(async(e,t)=>{q({namespace:e,name:t}),V(null);try{const s=await a.fetch(`${T}/api/kuadrant/requests?namespace=${e}`);if(!s.ok){const e=await(0,Y.T)(s);throw new Error(e)}const n=((await s.json()).items||[]).filter(a=>a.spec.apiName===t&&a.spec.apiNamespace===e),l=n.filter(e=>{var t;return"Approved"===(null===(t=e.status)||void 0===t?void 0:t.phase)}).length;V({requests:n.length,secrets:l})}catch(e){const t=e instanceof Error?e.message:"unknown error occurred";l.post({message:`Failed to delete access request: ${t}`,severity:"error",display:"transient"})}finally{W(!0)}})(e.metadata.namespace,e.metadata.name),title:"Delete API Product",children:(0,s.jsx)(m.A,{fontSize:"small"})})]})}}];return(0,s.jsxs)(f.Y,{themeId:"tool",children:[(0,s.jsx)(y.Y,{title:"API Products",subtitle:"Manage API products for Kubernetes",children:(0,s.jsx)(j.Y,{children:"Manage API products and plan policies"})}),(0,s.jsxs)(b.U,{children:[ke&&(0,s.jsxs)(i.A,{display:"flex",flexDirection:"column",alignItems:"center",justifyContent:"center",minHeight:300,children:[(0,s.jsx)(u.A,{}),(0,s.jsx)(o.A,{variant:"h6",style:{marginTop:16},children:"Loading data..."}),(0,s.jsx)(o.A,{variant:"body2",color:"textSecondary",children:"Preparing your data... This should only take a moment."})]}),Ie&&(0,s.jsx)(P._,{error:Ie}),Ce&&(0,s.jsxs)(i.A,{p:2,children:[(0,s.jsxs)(o.A,{color:"error",children:["unable to check permissions: ",Ce.message]}),(0,s.jsxs)(o.A,{variant:"body2",color:"textSecondary",children:["permission:"," ",ie?"kuadrant.apiproduct.create":ue?"kuadrant.apiproduct.delete":ve?"kuadrant.planpolicy.list":"unknown"]}),(0,s.jsx)(o.A,{variant:"body2",color:"textSecondary",children:"please try again or contact your administrator"})]}),!ke&&!Ie&&!Ce&&0===Te.length&&(0,s.jsx)(i.A,{className:e.emptyState,children:(0,s.jsxs)(i.A,{className:e.emptyStateContent,children:[(0,s.jsxs)(i.A,{className:e.emptyStateText,children:[(0,s.jsx)(o.A,{variant:"h4",className:e.emptyStateTitle,children:"API Product"}),(0,s.jsx)(o.A,{variant:"body1",className:e.emptyStateDescription,children:"Create API product by registering existing API, associate route and policy"}),ne&&(0,s.jsx)(c.A,{variant:"contained",color:"primary",startIcon:(0,s.jsx)(p.A,{}),onClick:()=>z(!0),children:"Create API Product"})]}),(0,s.jsx)("img",{src:Q,alt:"API Product illustration",className:e.emptyStateImage})]})}),!ke&&!Ie&&!Ce&&Te.length>0&&(0,s.jsxs)(i.A,{className:e.container,children:[(0,s.jsx)(v.Z,{sections:Ne,filters:ae,onChange:se}),(0,s.jsxs)(i.A,{className:e.tableContainer,children:[(0,s.jsx)(i.A,{display:"flex",justifyContent:"flex-end",mb:2,children:ne&&(0,s.jsx)(c.A,{variant:"contained",color:"primary",size:"small",startIcon:(0,s.jsx)(p.A,{}),onClick:()=>z(!0),children:"Create API Product"})}),0===$e.length?(0,s.jsx)(i.A,{p:4,textAlign:"center",children:(0,s.jsx)(o.A,{variant:"body1",color:"textSecondary",children:"No API products match the selected filters."})}):(0,s.jsx)(w.X,{options:{paging:$e.length>10,pageSize:20,search:!0,filtering:!1,debounceInterval:300,toolbar:!0,emptyRowsWhenPaging:!1},columns:Re,data:$e})]})]}),(0,s.jsx)(J,{open:R,onClose:()=>z(!1),onSuccess:e=>{H(e=>e+1),l.post({message:`"${e.displayName}" created successfully`,severity:"success",display:"transient"})}}),(0,s.jsx)(G.C,{open:D,onClose:()=>E(!1),onSuccess:()=>{H(e=>e+1);const e=(null==K?void 0:K.name)||"API Product";l.post({message:`"${e}" updated successfully`,severity:"success",display:"transient"})},namespace:(null==K?void 0:K.namespace)||"",name:(null==K?void 0:K.name)||""}),(0,s.jsx)(X.K,{open:M,title:"Delete API Product",description:_?`Deleting "${null==O?void 0:O.name}" will also remove:\n\n• ${_.requests} API Key(s)\n• ${_.secrets} API Key Secret(s)\n\nThis action cannot be undone.`:`Deleting "${null==O?void 0:O.name}" will also remove all associated API Keys and Secrets.\nThis action cannot be undone.`,confirmText:null==O?void 0:O.name,severity:"high",deleting:B,onConfirm:async()=>{if(O){F(!0);try{const e=await a.fetch(`${T}/api/kuadrant/apiproducts/${O.namespace}/${O.name}`,{method:"DELETE"});if(!e.ok){const t=await(0,Y.T)(e);throw new Error(t)}const t=(null==O?void 0:O.name)||"API Product";H(e=>e+1),l.post({message:`"${t}" deleted successfully`,severity:"success",display:"transient"})}catch(e){const t=e instanceof Error?e.message:"unknown error occurred";l.post({message:`Failed to delete API Product: ${t}`,severity:"error",display:"transient"})}finally{F(!1),W(!1),q(null)}}},onCancel:()=>{W(!1),q(null)}})]})]})},se=()=>(0,s.jsx)(T,{permission:Z.vs,errorMessage:"you don't have permission to view the Kuadrant page",children:(0,s.jsx)(ae,{})})}}]);
2
- //# sourceMappingURL=7005.72759857.chunk.js.map