@kuadrant/kuadrant-backstage-plugin-frontend 0.0.2-dev-7d09bfa → 0.0.2-dev-e6e0dbf

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 (152) hide show
  1. package/dist/index.d.ts +1 -4
  2. package/dist/index.esm.js +1 -2
  3. package/dist/index.esm.js.map +1 -1
  4. package/dist/plugin.esm.js +1 -9
  5. package/dist/plugin.esm.js.map +1 -1
  6. package/dist-scalprum/internal.plugin-kuadrant.1ec7b1dc71663b4b9a17.js +2 -0
  7. package/dist-scalprum/internal.plugin-kuadrant.1ec7b1dc71663b4b9a17.js.map +1 -0
  8. package/dist-scalprum/plugin-manifest.json +2 -2
  9. package/dist-scalprum/static/1085.d17e689a.chunk.js +3 -0
  10. package/dist-scalprum/static/1085.d17e689a.chunk.js.map +1 -0
  11. package/dist-scalprum/static/1645.db086648.chunk.js +3 -0
  12. package/dist-scalprum/static/{5568.5dbce633.chunk.js.LICENSE.txt → 1645.db086648.chunk.js.LICENSE.txt} +0 -9
  13. package/dist-scalprum/static/1645.db086648.chunk.js.map +1 -0
  14. package/dist-scalprum/static/2233.bf0c10d7.chunk.js +3 -0
  15. package/dist-scalprum/static/2233.bf0c10d7.chunk.js.LICENSE.txt +9 -0
  16. package/dist-scalprum/static/2233.bf0c10d7.chunk.js.map +1 -0
  17. package/dist-scalprum/static/2628.20156dc1.chunk.js +3 -0
  18. package/dist-scalprum/static/2628.20156dc1.chunk.js.map +1 -0
  19. package/dist-scalprum/static/2946.a35243f1.chunk.js +3 -0
  20. package/dist-scalprum/static/{9510.e4112e19.chunk.js.LICENSE.txt → 2946.a35243f1.chunk.js.LICENSE.txt} +10 -0
  21. package/dist-scalprum/static/2946.a35243f1.chunk.js.map +1 -0
  22. package/dist-scalprum/static/2952.b3c7e41c.chunk.js +2 -0
  23. package/dist-scalprum/static/2952.b3c7e41c.chunk.js.map +1 -0
  24. package/dist-scalprum/static/{2967.8d9c1b1f.chunk.js → 2967.6a1db3ad.chunk.js} +2 -2
  25. package/dist-scalprum/static/2967.6a1db3ad.chunk.js.map +1 -0
  26. package/dist-scalprum/static/3091.9a74ea5f.chunk.js +2 -0
  27. package/dist-scalprum/static/{65.0e01be7c.chunk.js.map → 3091.9a74ea5f.chunk.js.map} +1 -1
  28. package/dist-scalprum/static/{327.cd6b3fee.chunk.js → 327.a9a99299.chunk.js} +2 -2
  29. package/dist-scalprum/static/327.a9a99299.chunk.js.map +1 -0
  30. package/dist-scalprum/static/3504.1c2f2e54.chunk.js +3 -0
  31. package/dist-scalprum/static/3504.1c2f2e54.chunk.js.LICENSE.txt +9 -0
  32. package/dist-scalprum/static/3504.1c2f2e54.chunk.js.map +1 -0
  33. package/dist-scalprum/static/3657.882640ca.chunk.js +3 -0
  34. package/dist-scalprum/static/3657.882640ca.chunk.js.LICENSE.txt +8 -0
  35. package/dist-scalprum/static/3657.882640ca.chunk.js.map +1 -0
  36. package/dist-scalprum/static/3954.12dd8006.chunk.js +3 -0
  37. package/dist-scalprum/static/3954.12dd8006.chunk.js.LICENSE.txt +9 -0
  38. package/dist-scalprum/static/3954.12dd8006.chunk.js.map +1 -0
  39. package/dist-scalprum/static/3984.647ef00c.chunk.js +2 -0
  40. package/dist-scalprum/static/3984.647ef00c.chunk.js.map +1 -0
  41. package/dist-scalprum/static/4026.d652a7a1.chunk.js +2 -0
  42. package/dist-scalprum/static/4026.d652a7a1.chunk.js.map +1 -0
  43. package/dist-scalprum/static/{6386.6386.ce38cef3.css → 4093.4093.3597955d.css} +1 -1
  44. package/dist-scalprum/static/4093.4093.3597955d.css.map +1 -0
  45. package/dist-scalprum/static/4093.ee6f0452.chunk.js +3 -0
  46. package/dist-scalprum/static/4093.ee6f0452.chunk.js.map +1 -0
  47. package/dist-scalprum/static/4382.a7363415.chunk.js +3 -0
  48. package/dist-scalprum/static/4382.a7363415.chunk.js.map +1 -0
  49. package/dist-scalprum/static/4614.23f2a85b.chunk.js +2 -0
  50. package/dist-scalprum/static/4614.23f2a85b.chunk.js.map +1 -0
  51. package/dist-scalprum/static/{4682.3c7098a8.chunk.js → 4682.bd2dfccd.chunk.js} +2 -2
  52. package/dist-scalprum/static/4682.bd2dfccd.chunk.js.map +1 -0
  53. package/dist-scalprum/static/4935.e14465ff.chunk.js +3 -0
  54. package/dist-scalprum/static/4935.e14465ff.chunk.js.LICENSE.txt +9 -0
  55. package/dist-scalprum/static/4935.e14465ff.chunk.js.map +1 -0
  56. package/dist-scalprum/static/5010.0cd6c959.chunk.js +3 -0
  57. package/dist-scalprum/static/5010.0cd6c959.chunk.js.map +1 -0
  58. package/dist-scalprum/static/{5203.43732b3f.chunk.js → 5203.539ed3ec.chunk.js} +2 -2
  59. package/dist-scalprum/static/{5203.43732b3f.chunk.js.map → 5203.539ed3ec.chunk.js.map} +1 -1
  60. package/dist-scalprum/static/{5453.b3ee2392.chunk.js → 5453.ecdee66d.chunk.js} +2 -2
  61. package/dist-scalprum/static/{5453.b3ee2392.chunk.js.map → 5453.ecdee66d.chunk.js.map} +1 -1
  62. package/dist-scalprum/static/6038.63d1f832.chunk.js +3 -0
  63. package/dist-scalprum/static/6038.63d1f832.chunk.js.LICENSE.txt +8 -0
  64. package/dist-scalprum/static/6038.63d1f832.chunk.js.map +1 -0
  65. package/dist-scalprum/static/635.e030b18a.chunk.js +3 -0
  66. package/dist-scalprum/static/635.e030b18a.chunk.js.LICENSE.txt +9 -0
  67. package/dist-scalprum/static/635.e030b18a.chunk.js.map +1 -0
  68. package/dist-scalprum/static/6371.d45f37cc.chunk.js +2 -0
  69. package/dist-scalprum/static/6371.d45f37cc.chunk.js.map +1 -0
  70. package/dist-scalprum/static/{6800.8ec3a2eb.chunk.js → 6800.6e2a08fe.chunk.js} +2 -2
  71. package/dist-scalprum/static/{6800.8ec3a2eb.chunk.js.map → 6800.6e2a08fe.chunk.js.map} +1 -1
  72. package/dist-scalprum/static/{6840.6cc88a16.chunk.js → 6840.3bd67cf4.chunk.js} +2 -2
  73. package/dist-scalprum/static/{6840.6cc88a16.chunk.js.map → 6840.3bd67cf4.chunk.js.map} +1 -1
  74. package/dist-scalprum/static/7164.5dc4bc72.chunk.js +2 -0
  75. package/dist-scalprum/static/7164.5dc4bc72.chunk.js.map +1 -0
  76. package/dist-scalprum/static/{7791.55db7365.chunk.js → 7791.8135d440.chunk.js} +2 -2
  77. package/dist-scalprum/static/{7791.55db7365.chunk.js.map → 7791.8135d440.chunk.js.map} +1 -1
  78. package/dist-scalprum/static/8083.4763f88b.chunk.js +206 -0
  79. package/dist-scalprum/static/8083.4763f88b.chunk.js.LICENSE.txt +9 -0
  80. package/dist-scalprum/static/8083.4763f88b.chunk.js.map +1 -0
  81. package/dist-scalprum/static/{8172.e89bbae7.chunk.js → 8172.cbe1f2c4.chunk.js} +2 -2
  82. package/dist-scalprum/static/8172.cbe1f2c4.chunk.js.map +1 -0
  83. package/dist-scalprum/static/{8799.4ea4639c.chunk.js → 8799.83d049f3.chunk.js} +2 -2
  84. package/dist-scalprum/static/{8799.4ea4639c.chunk.js.map → 8799.83d049f3.chunk.js.map} +1 -1
  85. package/dist-scalprum/static/{3587.490690d6.chunk.js → 9316.d9f2fe18.chunk.js} +2 -2
  86. package/dist-scalprum/static/{3587.490690d6.chunk.js.map → 9316.d9f2fe18.chunk.js.map} +1 -1
  87. package/dist-scalprum/static/{5568.5dbce633.chunk.js → 9655.db0c82b2.chunk.js} +3 -3
  88. package/dist-scalprum/static/9655.db0c82b2.chunk.js.LICENSE.txt +8 -0
  89. package/dist-scalprum/static/9655.db0c82b2.chunk.js.map +1 -0
  90. package/dist-scalprum/static/9710.7ec34777.chunk.js +2 -0
  91. package/dist-scalprum/static/9710.7ec34777.chunk.js.map +1 -0
  92. package/dist-scalprum/static/{4302.9a59485e.chunk.js → 9738.b037daf3.chunk.js} +2 -2
  93. package/dist-scalprum/static/9738.b037daf3.chunk.js.map +1 -0
  94. package/dist-scalprum/static/exposed-PluginRoot.c3fd0cc1.chunk.js +2 -0
  95. package/dist-scalprum/static/exposed-PluginRoot.c3fd0cc1.chunk.js.map +1 -0
  96. package/dist-scalprum/static/react-syntax-highlighter_languages_highlight_http.a55e168b.chunk.js +2 -0
  97. package/dist-scalprum/static/{react-syntax-highlighter_languages_highlight_http.5859273c.chunk.js.map → react-syntax-highlighter_languages_highlight_http.a55e168b.chunk.js.map} +1 -1
  98. package/package.json +10 -11
  99. package/dist/components/ApprovalQueueCard/ApprovalQueueCard.esm.js +0 -761
  100. package/dist/components/ApprovalQueueCard/ApprovalQueueCard.esm.js.map +0 -1
  101. package/dist/components/ApprovalQueueCard/index.esm.js +0 -2
  102. package/dist/components/ApprovalQueueCard/index.esm.js.map +0 -1
  103. package/dist-scalprum/internal.plugin-kuadrant.5d9e1e73c7f21bbed39a.js +0 -2
  104. package/dist-scalprum/internal.plugin-kuadrant.5d9e1e73c7f21bbed39a.js.map +0 -1
  105. package/dist-scalprum/static/1994.19e6a1c5.chunk.js +0 -3
  106. package/dist-scalprum/static/1994.19e6a1c5.chunk.js.map +0 -1
  107. package/dist-scalprum/static/2628.0605e07f.chunk.js +0 -3
  108. package/dist-scalprum/static/2628.0605e07f.chunk.js.map +0 -1
  109. package/dist-scalprum/static/2752.df63f31c.chunk.js +0 -2
  110. package/dist-scalprum/static/2752.df63f31c.chunk.js.map +0 -1
  111. package/dist-scalprum/static/2967.8d9c1b1f.chunk.js.map +0 -1
  112. package/dist-scalprum/static/3097.582b68d3.chunk.js +0 -2
  113. package/dist-scalprum/static/3097.582b68d3.chunk.js.map +0 -1
  114. package/dist-scalprum/static/327.cd6b3fee.chunk.js.map +0 -1
  115. package/dist-scalprum/static/3584.c820a5c7.chunk.js +0 -2
  116. package/dist-scalprum/static/3584.c820a5c7.chunk.js.map +0 -1
  117. package/dist-scalprum/static/3647.67079e5f.chunk.js +0 -2
  118. package/dist-scalprum/static/3647.67079e5f.chunk.js.map +0 -1
  119. package/dist-scalprum/static/3984.7bc07774.chunk.js +0 -2
  120. package/dist-scalprum/static/3984.7bc07774.chunk.js.map +0 -1
  121. package/dist-scalprum/static/4302.9a59485e.chunk.js.map +0 -1
  122. package/dist-scalprum/static/4682.3c7098a8.chunk.js.map +0 -1
  123. package/dist-scalprum/static/5010.2228c754.chunk.js +0 -3
  124. package/dist-scalprum/static/5010.2228c754.chunk.js.map +0 -1
  125. package/dist-scalprum/static/5568.5dbce633.chunk.js.map +0 -1
  126. package/dist-scalprum/static/6272.ef31cb1c.chunk.js +0 -2
  127. package/dist-scalprum/static/6272.ef31cb1c.chunk.js.map +0 -1
  128. package/dist-scalprum/static/6371.c4899d73.chunk.js +0 -3
  129. package/dist-scalprum/static/6371.c4899d73.chunk.js.map +0 -1
  130. package/dist-scalprum/static/6386.6386.ce38cef3.css.map +0 -1
  131. package/dist-scalprum/static/6386.903891f3.chunk.js +0 -3
  132. package/dist-scalprum/static/6386.903891f3.chunk.js.map +0 -1
  133. package/dist-scalprum/static/65.0e01be7c.chunk.js +0 -2
  134. package/dist-scalprum/static/6753.76832e72.chunk.js +0 -2
  135. package/dist-scalprum/static/6753.76832e72.chunk.js.map +0 -1
  136. package/dist-scalprum/static/8172.e89bbae7.chunk.js.map +0 -1
  137. package/dist-scalprum/static/8627.111cbac9.chunk.js +0 -2
  138. package/dist-scalprum/static/8627.111cbac9.chunk.js.map +0 -1
  139. package/dist-scalprum/static/9510.e4112e19.chunk.js +0 -3
  140. package/dist-scalprum/static/9510.e4112e19.chunk.js.map +0 -1
  141. package/dist-scalprum/static/9644.7d342123.chunk.js +0 -2
  142. package/dist-scalprum/static/9644.7d342123.chunk.js.map +0 -1
  143. package/dist-scalprum/static/993.c164940e.chunk.js +0 -2
  144. package/dist-scalprum/static/993.c164940e.chunk.js.map +0 -1
  145. package/dist-scalprum/static/exposed-PluginRoot.5b6638e2.chunk.js +0 -2
  146. package/dist-scalprum/static/exposed-PluginRoot.5b6638e2.chunk.js.map +0 -1
  147. package/dist-scalprum/static/react-syntax-highlighter_languages_highlight_http.5859273c.chunk.js +0 -2
  148. /package/dist-scalprum/static/{6371.c4899d73.chunk.js.LICENSE.txt → 1085.d17e689a.chunk.js.LICENSE.txt} +0 -0
  149. /package/dist-scalprum/static/{2628.0605e07f.chunk.js.LICENSE.txt → 2628.20156dc1.chunk.js.LICENSE.txt} +0 -0
  150. /package/dist-scalprum/static/{6386.903891f3.chunk.js.LICENSE.txt → 4093.ee6f0452.chunk.js.LICENSE.txt} +0 -0
  151. /package/dist-scalprum/static/{1994.19e6a1c5.chunk.js.LICENSE.txt → 4382.a7363415.chunk.js.LICENSE.txt} +0 -0
  152. /package/dist-scalprum/static/{5010.2228c754.chunk.js.LICENSE.txt → 5010.0cd6c959.chunk.js.LICENSE.txt} +0 -0
@@ -1,2 +1,2 @@
1
- "use strict";(self.webpackChunkinternal_plugin_kuadrant=self.webpackChunkinternal_plugin_kuadrant||[]).push([[8172],{18466:(e,t,n)=>{var r=n(4293),a=n(78920);t.A=void 0;var i=a(n(95478)),o=(0,r(n(74044)).default)(i.createElement("path",{d:"M19 13h-6v6h-2v-6H5v-2h6V5h2v6h6v2z"}),"Add");t.A=o},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")},26343:(e,t,n)=>{n.d(t,{A:()=>h});var r=n(39850),a=n(24586),i=n(89575),o=n(95478),s=n(36274),l=n(7031),c=n(8109),d=o.forwardRef(function(e,t){var n,a=e.classes,l=e.className,d=e.component,h=void 0===d?"li":d,u=e.disableGutters,p=void 0!==u&&u,f=e.ListItemClasses,m=e.role,g=void 0===m?"menuitem":m,v=e.selected,x=e.tabIndex,y=(0,r.A)(e,["classes","className","component","disableGutters","ListItemClasses","role","selected","tabIndex"]);return e.disabled||(n=void 0!==x?x:-1),o.createElement(c.A,(0,i.A)({button:!0,role:g,tabIndex:n,component:h,selected:v,disableGutters:p,classes:(0,i.A)({dense:a.dense},f),className:(0,s.A)(a.root,l,v&&a.selected,!p&&a.gutters),ref:t},y))});const h=(0,l.A)(function(e){return{root:(0,i.A)({},e.typography.body1,(0,a.A)({minHeight:48,paddingTop:6,paddingBottom:6,boxSizing:"border-box",width:"auto",overflow:"hidden",whiteSpace:"nowrap"},e.breakpoints.up("sm"),{minHeight:"auto"})),gutters:{},selected:{},dense:(0,i.A)({},e.typography.body2,{minHeight:"auto"})}},{name:"MuiMenuItem"})(d)},32269:(e,t,n)=>{var r=n(4293),a=n(78920);t.A=void 0;var i=a(n(95478)),o=(0,r(n(74044)).default)(i.createElement("path",{d:"M12 4.5C7 4.5 2.73 7.61 1 12c1.73 4.39 6 7.5 11 7.5s9.27-3.11 11-7.5c-1.73-4.39-6-7.5-11-7.5zM12 17c-2.76 0-5-2.24-5-5s2.24-5 5-5 5 2.24 5 5-2.24 5-5 5zm0-8c-1.66 0-3 1.34-3 3s1.34 3 3 3 3-1.34 3-3-1.34-3-3-3z"}),"Visibility");t.A=o},35015:(e,t,n)=>{n.d(t,{A:()=>o});var r=n(95478),a=n(85608),i=n(71581);function o(e,t){void 0===t&&(t=[]);var n=function(e,t,n){void 0===t&&(t=[]),void 0===n&&(n={loading:!1});var o=(0,r.useRef)(0),s=(0,i.A)(),l=(0,r.useState)(n),c=l[0],d=l[1],h=(0,r.useCallback)(function(){for(var t=[],n=0;n<arguments.length;n++)t[n]=arguments[n];var r=++o.current;return c.loading||d(function(e){return(0,a.__assign)((0,a.__assign)({},e),{loading:!0})}),e.apply(void 0,t).then(function(e){return s()&&r===o.current&&d({value:e,loading:!1}),e},function(e){return s()&&r===o.current&&d({error:e,loading:!1}),e})},t);return[c,h]}(e,t,{loading:!0}),o=n[0],s=n[1];return(0,r.useEffect)(function(){s()},[s]),o}},37725:(e,t,n)=>{n.d(t,{N_:()=>A});var r=n(31085),a=n(22097),i=n(49203),o=n(37976),s=n(72501),l=n(53373),c=n.n(l),d=n(45250),h=n(95478),u=n(49634),p=n(39330);const f=(0,o.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"}),m=()=>{const e=(0,a.useApp)().getSystemIcon("externalLink")||p.A,t=f();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,x=window.open;if(x&&!x.__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 x.apply(this,e)};e.__backstage=!0,window.open=e}const y=e=>e instanceof Array?e.map(y).join(" ").trim():"object"==typeof e&&e?y(e?.props?.children):["string","number"].includes(typeof e)?String(e):"",A=(0,h.forwardRef)(({onClick:e,noTrack:t,externalLinkIcon:n,...o},l)=>{const h=f(),p=(0,a.useAnalytics)(),x=function(){const[e]=(0,u.createRoutesFromChildren)((0,r.jsx)(u.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,d.trimEnd)(t,"/")})(),r=g(t),i=t.startsWith(n);return r||i||(t=n.concat(t)),t})(o.to):o.to,A=y(o.children)||x,b=g(x),k=b&&!!/^https?:/.exec(x);if(v.test(x))throw new Error("Link component rejected javascript: URL as a security precaution");const w=n=>{e?.(n),t||p.captureEvent("click",A,{attributes:{to:x}})};return b?(0,r.jsxs)(i.A,{...k?{target:"_blank",rel:"noopener"}:{},...o,...o["aria-label"]?{"aria-label":`${o["aria-label"]}, Opens in a new window`}:{},ref:l,href:x,onClick:w,className:c()(h.externalLink,o.className),children:[o.children,n&&(0,r.jsx)(m,{}),(0,r.jsx)(s.A,{component:"span",className:h.visuallyHidden,children:", Opens in a new window"})]}):(0,r.jsx)(i.A,{...o,ref:l,component:u.Link,to:x,onClick:w})})},39053:(e,t,n)=>{var r=n(4293),a=n(78920);t.A=void 0;var i=a(n(95478)),o=(0,r(n(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 2zm1 15h-2v-6h2v6zm0-8h-2V7h2v2z"}),"Info");t.A=o},39330:(e,t,n)=>{var r=n(4293),a=n(78920);t.A=void 0;var i=a(n(95478)),o=(0,r(n(74044)).default)(i.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=o},52235:(e,t,n)=>{var r=n(4293),a=n(78920);t.A=void 0;var i=a(n(95478)),o=(0,r(n(74044)).default)(i.createElement("path",{d:"M12 4l-1.41 1.41L16.17 11H4v2h12.17l-5.58 5.59L12 20l8-8z"}),"ArrowForward");t.A=o},61524:(e,t,n)=>{var r=n(4293),a=n(78920);t.A=void 0;var i=a(n(95478)),o=(0,r(n(74044)).default)(i.createElement("path",{d:"M12 7c2.76 0 5 2.24 5 5 0 .65-.13 1.26-.36 1.83l2.92 2.92c1.51-1.26 2.7-2.89 3.43-4.75-1.73-4.39-6-7.5-11-7.5-1.4 0-2.74.25-3.98.7l2.16 2.16C10.74 7.13 11.35 7 12 7zM2 4.27l2.28 2.28.46.46C3.08 8.3 1.78 10.02 1 12c1.73 4.39 6 7.5 11 7.5 1.55 0 3.03-.3 4.38-.84l.42.42L19.73 22 21 20.73 3.27 3 2 4.27zM7.53 9.8l1.55 1.55c-.05.21-.08.43-.08.65 0 1.66 1.34 3 3 3 .22 0 .44-.03.65-.08l1.55 1.55c-.67.33-1.41.53-2.2.53-2.76 0-5-2.24-5-5 0-.79.2-1.53.53-2.2zm4.31-.78l3.15 3.15.02-.16c0-1.66-1.34-3-3-3l-.17.01z"}),"VisibilityOff");t.A=o},77225:(e,t,n)=>{var r=n(4293),a=n(78920);t.A=void 0;var i=a(n(95478)),o=(0,r(n(74044)).default)(i.createElement("path",{d:"M1 21h22L12 2 1 21zm12-3h-2v-2h2v2zm0-4h-2v-4h2v4z"}),"Warning");t.A=o},96040:(e,t,n)=>{n.d(t,{n:()=>S});var r=n(31085),a=n(40703),i=n(59469),o=n(48653),s=n(45685),l=n(37197),c=n(37976),d=n(53373),h=n.n(d),u=n(10394),p=n(72501),f=n(52235),m=n(37725);const g=(0,c.makeStyles)(e=>({root:{maxWidth:"fit-content",padding:e.spacing(2,2,2,2.5)},boxTitle:{margin:0,color:e.palette.textSubtle},arrow:{color:e.palette.textSubtle}}),{name:"BackstageBottomLink"});function v(e){const{link:t,title:n,onClick:a}=e,i=g();return(0,r.jsxs)(u.A,{children:[(0,r.jsx)(l.A,{}),(0,r.jsx)(m.N_,{to:t,onClick:a,underline:"none",children:(0,r.jsxs)(u.A,{display:"flex",alignItems:"center",className:i.root,children:[(0,r.jsx)(u.A,{className:i.boxTitle,fontWeight:"fontWeightBold",m:1,children:(0,r.jsx)(p.A,{children:(0,r.jsx)("strong",{children:n})})}),(0,r.jsx)(f.A,{className:i.arrow})]})})]})}var x=n(95478),y=n(64947);const A=(0,x.forwardRef)((e,t)=>(0,r.jsx)(m.N_,{ref:t,...e,color:"initial"})),b=(0,x.forwardRef)((e,t)=>(0,r.jsx)(y.A,{ref:t,component:A,...e}));var k=n(34169),w=n(21006),j=n(67550);const C=e=>{const{slackChannel:t}=e,{t:n}=(0,j.i)(w.O);return t?"string"==typeof t?(0,r.jsx)(p.A,{children:n("errorBoundary.title",{slackChannel:t})}):t.href?(0,r.jsx)(b,{to:t.href,variant:"contained",children:t.name}):(0,r.jsx)(p.A,{children:n("errorBoundary.title",{slackChannel:t.name})}):null},E=class extends x.Component{constructor(e){super(e),this.state={error:void 0,errorInfo:void 0}}componentDidCatch(e,t){console.error(`ErrorBoundary, error: ${e}`,{error:e,errorInfo:t}),this.setState({error:e,errorInfo:t})}render(){const{slackChannel:e,children:t}=this.props,{error:n}=this.state;return n?(0,r.jsx)(k.b,{title:"Something Went Wrong",error:n,children:(0,r.jsx)(C,{slackChannel:e})}):t}},I=(0,c.makeStyles)(e=>({noPadding:{padding:0,"&:last-child":{paddingBottom:0}},contentAlignBottom:{display:"flex",alignItems:"self-end"},header:{padding:e.spacing(2,2,2,2.5)},headerTitle:{fontWeight:e.typography.fontWeightBold},headerSubheader:{paddingTop:e.spacing(1)},headerAvatar:{},headerAction:{},headerContent:{},subheader:{display:"flex"}}),{name:"BackstageInfoCard"}),L=(0,c.withStyles)(e=>({root:{display:"inline-block",padding:e.spacing(8,8,0,0),float:"right"}}),{name:"BackstageInfoCardCardActionsTopRight"})(i.A),N={card:{flex:{display:"flex",flexDirection:"column"},fullHeight:{display:"flex",flexDirection:"column",height:"100%"},gridItem:{display:"flex",flexDirection:"column",height:"calc(100% - 10px)",marginBottom:"10px",breakInside:"avoid-page","@media print":{height:"auto"}}},cardContent:{fullHeight:{flex:1},gridItem:{flex:1}}};function S(e){const{title:t,subheader:n,divider:c=!0,deepLink:d,slackChannel:u,errorBoundaryProps:p,variant:f,alignContent:m="normal",children:g,headerStyle:x,headerProps:y,icon:A,action:b,actionsClassName:k,actions:w,cardClassName:j,actionsTopRight:C,className:S,noPadding:z,titleTypographyProps:_,subheaderTypographyProps:B}=e,R=I();let M={},T={};f&&f.split(/[\s]+/g).forEach(e=>{M={...M,...N.card[e]},T={...T,...N.cardContent[e]}});const H=p||(u?{slackChannel:u}:{});return(0,r.jsx)(a.A,{style:M,className:S,children:(0,r.jsxs)(E,{...H,children:[t&&(0,r.jsx)(s.A,{classes:{root:h()(R.header),title:R.headerTitle,subheader:R.headerSubheader,avatar:R.headerAvatar,action:R.headerAction,content:R.headerContent},title:t,subheader:n||A?(0,r.jsxs)("div",{"data-testid":"info-card-subheader",children:[n&&(0,r.jsx)("div",{className:R.subheader,children:n}),A]}):null,action:b,style:{...x},titleTypographyProps:_,subheaderTypographyProps:B,...y}),C&&(0,r.jsx)(L,{children:C}),c&&(0,r.jsx)(l.A,{}),(0,r.jsx)(o.A,{className:h()(j,{[R.noPadding]:z,[R.contentAlignBottom]:"bottom"===m}),style:T,children:g}),w&&(0,r.jsx)(i.A,{className:k,children:w}),d&&(0,r.jsx)(v,{...d})]})})}}}]);
2
- //# sourceMappingURL=8172.e89bbae7.chunk.js.map
1
+ "use strict";(self.webpackChunkinternal_plugin_kuadrant=self.webpackChunkinternal_plugin_kuadrant||[]).push([[8172],{18466:(e,t,n)=>{var r=n(4293),a=n(78920);t.A=void 0;var i=a(n(95478)),o=(0,r(n(74044)).default)(i.createElement("path",{d:"M19 13h-6v6h-2v-6H5v-2h6V5h2v6h6v2z"}),"Add");t.A=o},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")},26343:(e,t,n)=>{n.d(t,{A:()=>h});var r=n(39850),a=n(24586),i=n(89575),o=n(95478),s=n(36274),l=n(7031),c=n(8109),d=o.forwardRef(function(e,t){var n,a=e.classes,l=e.className,d=e.component,h=void 0===d?"li":d,u=e.disableGutters,p=void 0!==u&&u,f=e.ListItemClasses,m=e.role,g=void 0===m?"menuitem":m,v=e.selected,x=e.tabIndex,y=(0,r.A)(e,["classes","className","component","disableGutters","ListItemClasses","role","selected","tabIndex"]);return e.disabled||(n=void 0!==x?x:-1),o.createElement(c.A,(0,i.A)({button:!0,role:g,tabIndex:n,component:h,selected:v,disableGutters:p,classes:(0,i.A)({dense:a.dense},f),className:(0,s.A)(a.root,l,v&&a.selected,!p&&a.gutters),ref:t},y))});const h=(0,l.A)(function(e){return{root:(0,i.A)({},e.typography.body1,(0,a.A)({minHeight:48,paddingTop:6,paddingBottom:6,boxSizing:"border-box",width:"auto",overflow:"hidden",whiteSpace:"nowrap"},e.breakpoints.up("sm"),{minHeight:"auto"})),gutters:{},selected:{},dense:(0,i.A)({},e.typography.body2,{minHeight:"auto"})}},{name:"MuiMenuItem"})(d)},32269:(e,t,n)=>{var r=n(4293),a=n(78920);t.A=void 0;var i=a(n(95478)),o=(0,r(n(74044)).default)(i.createElement("path",{d:"M12 4.5C7 4.5 2.73 7.61 1 12c1.73 4.39 6 7.5 11 7.5s9.27-3.11 11-7.5c-1.73-4.39-6-7.5-11-7.5zM12 17c-2.76 0-5-2.24-5-5s2.24-5 5-5 5 2.24 5 5-2.24 5-5 5zm0-8c-1.66 0-3 1.34-3 3s1.34 3 3 3 3-1.34 3-3-1.34-3-3-3z"}),"Visibility");t.A=o},35015:(e,t,n)=>{n.d(t,{A:()=>o});var r=n(95478),a=n(85608),i=n(71581);function o(e,t){void 0===t&&(t=[]);var n=function(e,t,n){void 0===t&&(t=[]),void 0===n&&(n={loading:!1});var o=(0,r.useRef)(0),s=(0,i.A)(),l=(0,r.useState)(n),c=l[0],d=l[1],h=(0,r.useCallback)(function(){for(var t=[],n=0;n<arguments.length;n++)t[n]=arguments[n];var r=++o.current;return c.loading||d(function(e){return(0,a.__assign)((0,a.__assign)({},e),{loading:!0})}),e.apply(void 0,t).then(function(e){return s()&&r===o.current&&d({value:e,loading:!1}),e},function(e){return s()&&r===o.current&&d({error:e,loading:!1}),e})},t);return[c,h]}(e,t,{loading:!0}),o=n[0],s=n[1];return(0,r.useEffect)(function(){s()},[s]),o}},37725:(e,t,n)=>{n.d(t,{N_:()=>A});var r=n(31085),a=n(22097),i=n(49203),o=n(37976),s=n(72501),l=n(53373),c=n.n(l),d=n(45250),h=n(95478),u=n(49634),p=n(39330);const f=(0,o.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"}),m=()=>{const e=(0,a.useApp)().getSystemIcon("externalLink")||p.A,t=f();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,x=window.open;if(x&&!x.__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 x.apply(this,e)};e.__backstage=!0,window.open=e}const y=e=>e instanceof Array?e.map(y).join(" ").trim():"object"==typeof e&&e?y(e?.props?.children):["string","number"].includes(typeof e)?String(e):"",A=(0,h.forwardRef)(({onClick:e,noTrack:t,externalLinkIcon:n,...o},l)=>{const h=f(),p=(0,a.useAnalytics)(),x=function(){const[e]=(0,u.createRoutesFromChildren)((0,r.jsx)(u.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,d.trimEnd)(t,"/")})(),r=g(t),i=t.startsWith(n);return r||i||(t=n.concat(t)),t})(o.to):o.to,A=y(o.children)||x,b=g(x),k=b&&!!/^https?:/.exec(x);if(v.test(x))throw new Error("Link component rejected javascript: URL as a security precaution");const w=n=>{e?.(n),t||p.captureEvent("click",A,{attributes:{to:x}})};return b?(0,r.jsxs)(i.A,{...k?{target:"_blank",rel:"noopener"}:{},...o,...o["aria-label"]?{"aria-label":`${o["aria-label"]}, Opens in a new window`}:{},ref:l,href:x,onClick:w,className:c()(h.externalLink,o.className),children:[o.children,n&&(0,r.jsx)(m,{}),(0,r.jsx)(s.A,{component:"span",className:h.visuallyHidden,children:", Opens in a new window"})]}):(0,r.jsx)(i.A,{...o,ref:l,component:u.Link,to:x,onClick:w})})},39053:(e,t,n)=>{var r=n(4293),a=n(78920);t.A=void 0;var i=a(n(95478)),o=(0,r(n(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 2zm1 15h-2v-6h2v6zm0-8h-2V7h2v2z"}),"Info");t.A=o},39330:(e,t,n)=>{var r=n(4293),a=n(78920);t.A=void 0;var i=a(n(95478)),o=(0,r(n(74044)).default)(i.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=o},52235:(e,t,n)=>{var r=n(4293),a=n(78920);t.A=void 0;var i=a(n(95478)),o=(0,r(n(74044)).default)(i.createElement("path",{d:"M12 4l-1.41 1.41L16.17 11H4v2h12.17l-5.58 5.59L12 20l8-8z"}),"ArrowForward");t.A=o},61524:(e,t,n)=>{var r=n(4293),a=n(78920);t.A=void 0;var i=a(n(95478)),o=(0,r(n(74044)).default)(i.createElement("path",{d:"M12 7c2.76 0 5 2.24 5 5 0 .65-.13 1.26-.36 1.83l2.92 2.92c1.51-1.26 2.7-2.89 3.43-4.75-1.73-4.39-6-7.5-11-7.5-1.4 0-2.74.25-3.98.7l2.16 2.16C10.74 7.13 11.35 7 12 7zM2 4.27l2.28 2.28.46.46C3.08 8.3 1.78 10.02 1 12c1.73 4.39 6 7.5 11 7.5 1.55 0 3.03-.3 4.38-.84l.42.42L19.73 22 21 20.73 3.27 3 2 4.27zM7.53 9.8l1.55 1.55c-.05.21-.08.43-.08.65 0 1.66 1.34 3 3 3 .22 0 .44-.03.65-.08l1.55 1.55c-.67.33-1.41.53-2.2.53-2.76 0-5-2.24-5-5 0-.79.2-1.53.53-2.2zm4.31-.78l3.15 3.15.02-.16c0-1.66-1.34-3-3-3l-.17.01z"}),"VisibilityOff");t.A=o},77225:(e,t,n)=>{var r=n(4293),a=n(78920);t.A=void 0;var i=a(n(95478)),o=(0,r(n(74044)).default)(i.createElement("path",{d:"M1 21h22L12 2 1 21zm12-3h-2v-2h2v2zm0-4h-2v-4h2v4z"}),"Warning");t.A=o},96040:(e,t,n)=>{n.d(t,{n:()=>S});var r=n(31085),a=n(40703),i=n(59469),o=n(48653),s=n(45685),l=n(37197),c=n(37976),d=n(53373),h=n.n(d),u=n(10394),p=n(72501),f=n(52235),m=n(37725);const g=(0,c.makeStyles)(e=>({root:{maxWidth:"fit-content",padding:e.spacing(2,2,2,2.5)},boxTitle:{margin:0,color:e.palette.textSubtle},arrow:{color:e.palette.textSubtle}}),{name:"BackstageBottomLink"});function v(e){const{link:t,title:n,onClick:a}=e,i=g();return(0,r.jsxs)(u.A,{children:[(0,r.jsx)(l.A,{}),(0,r.jsx)(m.N_,{to:t,onClick:a,underline:"none",children:(0,r.jsxs)(u.A,{display:"flex",alignItems:"center",className:i.root,children:[(0,r.jsx)(u.A,{className:i.boxTitle,fontWeight:"fontWeightBold",m:1,children:(0,r.jsx)(p.A,{children:(0,r.jsx)("strong",{children:n})})}),(0,r.jsx)(f.A,{className:i.arrow})]})})]})}var x=n(95478),y=n(64947);const A=(0,x.forwardRef)((e,t)=>(0,r.jsx)(m.N_,{ref:t,...e,color:"initial"})),b=(0,x.forwardRef)((e,t)=>(0,r.jsx)(y.A,{ref:t,component:A,...e}));var k=n(56573),w=n(21006),j=n(75501);const C=e=>{const{slackChannel:t}=e,{t:n}=(0,j.useTranslationRef)(w.O);return t?"string"==typeof t?(0,r.jsx)(p.A,{children:n("errorBoundary.title",{slackChannel:t})}):t.href?(0,r.jsx)(b,{to:t.href,variant:"contained",children:t.name}):(0,r.jsx)(p.A,{children:n("errorBoundary.title",{slackChannel:t.name})}):null},E=class extends x.Component{constructor(e){super(e),this.state={error:void 0,errorInfo:void 0}}componentDidCatch(e,t){console.error(`ErrorBoundary, error: ${e}`,{error:e,errorInfo:t}),this.setState({error:e,errorInfo:t})}render(){const{slackChannel:e,children:t}=this.props,{error:n}=this.state;return n?(0,r.jsx)(k.b,{title:"Something Went Wrong",error:n,children:(0,r.jsx)(C,{slackChannel:e})}):t}},I=(0,c.makeStyles)(e=>({noPadding:{padding:0,"&:last-child":{paddingBottom:0}},contentAlignBottom:{display:"flex",alignItems:"self-end"},header:{padding:e.spacing(2,2,2,2.5)},headerTitle:{fontWeight:e.typography.fontWeightBold},headerSubheader:{paddingTop:e.spacing(1)},headerAvatar:{},headerAction:{},headerContent:{},subheader:{display:"flex"}}),{name:"BackstageInfoCard"}),L=(0,c.withStyles)(e=>({root:{display:"inline-block",padding:e.spacing(8,8,0,0),float:"right"}}),{name:"BackstageInfoCardCardActionsTopRight"})(i.A),N={card:{flex:{display:"flex",flexDirection:"column"},fullHeight:{display:"flex",flexDirection:"column",height:"100%"},gridItem:{display:"flex",flexDirection:"column",height:"calc(100% - 10px)",marginBottom:"10px",breakInside:"avoid-page","@media print":{height:"auto"}}},cardContent:{fullHeight:{flex:1},gridItem:{flex:1}}};function S(e){const{title:t,subheader:n,divider:c=!0,deepLink:d,slackChannel:u,errorBoundaryProps:p,variant:f,alignContent:m="normal",children:g,headerStyle:x,headerProps:y,icon:A,action:b,actionsClassName:k,actions:w,cardClassName:j,actionsTopRight:C,className:S,noPadding:z,titleTypographyProps:_,subheaderTypographyProps:B}=e,R=I();let T={},M={};f&&f.split(/[\s]+/g).forEach(e=>{T={...T,...N.card[e]},M={...M,...N.cardContent[e]}});const H=p||(u?{slackChannel:u}:{});return(0,r.jsx)(a.A,{style:T,className:S,children:(0,r.jsxs)(E,{...H,children:[t&&(0,r.jsx)(s.A,{classes:{root:h()(R.header),title:R.headerTitle,subheader:R.headerSubheader,avatar:R.headerAvatar,action:R.headerAction,content:R.headerContent},title:t,subheader:n||A?(0,r.jsxs)("div",{"data-testid":"info-card-subheader",children:[n&&(0,r.jsx)("div",{className:R.subheader,children:n}),A]}):null,action:b,style:{...x},titleTypographyProps:_,subheaderTypographyProps:B,...y}),C&&(0,r.jsx)(L,{children:C}),c&&(0,r.jsx)(l.A,{}),(0,r.jsx)(o.A,{className:h()(j,{[R.noPadding]:z,[R.contentAlignBottom]:"bottom"===m}),style:M,children:g}),w&&(0,r.jsx)(i.A,{className:k,children:w}),d&&(0,r.jsx)(v,{...d})]})})}}}]);
2
+ //# sourceMappingURL=8172.cbe1f2c4.chunk.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"static/8172.cbe1f2c4.chunk.js","mappings":"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","sources":["webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/icons/Add.js","webpack://internal.plugin-kuadrant/./node_modules/@backstage/plugin-catalog-react/dist/hooks/useEntity.esm.js","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/core/esm/MenuItem/MenuItem.js","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/icons/Visibility.js","webpack://internal.plugin-kuadrant/../../node_modules/react-use/esm/useAsync.js","webpack://internal.plugin-kuadrant/../../node_modules/react-use/esm/useAsyncFn.js","webpack://internal.plugin-kuadrant/./node_modules/@backstage/core-components/dist/components/Link/Link.esm.js","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/icons/Info.js","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/icons/OpenInNew.js","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/icons/ArrowForward.js","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/icons/VisibilityOff.js","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/icons/Warning.js","webpack://internal.plugin-kuadrant/./node_modules/@backstage/core-components/dist/layout/BottomLink/BottomLink.esm.js","webpack://internal.plugin-kuadrant/./node_modules/@backstage/core-components/dist/components/LinkButton/LinkButton.esm.js","webpack://internal.plugin-kuadrant/./node_modules/@backstage/core-components/dist/layout/ErrorBoundary/ErrorBoundary.esm.js","webpack://internal.plugin-kuadrant/./node_modules/@backstage/core-components/dist/layout/InfoCard/InfoCard.esm.js"],"sourcesContent":["\"use strict\";\n\nvar _interopRequireDefault = require(\"@babel/runtime/helpers/interopRequireDefault\");\n\nvar _interopRequireWildcard = require(\"@babel/runtime/helpers/interopRequireWildcard\");\n\nObject.defineProperty(exports, \"__esModule\", {\n value: true\n});\nexports.default = void 0;\n\nvar React = _interopRequireWildcard(require(\"react\"));\n\nvar _createSvgIcon = _interopRequireDefault(require(\"./utils/createSvgIcon\"));\n\nvar _default = (0, _createSvgIcon.default)( /*#__PURE__*/React.createElement(\"path\", {\n d: \"M19 13h-6v6h-2v-6H5v-2h6V5h2v6h6v2z\"\n}), 'Add');\n\nexports.default = _default;","import { 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 _objectWithoutProperties from \"@babel/runtime/helpers/esm/objectWithoutProperties\";\nimport _defineProperty from \"@babel/runtime/helpers/esm/defineProperty\";\nimport _extends from \"@babel/runtime/helpers/esm/extends\";\nimport * as React from 'react';\nimport PropTypes from 'prop-types';\nimport clsx from 'clsx';\nimport withStyles from '../styles/withStyles';\nimport ListItem from '../ListItem';\nexport var styles = function styles(theme) {\n return {\n /* Styles applied to the root element. */\n root: _extends({}, theme.typography.body1, _defineProperty({\n minHeight: 48,\n paddingTop: 6,\n paddingBottom: 6,\n boxSizing: 'border-box',\n width: 'auto',\n overflow: 'hidden',\n whiteSpace: 'nowrap'\n }, theme.breakpoints.up('sm'), {\n minHeight: 'auto'\n })),\n // TODO v5: remove\n\n /* Styles applied to the root element if `disableGutters={false}`. */\n gutters: {},\n\n /* Styles applied to the root element if `selected={true}`. */\n selected: {},\n\n /* Styles applied to the root element if dense. */\n dense: _extends({}, theme.typography.body2, {\n minHeight: 'auto'\n })\n };\n};\nvar MenuItem = /*#__PURE__*/React.forwardRef(function MenuItem(props, ref) {\n var classes = props.classes,\n className = props.className,\n _props$component = props.component,\n component = _props$component === void 0 ? 'li' : _props$component,\n _props$disableGutters = props.disableGutters,\n disableGutters = _props$disableGutters === void 0 ? false : _props$disableGutters,\n ListItemClasses = props.ListItemClasses,\n _props$role = props.role,\n role = _props$role === void 0 ? 'menuitem' : _props$role,\n selected = props.selected,\n tabIndexProp = props.tabIndex,\n other = _objectWithoutProperties(props, [\"classes\", \"className\", \"component\", \"disableGutters\", \"ListItemClasses\", \"role\", \"selected\", \"tabIndex\"]);\n\n var tabIndex;\n\n if (!props.disabled) {\n tabIndex = tabIndexProp !== undefined ? tabIndexProp : -1;\n }\n\n return /*#__PURE__*/React.createElement(ListItem, _extends({\n button: true,\n role: role,\n tabIndex: tabIndex,\n component: component,\n selected: selected,\n disableGutters: disableGutters,\n classes: _extends({\n dense: classes.dense\n }, ListItemClasses),\n className: clsx(classes.root, className, selected && classes.selected, !disableGutters && classes.gutters),\n ref: ref\n }, other));\n});\nprocess.env.NODE_ENV !== \"production\" ? MenuItem.propTypes = {\n /**\n * Menu item contents.\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.isRequired,\n\n /**\n * @ignore\n */\n className: PropTypes.string,\n\n /**\n * The component used for the root node.\n * Either a string to use a HTML element or a component.\n */\n component: PropTypes\n /* @typescript-to-proptypes-ignore */\n .elementType,\n\n /**\n * If `true`, compact vertical padding designed for keyboard and mouse input will be used.\n */\n dense: PropTypes.bool,\n\n /**\n * @ignore\n */\n disabled: PropTypes.bool,\n\n /**\n * If `true`, the left and right padding is removed.\n */\n disableGutters: PropTypes.bool,\n\n /**\n * `classes` prop applied to the [`ListItem`](/api/list-item/) element.\n */\n ListItemClasses: PropTypes.object,\n\n /**\n * @ignore\n */\n role: PropTypes.string,\n\n /**\n * @ignore\n */\n selected: PropTypes.bool,\n\n /**\n * @ignore\n */\n tabIndex: PropTypes.number\n} : void 0;\nexport default withStyles(styles, {\n name: 'MuiMenuItem'\n})(MenuItem);","\"use strict\";\n\nvar _interopRequireDefault = require(\"@babel/runtime/helpers/interopRequireDefault\");\n\nvar _interopRequireWildcard = require(\"@babel/runtime/helpers/interopRequireWildcard\");\n\nObject.defineProperty(exports, \"__esModule\", {\n value: true\n});\nexports.default = void 0;\n\nvar React = _interopRequireWildcard(require(\"react\"));\n\nvar _createSvgIcon = _interopRequireDefault(require(\"./utils/createSvgIcon\"));\n\nvar _default = (0, _createSvgIcon.default)( /*#__PURE__*/React.createElement(\"path\", {\n d: \"M12 4.5C7 4.5 2.73 7.61 1 12c1.73 4.39 6 7.5 11 7.5s9.27-3.11 11-7.5c-1.73-4.39-6-7.5-11-7.5zM12 17c-2.76 0-5-2.24-5-5s2.24-5 5-5 5 2.24 5 5-2.24 5-5 5zm0-8c-1.66 0-3 1.34-3 3s1.34 3 3 3 3-1.34 3-3-1.34-3-3-3z\"\n}), 'Visibility');\n\nexports.default = _default;","import { useEffect } from 'react';\nimport useAsyncFn from './useAsyncFn';\nexport default function useAsync(fn, deps) {\n if (deps === void 0) { deps = []; }\n var _a = useAsyncFn(fn, deps, {\n loading: true,\n }), state = _a[0], callback = _a[1];\n useEffect(function () {\n callback();\n }, [callback]);\n return state;\n}\n","import { __assign } from \"tslib\";\nimport { useCallback, useRef, useState } from 'react';\nimport useMountedState from './useMountedState';\nexport default function useAsyncFn(fn, deps, initialState) {\n if (deps === void 0) { deps = []; }\n if (initialState === void 0) { initialState = { loading: false }; }\n var lastCallId = useRef(0);\n var isMounted = useMountedState();\n var _a = useState(initialState), state = _a[0], set = _a[1];\n var callback = useCallback(function () {\n var args = [];\n for (var _i = 0; _i < arguments.length; _i++) {\n args[_i] = arguments[_i];\n }\n var callId = ++lastCallId.current;\n if (!state.loading) {\n set(function (prevState) { return (__assign(__assign({}, prevState), { loading: true })); });\n }\n return fn.apply(void 0, args).then(function (value) {\n isMounted() && callId === lastCallId.current && set({ value: value, loading: false });\n return value;\n }, function (error) {\n isMounted() && callId === lastCallId.current && set({ error: error, loading: false });\n return error;\n });\n }, deps);\n return [state, callback];\n}\n","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: \"M12 2C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10S17.52 2 12 2zm1 15h-2v-6h2v6zm0-8h-2V7h2v2z\"\n}), 'Info');\n\nexports.default = _default;","\"use strict\";\n\nvar _interopRequireDefault = require(\"@babel/runtime/helpers/interopRequireDefault\");\n\nvar _interopRequireWildcard = require(\"@babel/runtime/helpers/interopRequireWildcard\");\n\nObject.defineProperty(exports, \"__esModule\", {\n value: true\n});\nexports.default = void 0;\n\nvar React = _interopRequireWildcard(require(\"react\"));\n\nvar _createSvgIcon = _interopRequireDefault(require(\"./utils/createSvgIcon\"));\n\nvar _default = (0, _createSvgIcon.default)( /*#__PURE__*/React.createElement(\"path\", {\n d: \"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;","\"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 4l-1.41 1.41L16.17 11H4v2h12.17l-5.58 5.59L12 20l8-8z\"\n}), 'ArrowForward');\n\nexports.default = _default;","\"use strict\";\n\nvar _interopRequireDefault = require(\"@babel/runtime/helpers/interopRequireDefault\");\n\nvar _interopRequireWildcard = require(\"@babel/runtime/helpers/interopRequireWildcard\");\n\nObject.defineProperty(exports, \"__esModule\", {\n value: true\n});\nexports.default = void 0;\n\nvar React = _interopRequireWildcard(require(\"react\"));\n\nvar _createSvgIcon = _interopRequireDefault(require(\"./utils/createSvgIcon\"));\n\nvar _default = (0, _createSvgIcon.default)( /*#__PURE__*/React.createElement(\"path\", {\n d: \"M12 7c2.76 0 5 2.24 5 5 0 .65-.13 1.26-.36 1.83l2.92 2.92c1.51-1.26 2.7-2.89 3.43-4.75-1.73-4.39-6-7.5-11-7.5-1.4 0-2.74.25-3.98.7l2.16 2.16C10.74 7.13 11.35 7 12 7zM2 4.27l2.28 2.28.46.46C3.08 8.3 1.78 10.02 1 12c1.73 4.39 6 7.5 11 7.5 1.55 0 3.03-.3 4.38-.84l.42.42L19.73 22 21 20.73 3.27 3 2 4.27zM7.53 9.8l1.55 1.55c-.05.21-.08.43-.08.65 0 1.66 1.34 3 3 3 .22 0 .44-.03.65-.08l1.55 1.55c-.67.33-1.41.53-2.2.53-2.76 0-5-2.24-5-5 0-.79.2-1.53.53-2.2zm4.31-.78l3.15 3.15.02-.16c0-1.66-1.34-3-3-3l-.17.01z\"\n}), 'VisibilityOff');\n\nexports.default = _default;","\"use strict\";\n\nvar _interopRequireDefault = require(\"@babel/runtime/helpers/interopRequireDefault\");\n\nvar _interopRequireWildcard = require(\"@babel/runtime/helpers/interopRequireWildcard\");\n\nObject.defineProperty(exports, \"__esModule\", {\n value: true\n});\nexports.default = void 0;\n\nvar React = _interopRequireWildcard(require(\"react\"));\n\nvar _createSvgIcon = _interopRequireDefault(require(\"./utils/createSvgIcon\"));\n\nvar _default = (0, _createSvgIcon.default)( /*#__PURE__*/React.createElement(\"path\", {\n d: \"M1 21h22L12 2 1 21zm12-3h-2v-2h2v2zm0-4h-2v-4h2v4z\"\n}), 'Warning');\n\nexports.default = _default;","import { jsxs, jsx } from 'react/jsx-runtime';\nimport Box from '@material-ui/core/Box';\nimport Divider from '@material-ui/core/Divider';\nimport { makeStyles } from '@material-ui/core/styles';\nimport Typography from '@material-ui/core/Typography';\nimport ArrowIcon from '@material-ui/icons/ArrowForward';\nimport { Link } from '../../components/Link/Link.esm.js';\n\nconst useStyles = makeStyles(\n (theme) => ({\n root: {\n maxWidth: \"fit-content\",\n padding: theme.spacing(2, 2, 2, 2.5)\n },\n boxTitle: {\n margin: 0,\n color: theme.palette.textSubtle\n },\n arrow: {\n color: theme.palette.textSubtle\n }\n }),\n { name: \"BackstageBottomLink\" }\n);\nfunction BottomLink(props) {\n const { link, title, onClick } = props;\n const classes = useStyles();\n return /* @__PURE__ */ jsxs(Box, { children: [\n /* @__PURE__ */ jsx(Divider, {}),\n /* @__PURE__ */ jsx(Link, { to: link, onClick, underline: \"none\", children: /* @__PURE__ */ jsxs(Box, { display: \"flex\", alignItems: \"center\", className: classes.root, children: [\n /* @__PURE__ */ jsx(Box, { className: classes.boxTitle, fontWeight: \"fontWeightBold\", m: 1, children: /* @__PURE__ */ jsx(Typography, { children: /* @__PURE__ */ jsx(\"strong\", { children: title }) }) }),\n /* @__PURE__ */ jsx(ArrowIcon, { className: classes.arrow })\n ] }) })\n ] });\n}\n\nexport { BottomLink };\n//# sourceMappingURL=BottomLink.esm.js.map\n","import { jsx } from 'react/jsx-runtime';\nimport Button$1 from '@material-ui/core/Button';\nimport { forwardRef } from 'react';\nimport { Link } from '../Link/Link.esm.js';\n\nconst LinkWrapper = forwardRef((props, ref) => /* @__PURE__ */ jsx(Link, { ref, ...props, color: \"initial\" }));\nconst LinkButton = forwardRef((props, ref) => /* @__PURE__ */ jsx(Button$1, { ref, component: LinkWrapper, ...props }));\nconst Button = LinkButton;\n\nexport { Button, LinkButton };\n//# sourceMappingURL=LinkButton.esm.js.map\n","import { jsx } from 'react/jsx-runtime';\nimport Typography from '@material-ui/core/Typography';\nimport { Component } from 'react';\nimport { LinkButton } from '../../components/LinkButton/LinkButton.esm.js';\nimport { ErrorPanel } from '../../components/ErrorPanel/ErrorPanel.esm.js';\nimport { coreComponentsTranslationRef } from '../../translation.esm.js';\nimport { useTranslationRef } from '@backstage/core-plugin-api/alpha';\n\nconst SlackLink = (props) => {\n const { slackChannel } = props;\n const { t } = useTranslationRef(coreComponentsTranslationRef);\n if (!slackChannel) {\n return null;\n } else if (typeof slackChannel === \"string\") {\n return /* @__PURE__ */ jsx(Typography, { children: t(\"errorBoundary.title\", { slackChannel }) });\n } else if (!slackChannel.href) {\n return /* @__PURE__ */ jsx(Typography, { children: t(\"errorBoundary.title\", {\n slackChannel: slackChannel.name\n }) });\n }\n return /* @__PURE__ */ jsx(LinkButton, { to: slackChannel.href, variant: \"contained\", children: slackChannel.name });\n};\nconst ErrorBoundary = class ErrorBoundary2 extends Component {\n constructor(props) {\n super(props);\n this.state = {\n error: void 0,\n errorInfo: void 0\n };\n }\n componentDidCatch(error, errorInfo) {\n console.error(`ErrorBoundary, error: ${error}`, { error, errorInfo });\n this.setState({ error, errorInfo });\n }\n render() {\n const { slackChannel, children } = this.props;\n const { error } = this.state;\n if (!error) {\n return children;\n }\n return /* @__PURE__ */ jsx(ErrorPanel, { title: \"Something Went Wrong\", error, children: /* @__PURE__ */ jsx(SlackLink, { slackChannel }) });\n }\n};\n\nexport { ErrorBoundary };\n//# sourceMappingURL=ErrorBoundary.esm.js.map\n","import { jsx, jsxs } from 'react/jsx-runtime';\nimport Card from '@material-ui/core/Card';\nimport CardActions from '@material-ui/core/CardActions';\nimport CardContent from '@material-ui/core/CardContent';\nimport CardHeader from '@material-ui/core/CardHeader';\nimport Divider from '@material-ui/core/Divider';\nimport { makeStyles, withStyles } from '@material-ui/core/styles';\nimport classNames from 'classnames';\nimport { BottomLink } from '../BottomLink/BottomLink.esm.js';\nimport { ErrorBoundary } from '../ErrorBoundary/ErrorBoundary.esm.js';\n\nconst useStyles = makeStyles(\n (theme) => ({\n noPadding: {\n padding: 0,\n \"&:last-child\": {\n paddingBottom: 0\n }\n },\n contentAlignBottom: {\n display: \"flex\",\n alignItems: \"self-end\"\n },\n header: {\n padding: theme.spacing(2, 2, 2, 2.5)\n },\n headerTitle: {\n fontWeight: theme.typography.fontWeightBold\n },\n headerSubheader: {\n paddingTop: theme.spacing(1)\n },\n headerAvatar: {},\n headerAction: {},\n headerContent: {},\n subheader: {\n display: \"flex\"\n }\n }),\n { name: \"BackstageInfoCard\" }\n);\nconst CardActionsTopRight = withStyles(\n (theme) => ({\n root: {\n display: \"inline-block\",\n padding: theme.spacing(8, 8, 0, 0),\n float: \"right\"\n }\n }),\n { name: \"BackstageInfoCardCardActionsTopRight\" }\n)(CardActions);\nconst VARIANT_STYLES = {\n card: {\n flex: {\n display: \"flex\",\n flexDirection: \"column\"\n },\n fullHeight: {\n display: \"flex\",\n flexDirection: \"column\",\n height: \"100%\"\n },\n gridItem: {\n display: \"flex\",\n flexDirection: \"column\",\n height: \"calc(100% - 10px)\",\n // for pages without content header\n marginBottom: \"10px\",\n breakInside: \"avoid-page\",\n \"@media print\": {\n height: \"auto\"\n }\n }\n },\n cardContent: {\n fullHeight: {\n flex: 1\n },\n gridItem: {\n flex: 1\n }\n }\n};\nfunction InfoCard(props) {\n const {\n title,\n subheader,\n divider = true,\n deepLink,\n slackChannel,\n errorBoundaryProps,\n variant,\n alignContent = \"normal\",\n children,\n headerStyle,\n headerProps,\n icon,\n action,\n actionsClassName,\n actions,\n cardClassName,\n actionsTopRight,\n className,\n noPadding,\n titleTypographyProps,\n subheaderTypographyProps\n } = props;\n const classes = useStyles();\n let calculatedStyle = {};\n let calculatedCardStyle = {};\n if (variant) {\n const variants = variant.split(/[\\s]+/g);\n variants.forEach((name) => {\n calculatedStyle = {\n ...calculatedStyle,\n ...VARIANT_STYLES.card[name]\n };\n calculatedCardStyle = {\n ...calculatedCardStyle,\n ...VARIANT_STYLES.cardContent[name]\n };\n });\n }\n const cardSubTitle = () => {\n if (!subheader && !icon) {\n return null;\n }\n return /* @__PURE__ */ jsxs(\"div\", { \"data-testid\": \"info-card-subheader\", children: [\n subheader && /* @__PURE__ */ jsx(\"div\", { className: classes.subheader, children: subheader }),\n icon\n ] });\n };\n const errProps = errorBoundaryProps || (slackChannel ? { slackChannel } : {});\n return /* @__PURE__ */ jsx(Card, { style: calculatedStyle, className, children: /* @__PURE__ */ jsxs(ErrorBoundary, { ...errProps, children: [\n title && /* @__PURE__ */ jsx(\n CardHeader,\n {\n classes: {\n root: classNames(classes.header),\n title: classes.headerTitle,\n subheader: classes.headerSubheader,\n avatar: classes.headerAvatar,\n action: classes.headerAction,\n content: classes.headerContent\n },\n title,\n subheader: cardSubTitle(),\n action,\n style: { ...headerStyle },\n titleTypographyProps,\n subheaderTypographyProps,\n ...headerProps\n }\n ),\n actionsTopRight && /* @__PURE__ */ jsx(CardActionsTopRight, { children: actionsTopRight }),\n divider && /* @__PURE__ */ jsx(Divider, {}),\n /* @__PURE__ */ jsx(\n CardContent,\n {\n className: classNames(cardClassName, {\n [classes.noPadding]: noPadding,\n [classes.contentAlignBottom]: alignContent === \"bottom\"\n }),\n style: calculatedCardStyle,\n children\n }\n ),\n actions && /* @__PURE__ */ jsx(CardActions, { className: actionsClassName, children: actions }),\n deepLink && /* @__PURE__ */ jsx(BottomLink, { ...deepLink })\n ] }) });\n}\n\nexport { InfoCard };\n//# sourceMappingURL=InfoCard.esm.js.map\n"],"names":["_interopRequireDefault","_interopRequireWildcard","exports","React","_default","default","createElement","d","useEntity","versionedHolder","useVersionedContext","Error","value","atVersion","entity","createVersionedContext","MenuItem","props","ref","tabIndex","classes","className","_props$component","component","_props$disableGutters","disableGutters","ListItemClasses","_props$role","role","selected","tabIndexProp","other","disabled","undefined","button","dense","root","gutters","theme","typography","body1","minHeight","paddingTop","paddingBottom","boxSizing","width","overflow","whiteSpace","breakpoints","up","body2","name","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","useStyles","makeStyles","visuallyHidden","clip","clipPath","position","userSelect","height","externalLink","externalLinkIcon","verticalAlign","marginLeft","spacing","ExternalLinkIcon","Icon","useApp","getSystemIcon","jsx","isExternalUri","uri","test","scriptProtocolPattern","originalWindowOpen","window","open","__backstage","newOpen","url","String","this","getNodeText","node","Array","map","join","trim","children","includes","Link","forwardRef","onClick","noTrack","analytics","useAnalytics","to","obj","createRoutesFromChildren","Route","index","element","isReactRouterBeta","resolvedPath","basePath","useApi","configApiRef","getOptionalString","useBaseUrl","pathname","URL","trimEnd","useBasePath","external","startsWithBasePath","startsWith","concat","useResolvedPath","linkText","newWindow","exec","handleClick","event","captureEvent","attributes","jsxs","target","rel","href","maxWidth","padding","boxTitle","margin","color","palette","textSubtle","arrow","BottomLink","link","title","Box","Divider","underline","display","alignItems","fontWeight","m","Typography","A","LinkWrapper","LinkButton","Button","SlackLink","slackChannel","t","useTranslationRef","variant","ErrorBoundary","Component","constructor","super","errorInfo","componentDidCatch","console","setState","render","noPadding","contentAlignBottom","header","headerTitle","fontWeightBold","headerSubheader","headerAvatar","headerAction","headerContent","subheader","CardActionsTopRight","withStyles","float","CardActions","VARIANT_STYLES","card","flex","flexDirection","fullHeight","gridItem","marginBottom","breakInside","cardContent","InfoCard","divider","deepLink","errorBoundaryProps","alignContent","headerStyle","headerProps","icon","action","actionsClassName","actions","cardClassName","actionsTopRight","titleTypographyProps","subheaderTypographyProps","calculatedStyle","calculatedCardStyle","split","forEach","errProps","Card","style","CardHeader","avatar","content","CardContent"],"sourceRoot":""}
@@ -1,2 +1,2 @@
1
- "use strict";(self.webpackChunkinternal_plugin_kuadrant=self.webpackChunkinternal_plugin_kuadrant||[]).push([[8799],{24170:(e,a,t)=>{t.d(a,{o:()=>i,q:()=>n});const i=e=>e&&e.trim()?e.length>253?"Must be 253 characters or less":/^[a-z0-9]([-a-z0-9]*[a-z0-9])?$/.test(e)?null:"Must be lowercase alphanumeric with hyphens, start and end with alphanumeric":"Name is required",n=e=>{if(!e)return null;try{const a=new URL(e);return["http:","https:"].includes(a.protocol)?null:"Must be a valid HTTP or HTTPS URL"}catch{return"Must be a valid URL"}}},26997:(e,a,t)=>{t.d(a,{C:()=>$});var i=t(31085),n=t(95478),r=t.n(n),l=t(76891),s=t(61477),o=t(46805),d=t(72501),c=t(10394),u=t(42899),p=t(16249),h=t(34839),m=t(29365),x=t(67720),A=t(71677),g=t(95061),v=t(29635),j=t(86901),y=t(30285),b=t(93453),f=t(64947),k=t(78467),P=t(58837),T=t(89031),S=t(18466),w=t(22097),I=t(84441),C=t(86687),R=t(91638),H=t(65867),N=t(24170);const W=(0,P.A)(e=>({asterisk:{color:"#f44336"},sectionHeader:{display:"flex",alignItems:"center",gap:e.spacing(.5),marginTop:e.spacing(2),marginBottom:e.spacing(1)},infoIcon:{fontSize:18,color:e.palette.text.secondary},tagChip:{marginRight:e.spacing(.5),marginBottom:e.spacing(.5)}})),$=({open:e,onClose:a,onSuccess:t,namespace:P,name:$})=>{const L=W(),M=(0,w.useApi)(w.configApiRef),z=(0,w.useApi)(w.fetchApiRef),U=M.getString("backend.baseUrl"),[q,E]=(0,n.useState)(!1),[D,F]=(0,n.useState)(""),[B,K]=(0,n.useState)(""),[_,J]=(0,n.useState)("v1"),[G,O]=(0,n.useState)("Draft"),[V,Q]=(0,n.useState)("manual"),[X,Y]=(0,n.useState)([]),[Z,ee]=(0,n.useState)(null),[ae,te]=(0,n.useState)(""),[ie,ne]=(0,n.useState)(""),[re,le]=(0,n.useState)(""),[se,oe]=(0,n.useState)(""),[de,ce]=(0,n.useState)(""),[ue,pe]=(0,n.useState)(""),[he,me]=(0,n.useState)(!1),[xe,Ae]=(0,n.useState)(null);(0,n.useEffect)(()=>{e&&P&&$&&(E(!0),pe(""),z.fetch(`${U}/api/kuadrant/apiproducts/${P}/${$}`).then(async e=>{if(!e.ok){const a=await e.json();throw new Error(a.error||`Failed to fetch API product: ${e.status}`)}return e.json()}).then(e=>{var a,t,i,n;F(e.spec.displayName||""),K(e.spec.description||""),J(e.spec.version||"v1"),O(e.spec.publishStatus||"Draft"),Q(e.spec.approvalMode||"manual"),Y(e.spec.tags||[]),ee(e.spec.targetRef||null),ne((null===(a=e.spec.contact)||void 0===a?void 0:a.email)||""),le((null===(t=e.spec.contact)||void 0===t?void 0:t.team)||""),oe((null===(i=e.spec.documentation)||void 0===i?void 0:i.docsURL)||""),ce((null===(n=e.spec.documentation)||void 0===n?void 0:n.openAPISpecURL)||""),Ae(null),E(!1)}).catch(e=>{pe(e.message||"Failed to load API product"),E(!1)}))},[e,P,$,U,z]);const{value:ge,error:ve}=(0,R.A)(async()=>{if(!e)return null;const a=await z.fetch(`${U}/api/kuadrant/planpolicies`);return await a.json()},[U,z,e]),je=r().useMemo(()=>(null==ge?void 0:ge.items)&&Z?ge.items.find(e=>{const a=e.targetRef;return"HTTPRoute"===(null==a?void 0:a.kind)&&(null==a?void 0:a.name)===Z.name&&(!(null==a?void 0:a.namespace)||(null==a?void 0:a.namespace)===(Z.namespace||P))}):null,[ge,Z,P]);(0,n.useEffect)(()=>{e&&Ae(null)},[e]);const ye=()=>{ae.trim()&&!X.includes(ae.trim())&&(Y([...X,ae.trim()]),te(""))};return(0,i.jsxs)(l.A,{open:e,onClose:a,maxWidth:"md",fullWidth:!0,children:[(0,i.jsx)(s.A,{children:"Edit API Product"}),(0,i.jsxs)(o.A,{children:[ue&&(0,i.jsx)(I.A,{severity:"error",style:{marginBottom:16},children:ue}),ve&&(0,i.jsxs)(I.A,{severity:"warning",style:{marginBottom:16},children:[(0,i.jsx)("strong",{children:"Failed to load PlanPolicies:"})," ",ve.message,(0,i.jsx)(d.A,{variant:"body2",style:{marginTop:8},children:"Plan information may be incomplete."})]}),q?(0,i.jsx)(C.k,{}):(0,i.jsxs)(i.Fragment,{children:[(0,i.jsx)(c.A,{className:L.sectionHeader,children:(0,i.jsx)(d.A,{variant:"subtitle1",children:(0,i.jsx)("strong",{children:"API product info"})})}),(0,i.jsxs)(u.A,{container:!0,spacing:2,children:[(0,i.jsx)(u.A,{item:!0,xs:6,children:(0,i.jsx)(p.A,{fullWidth:!0,label:"API product name",value:D,onChange:e=>F(e.target.value),placeholder:"My API",helperText:"Give a unique name for your API product",margin:"normal",required:!0,disabled:he,InputLabelProps:{classes:{asterisk:L.asterisk}}})}),(0,i.jsx)(u.A,{item:!0,xs:6,children:(0,i.jsx)(p.A,{fullWidth:!0,label:"Resource name",value:$,disabled:!0,helperText:"Kubernetes resource name (immutable)",margin:"normal"})}),(0,i.jsx)(u.A,{item:!0,xs:6,children:(0,i.jsx)(p.A,{fullWidth:!0,label:"Version",value:_,onChange:e=>J(e.target.value),placeholder:"v1",helperText:"Give a version to your API product",margin:"normal",disabled:he})}),(0,i.jsx)(u.A,{item:!0,xs:6,children:(0,i.jsx)(p.A,{fullWidth:!0,label:"Tag",value:ae,onChange:e=>te(e.target.value),onKeyPress:e=>{"Enter"===e.key&&(e.preventDefault(),ye())},placeholder:"Add tag",helperText:"Add a tag to your API product",margin:"normal",disabled:he,InputProps:{endAdornment:ae?(0,i.jsx)(h.A,{position:"end",children:(0,i.jsx)(m.A,{size:"small",onClick:ye,disabled:he,children:(0,i.jsx)(S.A,{fontSize:"small"})})}):void 0}})}),X.length>0&&(0,i.jsx)(u.A,{item:!0,xs:12,children:(0,i.jsx)(c.A,{display:"flex",flexWrap:"wrap",children:X.map(e=>(0,i.jsx)(x.A,{label:e,onDelete:he?void 0:()=>{return a=e,void Y(X.filter(e=>e!==a));var a},size:"small",className:L.tagChip,disabled:he},e))})}),(0,i.jsx)(u.A,{item:!0,xs:12,children:(0,i.jsx)(p.A,{fullWidth:!0,label:"Description",value:B,onChange:e=>K(e.target.value),placeholder:"API description",margin:"normal",multiline:!0,rows:2,required:!0,disabled:he,InputLabelProps:{classes:{asterisk:L.asterisk}}})})]}),(0,i.jsxs)(c.A,{className:L.sectionHeader,children:[(0,i.jsx)(d.A,{variant:"subtitle1",children:(0,i.jsx)("strong",{children:"Associated route"})}),(0,i.jsx)(A.Ay,{title:"The HTTPRoute this API product is associated with",children:(0,i.jsx)(T.A,{className:L.infoIcon})})]}),(0,i.jsxs)(u.A,{container:!0,spacing:2,children:[(0,i.jsx)(u.A,{item:!0,xs:12,children:(0,i.jsx)(p.A,{fullWidth:!0,label:"OpenAPI Spec URL",value:de,onChange:e=>{return a=e.target.value,ce(a),void Ae((0,N.q)(a));var a},placeholder:"https://api.example.com/openapi.json",helperText:xe||"Enter the full path to your API spec file",error:!!xe,margin:"normal",disabled:he})}),(0,i.jsx)(u.A,{item:!0,xs:12,children:(0,i.jsx)(p.A,{fullWidth:!0,label:"Documentation URL",value:se,onChange:e=>oe(e.target.value),placeholder:"https://docs.example.com/api",helperText:"Link to external documentation for this API",margin:"normal",disabled:he})}),Z&&(0,i.jsx)(u.A,{item:!0,xs:12,children:(0,i.jsx)(p.A,{fullWidth:!0,label:"HTTPRoute",value:`${Z.namespace||P}/${Z.name}`,disabled:!0,helperText:"Target HTTPRoute (immutable)",margin:"normal"})})]}),Z&&(0,i.jsxs)(i.Fragment,{children:[(0,i.jsxs)(c.A,{className:L.sectionHeader,children:[(0,i.jsx)(d.A,{variant:"subtitle1",children:(0,i.jsx)("strong",{children:"HTTPRoute policies"})}),(0,i.jsx)(A.Ay,{title:"Shows the associated policies and rate limit tiers for the HTTPRoute",children:(0,i.jsx)(T.A,{className:L.infoIcon})})]}),(0,i.jsx)(H.g,{selectedPolicy:je,alertSeverity:"info",alertMessage:"No PlanPolicy found for this HTTPRoute.",includeTopMargin:!1})]}),(0,i.jsxs)(c.A,{className:L.sectionHeader,children:[(0,i.jsx)(d.A,{variant:"subtitle1",children:(0,i.jsx)("strong",{children:"API Key approval"})}),(0,i.jsx)(A.Ay,{title:"Choose how API key requests are handled for this product",children:(0,i.jsx)(T.A,{className:L.infoIcon})})]}),(0,i.jsx)(g.A,{component:"fieldset",disabled:he,children:(0,i.jsxs)(v.A,{row:!0,value:V,onChange:e=>Q(e.target.value),children:[(0,i.jsx)(j.A,{value:"manual",control:(0,i.jsx)(y.A,{color:"primary"}),label:(0,i.jsxs)(c.A,{children:[(0,i.jsx)(d.A,{variant:"body2",children:"Need manual approval"}),(0,i.jsx)(d.A,{variant:"caption",color:"textSecondary",children:"Requires approval for requesting this API"})]})}),(0,i.jsx)(j.A,{value:"automatic",control:(0,i.jsx)(y.A,{color:"primary"}),label:(0,i.jsxs)(c.A,{children:[(0,i.jsx)(d.A,{variant:"body2",children:"Automatic"}),(0,i.jsx)(d.A,{variant:"caption",color:"textSecondary",children:"Keys are created without need to be approved"})]})})]})})]})]}),(0,i.jsxs)(b.A,{children:[(0,i.jsx)(f.A,{onClick:a,disabled:he,children:"Cancel"}),(0,i.jsx)(f.A,{onClick:async()=>{pe(""),me(!0);try{const e={spec:{displayName:D,description:B,version:_,publishStatus:G,approvalMode:V,tags:X,targetRef:Z,...ie||re?{contact:{...ie&&{email:ie},...re&&{team:re}}}:{},...se||de?{documentation:{...se&&{docsURL:se},...de&&{openAPISpecURL:de}}}:{}}},i=await z.fetch(`${U}/api/kuadrant/apiproducts/${P}/${$}`,{method:"PATCH",headers:{"Content-Type":"application/json"},body:JSON.stringify(e)});if(!i.ok){const e=await i.json();throw new Error(e.error||"failed to update apiproduct")}t(),a()}catch(e){pe(e instanceof Error?e.message:String(e))}finally{me(!1)}},color:"primary",variant:"contained",disabled:he||q||!D||!B||!!xe,startIcon:he?(0,i.jsx)(k.A,{size:16,color:"inherit"}):void 0,children:he?"Saving...":"Save"})]})]})}},34955:(e,a,t)=>{t.d(a,{Al:()=>l,DS:()=>h,EM:()=>d,FL:()=>r,J:()=>n,KV:()=>v,R_:()=>c,U3:()=>s,dp:()=>p,jH:()=>g,q0:()=>m,uL:()=>A,v_:()=>o,vs:()=>u,z4:()=>x});var i=t(83572);(0,i.i)({name:"kuadrant.planpolicy.create",attributes:{action:"create"}}),(0,i.i)({name:"kuadrant.planpolicy.read",attributes:{action:"read"}}),(0,i.i)({name:"kuadrant.planpolicy.update",attributes:{action:"update"}}),(0,i.i)({name:"kuadrant.planpolicy.delete",attributes:{action:"delete"}});const n=(0,i.i)({name:"kuadrant.planpolicy.list",attributes:{action:"read"}}),r=(0,i.i)({name:"kuadrant.apiproduct.create",attributes:{action:"create"}}),l=((0,i.i)({name:"kuadrant.apiproduct.read.own",attributes:{action:"read"}}),(0,i.i)({name:"kuadrant.apiproduct.read.all",attributes:{action:"read"}})),s=(0,i.i)({name:"kuadrant.apiproduct.update.own",attributes:{action:"update"}}),o=(0,i.i)({name:"kuadrant.apiproduct.update.all",attributes:{action:"update"}}),d=(0,i.i)({name:"kuadrant.apiproduct.delete.own",attributes:{action:"delete"}}),c=(0,i.i)({name:"kuadrant.apiproduct.delete.all",attributes:{action:"delete"}}),u=(0,i.i)({name:"kuadrant.apiproduct.list",attributes:{action:"read"}}),p=(0,i.i)({name:"kuadrant.apikey.create",attributes:{action:"create"},resourceType:"apiproduct"}),h=(0,i.i)({name:"kuadrant.apikey.read.own",attributes:{action:"read"}}),m=((0,i.i)({name:"kuadrant.apikey.read.all",attributes:{action:"read"}}),(0,i.i)({name:"kuadrant.apikey.update.own",attributes:{action:"update"}})),x=(0,i.i)({name:"kuadrant.apikey.update.all",attributes:{action:"update"}}),A=(0,i.i)({name:"kuadrant.apikey.delete.own",attributes:{action:"delete"}}),g=(0,i.i)({name:"kuadrant.apikey.delete.all",attributes:{action:"delete"}}),v=(0,i.i)({name:"kuadrant.apikey.approve",attributes:{action:"update"}})},46205:(e,a,t)=>{t.d(a,{W:()=>r,l:()=>n});var i=t(87421);function n(e,a){const t="resourceType"in e?{permission:e,resourceRef:a}:{permission:e},n=(0,i.J)(t);return{allowed:n.allowed,loading:n.loading,error:n.error}}function r(e,a,t,i){return!!i||!(!t||e!==a)}},63221:(e,a,t)=>{t.d(a,{K:()=>A});var i=t(31085),n=t(95478),r=t(76891),l=t(61477),s=t(10394),o=t(46805),d=t(59461),c=t(72501),u=t(16249),p=t(93453),h=t(64947),m=t(78467),x=t(77225);const A=({open:e,title:a,description:t,confirmText:A,severity:g="normal",deleting:v=!1,onConfirm:j,onCancel:y})=>{const[b,f]=(0,n.useState)("");(0,n.useEffect)(()=>{e||f("")},[e]);const k="high"===g&&A,P=!k||b===A;return(0,i.jsxs)(r.A,{open:e,onClose:v?void 0:y,maxWidth:"sm",fullWidth:!0,children:[(0,i.jsxs)(l.A,{children:["high"===g&&(0,i.jsxs)(s.A,{display:"flex",alignItems:"center",style:{gap:8},children:[(0,i.jsx)(x.A,{color:"error"}),(0,i.jsx)("span",{children:a})]}),"high"!==g&&a]}),(0,i.jsxs)(o.A,{children:[(0,i.jsx)(d.A,{style:{whiteSpace:"pre-line"},children:t}),k&&(0,i.jsxs)(s.A,{mt:2,children:[(0,i.jsxs)(c.A,{variant:"body2",color:"textSecondary",gutterBottom:!0,children:["Type ",(0,i.jsx)("strong",{children:A})," to confirm:"]}),(0,i.jsx)(u.A,{fullWidth:!0,variant:"outlined",size:"small",value:b,onChange:e=>f(e.target.value),disabled:v,autoFocus:!0,placeholder:A})]})]}),(0,i.jsxs)(p.A,{children:[(0,i.jsx)(h.A,{onClick:y,disabled:v,children:"Cancel"}),(0,i.jsx)(h.A,{onClick:()=>{P&&j()},color:"secondary",variant:"contained",disabled:v||!P,startIcon:v?(0,i.jsx)(m.A,{size:16,color:"inherit"}):void 0,children:v?"Deleting...":"Delete"})]})]})}},65867:(e,a,t)=>{t.d(a,{g:()=>d});var i=t(31085),n=(t(95478),t(10394)),r=t(72501),l=t(67720),s=t(54917),o=t(84441);const d=({selectedPolicy:e,alertSeverity:a="warning",alertMessage:t="No PlanPolicy found for this HTTPRoute. API keys and rate limiting may not be available.",includeTopMargin:d=!0})=>{const c=(0,s.A)();return(0,i.jsx)(n.A,{mt:d?1:0,p:2,bgcolor:c.palette.background.default,borderRadius:1,border:`1px solid ${c.palette.divider}`,children:e?(0,i.jsxs)(i.Fragment,{children:[(0,i.jsxs)(r.A,{variant:"subtitle2",gutterBottom:!0,style:{fontWeight:600},children:["Associated PlanPolicy: ",(0,i.jsx)("strong",{children:e.metadata.name})]}),e.plans&&e.plans.length>0?(0,i.jsxs)(i.Fragment,{children:[(0,i.jsx)(r.A,{variant:"caption",display:"block",gutterBottom:!0,color:"textSecondary",style:{marginTop:8},children:"Available Tiers:"}),(0,i.jsx)(n.A,{display:"flex",flexWrap:"wrap",mt:1,style:{gap:8},children:e.plans.map((e,a)=>{var t,n,r;const s=(null===(t=e.limits)||void 0===t?void 0:t.daily)?`${e.limits.daily}/day`:(null===(n=e.limits)||void 0===n?void 0:n.monthly)?`${e.limits.monthly}/month`:(null===(r=e.limits)||void 0===r?void 0:r.yearly)?`${e.limits.yearly}/year`:"No limit";return(0,i.jsx)(l.A,{label:`${e.tier}: ${s}`,size:"small",variant:"outlined",color:"primary"},a)})})]}):(0,i.jsx)(r.A,{variant:"caption",color:"textSecondary",children:"No plans defined in this PlanPolicy"})]}):(0,i.jsx)(o.A,{severity:a,children:t})})}}}]);
2
- //# sourceMappingURL=8799.4ea4639c.chunk.js.map
1
+ "use strict";(self.webpackChunkinternal_plugin_kuadrant=self.webpackChunkinternal_plugin_kuadrant||[]).push([[8799],{24170:(e,a,t)=>{t.d(a,{o:()=>i,q:()=>n});const i=e=>e&&e.trim()?e.length>253?"Must be 253 characters or less":/^[a-z0-9]([-a-z0-9]*[a-z0-9])?$/.test(e)?null:"Must be lowercase alphanumeric with hyphens, start and end with alphanumeric":"Name is required",n=e=>{if(!e)return null;try{const a=new URL(e);return["http:","https:"].includes(a.protocol)?null:"Must be a valid HTTP or HTTPS URL"}catch{return"Must be a valid URL"}}},26997:(e,a,t)=>{t.d(a,{C:()=>$});var i=t(31085),n=t(95478),r=t.n(n),l=t(76891),s=t(61477),o=t(46805),d=t(72501),c=t(10394),u=t(42899),p=t(16249),h=t(34839),m=t(29365),x=t(67720),A=t(71677),g=t(95061),v=t(29635),j=t(86901),y=t(30285),b=t(93453),f=t(64947),k=t(78467),P=t(58837),T=t(89031),S=t(18466),w=t(22097),I=t(84441),C=t(86687),R=t(91638),H=t(65867),N=t(24170);const W=(0,P.A)(e=>({asterisk:{color:"#f44336"},sectionHeader:{display:"flex",alignItems:"center",gap:e.spacing(.5),marginTop:e.spacing(2),marginBottom:e.spacing(1)},infoIcon:{fontSize:18,color:e.palette.text.secondary},tagChip:{marginRight:e.spacing(.5),marginBottom:e.spacing(.5)}})),$=({open:e,onClose:a,onSuccess:t,namespace:P,name:$})=>{const L=W(),M=(0,w.useApi)(w.configApiRef),z=(0,w.useApi)(w.fetchApiRef),U=M.getString("backend.baseUrl"),[q,E]=(0,n.useState)(!1),[D,F]=(0,n.useState)(""),[B,K]=(0,n.useState)(""),[_,J]=(0,n.useState)("v1"),[G,O]=(0,n.useState)("Draft"),[V,Q]=(0,n.useState)("manual"),[X,Y]=(0,n.useState)([]),[Z,ee]=(0,n.useState)(null),[ae,te]=(0,n.useState)(""),[ie,ne]=(0,n.useState)(""),[re,le]=(0,n.useState)(""),[se,oe]=(0,n.useState)(""),[de,ce]=(0,n.useState)(""),[ue,pe]=(0,n.useState)(""),[he,me]=(0,n.useState)(!1),[xe,Ae]=(0,n.useState)(null);(0,n.useEffect)(()=>{e&&P&&$&&(E(!0),pe(""),z.fetch(`${U}/api/kuadrant/apiproducts/${P}/${$}`).then(async e=>{if(!e.ok){const a=await e.json();throw new Error(a.error||`Failed to fetch API product: ${e.status}`)}return e.json()}).then(e=>{var a,t,i,n;F(e.spec.displayName||""),K(e.spec.description||""),J(e.spec.version||"v1"),O(e.spec.publishStatus||"Draft"),Q(e.spec.approvalMode||"manual"),Y(e.spec.tags||[]),ee(e.spec.targetRef||null),ne((null===(a=e.spec.contact)||void 0===a?void 0:a.email)||""),le((null===(t=e.spec.contact)||void 0===t?void 0:t.team)||""),oe((null===(i=e.spec.documentation)||void 0===i?void 0:i.docsURL)||""),ce((null===(n=e.spec.documentation)||void 0===n?void 0:n.openAPISpecURL)||""),Ae(null),E(!1)}).catch(e=>{pe(e.message||"Failed to load API product"),E(!1)}))},[e,P,$,U,z]);const{value:ge,error:ve}=(0,R.A)(async()=>{if(!e)return null;const a=await z.fetch(`${U}/api/kuadrant/planpolicies`);return await a.json()},[U,z,e]),je=r().useMemo(()=>(null==ge?void 0:ge.items)&&Z?ge.items.find(e=>{const a=e.targetRef;return"HTTPRoute"===(null==a?void 0:a.kind)&&(null==a?void 0:a.name)===Z.name&&(!(null==a?void 0:a.namespace)||(null==a?void 0:a.namespace)===(Z.namespace||P))}):null,[ge,Z,P]);(0,n.useEffect)(()=>{e&&Ae(null)},[e]);const ye=()=>{ae.trim()&&!X.includes(ae.trim())&&(Y([...X,ae.trim()]),te(""))};return(0,i.jsxs)(l.A,{open:e,onClose:a,maxWidth:"md",fullWidth:!0,children:[(0,i.jsx)(s.A,{children:"Edit API Product"}),(0,i.jsxs)(o.A,{children:[ue&&(0,i.jsx)(I.A,{severity:"error",style:{marginBottom:16},children:ue}),ve&&(0,i.jsxs)(I.A,{severity:"warning",style:{marginBottom:16},children:[(0,i.jsx)("strong",{children:"Failed to load PlanPolicies:"})," ",ve.message,(0,i.jsx)(d.A,{variant:"body2",style:{marginTop:8},children:"Plan information may be incomplete."})]}),q?(0,i.jsx)(C.k,{}):(0,i.jsxs)(i.Fragment,{children:[(0,i.jsx)(c.A,{className:L.sectionHeader,children:(0,i.jsx)(d.A,{variant:"subtitle1",children:(0,i.jsx)("strong",{children:"API product info"})})}),(0,i.jsxs)(u.A,{container:!0,spacing:2,children:[(0,i.jsx)(u.A,{item:!0,xs:6,children:(0,i.jsx)(p.A,{fullWidth:!0,label:"API product name",value:D,onChange:e=>F(e.target.value),placeholder:"My API",helperText:"Give a unique name for your API product",margin:"normal",required:!0,disabled:he,InputLabelProps:{classes:{asterisk:L.asterisk}}})}),(0,i.jsx)(u.A,{item:!0,xs:6,children:(0,i.jsx)(p.A,{fullWidth:!0,label:"Resource name",value:$,disabled:!0,helperText:"Kubernetes resource name (immutable)",margin:"normal"})}),(0,i.jsx)(u.A,{item:!0,xs:6,children:(0,i.jsx)(p.A,{fullWidth:!0,label:"Version",value:_,onChange:e=>J(e.target.value),placeholder:"v1",helperText:"Give a version to your API product",margin:"normal",disabled:he})}),(0,i.jsx)(u.A,{item:!0,xs:6,children:(0,i.jsx)(p.A,{fullWidth:!0,label:"Tag",value:ae,onChange:e=>te(e.target.value),onKeyPress:e=>{"Enter"===e.key&&(e.preventDefault(),ye())},placeholder:"Add tag",helperText:"Add a tag to your API product",margin:"normal",disabled:he,InputProps:{endAdornment:ae?(0,i.jsx)(h.A,{position:"end",children:(0,i.jsx)(m.A,{size:"small",onClick:ye,disabled:he,children:(0,i.jsx)(S.A,{fontSize:"small"})})}):void 0}})}),X.length>0&&(0,i.jsx)(u.A,{item:!0,xs:12,children:(0,i.jsx)(c.A,{display:"flex",flexWrap:"wrap",children:X.map(e=>(0,i.jsx)(x.A,{label:e,onDelete:he?void 0:()=>{return a=e,void Y(X.filter(e=>e!==a));var a},size:"small",className:L.tagChip,disabled:he},e))})}),(0,i.jsx)(u.A,{item:!0,xs:12,children:(0,i.jsx)(p.A,{fullWidth:!0,label:"Description",value:B,onChange:e=>K(e.target.value),placeholder:"API description",margin:"normal",multiline:!0,rows:2,required:!0,disabled:he,InputLabelProps:{classes:{asterisk:L.asterisk}}})})]}),(0,i.jsxs)(c.A,{className:L.sectionHeader,children:[(0,i.jsx)(d.A,{variant:"subtitle1",children:(0,i.jsx)("strong",{children:"Associated route"})}),(0,i.jsx)(A.Ay,{title:"The HTTPRoute this API product is associated with",children:(0,i.jsx)(T.A,{className:L.infoIcon})})]}),(0,i.jsxs)(u.A,{container:!0,spacing:2,children:[(0,i.jsx)(u.A,{item:!0,xs:12,children:(0,i.jsx)(p.A,{fullWidth:!0,label:"OpenAPI Spec URL",value:de,onChange:e=>{return a=e.target.value,ce(a),void Ae((0,N.q)(a));var a},placeholder:"https://api.example.com/openapi.json",helperText:xe||"Enter the full path to your API spec file",error:!!xe,margin:"normal",disabled:he})}),(0,i.jsx)(u.A,{item:!0,xs:12,children:(0,i.jsx)(p.A,{fullWidth:!0,label:"Documentation URL",value:se,onChange:e=>oe(e.target.value),placeholder:"https://docs.example.com/api",helperText:"Link to external documentation for this API",margin:"normal",disabled:he})}),Z&&(0,i.jsx)(u.A,{item:!0,xs:12,children:(0,i.jsx)(p.A,{fullWidth:!0,label:"HTTPRoute",value:`${Z.namespace||P}/${Z.name}`,disabled:!0,helperText:"Target HTTPRoute (immutable)",margin:"normal"})})]}),Z&&(0,i.jsxs)(i.Fragment,{children:[(0,i.jsxs)(c.A,{className:L.sectionHeader,children:[(0,i.jsx)(d.A,{variant:"subtitle1",children:(0,i.jsx)("strong",{children:"HTTPRoute policies"})}),(0,i.jsx)(A.Ay,{title:"Shows the associated policies and rate limit tiers for the HTTPRoute",children:(0,i.jsx)(T.A,{className:L.infoIcon})})]}),(0,i.jsx)(H.g,{selectedPolicy:je,alertSeverity:"info",alertMessage:"No PlanPolicy found for this HTTPRoute.",includeTopMargin:!1})]}),(0,i.jsxs)(c.A,{className:L.sectionHeader,children:[(0,i.jsx)(d.A,{variant:"subtitle1",children:(0,i.jsx)("strong",{children:"API Key approval"})}),(0,i.jsx)(A.Ay,{title:"Choose how API key requests are handled for this product",children:(0,i.jsx)(T.A,{className:L.infoIcon})})]}),(0,i.jsx)(g.A,{component:"fieldset",disabled:he,children:(0,i.jsxs)(v.A,{row:!0,value:V,onChange:e=>Q(e.target.value),children:[(0,i.jsx)(j.A,{value:"manual",control:(0,i.jsx)(y.A,{color:"primary"}),label:(0,i.jsxs)(c.A,{children:[(0,i.jsx)(d.A,{variant:"body2",children:"Need manual approval"}),(0,i.jsx)(d.A,{variant:"caption",color:"textSecondary",children:"Requires approval for requesting this API"})]})}),(0,i.jsx)(j.A,{value:"automatic",control:(0,i.jsx)(y.A,{color:"primary"}),label:(0,i.jsxs)(c.A,{children:[(0,i.jsx)(d.A,{variant:"body2",children:"Automatic"}),(0,i.jsx)(d.A,{variant:"caption",color:"textSecondary",children:"Keys are created without need to be approved"})]})})]})})]})]}),(0,i.jsxs)(b.A,{children:[(0,i.jsx)(f.A,{onClick:a,disabled:he,children:"Cancel"}),(0,i.jsx)(f.A,{onClick:async()=>{pe(""),me(!0);try{const e={spec:{displayName:D,description:B,version:_,publishStatus:G,approvalMode:V,tags:X,targetRef:Z,...ie||re?{contact:{...ie&&{email:ie},...re&&{team:re}}}:{},...se||de?{documentation:{...se&&{docsURL:se},...de&&{openAPISpecURL:de}}}:{}}},i=await z.fetch(`${U}/api/kuadrant/apiproducts/${P}/${$}`,{method:"PATCH",headers:{"Content-Type":"application/json"},body:JSON.stringify(e)});if(!i.ok){const e=await i.json();throw new Error(e.error||"failed to update apiproduct")}t(),a()}catch(e){pe(e instanceof Error?e.message:String(e))}finally{me(!1)}},color:"primary",variant:"contained",disabled:he||q||!D||!B||!!xe,startIcon:he?(0,i.jsx)(k.A,{size:16,color:"inherit"}):void 0,children:he?"Saving...":"Save"})]})]})}},34955:(e,a,t)=>{t.d(a,{Al:()=>l,DS:()=>h,EM:()=>d,FL:()=>r,J:()=>n,KV:()=>v,R_:()=>c,U3:()=>s,dp:()=>p,jH:()=>g,q0:()=>m,uL:()=>A,v_:()=>o,vs:()=>u,z4:()=>x});var i=t(83572);(0,i.i)({name:"kuadrant.planpolicy.create",attributes:{action:"create"}}),(0,i.i)({name:"kuadrant.planpolicy.read",attributes:{action:"read"}}),(0,i.i)({name:"kuadrant.planpolicy.update",attributes:{action:"update"}}),(0,i.i)({name:"kuadrant.planpolicy.delete",attributes:{action:"delete"}});const n=(0,i.i)({name:"kuadrant.planpolicy.list",attributes:{action:"read"}}),r=(0,i.i)({name:"kuadrant.apiproduct.create",attributes:{action:"create"}}),l=((0,i.i)({name:"kuadrant.apiproduct.read.own",attributes:{action:"read"}}),(0,i.i)({name:"kuadrant.apiproduct.read.all",attributes:{action:"read"}})),s=(0,i.i)({name:"kuadrant.apiproduct.update.own",attributes:{action:"update"}}),o=(0,i.i)({name:"kuadrant.apiproduct.update.all",attributes:{action:"update"}}),d=(0,i.i)({name:"kuadrant.apiproduct.delete.own",attributes:{action:"delete"}}),c=(0,i.i)({name:"kuadrant.apiproduct.delete.all",attributes:{action:"delete"}}),u=(0,i.i)({name:"kuadrant.apiproduct.list",attributes:{action:"read"}}),p=(0,i.i)({name:"kuadrant.apikey.create",attributes:{action:"create"},resourceType:"apiproduct"}),h=(0,i.i)({name:"kuadrant.apikey.read.own",attributes:{action:"read"}}),m=((0,i.i)({name:"kuadrant.apikey.read.all",attributes:{action:"read"}}),(0,i.i)({name:"kuadrant.apikey.update.own",attributes:{action:"update"}})),x=(0,i.i)({name:"kuadrant.apikey.update.all",attributes:{action:"update"}}),A=(0,i.i)({name:"kuadrant.apikey.delete.own",attributes:{action:"delete"}}),g=(0,i.i)({name:"kuadrant.apikey.delete.all",attributes:{action:"delete"}}),v=(0,i.i)({name:"kuadrant.apikey.approve",attributes:{action:"update"}})},46205:(e,a,t)=>{t.d(a,{W:()=>r,l:()=>n});var i=t(24217);function n(e,a){const t="resourceType"in e?{permission:e,resourceRef:a}:{permission:e},n=(0,i.J)(t);return{allowed:n.allowed,loading:n.loading,error:n.error}}function r(e,a,t,i){return!!i||!(!t||e!==a)}},63221:(e,a,t)=>{t.d(a,{K:()=>A});var i=t(31085),n=t(95478),r=t(76891),l=t(61477),s=t(10394),o=t(46805),d=t(59461),c=t(72501),u=t(16249),p=t(93453),h=t(64947),m=t(78467),x=t(77225);const A=({open:e,title:a,description:t,confirmText:A,severity:g="normal",deleting:v=!1,onConfirm:j,onCancel:y})=>{const[b,f]=(0,n.useState)("");(0,n.useEffect)(()=>{e||f("")},[e]);const k="high"===g&&A,P=!k||b===A;return(0,i.jsxs)(r.A,{open:e,onClose:v?void 0:y,maxWidth:"sm",fullWidth:!0,children:[(0,i.jsxs)(l.A,{children:["high"===g&&(0,i.jsxs)(s.A,{display:"flex",alignItems:"center",style:{gap:8},children:[(0,i.jsx)(x.A,{color:"error"}),(0,i.jsx)("span",{children:a})]}),"high"!==g&&a]}),(0,i.jsxs)(o.A,{children:[(0,i.jsx)(d.A,{style:{whiteSpace:"pre-line"},children:t}),k&&(0,i.jsxs)(s.A,{mt:2,children:[(0,i.jsxs)(c.A,{variant:"body2",color:"textSecondary",gutterBottom:!0,children:["Type ",(0,i.jsx)("strong",{children:A})," to confirm:"]}),(0,i.jsx)(u.A,{fullWidth:!0,variant:"outlined",size:"small",value:b,onChange:e=>f(e.target.value),disabled:v,autoFocus:!0,placeholder:A})]})]}),(0,i.jsxs)(p.A,{children:[(0,i.jsx)(h.A,{onClick:y,disabled:v,children:"Cancel"}),(0,i.jsx)(h.A,{onClick:()=>{P&&j()},color:"secondary",variant:"contained",disabled:v||!P,startIcon:v?(0,i.jsx)(m.A,{size:16,color:"inherit"}):void 0,children:v?"Deleting...":"Delete"})]})]})}},65867:(e,a,t)=>{t.d(a,{g:()=>d});var i=t(31085),n=(t(95478),t(10394)),r=t(72501),l=t(67720),s=t(54917),o=t(84441);const d=({selectedPolicy:e,alertSeverity:a="warning",alertMessage:t="No PlanPolicy found for this HTTPRoute. API keys and rate limiting may not be available.",includeTopMargin:d=!0})=>{const c=(0,s.A)();return(0,i.jsx)(n.A,{mt:d?1:0,p:2,bgcolor:c.palette.background.default,borderRadius:1,border:`1px solid ${c.palette.divider}`,children:e?(0,i.jsxs)(i.Fragment,{children:[(0,i.jsxs)(r.A,{variant:"subtitle2",gutterBottom:!0,style:{fontWeight:600},children:["Associated PlanPolicy: ",(0,i.jsx)("strong",{children:e.metadata.name})]}),e.plans&&e.plans.length>0?(0,i.jsxs)(i.Fragment,{children:[(0,i.jsx)(r.A,{variant:"caption",display:"block",gutterBottom:!0,color:"textSecondary",style:{marginTop:8},children:"Available Tiers:"}),(0,i.jsx)(n.A,{display:"flex",flexWrap:"wrap",mt:1,style:{gap:8},children:e.plans.map((e,a)=>{var t,n,r;const s=(null===(t=e.limits)||void 0===t?void 0:t.daily)?`${e.limits.daily}/day`:(null===(n=e.limits)||void 0===n?void 0:n.monthly)?`${e.limits.monthly}/month`:(null===(r=e.limits)||void 0===r?void 0:r.yearly)?`${e.limits.yearly}/year`:"No limit";return(0,i.jsx)(l.A,{label:`${e.tier}: ${s}`,size:"small",variant:"outlined",color:"primary"},a)})})]}):(0,i.jsx)(r.A,{variant:"caption",color:"textSecondary",children:"No plans defined in this PlanPolicy"})]}):(0,i.jsx)(o.A,{severity:a,children:t})})}}}]);
2
+ //# sourceMappingURL=8799.83d049f3.chunk.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"static/8799.4ea4639c.chunk.js","mappings":"8JACO,MAAMA,EAA0BC,GAChCA,GAAUA,EAAMC,OAGjBD,EAAME,OAAS,IACV,iCAGY,kCAEHC,KAAKH,GAIhB,KAHE,+EATA,mBAgBEI,EAAeJ,IAC1B,IAAKA,EACH,OAAO,KAGT,IACE,MAAMK,EAAM,IAAIC,IAAIN,GACpB,MAAK,CAAC,QAAS,UAAUO,SAASF,EAAIG,UAG/B,KAFE,mCAGX,CAAE,MACA,MAAO,qBACT,E,+WCDF,MAAMC,GAAYC,EAAAA,EAAAA,GAAYC,IAAW,CACvCC,SAAU,CACRC,MAAO,WAETC,cAAe,CACbC,QAAS,OACTC,WAAY,SACZC,IAAKN,EAAMO,QAAQ,IACnBC,UAAWR,EAAMO,QAAQ,GACzBE,aAAcT,EAAMO,QAAQ,IAE9BG,SAAU,CACRC,SAAU,GACVT,MAAOF,EAAMY,QAAQC,KAAKC,WAE5BC,QAAS,CACPC,YAAahB,EAAMO,QAAQ,IAC3BE,aAAcT,EAAMO,QAAQ,QAYnBU,EAAuB,EAAEC,OAAMC,UAASC,YAAWC,YAAWC,WACzE,MAAMC,EAAUzB,IACV0B,GAASC,EAAAA,EAAAA,QAAOC,EAAAA,cAChBC,GAAWF,EAAAA,EAAAA,QAAOG,EAAAA,aAClBC,EAAaL,EAAOM,UAAU,oBAC7BC,EAASC,IAAcC,EAAAA,EAAAA,WAAS,IAChCC,EAAaC,IAAkBF,EAAAA,EAAAA,UAAS,KACxCG,EAAaC,IAAkBJ,EAAAA,EAAAA,UAAS,KACxCK,EAASC,IAAcN,EAAAA,EAAAA,UAAS,OAChCO,EAAeC,IAAoBR,EAAAA,EAAAA,UAAgC,UACnES,EAAcC,IAAmBV,EAAAA,EAAAA,UAAiC,WAClEW,EAAMC,IAAWZ,EAAAA,EAAAA,UAAmB,KACpCa,EAAWC,KAAgBd,EAAAA,EAAAA,UAAc,OACzCe,GAAUC,KAAehB,EAAAA,EAAAA,UAAS,KAClCiB,GAAcC,KAAmBlB,EAAAA,EAAAA,UAAS,KAC1CmB,GAAaC,KAAkBpB,EAAAA,EAAAA,UAAS,KACxCqB,GAASC,KAActB,EAAAA,EAAAA,UAAS,KAChCuB,GAAaC,KAAkBxB,EAAAA,EAAAA,UAAS,KACxCyB,GAAOC,KAAY1B,EAAAA,EAAAA,UAAS,KAC5B2B,GAAQC,KAAa5B,EAAAA,EAAAA,WAAS,IAC9B6B,GAAkBC,KAAuB9B,EAAAA,EAAAA,UAAwB,OAGxE+B,EAAAA,EAAAA,WAAU,KACJ9C,GAAQG,GAAaC,IACvBU,GAAW,GACX2B,GAAS,IAEThC,EAASsC,MAAM,GAAGpC,8BAAuCR,KAAaC,KACnE4C,KAAKC,MAAMC,IACV,IAAKA,EAAIC,GAAI,CACX,MAAMC,QAAkBF,EAAIG,OAC5B,MAAM,IAAIC,MAAMF,EAAUZ,OAAS,gCAAgCU,EAAIK,SACzE,CACA,OAAOL,EAAIG,SAEZL,KAAKQ,I,IAQYA,EACDA,EACJA,EACIA,EAVfvC,EAAeuC,EAAKC,KAAKzC,aAAe,IACxCG,EAAeqC,EAAKC,KAAKvC,aAAe,IACxCG,EAAWmC,EAAKC,KAAKrC,SAAW,MAChCG,EAAiBiC,EAAKC,KAAKnC,eAAiB,SAC5CG,EAAgB+B,EAAKC,KAAKjC,cAAgB,UAC1CG,EAAQ6B,EAAKC,KAAK/B,MAAQ,IAC1BG,GAAa2B,EAAKC,KAAK7B,WAAa,MACpCK,IAAiC,QAAjBuB,EAAAA,EAAKC,KAAKC,eAAVF,IAAAA,OAAAA,EAAAA,EAAmBG,QAAS,IAC5CxB,IAAgC,QAAjBqB,EAAAA,EAAKC,KAAKC,eAAVF,IAAAA,OAAAA,EAAAA,EAAmBI,OAAQ,IAC1CvB,IAAkC,QAAvBmB,EAAAA,EAAKC,KAAKI,qBAAVL,IAAAA,OAAAA,EAAAA,EAAyBpB,UAAW,IAC/CG,IAAsC,QAAvBiB,EAAAA,EAAKC,KAAKI,qBAAVL,IAAAA,OAAAA,EAAAA,EAAyBM,iBAAkB,IAC1DjB,GAAoB,MACpB/B,GAAW,KAEZiD,MAAMC,IACLvB,GAASuB,EAAIC,SAAW,8BACxBnD,GAAW,OAGhB,CAACd,EAAMG,EAAWC,EAAMO,EAAYF,IAGvC,MACEtC,MAAO+F,GACP1B,MAAO2B,KACLC,EAAAA,EAAAA,GAASnB,UACX,IAAKjD,EAAM,OAAO,KAClB,MAAMqE,QAAiB5D,EAASsC,MAAM,GAAGpC,+BACzC,aAAa0D,EAAShB,QACrB,CAAC1C,EAAYF,EAAUT,IAGpBsE,GAAiBC,IAAAA,QAAc,KAC9BL,cAAAA,EAAAA,GAAcM,QAAU5C,EAEtBsC,GAAaM,MAAMC,KAAMC,IAC9B,MAAMC,EAAMD,EAAG9C,UACf,MACgB,eAAd+C,aAAAA,EAAAA,EAAKC,QACLD,aAAAA,EAAAA,EAAKvE,QAASwB,EAAUxB,SACtBuE,aAAAA,EAAAA,EAAKxE,aAAawE,aAAAA,EAAAA,EAAKxE,cAAeyB,EAAUzB,WAAaA,MAPpB,KAU9C,CAAC+D,GAActC,EAAWzB,KAE7B2C,EAAAA,EAAAA,WAAU,KACJ9C,GACF6C,GAAoB,OAErB,CAAC7C,IAEJ,MAKM6E,GAAe,KACf/C,GAAS1D,SAAWsD,EAAKhD,SAASoD,GAAS1D,UAC7CuD,EAAQ,IAAID,EAAMI,GAAS1D,SAC3B2D,GAAY,MA8DhB,OACE,UAAC+C,EAAAA,EAAMA,CAAC9E,KAAMA,EAAMC,QAASA,EAAS8E,SAAS,KAAKC,WAAS,E,WAC3D,SAACC,EAAAA,EAAWA,C,SAAC,sBACb,UAACC,EAAAA,EAAaA,C,UACX1C,KACC,SAAC2C,EAAAA,EAAKA,CAACC,SAAS,QAAQC,MAAO,CAAE9F,aAAc,I,SAC5CiD,KAGJ2B,KACC,UAACgB,EAAAA,EAAKA,CAACC,SAAS,UAAUC,MAAO,CAAE9F,aAAc,I,WAC/C,SAAC+F,SAAAA,C,SAAO,iCAAqC,IAAEnB,GAAkBF,SACjE,SAACsB,EAAAA,EAAUA,CAACC,QAAQ,QAAQH,MAAO,CAAE/F,UAAW,G,SAAK,2CAKxDuB,GACC,SAAC4E,EAAAA,EAAQA,CAAAA,IAET,sB,WAEE,SAACC,EAAAA,EAAGA,CAACC,UAAWtF,EAAQpB,c,UACtB,SAACsG,EAAAA,EAAUA,CAACC,QAAQ,Y,UAAY,SAACF,SAAAA,C,SAAO,0BAE1C,UAACM,EAAAA,EAAIA,CAACC,WAAS,EAACxG,QAAS,E,WACvB,SAACuG,EAAAA,EAAIA,CAACE,MAAI,EAACC,GAAI,E,UACb,SAACC,EAAAA,EAASA,CACRhB,WAAS,EACTiB,MAAM,mBACN9H,MAAO6C,EACPkF,SAAUC,GAAKlF,EAAekF,EAAEC,OAAOjI,OACvCkI,YAAY,SACZC,WAAW,0CACXC,OAAO,SACPC,UAAQ,EACRC,SAAU/D,GACVgE,gBAAiB,CACfrG,QAAS,CACPtB,SAAUsB,EAAQtB,gBAK1B,SAAC6G,EAAAA,EAAIA,CAACE,MAAI,EAACC,GAAI,E,UACb,SAACC,EAAAA,EAASA,CACRhB,WAAS,EACTiB,MAAM,gBACN9H,MAAOiC,EACPqG,UAAQ,EACRH,WAAW,uCACXC,OAAO,cAGX,SAACX,EAAAA,EAAIA,CAACE,MAAI,EAACC,GAAI,E,UACb,SAACC,EAAAA,EAASA,CACRhB,WAAS,EACTiB,MAAM,UACN9H,MAAOiD,EACP8E,SAAUC,GAAK9E,EAAW8E,EAAEC,OAAOjI,OACnCkI,YAAY,KACZC,WAAW,qCACXC,OAAO,SACPE,SAAU/D,QAGd,SAACkD,EAAAA,EAAIA,CAACE,MAAI,EAACC,GAAI,E,UACb,SAACC,EAAAA,EAASA,CACRhB,WAAS,EACTiB,MAAM,MACN9H,MAAO2D,GACPoE,SAAUC,GAAKpE,GAAYoE,EAAEC,OAAOjI,OACpCwI,WAAYR,IACI,UAAVA,EAAES,MACJT,EAAEU,iBACFhC,OAGJwB,YAAY,UACZC,WAAW,gCACXC,OAAO,SACPE,SAAU/D,GACVoE,WAAY,CACVC,aAAcjF,IACZ,SAACkF,EAAAA,EAAcA,CAACC,SAAS,M,UACvB,SAACC,EAAAA,EAAUA,CAACC,KAAK,QAAQC,QAASvC,GAAc4B,SAAU/D,G,UACxD,SAAC2E,EAAAA,EAAOA,CAAC5H,SAAS,mBAGpB6H,OAIT5F,EAAKrD,OAAS,IACb,SAACuH,EAAAA,EAAIA,CAACE,MAAI,EAACC,GAAI,G,UACb,SAACL,EAAAA,EAAGA,CAACxG,QAAQ,OAAOqI,SAAS,O,SAC1B7F,EAAK8F,IAAIC,IACR,SAACC,EAAAA,EAAIA,CAEHzB,MAAOwB,EACPE,SAAUjF,QAAS4E,EAAY,KAAMM,OA9JlCC,EA8JkDJ,OA7JzE9F,EAAQD,EAAKoG,OAAOL,GAAOA,IAAQI,IADb,IAACA,GA+JHV,KAAK,QACLxB,UAAWtF,EAAQR,QACnB4G,SAAU/D,IALL+E,SAWf,SAAC7B,EAAAA,EAAIA,CAACE,MAAI,EAACC,GAAI,G,UACb,SAACC,EAAAA,EAASA,CACRhB,WAAS,EACTiB,MAAM,cACN9H,MAAO+C,EACPgF,SAAUC,GAAKhF,EAAegF,EAAEC,OAAOjI,OACvCkI,YAAY,kBACZE,OAAO,SACPwB,WAAS,EACTC,KAAM,EACNxB,UAAQ,EACRC,SAAU/D,GACVgE,gBAAiB,CACfrG,QAAS,CACPtB,SAAUsB,EAAQtB,mBAQ5B,UAAC2G,EAAAA,EAAGA,CAACC,UAAWtF,EAAQpB,c,WACtB,SAACsG,EAAAA,EAAUA,CAACC,QAAQ,Y,UAAY,SAACF,SAAAA,C,SAAO,wBACxC,SAAC2C,EAAAA,GAAOA,CAACC,MAAM,oD,UACb,SAACC,EAAAA,EAAgBA,CAACxC,UAAWtF,EAAQb,iBAGzC,UAACoG,EAAAA,EAAIA,CAACC,WAAS,EAACxG,QAAS,E,WACvB,SAACuG,EAAAA,EAAIA,CAACE,MAAI,EAACC,GAAI,G,UACb,SAACC,EAAAA,EAASA,CACRhB,WAAS,EACTiB,MAAM,mBACN9H,MAAOmE,GACP4D,SAAUC,IAAKiC,OArNEjK,EAqNsBgI,EAAEC,OAAOjI,MApN9DoE,GAAepE,QACf0E,IAAoBtE,EAAAA,EAAAA,GAAYJ,IAFF,IAACA,GAsNjBkI,YAAY,uCACZC,WAAY1D,IAAoB,4CAChCJ,QAASI,GACT2D,OAAO,SACPE,SAAU/D,QAGd,SAACkD,EAAAA,EAAIA,CAACE,MAAI,EAACC,GAAI,G,UACb,SAACC,EAAAA,EAASA,CACRhB,WAAS,EACTiB,MAAM,oBACN9H,MAAOiE,GACP8D,SAAUC,GAAK9D,GAAW8D,EAAEC,OAAOjI,OACnCkI,YAAY,+BACZC,WAAW,8CACXC,OAAO,SACPE,SAAU/D,OAGbd,IACC,SAACgE,EAAAA,EAAIA,CAACE,MAAI,EAACC,GAAI,G,UACb,SAACC,EAAAA,EAASA,CACRhB,WAAS,EACTiB,MAAM,YACN9H,MAAO,GAAGyD,EAAUzB,WAAaA,KAAayB,EAAUxB,OACxDqG,UAAQ,EACRH,WAAW,+BACXC,OAAO,gBAOd3E,IACC,sB,WACE,UAAC8D,EAAAA,EAAGA,CAACC,UAAWtF,EAAQpB,c,WACtB,SAACsG,EAAAA,EAAUA,CAACC,QAAQ,Y,UAAY,SAACF,SAAAA,C,SAAO,0BACxC,SAAC2C,EAAAA,GAAOA,CAACC,MAAM,uE,UACb,SAACC,EAAAA,EAAgBA,CAACxC,UAAWtF,EAAQb,iBAGzC,SAAC6I,EAAAA,EAAiBA,CAChB/D,eAAgBA,GAChBgE,cAAc,OACdC,aAAa,0CACbC,kBAAkB,QAMxB,UAAC9C,EAAAA,EAAGA,CAACC,UAAWtF,EAAQpB,c,WACtB,SAACsG,EAAAA,EAAUA,CAACC,QAAQ,Y,UAAY,SAACF,SAAAA,C,SAAO,wBACxC,SAAC2C,EAAAA,GAAOA,CAACC,MAAM,2D,UACb,SAACC,EAAAA,EAAgBA,CAACxC,UAAWtF,EAAQb,iBAGzC,SAACiJ,EAAAA,EAAWA,CAACC,UAAU,WAAWjC,SAAU/D,G,UAC1C,UAACiG,EAAAA,EAAUA,CACTC,KAAG,EACHzK,MAAOqD,EACP0E,SAAUC,GAAK1E,EAAgB0E,EAAEC,OAAOjI,O,WAExC,SAAC0K,EAAAA,EAAgBA,CACf1K,MAAM,SACN2K,SAAS,SAACC,EAAAA,EAAKA,CAAC/J,MAAM,YACtBiH,OACE,UAACP,EAAAA,EAAGA,C,WACF,SAACH,EAAAA,EAAUA,CAACC,QAAQ,Q,SAAQ,0BAC5B,SAACD,EAAAA,EAAUA,CAACC,QAAQ,UAAUxG,MAAM,gB,SAAgB,oDAM1D,SAAC6J,EAAAA,EAAgBA,CACf1K,MAAM,YACN2K,SAAS,SAACC,EAAAA,EAAKA,CAAC/J,MAAM,YACtBiH,OACE,UAACP,EAAAA,EAAGA,C,WACF,SAACH,EAAAA,EAAUA,CAACC,QAAQ,Q,SAAQ,eAC5B,SAACD,EAAAA,EAAUA,CAACC,QAAQ,UAAUxG,MAAM,gB,SAAgB,kEAWpE,UAACgK,EAAAA,EAAaA,C,WACZ,SAACC,EAAAA,EAAMA,CAAC7B,QAASnH,EAASwG,SAAU/D,G,SAAQ,YAC5C,SAACuG,EAAAA,EAAMA,CACL7B,QAtSWnE,UACjBR,GAAS,IACTE,IAAU,GAEV,IACE,MAAMuG,EAAQ,CACZzF,KAAM,CACJzC,cACAE,cACAE,UACFE,gBACAE,eACAE,OACAE,eACII,IAAgBE,GAAc,CAC9BwB,QAAS,IACH1B,IAAgB,CAAE2B,MAAO3B,OACzBE,IAAe,CAAE0B,KAAM1B,MAE3B,CAAC,KACDE,IAAWE,GAAc,CAC3BuB,cAAe,IACTzB,IAAW,CAAEA,eACbE,IAAe,CAAEwB,eAAgBxB,MAErC,CAAC,IAIH+B,QAAiB5D,EAASsC,MAC9B,GAAGpC,8BAAuCR,KAAaC,IACvD,CACE+I,OAAQ,QACRC,QAAS,CACP,eAAgB,oBAElBC,KAAMC,KAAKC,UAAUL,KAIzB,IAAK7E,EAASlB,GAAI,CAChB,MAAMC,QAAkBiB,EAAShB,OACjC,MAAM,IAAIC,MAAMF,EAAUZ,OAAS,8BACrC,CAEAtC,IACAD,GACF,CAAE,MAAO+D,GACPvB,GAASuB,aAAeV,MAAQU,EAAIC,QAAUuF,OAAOxF,GACvD,CAAE,QACArB,IAAU,EACZ,GAoPM3D,MAAM,UACNwG,QAAQ,YACRiB,SAAU/D,IAAU7B,IAAYG,IAAgBE,KAAiB0B,GACjE6G,UAAW/G,IAAS,SAACgH,EAAAA,EAAgBA,CAACvC,KAAM,GAAInI,MAAM,iBAAesI,E,SAEpE5E,GAAS,YAAc,e,iLCvbgBiH,EAAAA,EAAAA,GAAiB,CACjEvJ,KAAM,6BACNwJ,WAAY,CAAEC,OAAQ,aAGwBF,EAAAA,EAAAA,GAAiB,CAC/DvJ,KAAM,2BACNwJ,WAAY,CAAEC,OAAQ,WAG0BF,EAAAA,EAAAA,GAAiB,CACjEvJ,KAAM,6BACNwJ,WAAY,CAAEC,OAAQ,aAG0BF,EAAAA,EAAAA,GAAiB,CACjEvJ,KAAM,6BACNwJ,WAAY,CAAEC,OAAQ,YAjBjB,MAoBMC,GAAmCH,EAAAA,EAAAA,GAAiB,CAC/DvJ,KAAM,2BACNwJ,WAAY,CAAEC,OAAQ,UASXE,GAAqCJ,EAAAA,EAAAA,GAAiB,CACjEvJ,KAAM,6BACNwJ,WAAY,CAAEC,OAAQ,YAgBXG,IATsCL,EAAAA,EAAAA,GAAiB,CAClEvJ,KAAM,+BACNwJ,WAAY,CAAEC,OAAQ,WAO2BF,EAAAA,EAAAA,GAAiB,CAClEvJ,KAAM,+BACNwJ,WAAY,CAAEC,OAAQ,WAOXI,GAAwCN,EAAAA,EAAAA,GAAiB,CACpEvJ,KAAM,iCACNwJ,WAAY,CAAEC,OAAQ,YAOXK,GAAwCP,EAAAA,EAAAA,GAAiB,CACpEvJ,KAAM,iCACNwJ,WAAY,CAAEC,OAAQ,YAOXM,GAAwCR,EAAAA,EAAAA,GAAiB,CACpEvJ,KAAM,iCACNwJ,WAAY,CAAEC,OAAQ,YAOXO,GAAwCT,EAAAA,EAAAA,GAAiB,CACpEvJ,KAAM,iCACNwJ,WAAY,CAAEC,OAAQ,YAOXQ,GAAmCV,EAAAA,EAAAA,GAAiB,CAC/DvJ,KAAM,2BACNwJ,WAAY,CAAEC,OAAQ,UAcXS,GAAiCX,EAAAA,EAAAA,GAAiB,CAC7DvJ,KAAM,yBACNwJ,WAAY,CAAEC,OAAQ,UACtBU,aAAc,eAOHC,GAAkCb,EAAAA,EAAAA,GAAiB,CAC9DvJ,KAAM,2BACNwJ,WAAY,CAAEC,OAAQ,UAgBXY,IATkCd,EAAAA,EAAAA,GAAiB,CAC9DvJ,KAAM,2BACNwJ,WAAY,CAAEC,OAAQ,WAOyBF,EAAAA,EAAAA,GAAiB,CAChEvJ,KAAM,6BACNwJ,WAAY,CAAEC,OAAQ,aAOXa,GAAoCf,EAAAA,EAAAA,GAAiB,CAChEvJ,KAAM,6BACNwJ,WAAY,CAAEC,OAAQ,YAOXc,GAAoChB,EAAAA,EAAAA,GAAiB,CAChEvJ,KAAM,6BACNwJ,WAAY,CAAEC,OAAQ,YAOXe,GAAoCjB,EAAAA,EAAAA,GAAiB,CAChEvJ,KAAM,6BACNwJ,WAAY,CAAEC,OAAQ,YAQXgB,GAAkClB,EAAAA,EAAAA,GAAiB,CAC9DvJ,KAAM,0BACNwJ,WAAY,CAAEC,OAAQ,W,0DCzKjB,SAASiB,EACdC,EACAC,GAGA,MAAMC,EAAoB,iBAAkBF,EACxC,CAAEA,WAAYA,EAAkCC,eAChD,CAAED,cAEAG,GAASC,EAAAA,EAAAA,GAAcF,GAE7B,MAAO,CACLG,QAASF,EAAOE,QAChBvK,QAASqK,EAAOrK,QAChB2B,MAAO0I,EAAO1I,MAElB,CAWO,SAAS6I,EACdC,EACAC,EACAC,EACAC,GAEA,QAAIA,MACAD,GAAgBF,IAAYC,EAElC,C,sLCzCO,MAAMG,EAAsB,EACjC1L,OACAkI,QACAhH,cACAyK,cACAvG,WAAW,SACXwG,YAAW,EACXC,YACAC,eAEA,MAAOC,EAAYC,IAAiBjL,EAAAA,EAAAA,UAAS,KAG7C+B,EAAAA,EAAAA,WAAU,KACH9C,GACHgM,EAAc,KAEf,CAAChM,IAEJ,MAAMiM,EAAwC,SAAb7G,GAAuBuG,EAClDO,GAAaD,GAA2BF,IAAeJ,EAQ7D,OACE,UAAC7G,EAAAA,EAAMA,CACL9E,KAAMA,EACNC,QAAS2L,OAAWtE,EAAYwE,EAChC/G,SAAS,KACTC,WAAS,E,WAET,UAACC,EAAAA,EAAWA,C,UACI,SAAbG,IACC,UAACM,EAAAA,EAAGA,CAACxG,QAAQ,OAAOC,WAAW,SAASkG,MAAO,CAAEjG,IAAK,G,WACpD,SAAC+M,EAAAA,EAAWA,CAACnN,MAAM,WACnB,SAACoN,OAAAA,C,SAAMlE,OAGG,SAAb9C,GAAuB8C,MAE1B,UAAChD,EAAAA,EAAaA,C,WACZ,SAACmH,EAAAA,EAAiBA,CAAChH,MAAO,CAAEiH,WAAY,Y,SACrCpL,IAEF+K,IACC,UAACvG,EAAAA,EAAGA,CAAC6G,GAAI,E,WACP,UAAChH,EAAAA,EAAUA,CAACC,QAAQ,QAAQxG,MAAM,gBAAgBwN,cAAY,E,UAAC,SACxD,SAAClH,SAAAA,C,SAAQqG,IAAqB,mBAErC,SAAC3F,EAAAA,EAASA,CACRhB,WAAS,EACTQ,QAAQ,WACR2B,KAAK,QACLhJ,MAAO4N,EACP7F,SAAUC,GAAK6F,EAAc7F,EAAEC,OAAOjI,OACtCsI,SAAUmF,EACVa,WAAS,EACTpG,YAAasF,WAKrB,UAAC3C,EAAAA,EAAaA,C,WACZ,SAACC,EAAAA,EAAMA,CAAC7B,QAAS0E,EAAUrF,SAAUmF,E,SAAU,YAG/C,SAAC3C,EAAAA,EAAMA,CACL7B,QAjDc,KAChB8E,GACFL,KAgDI7M,MAAM,YACNwG,QAAQ,YACRiB,SAAUmF,IAAaM,EACvBzC,UAAWmC,GAAW,SAAClC,EAAAA,EAAgBA,CAACvC,KAAM,GAAInI,MAAM,iBAAesI,E,SAEtEsE,EAAW,cAAgB,iB,oHCjF/B,MAAMvD,EAAsD,EACjE/D,iBACAgE,gBAAgB,UAChBC,eAAe,2FACfC,oBAAmB,MAEnB,MAAM1J,GAAQ4N,EAAAA,EAAAA,KACd,OACE,SAAChH,EAAAA,EAAGA,CACF6G,GAAI/D,EAAmB,EAAI,EAC3BmE,EAAG,EACHC,QAAS9N,EAAMY,QAAQmN,WAAWC,QAClCC,aAAc,EACdC,OAAQ,aAAalO,EAAMY,QAAQuN,U,SAElC3I,GACC,sB,WACE,UAACiB,EAAAA,EAAUA,CAACC,QAAQ,YAAYgH,cAAY,EAACnH,MAAO,CAAE6H,WAAY,K,UAAO,2BAChD,SAAC5H,SAAAA,C,SAAQhB,EAAe6I,SAAS/M,UAGzDkE,EAAe8I,OAAS9I,EAAe8I,MAAM/O,OAAS,GACrD,sB,WACE,SAACkH,EAAAA,EAAUA,CACTC,QAAQ,UACRtG,QAAQ,QACRsN,cAAY,EACZxN,MAAM,gBACNqG,MAAO,CAAE/F,UAAW,G,SACrB,sBAGD,SAACoG,EAAAA,EAAGA,CAACxG,QAAQ,OAAOqI,SAAS,OAAOgF,GAAI,EAAGlH,MAAO,CAAEjG,IAAK,G,SACtDkF,EAAe8I,MAAM5F,IAAI,CAAC6F,EAAWC,K,IAClBD,EAEdA,EAEEA,EAJN,MAAME,GAAuB,QAAXF,EAAAA,EAAKG,cAALH,IAAAA,OAAAA,EAAAA,EAAaI,OAC3B,GAAGJ,EAAKG,OAAOC,aACJ,QAAXJ,EAAAA,EAAKG,cAALH,IAAAA,OAAAA,EAAAA,EAAaK,SACX,GAAGL,EAAKG,OAAOE,iBACJ,QAAXL,EAAAA,EAAKG,cAALH,IAAAA,OAAAA,EAAAA,EAAaM,QACX,GAAGN,EAAKG,OAAOG,cACf,WAER,OACE,SAACjG,EAAAA,EAAIA,CAEHzB,MAAO,GAAGoH,EAAKO,SAASL,IACxBpG,KAAK,QACL3B,QAAQ,WACRxG,MAAM,WAJDsO,WAWf,SAAC/H,EAAAA,EAAUA,CAACC,QAAQ,UAAUxG,MAAM,gB,SAAgB,4CAMxD,SAACmG,EAAAA,EAAKA,CAACC,SAAUkD,E,SAAgBC,M","sources":["webpack://internal.plugin-kuadrant/./src/utils/validation.ts","webpack://internal.plugin-kuadrant/./src/components/EditAPIProductDialog/EditAPIProductDialog.tsx","webpack://internal.plugin-kuadrant/./src/permissions.ts","webpack://internal.plugin-kuadrant/./src/utils/permissions.ts","webpack://internal.plugin-kuadrant/./src/components/ConfirmDeleteDialog/ConfirmDeleteDialog.tsx","webpack://internal.plugin-kuadrant/./src/components/PlanPolicyDetailsCard/PlanPolicyDetails.tsx"],"sourcesContent":["// Kubernetes name validation\nexport const validateKubernetesName = (value: string): string | null => {\n if (!value || !value.trim()) {\n return 'Name is required';\n }\n if (value.length > 253) {\n return 'Must be 253 characters or less';\n }\n\n const dns1123Regex = /^[a-z0-9]([-a-z0-9]*[a-z0-9])?$/;\n\n if (!dns1123Regex.test(value)) {\n return 'Must be lowercase alphanumeric with hyphens, start and end with alphanumeric';\n }\n\n return null;\n};\n\n// URL validation\nexport const validateURL = (value: string): string | null => {\n if (!value) {\n return null;\n }\n\n try {\n const url = new URL(value);\n if (!['http:', 'https:'].includes(url.protocol)) {\n return 'Must be a valid HTTP or HTTPS URL';\n }\n return null;\n } catch {\n return 'Must be a valid URL';\n }\n};\n","import React, { useState, useEffect } from 'react';\nimport {\n Dialog,\n DialogTitle,\n DialogContent,\n DialogActions,\n Button,\n TextField,\n Box,\n Typography,\n Chip,\n Grid,\n CircularProgress,\n makeStyles,\n FormControl,\n RadioGroup,\n FormControlLabel,\n Radio,\n Tooltip,\n IconButton,\n InputAdornment,\n} from '@material-ui/core';\nimport InfoOutlinedIcon from '@material-ui/icons/InfoOutlined';\nimport AddIcon from '@material-ui/icons/Add';\nimport { useApi, configApiRef, fetchApiRef } from '@backstage/core-plugin-api';\nimport { Alert } from '@material-ui/lab';\nimport { Progress } from '@backstage/core-components';\nimport useAsync from 'react-use/lib/useAsync';\nimport { PlanPolicyDetails } from '../PlanPolicyDetailsCard';\nimport { validateURL } from '../../utils/validation';\n\nconst useStyles = makeStyles((theme) => ({\n asterisk: {\n color: '#f44336',\n },\n sectionHeader: {\n display: 'flex',\n alignItems: 'center',\n gap: theme.spacing(0.5),\n marginTop: theme.spacing(2),\n marginBottom: theme.spacing(1),\n },\n infoIcon: {\n fontSize: 18,\n color: theme.palette.text.secondary,\n },\n tagChip: {\n marginRight: theme.spacing(0.5),\n marginBottom: theme.spacing(0.5),\n },\n}));\n\ninterface EditAPIProductDialogProps {\n open: boolean;\n onClose: () => void;\n onSuccess: () => void;\n namespace: string;\n name: string;\n}\n\nexport const EditAPIProductDialog = ({open, onClose, onSuccess, namespace, name}: EditAPIProductDialogProps) => {\n const classes = useStyles();\n const config = useApi(configApiRef);\n const fetchApi = useApi(fetchApiRef);\n const backendUrl = config.getString('backend.baseUrl');\n const [loading, setLoading] = useState(false);\n const [displayName, setDisplayName] = useState('');\n const [description, setDescription] = useState('');\n const [version, setVersion] = useState('v1');\n const [publishStatus, setPublishStatus] = useState<'Draft' | 'Published'>('Draft');\n const [approvalMode, setApprovalMode] = useState<'automatic' | 'manual'>('manual');\n const [tags, setTags] = useState<string[]>([]);\n const [targetRef, setTargetRef] = useState<any>(null);\n const [tagInput, setTagInput] = useState('');\n const [contactEmail, setContactEmail] = useState('');\n const [contactTeam, setContactTeam] = useState('');\n const [docsURL, setDocsURL] = useState('');\n const [openAPISpec, setOpenAPISpec] = useState('');\n const [error, setError] = useState('');\n const [saving, setSaving] = useState(false);\n const [openAPISpecError, setOpenAPISpecError] = useState<string | null>(null);\n\n // Load APIProduct data when dialog opens\n useEffect(() => {\n if (open && namespace && name) {\n setLoading(true);\n setError('');\n\n fetchApi.fetch(`${backendUrl}/api/kuadrant/apiproducts/${namespace}/${name}`)\n .then(async res => {\n if (!res.ok) {\n const errorData = await res.json();\n throw new Error(errorData.error || `Failed to fetch API product: ${res.status}`);\n }\n return res.json();\n })\n .then(data => {\n setDisplayName(data.spec.displayName || '');\n setDescription(data.spec.description || '');\n setVersion(data.spec.version || 'v1');\n setPublishStatus(data.spec.publishStatus || 'Draft');\n setApprovalMode(data.spec.approvalMode || 'manual');\n setTags(data.spec.tags || []);\n setTargetRef(data.spec.targetRef || null);\n setContactEmail(data.spec.contact?.email || '');\n setContactTeam(data.spec.contact?.team || '');\n setDocsURL(data.spec.documentation?.docsURL || '');\n setOpenAPISpec(data.spec.documentation?.openAPISpecURL || '');\n setOpenAPISpecError(null);\n setLoading(false);\n })\n .catch(err => {\n setError(err.message || 'Failed to load API product');\n setLoading(false);\n });\n }\n }, [open, namespace, name, backendUrl, fetchApi]);\n\n // load planpolicies with full details to show associated plans\n const {\n value: planPolicies,\n error: planPoliciesError\n } = useAsync(async () => {\n if (!open) return null;\n const response = await fetchApi.fetch(`${backendUrl}/api/kuadrant/planpolicies`);\n return await response.json();\n }, [backendUrl, fetchApi, open]);\n\n // find planpolicy associated with targetRef\n const selectedPolicy = React.useMemo(() => {\n if (!planPolicies?.items || !targetRef) return null;\n\n return planPolicies.items.find((pp: any) => {\n const ref = pp.targetRef;\n return (\n ref?.kind === 'HTTPRoute' &&\n ref?.name === targetRef.name &&\n (!ref?.namespace || ref?.namespace === (targetRef.namespace || namespace))\n );\n });\n }, [planPolicies, targetRef, namespace]);\n\n useEffect(() => {\n if (open) {\n setOpenAPISpecError(null);\n }\n }, [open]);\n\n const handleOpenAPISpecChange = (value: string) => {\n setOpenAPISpec(value);\n setOpenAPISpecError(validateURL(value));\n };\n\n const handleAddTag = () => {\n if (tagInput.trim() && !tags.includes(tagInput.trim())) {\n setTags([...tags, tagInput.trim()]);\n setTagInput('');\n }\n };\n\n const handleDeleteTag = (tagToDelete: string) => {\n setTags(tags.filter(tag => tag !== tagToDelete));\n };\n\n const handleSave = async () => {\n setError('');\n setSaving(true);\n\n try {\n const patch = {\n spec: {\n displayName,\n description,\n version,\n publishStatus,\n approvalMode,\n tags,\n targetRef,\n ...(contactEmail || contactTeam ? {\n contact: {\n ...(contactEmail && { email: contactEmail }),\n ...(contactTeam && { team: contactTeam }),\n },\n } : {}),\n ...(docsURL || openAPISpec ? {\n documentation: {\n ...(docsURL && { docsURL }),\n ...(openAPISpec && { openAPISpecURL: openAPISpec }),\n },\n } : {}),\n },\n };\n\n const response = await fetchApi.fetch(\n `${backendUrl}/api/kuadrant/apiproducts/${namespace}/${name}`,\n {\n method: 'PATCH',\n headers: {\n 'Content-Type': 'application/json',\n },\n body: JSON.stringify(patch),\n }\n );\n\n if (!response.ok) {\n const errorData = await response.json();\n throw new Error(errorData.error || 'failed to update apiproduct');\n }\n\n onSuccess();\n onClose();\n } catch (err) {\n setError(err instanceof Error ? err.message : String(err));\n } finally {\n setSaving(false);\n }\n };\n\n return (\n <Dialog open={open} onClose={onClose} maxWidth=\"md\" fullWidth>\n <DialogTitle>Edit API Product</DialogTitle>\n <DialogContent>\n {error && (\n <Alert severity=\"error\" style={{ marginBottom: 16 }}>\n {error}\n </Alert>\n )}\n {planPoliciesError && (\n <Alert severity=\"warning\" style={{ marginBottom: 16 }}>\n <strong>Failed to load PlanPolicies:</strong> {planPoliciesError.message}\n <Typography variant=\"body2\" style={{ marginTop: 8 }}>\n Plan information may be incomplete.\n </Typography>\n </Alert>\n )}\n {loading ? (\n <Progress />\n ) : (\n <>\n {/* API product info section */}\n <Box className={classes.sectionHeader}>\n <Typography variant=\"subtitle1\"><strong>API product info</strong></Typography>\n </Box>\n <Grid container spacing={2}>\n <Grid item xs={6}>\n <TextField\n fullWidth\n label=\"API product name\"\n value={displayName}\n onChange={e => setDisplayName(e.target.value)}\n placeholder=\"My API\"\n helperText=\"Give a unique name for your API product\"\n margin=\"normal\"\n required\n disabled={saving}\n InputLabelProps={{\n classes: {\n asterisk: classes.asterisk,\n },\n }}\n />\n </Grid>\n <Grid item xs={6}>\n <TextField\n fullWidth\n label=\"Resource name\"\n value={name}\n disabled\n helperText=\"Kubernetes resource name (immutable)\"\n margin=\"normal\"\n />\n </Grid>\n <Grid item xs={6}>\n <TextField\n fullWidth\n label=\"Version\"\n value={version}\n onChange={e => setVersion(e.target.value)}\n placeholder=\"v1\"\n helperText=\"Give a version to your API product\"\n margin=\"normal\"\n disabled={saving}\n />\n </Grid>\n <Grid item xs={6}>\n <TextField\n fullWidth\n label=\"Tag\"\n value={tagInput}\n onChange={e => setTagInput(e.target.value)}\n onKeyPress={e => {\n if (e.key === 'Enter') {\n e.preventDefault();\n handleAddTag();\n }\n }}\n placeholder=\"Add tag\"\n helperText=\"Add a tag to your API product\"\n margin=\"normal\"\n disabled={saving}\n InputProps={{\n endAdornment: tagInput ? (\n <InputAdornment position=\"end\">\n <IconButton size=\"small\" onClick={handleAddTag} disabled={saving}>\n <AddIcon fontSize=\"small\" />\n </IconButton>\n </InputAdornment>\n ) : undefined,\n }}\n />\n </Grid>\n {tags.length > 0 && (\n <Grid item xs={12}>\n <Box display=\"flex\" flexWrap=\"wrap\">\n {tags.map(tag => (\n <Chip\n key={tag}\n label={tag}\n onDelete={saving ? undefined : () => handleDeleteTag(tag)}\n size=\"small\"\n className={classes.tagChip}\n disabled={saving}\n />\n ))}\n </Box>\n </Grid>\n )}\n <Grid item xs={12}>\n <TextField\n fullWidth\n label=\"Description\"\n value={description}\n onChange={e => setDescription(e.target.value)}\n placeholder=\"API description\"\n margin=\"normal\"\n multiline\n rows={2}\n required\n disabled={saving}\n InputLabelProps={{\n classes: {\n asterisk: classes.asterisk,\n },\n }}\n />\n </Grid>\n </Grid>\n\n {/* Associated route section */}\n <Box className={classes.sectionHeader}>\n <Typography variant=\"subtitle1\"><strong>Associated route</strong></Typography>\n <Tooltip title=\"The HTTPRoute this API product is associated with\">\n <InfoOutlinedIcon className={classes.infoIcon} />\n </Tooltip>\n </Box>\n <Grid container spacing={2}>\n <Grid item xs={12}>\n <TextField\n fullWidth\n label=\"OpenAPI Spec URL\"\n value={openAPISpec}\n onChange={e => handleOpenAPISpecChange(e.target.value)}\n placeholder=\"https://api.example.com/openapi.json\"\n helperText={openAPISpecError || \"Enter the full path to your API spec file\"}\n error={!!openAPISpecError}\n margin=\"normal\"\n disabled={saving}\n />\n </Grid>\n <Grid item xs={12}>\n <TextField\n fullWidth\n label=\"Documentation URL\"\n value={docsURL}\n onChange={e => setDocsURL(e.target.value)}\n placeholder=\"https://docs.example.com/api\"\n helperText=\"Link to external documentation for this API\"\n margin=\"normal\"\n disabled={saving}\n />\n </Grid>\n {targetRef && (\n <Grid item xs={12}>\n <TextField\n fullWidth\n label=\"HTTPRoute\"\n value={`${targetRef.namespace || namespace}/${targetRef.name}`}\n disabled\n helperText=\"Target HTTPRoute (immutable)\"\n margin=\"normal\"\n />\n </Grid>\n )}\n </Grid>\n\n {/* HTTPRoute policies section */}\n {targetRef && (\n <>\n <Box className={classes.sectionHeader}>\n <Typography variant=\"subtitle1\"><strong>HTTPRoute policies</strong></Typography>\n <Tooltip title=\"Shows the associated policies and rate limit tiers for the HTTPRoute\">\n <InfoOutlinedIcon className={classes.infoIcon} />\n </Tooltip>\n </Box>\n <PlanPolicyDetails\n selectedPolicy={selectedPolicy}\n alertSeverity=\"info\"\n alertMessage=\"No PlanPolicy found for this HTTPRoute.\"\n includeTopMargin={false}\n />\n </>\n )}\n\n {/* API Key approval section */}\n <Box className={classes.sectionHeader}>\n <Typography variant=\"subtitle1\"><strong>API Key approval</strong></Typography>\n <Tooltip title=\"Choose how API key requests are handled for this product\">\n <InfoOutlinedIcon className={classes.infoIcon} />\n </Tooltip>\n </Box>\n <FormControl component=\"fieldset\" disabled={saving}>\n <RadioGroup\n row\n value={approvalMode}\n onChange={e => setApprovalMode(e.target.value as 'automatic' | 'manual')}\n >\n <FormControlLabel\n value=\"manual\"\n control={<Radio color=\"primary\" />}\n label={\n <Box>\n <Typography variant=\"body2\">Need manual approval</Typography>\n <Typography variant=\"caption\" color=\"textSecondary\">\n Requires approval for requesting this API\n </Typography>\n </Box>\n }\n />\n <FormControlLabel\n value=\"automatic\"\n control={<Radio color=\"primary\" />}\n label={\n <Box>\n <Typography variant=\"body2\">Automatic</Typography>\n <Typography variant=\"caption\" color=\"textSecondary\">\n Keys are created without need to be approved\n </Typography>\n </Box>\n }\n />\n </RadioGroup>\n </FormControl>\n </>\n )}\n </DialogContent>\n <DialogActions>\n <Button onClick={onClose} disabled={saving}>Cancel</Button>\n <Button\n onClick={handleSave}\n color=\"primary\"\n variant=\"contained\"\n disabled={saving || loading || !displayName || !description || !!openAPISpecError}\n startIcon={saving ? <CircularProgress size={16} color=\"inherit\" /> : undefined}\n >\n {saving ? 'Saving...' : 'Save'}\n </Button>\n </DialogActions>\n </Dialog>\n );\n};\n","import { createPermission } from '@backstage/plugin-permission-common';\n\n/**\n * permission definitions for the kuadrant plugin\n *\n * these permissions control access to kuadrant resources and operations.\n * they must match the permissions defined in the backend plugin.\n *\n * permission types:\n * - BasicPermission: standard permission that applies globally\n * - ResourcePermission: permission scoped to specific resource types (e.g., apiproduct)\n *\n * permission patterns:\n * - `.create` - create new resources\n * - `.read` - read resource details\n * - `.read.own` - read only resources owned by the user\n * - `.read.all` - read all resources regardless of ownership\n * - `.update` - modify existing resources\n * - `.delete` - delete resources\n * - `.delete.own` - delete only resources owned by the user\n * - `.delete.all` - delete any resource regardless of ownership\n * - `.list` - list/view collections of resources\n */\n\n// planpolicy permissions\nexport const kuadrantPlanPolicyCreatePermission = createPermission({\n name: 'kuadrant.planpolicy.create',\n attributes: { action: 'create' },\n});\n\nexport const kuadrantPlanPolicyReadPermission = createPermission({\n name: 'kuadrant.planpolicy.read',\n attributes: { action: 'read' },\n});\n\nexport const kuadrantPlanPolicyUpdatePermission = createPermission({\n name: 'kuadrant.planpolicy.update',\n attributes: { action: 'update' },\n});\n\nexport const kuadrantPlanPolicyDeletePermission = createPermission({\n name: 'kuadrant.planpolicy.delete',\n attributes: { action: 'delete' },\n});\n\nexport const kuadrantPlanPolicyListPermission = createPermission({\n name: 'kuadrant.planpolicy.list',\n attributes: { action: 'read' },\n});\n\n// apiproduct permissions\n\n/**\n * permission to create new API products\n * granted to api owners and admins\n */\nexport const kuadrantApiProductCreatePermission = createPermission({\n name: 'kuadrant.apiproduct.create',\n attributes: { action: 'create' },\n});\n\n/**\n * permission to read API products owned by the current user\n * for api owners to view their own products\n */\nexport const kuadrantApiProductReadOwnPermission = createPermission({\n name: 'kuadrant.apiproduct.read.own',\n attributes: { action: 'read' },\n});\n\n/**\n * permission to read all API products regardless of ownership\n * for platform engineers/admins who need to view all products\n */\nexport const kuadrantApiProductReadAllPermission = createPermission({\n name: 'kuadrant.apiproduct.read.all',\n attributes: { action: 'read' },\n});\n\n/**\n * permission to update API products owned by the current user\n * for api owners to modify their own products\n */\nexport const kuadrantApiProductUpdateOwnPermission = createPermission({\n name: 'kuadrant.apiproduct.update.own',\n attributes: { action: 'update' },\n});\n\n/**\n * permission to update any API product regardless of ownership\n * for platform engineers/admins\n */\nexport const kuadrantApiProductUpdateAllPermission = createPermission({\n name: 'kuadrant.apiproduct.update.all',\n attributes: { action: 'update' },\n});\n\n/**\n * permission to delete API products owned by the current user\n * for api owners to remove their own products\n */\nexport const kuadrantApiProductDeleteOwnPermission = createPermission({\n name: 'kuadrant.apiproduct.delete.own',\n attributes: { action: 'delete' },\n});\n\n/**\n * permission to delete any API product regardless of ownership\n * for platform engineers/admins\n */\nexport const kuadrantApiProductDeleteAllPermission = createPermission({\n name: 'kuadrant.apiproduct.delete.all',\n attributes: { action: 'delete' },\n});\n\n/**\n * permission to list API products\n * backend filters results based on .own vs .all read permissions\n */\nexport const kuadrantApiProductListPermission = createPermission({\n name: 'kuadrant.apiproduct.list',\n attributes: { action: 'read' },\n});\n\n// apikey permissions\n\n/**\n * permission to create API keys (request API access)\n *\n * this is a ResourcePermission scoped to 'apiproduct', allowing\n * fine-grained control over which API products users can request access to.\n *\n * use in frontend: useKuadrantPermission(kuadrantApiKeyCreatePermission)\n * use in backend with resource: { permission, resourceRef: 'apiproduct:namespace/name' }\n */\nexport const kuadrantApiKeyCreatePermission = createPermission({\n name: 'kuadrant.apikey.create',\n attributes: { action: 'create' },\n resourceType: 'apiproduct',\n});\n\n/**\n * permission to read API keys owned by the current user\n * allows users to view their own API keys and request history\n */\nexport const kuadrantApiKeyReadOwnPermission = createPermission({\n name: 'kuadrant.apikey.read.own',\n attributes: { action: 'read' },\n});\n\n/**\n * permission to read all API keys regardless of ownership\n * for platform engineers/admins who need to view the approval queue and audit keys\n */\nexport const kuadrantApiKeyReadAllPermission = createPermission({\n name: 'kuadrant.apikey.read.all',\n attributes: { action: 'read' },\n});\n\n/**\n * permission to update API keys owned by the current user\n * allows users to edit their own pending requests (change plan tier, use case)\n */\nexport const kuadrantApiKeyUpdateOwnPermission = createPermission({\n name: 'kuadrant.apikey.update.own',\n attributes: { action: 'update' },\n});\n\n/**\n * permission to update any API key regardless of ownership\n * typically granted to API owners and platform engineers for approving/rejecting requests\n */\nexport const kuadrantApiKeyUpdateAllPermission = createPermission({\n name: 'kuadrant.apikey.update.all',\n attributes: { action: 'update' },\n});\n\n/**\n * permission to delete API keys owned by the current user\n * allows users to cancel their own requests or revoke their own access\n */\nexport const kuadrantApiKeyDeleteOwnPermission = createPermission({\n name: 'kuadrant.apikey.delete.own',\n attributes: { action: 'delete' },\n});\n\n/**\n * permission to delete any API key regardless of ownership\n * for platform engineers/admins who need to revoke access\n */\nexport const kuadrantApiKeyDeleteAllPermission = createPermission({\n name: 'kuadrant.apikey.delete.all',\n attributes: { action: 'delete' },\n});\n\n/**\n * permission to approve/reject API key requests\n * grants access to the approval queue - for API owners and admins only\n * separate from update.own which consumers use to edit their pending requests\n */\nexport const kuadrantApiKeyApprovePermission = createPermission({\n name: 'kuadrant.apikey.approve',\n attributes: { action: 'update' },\n});\n\nexport const kuadrantPermissions = [\n kuadrantPlanPolicyCreatePermission,\n kuadrantPlanPolicyReadPermission,\n kuadrantPlanPolicyUpdatePermission,\n kuadrantPlanPolicyDeletePermission,\n kuadrantPlanPolicyListPermission,\n kuadrantApiProductCreatePermission,\n kuadrantApiProductReadOwnPermission,\n kuadrantApiProductReadAllPermission,\n kuadrantApiProductUpdateOwnPermission,\n kuadrantApiProductUpdateAllPermission,\n kuadrantApiProductDeleteOwnPermission,\n kuadrantApiProductDeleteAllPermission,\n kuadrantApiProductListPermission,\n kuadrantApiKeyCreatePermission,\n kuadrantApiKeyReadOwnPermission,\n kuadrantApiKeyReadAllPermission,\n kuadrantApiKeyUpdateOwnPermission,\n kuadrantApiKeyUpdateAllPermission,\n kuadrantApiKeyDeleteOwnPermission,\n kuadrantApiKeyDeleteAllPermission,\n kuadrantApiKeyApprovePermission,\n];\n","import { usePermission } from '@backstage/plugin-permission-react';\nimport { Permission, ResourcePermission } from '@backstage/plugin-permission-common';\n\n/**\n * result of a permission check including error state\n */\nexport interface PermissionCheckResult {\n allowed: boolean;\n loading: boolean;\n error?: Error;\n}\n\n/**\n * custom hook for checking kuadrant permissions that handles both\n * BasicPermission and ResourcePermission types without type bypasses\n *\n * @param permission - the permission to check\n * @param resourceRef - optional resource reference for ResourcePermissions\n * @returns permission check result with error handling\n *\n * @example\n * // basic permission\n * const { allowed, loading, error } = useKuadrantPermission(\n * kuadrantApiProductListPermission\n * );\n *\n * @example\n * // resource permission\n * const { allowed, loading, error } = useKuadrantPermission(\n * kuadrantApiKeyCreatePermission,\n * 'apiproduct:namespace/name'\n * );\n */\nexport function useKuadrantPermission(\n permission: Permission,\n resourceRef?: string,\n): PermissionCheckResult {\n // construct the permission request based on whether it's a ResourcePermission\n const permissionRequest = 'resourceType' in permission\n ? { permission: permission as ResourcePermission, resourceRef }\n : { permission };\n\n const result = usePermission(permissionRequest as any);\n\n return {\n allowed: result.allowed,\n loading: result.loading,\n error: result.error,\n };\n}\n\n/**\n * helper to determine if a user can delete a specific API key or request\n *\n * @param ownerId - the user id who owns the key/request\n * @param currentUserId - the current user's id\n * @param canDeleteOwn - whether user has permission to delete their own keys\n * @param canDeleteAll - whether user has permission to delete all keys\n * @returns true if user can delete this specific key/request\n */\nexport function canDeleteResource(\n ownerId: string,\n currentUserId: string,\n canDeleteOwn: boolean,\n canDeleteAll: boolean,\n): boolean {\n if (canDeleteAll) return true;\n if (canDeleteOwn && ownerId === currentUserId) return true;\n return false;\n}\n","import React, { useState, useEffect } from 'react';\nimport {\n Dialog,\n DialogTitle,\n DialogContent,\n DialogContentText,\n DialogActions,\n Button,\n TextField,\n Typography,\n Box,\n CircularProgress,\n} from '@material-ui/core';\nimport WarningIcon from '@material-ui/icons/Warning';\n\nexport interface ConfirmDeleteDialogProps {\n open: boolean;\n title: string;\n description: string;\n // for dangerous deletes, require typing this text to confirm\n confirmText?: string;\n // severity affects styling - 'high' shows warning icon and requires text confirmation\n severity?: 'normal' | 'high';\n deleting?: boolean;\n onConfirm: () => void;\n onCancel: () => void;\n}\n\nexport const ConfirmDeleteDialog = ({\n open,\n title,\n description,\n confirmText,\n severity = 'normal',\n deleting = false,\n onConfirm,\n onCancel,\n}: ConfirmDeleteDialogProps) => {\n const [inputValue, setInputValue] = useState('');\n\n // reset input when dialog opens/closes\n useEffect(() => {\n if (!open) {\n setInputValue('');\n }\n }, [open]);\n\n const requiresTextConfirmation = severity === 'high' && confirmText;\n const canConfirm = requiresTextConfirmation ? inputValue === confirmText : true;\n\n const handleConfirm = () => {\n if (canConfirm) {\n onConfirm();\n }\n };\n\n return (\n <Dialog\n open={open}\n onClose={deleting ? undefined : onCancel}\n maxWidth=\"sm\"\n fullWidth\n >\n <DialogTitle>\n {severity === 'high' && (\n <Box display=\"flex\" alignItems=\"center\" style={{ gap: 8 }}>\n <WarningIcon color=\"error\" />\n <span>{title}</span>\n </Box>\n )}\n {severity !== 'high' && title}\n </DialogTitle>\n <DialogContent>\n <DialogContentText style={{ whiteSpace: 'pre-line' }}>\n {description}\n </DialogContentText>\n {requiresTextConfirmation && (\n <Box mt={2}>\n <Typography variant=\"body2\" color=\"textSecondary\" gutterBottom>\n Type <strong>{confirmText}</strong> to confirm:\n </Typography>\n <TextField\n fullWidth\n variant=\"outlined\"\n size=\"small\"\n value={inputValue}\n onChange={e => setInputValue(e.target.value)}\n disabled={deleting}\n autoFocus\n placeholder={confirmText}\n />\n </Box>\n )}\n </DialogContent>\n <DialogActions>\n <Button onClick={onCancel} disabled={deleting}>\n Cancel\n </Button>\n <Button\n onClick={handleConfirm}\n color=\"secondary\"\n variant=\"contained\"\n disabled={deleting || !canConfirm}\n startIcon={deleting ? <CircularProgress size={16} color=\"inherit\" /> : undefined}\n >\n {deleting ? 'Deleting...' : 'Delete'}\n </Button>\n </DialogActions>\n </Dialog>\n );\n};\n","import React from 'react';\nimport { Box, Typography, Chip, useTheme } from '@material-ui/core';\nimport { Alert } from '@material-ui/lab';\n\ninterface PlanPolicyDetailsProps {\n selectedPolicy: {\n metadata: {\n name: string;\n };\n plans?: Array<{\n tier: string;\n description?: string;\n limits?: {\n daily?: number;\n monthly?: number;\n yearly?: number;\n };\n }>;\n } | null;\n alertSeverity?: 'warning' | 'info';\n alertMessage?: string;\n includeTopMargin?: boolean;\n}\n\nexport const PlanPolicyDetails: React.FC<PlanPolicyDetailsProps> = ({\n selectedPolicy,\n alertSeverity = 'warning',\n alertMessage = 'No PlanPolicy found for this HTTPRoute. API keys and rate limiting may not be available.',\n includeTopMargin = true,\n}) => {\n const theme = useTheme();\n return (\n <Box\n mt={includeTopMargin ? 1 : 0}\n p={2}\n bgcolor={theme.palette.background.default}\n borderRadius={1}\n border={`1px solid ${theme.palette.divider}`}\n >\n {selectedPolicy ? (\n <>\n <Typography variant=\"subtitle2\" gutterBottom style={{ fontWeight: 600 }}>\n Associated PlanPolicy: <strong>{selectedPolicy.metadata.name}</strong>\n </Typography>\n\n {selectedPolicy.plans && selectedPolicy.plans.length > 0 ? (\n <>\n <Typography\n variant=\"caption\"\n display=\"block\"\n gutterBottom\n color=\"textSecondary\"\n style={{ marginTop: 8 }}\n >\n Available Tiers:\n </Typography>\n <Box display=\"flex\" flexWrap=\"wrap\" mt={1} style={{ gap: 8 }}>\n {selectedPolicy.plans.map((plan: any, idx: number) => {\n const limitText = plan.limits?.daily\n ? `${plan.limits.daily}/day`\n : plan.limits?.monthly\n ? `${plan.limits.monthly}/month`\n : plan.limits?.yearly\n ? `${plan.limits.yearly}/year`\n : 'No limit';\n\n return (\n <Chip\n key={idx}\n label={`${plan.tier}: ${limitText}`}\n size=\"small\"\n variant=\"outlined\"\n color=\"primary\"\n />\n );\n })}\n </Box>\n </>\n ) : (\n <Typography variant=\"caption\" color=\"textSecondary\">\n No plans defined in this PlanPolicy\n </Typography>\n )}\n </>\n ) : (\n <Alert severity={alertSeverity}>{alertMessage}</Alert>\n )}\n </Box>\n );\n};\n"],"names":["validateKubernetesName","value","trim","length","test","validateURL","url","URL","includes","protocol","useStyles","makeStyles","theme","asterisk","color","sectionHeader","display","alignItems","gap","spacing","marginTop","marginBottom","infoIcon","fontSize","palette","text","secondary","tagChip","marginRight","EditAPIProductDialog","open","onClose","onSuccess","namespace","name","classes","config","useApi","configApiRef","fetchApi","fetchApiRef","backendUrl","getString","loading","setLoading","useState","displayName","setDisplayName","description","setDescription","version","setVersion","publishStatus","setPublishStatus","approvalMode","setApprovalMode","tags","setTags","targetRef","setTargetRef","tagInput","setTagInput","contactEmail","setContactEmail","contactTeam","setContactTeam","docsURL","setDocsURL","openAPISpec","setOpenAPISpec","error","setError","saving","setSaving","openAPISpecError","setOpenAPISpecError","useEffect","fetch","then","async","res","ok","errorData","json","Error","status","data","spec","contact","email","team","documentation","openAPISpecURL","catch","err","message","planPolicies","planPoliciesError","useAsync","response","selectedPolicy","React","items","find","pp","ref","kind","handleAddTag","Dialog","maxWidth","fullWidth","DialogTitle","DialogContent","Alert","severity","style","strong","Typography","variant","Progress","Box","className","Grid","container","item","xs","TextField","label","onChange","e","target","placeholder","helperText","margin","required","disabled","InputLabelProps","onKeyPress","key","preventDefault","InputProps","endAdornment","InputAdornment","position","IconButton","size","onClick","AddIcon","undefined","flexWrap","map","tag","Chip","onDelete","handleDeleteTag","tagToDelete","filter","multiline","rows","Tooltip","title","InfoOutlinedIcon","handleOpenAPISpecChange","PlanPolicyDetails","alertSeverity","alertMessage","includeTopMargin","FormControl","component","RadioGroup","row","FormControlLabel","control","Radio","DialogActions","Button","patch","method","headers","body","JSON","stringify","String","startIcon","CircularProgress","createPermission","attributes","action","kuadrantPlanPolicyListPermission","kuadrantApiProductCreatePermission","kuadrantApiProductReadAllPermission","kuadrantApiProductUpdateOwnPermission","kuadrantApiProductUpdateAllPermission","kuadrantApiProductDeleteOwnPermission","kuadrantApiProductDeleteAllPermission","kuadrantApiProductListPermission","kuadrantApiKeyCreatePermission","resourceType","kuadrantApiKeyReadOwnPermission","kuadrantApiKeyUpdateOwnPermission","kuadrantApiKeyUpdateAllPermission","kuadrantApiKeyDeleteOwnPermission","kuadrantApiKeyDeleteAllPermission","kuadrantApiKeyApprovePermission","useKuadrantPermission","permission","resourceRef","permissionRequest","result","usePermission","allowed","canDeleteResource","ownerId","currentUserId","canDeleteOwn","canDeleteAll","ConfirmDeleteDialog","confirmText","deleting","onConfirm","onCancel","inputValue","setInputValue","requiresTextConfirmation","canConfirm","WarningIcon","span","DialogContentText","whiteSpace","mt","gutterBottom","autoFocus","useTheme","p","bgcolor","background","default","borderRadius","border","divider","fontWeight","metadata","plans","plan","idx","limitText","limits","daily","monthly","yearly","tier"],"sourceRoot":""}
1
+ {"version":3,"file":"static/8799.83d049f3.chunk.js","mappings":"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","sources":["webpack://internal.plugin-kuadrant/./src/utils/validation.ts","webpack://internal.plugin-kuadrant/./src/components/EditAPIProductDialog/EditAPIProductDialog.tsx","webpack://internal.plugin-kuadrant/./src/permissions.ts","webpack://internal.plugin-kuadrant/./src/utils/permissions.ts","webpack://internal.plugin-kuadrant/./src/components/ConfirmDeleteDialog/ConfirmDeleteDialog.tsx","webpack://internal.plugin-kuadrant/./src/components/PlanPolicyDetailsCard/PlanPolicyDetails.tsx"],"sourcesContent":["// Kubernetes name validation\nexport const validateKubernetesName = (value: string): string | null => {\n if (!value || !value.trim()) {\n return 'Name is required';\n }\n if (value.length > 253) {\n return 'Must be 253 characters or less';\n }\n\n const dns1123Regex = /^[a-z0-9]([-a-z0-9]*[a-z0-9])?$/;\n\n if (!dns1123Regex.test(value)) {\n return 'Must be lowercase alphanumeric with hyphens, start and end with alphanumeric';\n }\n\n return null;\n};\n\n// URL validation\nexport const validateURL = (value: string): string | null => {\n if (!value) {\n return null;\n }\n\n try {\n const url = new URL(value);\n if (!['http:', 'https:'].includes(url.protocol)) {\n return 'Must be a valid HTTP or HTTPS URL';\n }\n return null;\n } catch {\n return 'Must be a valid URL';\n }\n};\n","import React, { useState, useEffect } from 'react';\nimport {\n Dialog,\n DialogTitle,\n DialogContent,\n DialogActions,\n Button,\n TextField,\n Box,\n Typography,\n Chip,\n Grid,\n CircularProgress,\n makeStyles,\n FormControl,\n RadioGroup,\n FormControlLabel,\n Radio,\n Tooltip,\n IconButton,\n InputAdornment,\n} from '@material-ui/core';\nimport InfoOutlinedIcon from '@material-ui/icons/InfoOutlined';\nimport AddIcon from '@material-ui/icons/Add';\nimport { useApi, configApiRef, fetchApiRef } from '@backstage/core-plugin-api';\nimport { Alert } from '@material-ui/lab';\nimport { Progress } from '@backstage/core-components';\nimport useAsync from 'react-use/lib/useAsync';\nimport { PlanPolicyDetails } from '../PlanPolicyDetailsCard';\nimport { validateURL } from '../../utils/validation';\n\nconst useStyles = makeStyles((theme) => ({\n asterisk: {\n color: '#f44336',\n },\n sectionHeader: {\n display: 'flex',\n alignItems: 'center',\n gap: theme.spacing(0.5),\n marginTop: theme.spacing(2),\n marginBottom: theme.spacing(1),\n },\n infoIcon: {\n fontSize: 18,\n color: theme.palette.text.secondary,\n },\n tagChip: {\n marginRight: theme.spacing(0.5),\n marginBottom: theme.spacing(0.5),\n },\n}));\n\ninterface EditAPIProductDialogProps {\n open: boolean;\n onClose: () => void;\n onSuccess: () => void;\n namespace: string;\n name: string;\n}\n\nexport const EditAPIProductDialog = ({open, onClose, onSuccess, namespace, name}: EditAPIProductDialogProps) => {\n const classes = useStyles();\n const config = useApi(configApiRef);\n const fetchApi = useApi(fetchApiRef);\n const backendUrl = config.getString('backend.baseUrl');\n const [loading, setLoading] = useState(false);\n const [displayName, setDisplayName] = useState('');\n const [description, setDescription] = useState('');\n const [version, setVersion] = useState('v1');\n const [publishStatus, setPublishStatus] = useState<'Draft' | 'Published'>('Draft');\n const [approvalMode, setApprovalMode] = useState<'automatic' | 'manual'>('manual');\n const [tags, setTags] = useState<string[]>([]);\n const [targetRef, setTargetRef] = useState<any>(null);\n const [tagInput, setTagInput] = useState('');\n const [contactEmail, setContactEmail] = useState('');\n const [contactTeam, setContactTeam] = useState('');\n const [docsURL, setDocsURL] = useState('');\n const [openAPISpec, setOpenAPISpec] = useState('');\n const [error, setError] = useState('');\n const [saving, setSaving] = useState(false);\n const [openAPISpecError, setOpenAPISpecError] = useState<string | null>(null);\n\n // Load APIProduct data when dialog opens\n useEffect(() => {\n if (open && namespace && name) {\n setLoading(true);\n setError('');\n\n fetchApi.fetch(`${backendUrl}/api/kuadrant/apiproducts/${namespace}/${name}`)\n .then(async res => {\n if (!res.ok) {\n const errorData = await res.json();\n throw new Error(errorData.error || `Failed to fetch API product: ${res.status}`);\n }\n return res.json();\n })\n .then(data => {\n setDisplayName(data.spec.displayName || '');\n setDescription(data.spec.description || '');\n setVersion(data.spec.version || 'v1');\n setPublishStatus(data.spec.publishStatus || 'Draft');\n setApprovalMode(data.spec.approvalMode || 'manual');\n setTags(data.spec.tags || []);\n setTargetRef(data.spec.targetRef || null);\n setContactEmail(data.spec.contact?.email || '');\n setContactTeam(data.spec.contact?.team || '');\n setDocsURL(data.spec.documentation?.docsURL || '');\n setOpenAPISpec(data.spec.documentation?.openAPISpecURL || '');\n setOpenAPISpecError(null);\n setLoading(false);\n })\n .catch(err => {\n setError(err.message || 'Failed to load API product');\n setLoading(false);\n });\n }\n }, [open, namespace, name, backendUrl, fetchApi]);\n\n // load planpolicies with full details to show associated plans\n const {\n value: planPolicies,\n error: planPoliciesError\n } = useAsync(async () => {\n if (!open) return null;\n const response = await fetchApi.fetch(`${backendUrl}/api/kuadrant/planpolicies`);\n return await response.json();\n }, [backendUrl, fetchApi, open]);\n\n // find planpolicy associated with targetRef\n const selectedPolicy = React.useMemo(() => {\n if (!planPolicies?.items || !targetRef) return null;\n\n return planPolicies.items.find((pp: any) => {\n const ref = pp.targetRef;\n return (\n ref?.kind === 'HTTPRoute' &&\n ref?.name === targetRef.name &&\n (!ref?.namespace || ref?.namespace === (targetRef.namespace || namespace))\n );\n });\n }, [planPolicies, targetRef, namespace]);\n\n useEffect(() => {\n if (open) {\n setOpenAPISpecError(null);\n }\n }, [open]);\n\n const handleOpenAPISpecChange = (value: string) => {\n setOpenAPISpec(value);\n setOpenAPISpecError(validateURL(value));\n };\n\n const handleAddTag = () => {\n if (tagInput.trim() && !tags.includes(tagInput.trim())) {\n setTags([...tags, tagInput.trim()]);\n setTagInput('');\n }\n };\n\n const handleDeleteTag = (tagToDelete: string) => {\n setTags(tags.filter(tag => tag !== tagToDelete));\n };\n\n const handleSave = async () => {\n setError('');\n setSaving(true);\n\n try {\n const patch = {\n spec: {\n displayName,\n description,\n version,\n publishStatus,\n approvalMode,\n tags,\n targetRef,\n ...(contactEmail || contactTeam ? {\n contact: {\n ...(contactEmail && { email: contactEmail }),\n ...(contactTeam && { team: contactTeam }),\n },\n } : {}),\n ...(docsURL || openAPISpec ? {\n documentation: {\n ...(docsURL && { docsURL }),\n ...(openAPISpec && { openAPISpecURL: openAPISpec }),\n },\n } : {}),\n },\n };\n\n const response = await fetchApi.fetch(\n `${backendUrl}/api/kuadrant/apiproducts/${namespace}/${name}`,\n {\n method: 'PATCH',\n headers: {\n 'Content-Type': 'application/json',\n },\n body: JSON.stringify(patch),\n }\n );\n\n if (!response.ok) {\n const errorData = await response.json();\n throw new Error(errorData.error || 'failed to update apiproduct');\n }\n\n onSuccess();\n onClose();\n } catch (err) {\n setError(err instanceof Error ? err.message : String(err));\n } finally {\n setSaving(false);\n }\n };\n\n return (\n <Dialog open={open} onClose={onClose} maxWidth=\"md\" fullWidth>\n <DialogTitle>Edit API Product</DialogTitle>\n <DialogContent>\n {error && (\n <Alert severity=\"error\" style={{ marginBottom: 16 }}>\n {error}\n </Alert>\n )}\n {planPoliciesError && (\n <Alert severity=\"warning\" style={{ marginBottom: 16 }}>\n <strong>Failed to load PlanPolicies:</strong> {planPoliciesError.message}\n <Typography variant=\"body2\" style={{ marginTop: 8 }}>\n Plan information may be incomplete.\n </Typography>\n </Alert>\n )}\n {loading ? (\n <Progress />\n ) : (\n <>\n {/* API product info section */}\n <Box className={classes.sectionHeader}>\n <Typography variant=\"subtitle1\"><strong>API product info</strong></Typography>\n </Box>\n <Grid container spacing={2}>\n <Grid item xs={6}>\n <TextField\n fullWidth\n label=\"API product name\"\n value={displayName}\n onChange={e => setDisplayName(e.target.value)}\n placeholder=\"My API\"\n helperText=\"Give a unique name for your API product\"\n margin=\"normal\"\n required\n disabled={saving}\n InputLabelProps={{\n classes: {\n asterisk: classes.asterisk,\n },\n }}\n />\n </Grid>\n <Grid item xs={6}>\n <TextField\n fullWidth\n label=\"Resource name\"\n value={name}\n disabled\n helperText=\"Kubernetes resource name (immutable)\"\n margin=\"normal\"\n />\n </Grid>\n <Grid item xs={6}>\n <TextField\n fullWidth\n label=\"Version\"\n value={version}\n onChange={e => setVersion(e.target.value)}\n placeholder=\"v1\"\n helperText=\"Give a version to your API product\"\n margin=\"normal\"\n disabled={saving}\n />\n </Grid>\n <Grid item xs={6}>\n <TextField\n fullWidth\n label=\"Tag\"\n value={tagInput}\n onChange={e => setTagInput(e.target.value)}\n onKeyPress={e => {\n if (e.key === 'Enter') {\n e.preventDefault();\n handleAddTag();\n }\n }}\n placeholder=\"Add tag\"\n helperText=\"Add a tag to your API product\"\n margin=\"normal\"\n disabled={saving}\n InputProps={{\n endAdornment: tagInput ? (\n <InputAdornment position=\"end\">\n <IconButton size=\"small\" onClick={handleAddTag} disabled={saving}>\n <AddIcon fontSize=\"small\" />\n </IconButton>\n </InputAdornment>\n ) : undefined,\n }}\n />\n </Grid>\n {tags.length > 0 && (\n <Grid item xs={12}>\n <Box display=\"flex\" flexWrap=\"wrap\">\n {tags.map(tag => (\n <Chip\n key={tag}\n label={tag}\n onDelete={saving ? undefined : () => handleDeleteTag(tag)}\n size=\"small\"\n className={classes.tagChip}\n disabled={saving}\n />\n ))}\n </Box>\n </Grid>\n )}\n <Grid item xs={12}>\n <TextField\n fullWidth\n label=\"Description\"\n value={description}\n onChange={e => setDescription(e.target.value)}\n placeholder=\"API description\"\n margin=\"normal\"\n multiline\n rows={2}\n required\n disabled={saving}\n InputLabelProps={{\n classes: {\n asterisk: classes.asterisk,\n },\n }}\n />\n </Grid>\n </Grid>\n\n {/* Associated route section */}\n <Box className={classes.sectionHeader}>\n <Typography variant=\"subtitle1\"><strong>Associated route</strong></Typography>\n <Tooltip title=\"The HTTPRoute this API product is associated with\">\n <InfoOutlinedIcon className={classes.infoIcon} />\n </Tooltip>\n </Box>\n <Grid container spacing={2}>\n <Grid item xs={12}>\n <TextField\n fullWidth\n label=\"OpenAPI Spec URL\"\n value={openAPISpec}\n onChange={e => handleOpenAPISpecChange(e.target.value)}\n placeholder=\"https://api.example.com/openapi.json\"\n helperText={openAPISpecError || \"Enter the full path to your API spec file\"}\n error={!!openAPISpecError}\n margin=\"normal\"\n disabled={saving}\n />\n </Grid>\n <Grid item xs={12}>\n <TextField\n fullWidth\n label=\"Documentation URL\"\n value={docsURL}\n onChange={e => setDocsURL(e.target.value)}\n placeholder=\"https://docs.example.com/api\"\n helperText=\"Link to external documentation for this API\"\n margin=\"normal\"\n disabled={saving}\n />\n </Grid>\n {targetRef && (\n <Grid item xs={12}>\n <TextField\n fullWidth\n label=\"HTTPRoute\"\n value={`${targetRef.namespace || namespace}/${targetRef.name}`}\n disabled\n helperText=\"Target HTTPRoute (immutable)\"\n margin=\"normal\"\n />\n </Grid>\n )}\n </Grid>\n\n {/* HTTPRoute policies section */}\n {targetRef && (\n <>\n <Box className={classes.sectionHeader}>\n <Typography variant=\"subtitle1\"><strong>HTTPRoute policies</strong></Typography>\n <Tooltip title=\"Shows the associated policies and rate limit tiers for the HTTPRoute\">\n <InfoOutlinedIcon className={classes.infoIcon} />\n </Tooltip>\n </Box>\n <PlanPolicyDetails\n selectedPolicy={selectedPolicy}\n alertSeverity=\"info\"\n alertMessage=\"No PlanPolicy found for this HTTPRoute.\"\n includeTopMargin={false}\n />\n </>\n )}\n\n {/* API Key approval section */}\n <Box className={classes.sectionHeader}>\n <Typography variant=\"subtitle1\"><strong>API Key approval</strong></Typography>\n <Tooltip title=\"Choose how API key requests are handled for this product\">\n <InfoOutlinedIcon className={classes.infoIcon} />\n </Tooltip>\n </Box>\n <FormControl component=\"fieldset\" disabled={saving}>\n <RadioGroup\n row\n value={approvalMode}\n onChange={e => setApprovalMode(e.target.value as 'automatic' | 'manual')}\n >\n <FormControlLabel\n value=\"manual\"\n control={<Radio color=\"primary\" />}\n label={\n <Box>\n <Typography variant=\"body2\">Need manual approval</Typography>\n <Typography variant=\"caption\" color=\"textSecondary\">\n Requires approval for requesting this API\n </Typography>\n </Box>\n }\n />\n <FormControlLabel\n value=\"automatic\"\n control={<Radio color=\"primary\" />}\n label={\n <Box>\n <Typography variant=\"body2\">Automatic</Typography>\n <Typography variant=\"caption\" color=\"textSecondary\">\n Keys are created without need to be approved\n </Typography>\n </Box>\n }\n />\n </RadioGroup>\n </FormControl>\n </>\n )}\n </DialogContent>\n <DialogActions>\n <Button onClick={onClose} disabled={saving}>Cancel</Button>\n <Button\n onClick={handleSave}\n color=\"primary\"\n variant=\"contained\"\n disabled={saving || loading || !displayName || !description || !!openAPISpecError}\n startIcon={saving ? <CircularProgress size={16} color=\"inherit\" /> : undefined}\n >\n {saving ? 'Saving...' : 'Save'}\n </Button>\n </DialogActions>\n </Dialog>\n );\n};\n","import { createPermission } from '@backstage/plugin-permission-common';\n\n/**\n * permission definitions for the kuadrant plugin\n *\n * these permissions control access to kuadrant resources and operations.\n * they must match the permissions defined in the backend plugin.\n *\n * permission types:\n * - BasicPermission: standard permission that applies globally\n * - ResourcePermission: permission scoped to specific resource types (e.g., apiproduct)\n *\n * permission patterns:\n * - `.create` - create new resources\n * - `.read` - read resource details\n * - `.read.own` - read only resources owned by the user\n * - `.read.all` - read all resources regardless of ownership\n * - `.update` - modify existing resources\n * - `.delete` - delete resources\n * - `.delete.own` - delete only resources owned by the user\n * - `.delete.all` - delete any resource regardless of ownership\n * - `.list` - list/view collections of resources\n */\n\n// planpolicy permissions\nexport const kuadrantPlanPolicyCreatePermission = createPermission({\n name: 'kuadrant.planpolicy.create',\n attributes: { action: 'create' },\n});\n\nexport const kuadrantPlanPolicyReadPermission = createPermission({\n name: 'kuadrant.planpolicy.read',\n attributes: { action: 'read' },\n});\n\nexport const kuadrantPlanPolicyUpdatePermission = createPermission({\n name: 'kuadrant.planpolicy.update',\n attributes: { action: 'update' },\n});\n\nexport const kuadrantPlanPolicyDeletePermission = createPermission({\n name: 'kuadrant.planpolicy.delete',\n attributes: { action: 'delete' },\n});\n\nexport const kuadrantPlanPolicyListPermission = createPermission({\n name: 'kuadrant.planpolicy.list',\n attributes: { action: 'read' },\n});\n\n// apiproduct permissions\n\n/**\n * permission to create new API products\n * granted to api owners and admins\n */\nexport const kuadrantApiProductCreatePermission = createPermission({\n name: 'kuadrant.apiproduct.create',\n attributes: { action: 'create' },\n});\n\n/**\n * permission to read API products owned by the current user\n * for api owners to view their own products\n */\nexport const kuadrantApiProductReadOwnPermission = createPermission({\n name: 'kuadrant.apiproduct.read.own',\n attributes: { action: 'read' },\n});\n\n/**\n * permission to read all API products regardless of ownership\n * for platform engineers/admins who need to view all products\n */\nexport const kuadrantApiProductReadAllPermission = createPermission({\n name: 'kuadrant.apiproduct.read.all',\n attributes: { action: 'read' },\n});\n\n/**\n * permission to update API products owned by the current user\n * for api owners to modify their own products\n */\nexport const kuadrantApiProductUpdateOwnPermission = createPermission({\n name: 'kuadrant.apiproduct.update.own',\n attributes: { action: 'update' },\n});\n\n/**\n * permission to update any API product regardless of ownership\n * for platform engineers/admins\n */\nexport const kuadrantApiProductUpdateAllPermission = createPermission({\n name: 'kuadrant.apiproduct.update.all',\n attributes: { action: 'update' },\n});\n\n/**\n * permission to delete API products owned by the current user\n * for api owners to remove their own products\n */\nexport const kuadrantApiProductDeleteOwnPermission = createPermission({\n name: 'kuadrant.apiproduct.delete.own',\n attributes: { action: 'delete' },\n});\n\n/**\n * permission to delete any API product regardless of ownership\n * for platform engineers/admins\n */\nexport const kuadrantApiProductDeleteAllPermission = createPermission({\n name: 'kuadrant.apiproduct.delete.all',\n attributes: { action: 'delete' },\n});\n\n/**\n * permission to list API products\n * backend filters results based on .own vs .all read permissions\n */\nexport const kuadrantApiProductListPermission = createPermission({\n name: 'kuadrant.apiproduct.list',\n attributes: { action: 'read' },\n});\n\n// apikey permissions\n\n/**\n * permission to create API keys (request API access)\n *\n * this is a ResourcePermission scoped to 'apiproduct', allowing\n * fine-grained control over which API products users can request access to.\n *\n * use in frontend: useKuadrantPermission(kuadrantApiKeyCreatePermission)\n * use in backend with resource: { permission, resourceRef: 'apiproduct:namespace/name' }\n */\nexport const kuadrantApiKeyCreatePermission = createPermission({\n name: 'kuadrant.apikey.create',\n attributes: { action: 'create' },\n resourceType: 'apiproduct',\n});\n\n/**\n * permission to read API keys owned by the current user\n * allows users to view their own API keys and request history\n */\nexport const kuadrantApiKeyReadOwnPermission = createPermission({\n name: 'kuadrant.apikey.read.own',\n attributes: { action: 'read' },\n});\n\n/**\n * permission to read all API keys regardless of ownership\n * for platform engineers/admins who need to view the approval queue and audit keys\n */\nexport const kuadrantApiKeyReadAllPermission = createPermission({\n name: 'kuadrant.apikey.read.all',\n attributes: { action: 'read' },\n});\n\n/**\n * permission to update API keys owned by the current user\n * allows users to edit their own pending requests (change plan tier, use case)\n */\nexport const kuadrantApiKeyUpdateOwnPermission = createPermission({\n name: 'kuadrant.apikey.update.own',\n attributes: { action: 'update' },\n});\n\n/**\n * permission to update any API key regardless of ownership\n * typically granted to API owners and platform engineers for approving/rejecting requests\n */\nexport const kuadrantApiKeyUpdateAllPermission = createPermission({\n name: 'kuadrant.apikey.update.all',\n attributes: { action: 'update' },\n});\n\n/**\n * permission to delete API keys owned by the current user\n * allows users to cancel their own requests or revoke their own access\n */\nexport const kuadrantApiKeyDeleteOwnPermission = createPermission({\n name: 'kuadrant.apikey.delete.own',\n attributes: { action: 'delete' },\n});\n\n/**\n * permission to delete any API key regardless of ownership\n * for platform engineers/admins who need to revoke access\n */\nexport const kuadrantApiKeyDeleteAllPermission = createPermission({\n name: 'kuadrant.apikey.delete.all',\n attributes: { action: 'delete' },\n});\n\n/**\n * permission to approve/reject API key requests\n * grants access to the approval queue - for API owners and admins only\n * separate from update.own which consumers use to edit their pending requests\n */\nexport const kuadrantApiKeyApprovePermission = createPermission({\n name: 'kuadrant.apikey.approve',\n attributes: { action: 'update' },\n});\n\nexport const kuadrantPermissions = [\n kuadrantPlanPolicyCreatePermission,\n kuadrantPlanPolicyReadPermission,\n kuadrantPlanPolicyUpdatePermission,\n kuadrantPlanPolicyDeletePermission,\n kuadrantPlanPolicyListPermission,\n kuadrantApiProductCreatePermission,\n kuadrantApiProductReadOwnPermission,\n kuadrantApiProductReadAllPermission,\n kuadrantApiProductUpdateOwnPermission,\n kuadrantApiProductUpdateAllPermission,\n kuadrantApiProductDeleteOwnPermission,\n kuadrantApiProductDeleteAllPermission,\n kuadrantApiProductListPermission,\n kuadrantApiKeyCreatePermission,\n kuadrantApiKeyReadOwnPermission,\n kuadrantApiKeyReadAllPermission,\n kuadrantApiKeyUpdateOwnPermission,\n kuadrantApiKeyUpdateAllPermission,\n kuadrantApiKeyDeleteOwnPermission,\n kuadrantApiKeyDeleteAllPermission,\n kuadrantApiKeyApprovePermission,\n];\n","import { usePermission } from '@backstage/plugin-permission-react';\nimport { Permission, ResourcePermission } from '@backstage/plugin-permission-common';\n\n/**\n * result of a permission check including error state\n */\nexport interface PermissionCheckResult {\n allowed: boolean;\n loading: boolean;\n error?: Error;\n}\n\n/**\n * custom hook for checking kuadrant permissions that handles both\n * BasicPermission and ResourcePermission types without type bypasses\n *\n * @param permission - the permission to check\n * @param resourceRef - optional resource reference for ResourcePermissions\n * @returns permission check result with error handling\n *\n * @example\n * // basic permission\n * const { allowed, loading, error } = useKuadrantPermission(\n * kuadrantApiProductListPermission\n * );\n *\n * @example\n * // resource permission\n * const { allowed, loading, error } = useKuadrantPermission(\n * kuadrantApiKeyCreatePermission,\n * 'apiproduct:namespace/name'\n * );\n */\nexport function useKuadrantPermission(\n permission: Permission,\n resourceRef?: string,\n): PermissionCheckResult {\n // construct the permission request based on whether it's a ResourcePermission\n const permissionRequest = 'resourceType' in permission\n ? { permission: permission as ResourcePermission, resourceRef }\n : { permission };\n\n const result = usePermission(permissionRequest as any);\n\n return {\n allowed: result.allowed,\n loading: result.loading,\n error: result.error,\n };\n}\n\n/**\n * helper to determine if a user can delete a specific API key or request\n *\n * @param ownerId - the user id who owns the key/request\n * @param currentUserId - the current user's id\n * @param canDeleteOwn - whether user has permission to delete their own keys\n * @param canDeleteAll - whether user has permission to delete all keys\n * @returns true if user can delete this specific key/request\n */\nexport function canDeleteResource(\n ownerId: string,\n currentUserId: string,\n canDeleteOwn: boolean,\n canDeleteAll: boolean,\n): boolean {\n if (canDeleteAll) return true;\n if (canDeleteOwn && ownerId === currentUserId) return true;\n return false;\n}\n","import React, { useState, useEffect } from 'react';\nimport {\n Dialog,\n DialogTitle,\n DialogContent,\n DialogContentText,\n DialogActions,\n Button,\n TextField,\n Typography,\n Box,\n CircularProgress,\n} from '@material-ui/core';\nimport WarningIcon from '@material-ui/icons/Warning';\n\nexport interface ConfirmDeleteDialogProps {\n open: boolean;\n title: string;\n description: string;\n // for dangerous deletes, require typing this text to confirm\n confirmText?: string;\n // severity affects styling - 'high' shows warning icon and requires text confirmation\n severity?: 'normal' | 'high';\n deleting?: boolean;\n onConfirm: () => void;\n onCancel: () => void;\n}\n\nexport const ConfirmDeleteDialog = ({\n open,\n title,\n description,\n confirmText,\n severity = 'normal',\n deleting = false,\n onConfirm,\n onCancel,\n}: ConfirmDeleteDialogProps) => {\n const [inputValue, setInputValue] = useState('');\n\n // reset input when dialog opens/closes\n useEffect(() => {\n if (!open) {\n setInputValue('');\n }\n }, [open]);\n\n const requiresTextConfirmation = severity === 'high' && confirmText;\n const canConfirm = requiresTextConfirmation ? inputValue === confirmText : true;\n\n const handleConfirm = () => {\n if (canConfirm) {\n onConfirm();\n }\n };\n\n return (\n <Dialog\n open={open}\n onClose={deleting ? undefined : onCancel}\n maxWidth=\"sm\"\n fullWidth\n >\n <DialogTitle>\n {severity === 'high' && (\n <Box display=\"flex\" alignItems=\"center\" style={{ gap: 8 }}>\n <WarningIcon color=\"error\" />\n <span>{title}</span>\n </Box>\n )}\n {severity !== 'high' && title}\n </DialogTitle>\n <DialogContent>\n <DialogContentText style={{ whiteSpace: 'pre-line' }}>\n {description}\n </DialogContentText>\n {requiresTextConfirmation && (\n <Box mt={2}>\n <Typography variant=\"body2\" color=\"textSecondary\" gutterBottom>\n Type <strong>{confirmText}</strong> to confirm:\n </Typography>\n <TextField\n fullWidth\n variant=\"outlined\"\n size=\"small\"\n value={inputValue}\n onChange={e => setInputValue(e.target.value)}\n disabled={deleting}\n autoFocus\n placeholder={confirmText}\n />\n </Box>\n )}\n </DialogContent>\n <DialogActions>\n <Button onClick={onCancel} disabled={deleting}>\n Cancel\n </Button>\n <Button\n onClick={handleConfirm}\n color=\"secondary\"\n variant=\"contained\"\n disabled={deleting || !canConfirm}\n startIcon={deleting ? <CircularProgress size={16} color=\"inherit\" /> : undefined}\n >\n {deleting ? 'Deleting...' : 'Delete'}\n </Button>\n </DialogActions>\n </Dialog>\n );\n};\n","import React from 'react';\nimport { Box, Typography, Chip, useTheme } from '@material-ui/core';\nimport { Alert } from '@material-ui/lab';\n\ninterface PlanPolicyDetailsProps {\n selectedPolicy: {\n metadata: {\n name: string;\n };\n plans?: Array<{\n tier: string;\n description?: string;\n limits?: {\n daily?: number;\n monthly?: number;\n yearly?: number;\n };\n }>;\n } | null;\n alertSeverity?: 'warning' | 'info';\n alertMessage?: string;\n includeTopMargin?: boolean;\n}\n\nexport const PlanPolicyDetails: React.FC<PlanPolicyDetailsProps> = ({\n selectedPolicy,\n alertSeverity = 'warning',\n alertMessage = 'No PlanPolicy found for this HTTPRoute. API keys and rate limiting may not be available.',\n includeTopMargin = true,\n}) => {\n const theme = useTheme();\n return (\n <Box\n mt={includeTopMargin ? 1 : 0}\n p={2}\n bgcolor={theme.palette.background.default}\n borderRadius={1}\n border={`1px solid ${theme.palette.divider}`}\n >\n {selectedPolicy ? (\n <>\n <Typography variant=\"subtitle2\" gutterBottom style={{ fontWeight: 600 }}>\n Associated PlanPolicy: <strong>{selectedPolicy.metadata.name}</strong>\n </Typography>\n\n {selectedPolicy.plans && selectedPolicy.plans.length > 0 ? (\n <>\n <Typography\n variant=\"caption\"\n display=\"block\"\n gutterBottom\n color=\"textSecondary\"\n style={{ marginTop: 8 }}\n >\n Available Tiers:\n </Typography>\n <Box display=\"flex\" flexWrap=\"wrap\" mt={1} style={{ gap: 8 }}>\n {selectedPolicy.plans.map((plan: any, idx: number) => {\n const limitText = plan.limits?.daily\n ? `${plan.limits.daily}/day`\n : plan.limits?.monthly\n ? `${plan.limits.monthly}/month`\n : plan.limits?.yearly\n ? `${plan.limits.yearly}/year`\n : 'No limit';\n\n return (\n <Chip\n key={idx}\n label={`${plan.tier}: ${limitText}`}\n size=\"small\"\n variant=\"outlined\"\n color=\"primary\"\n />\n );\n })}\n </Box>\n </>\n ) : (\n <Typography variant=\"caption\" color=\"textSecondary\">\n No plans defined in this PlanPolicy\n </Typography>\n )}\n </>\n ) : (\n <Alert severity={alertSeverity}>{alertMessage}</Alert>\n )}\n </Box>\n );\n};\n"],"names":["validateKubernetesName","value","trim","length","test","validateURL","url","URL","includes","protocol","useStyles","makeStyles","theme","asterisk","color","sectionHeader","display","alignItems","gap","spacing","marginTop","marginBottom","infoIcon","fontSize","palette","text","secondary","tagChip","marginRight","EditAPIProductDialog","open","onClose","onSuccess","namespace","name","classes","config","useApi","configApiRef","fetchApi","fetchApiRef","backendUrl","getString","loading","setLoading","useState","displayName","setDisplayName","description","setDescription","version","setVersion","publishStatus","setPublishStatus","approvalMode","setApprovalMode","tags","setTags","targetRef","setTargetRef","tagInput","setTagInput","contactEmail","setContactEmail","contactTeam","setContactTeam","docsURL","setDocsURL","openAPISpec","setOpenAPISpec","error","setError","saving","setSaving","openAPISpecError","setOpenAPISpecError","useEffect","fetch","then","async","res","ok","errorData","json","Error","status","data","spec","contact","email","team","documentation","openAPISpecURL","catch","err","message","planPolicies","planPoliciesError","useAsync","response","selectedPolicy","React","items","find","pp","ref","kind","handleAddTag","Dialog","maxWidth","fullWidth","DialogTitle","DialogContent","Alert","severity","style","strong","Typography","variant","Progress","Box","className","Grid","container","item","xs","TextField","label","onChange","e","target","placeholder","helperText","margin","required","disabled","InputLabelProps","onKeyPress","key","preventDefault","InputProps","endAdornment","InputAdornment","position","IconButton","size","onClick","AddIcon","undefined","flexWrap","map","tag","Chip","onDelete","handleDeleteTag","tagToDelete","filter","multiline","rows","Tooltip","title","InfoOutlinedIcon","handleOpenAPISpecChange","PlanPolicyDetails","alertSeverity","alertMessage","includeTopMargin","FormControl","component","RadioGroup","row","FormControlLabel","control","Radio","DialogActions","Button","patch","method","headers","body","JSON","stringify","String","startIcon","CircularProgress","createPermission","attributes","action","kuadrantPlanPolicyListPermission","kuadrantApiProductCreatePermission","kuadrantApiProductReadAllPermission","kuadrantApiProductUpdateOwnPermission","kuadrantApiProductUpdateAllPermission","kuadrantApiProductDeleteOwnPermission","kuadrantApiProductDeleteAllPermission","kuadrantApiProductListPermission","kuadrantApiKeyCreatePermission","resourceType","kuadrantApiKeyReadOwnPermission","kuadrantApiKeyUpdateOwnPermission","kuadrantApiKeyUpdateAllPermission","kuadrantApiKeyDeleteOwnPermission","kuadrantApiKeyDeleteAllPermission","kuadrantApiKeyApprovePermission","useKuadrantPermission","permission","resourceRef","permissionRequest","result","usePermission","allowed","canDeleteResource","ownerId","currentUserId","canDeleteOwn","canDeleteAll","ConfirmDeleteDialog","confirmText","deleting","onConfirm","onCancel","inputValue","setInputValue","requiresTextConfirmation","canConfirm","WarningIcon","span","DialogContentText","whiteSpace","mt","gutterBottom","autoFocus","useTheme","p","bgcolor","background","default","borderRadius","border","divider","fontWeight","metadata","plans","plan","idx","limitText","limits","daily","monthly","yearly","tier"],"sourceRoot":""}