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

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (73) hide show
  1. package/dist/api.esm.js +16 -0
  2. package/dist/api.esm.js.map +1 -1
  3. package/dist/components/ApiAccessCard/ApiAccessCard.esm.js.map +1 -1
  4. package/dist/components/ApiKeyManagementTab/ApiKeyManagementTab.esm.js.map +1 -1
  5. package/dist/components/ApiProductDetails/ApiProductDetails.esm.js.map +1 -1
  6. package/dist/components/ApiProductPolicies/ApiProductPolicies.esm.js +113 -0
  7. package/dist/components/ApiProductPolicies/ApiProductPolicies.esm.js.map +1 -0
  8. package/dist/components/CreateAPIProductDialog/CreateAPIProductDialog.esm.js +57 -13
  9. package/dist/components/CreateAPIProductDialog/CreateAPIProductDialog.esm.js.map +1 -1
  10. package/dist/components/EditAPIProductDialog/EditAPIProductDialog.esm.js +49 -8
  11. package/dist/components/EditAPIProductDialog/EditAPIProductDialog.esm.js.map +1 -1
  12. package/dist/components/KuadrantPage/ApiProductsPage.esm.js +1 -1
  13. package/dist/components/KuadrantPage/ApiProductsPage.esm.js.map +1 -1
  14. package/dist/components/RequestAccessDialog/RequestAccessDialog.esm.js.map +1 -1
  15. package/dist/index.d.ts +59 -8
  16. package/dist/index.esm.js +1 -1
  17. package/dist/permissions.esm.js +12 -2
  18. package/dist/permissions.esm.js.map +1 -1
  19. package/dist/utils/policies.esm.js +11 -0
  20. package/dist/utils/policies.esm.js.map +1 -0
  21. package/dist-scalprum/{internal.plugin-kuadrant.8bf02d5482a5a3ed6337.js → internal.plugin-kuadrant.1ed3b3b414caa08ac5be.js} +2 -2
  22. package/dist-scalprum/{internal.plugin-kuadrant.8bf02d5482a5a3ed6337.js.map → internal.plugin-kuadrant.1ed3b3b414caa08ac5be.js.map} +1 -1
  23. package/dist-scalprum/plugin-manifest.json +3 -3
  24. package/dist-scalprum/static/{2967.28e7c7f8.chunk.js → 2967.17aec2fb.chunk.js} +1 -1
  25. package/dist-scalprum/static/2967.17aec2fb.chunk.js.map +1 -0
  26. package/dist-scalprum/static/{3947.ff1c25cf.chunk.js → 3947.ad129ba4.chunk.js} +1 -1
  27. package/dist-scalprum/static/{3947.ff1c25cf.chunk.js.map → 3947.ad129ba4.chunk.js.map} +1 -1
  28. package/dist-scalprum/static/{3976.cf3ec7be.chunk.js → 3976.c8138b52.chunk.js} +1 -1
  29. package/dist-scalprum/static/{3976.cf3ec7be.chunk.js.map → 3976.c8138b52.chunk.js.map} +1 -1
  30. package/dist-scalprum/static/4447.d924bc59.chunk.js +2 -0
  31. package/dist-scalprum/static/4447.d924bc59.chunk.js.map +1 -0
  32. package/dist-scalprum/static/{5203.e92a3353.chunk.js → 5203.11ef2211.chunk.js} +1 -1
  33. package/dist-scalprum/static/5203.11ef2211.chunk.js.map +1 -0
  34. package/dist-scalprum/static/{6371.6ee1f11d.chunk.js → 6371.d849c508.chunk.js} +1 -1
  35. package/dist-scalprum/static/6371.d849c508.chunk.js.map +1 -0
  36. package/dist-scalprum/static/6387.79be6155.chunk.js +2 -0
  37. package/dist-scalprum/static/6387.79be6155.chunk.js.map +1 -0
  38. package/dist-scalprum/static/{6800.a7645f4c.chunk.js → 6800.6faeb7c6.chunk.js} +1 -1
  39. package/dist-scalprum/static/{6800.a7645f4c.chunk.js.map → 6800.6faeb7c6.chunk.js.map} +1 -1
  40. package/dist-scalprum/static/69.b6afd1fe.chunk.js +2 -0
  41. package/dist-scalprum/static/69.b6afd1fe.chunk.js.map +1 -0
  42. package/dist-scalprum/static/7005.98c5e400.chunk.js +2 -0
  43. package/dist-scalprum/static/7005.98c5e400.chunk.js.map +1 -0
  44. package/dist-scalprum/static/{7270.b0e185f1.chunk.js → 7270.4a71807b.chunk.js} +1 -1
  45. package/dist-scalprum/static/{7270.b0e185f1.chunk.js.map → 7270.4a71807b.chunk.js.map} +1 -1
  46. package/dist-scalprum/static/{7791.5c1eea8a.chunk.js → 7791.0a1d55bc.chunk.js} +1 -1
  47. package/dist-scalprum/static/{7791.5c1eea8a.chunk.js.map → 7791.0a1d55bc.chunk.js.map} +1 -1
  48. package/dist-scalprum/static/8789.2f437443.chunk.js +2 -0
  49. package/dist-scalprum/static/8789.2f437443.chunk.js.map +1 -0
  50. package/dist-scalprum/static/{9051.db875198.chunk.js → 9051.d45ac154.chunk.js} +2 -2
  51. package/dist-scalprum/static/9051.d45ac154.chunk.js.map +1 -0
  52. package/dist-scalprum/static/{9838.5df9b1de.chunk.js → 9838.966ce0a0.chunk.js} +1 -1
  53. package/dist-scalprum/static/9838.966ce0a0.chunk.js.map +1 -0
  54. package/dist-scalprum/static/{exposed-PluginRoot.8a8dd91f.chunk.js → exposed-PluginRoot.3b24e5af.chunk.js} +2 -2
  55. package/dist-scalprum/static/{exposed-PluginRoot.8a8dd91f.chunk.js.map → exposed-PluginRoot.3b24e5af.chunk.js.map} +1 -1
  56. package/package.json +1 -1
  57. package/dist/components/PlanPolicyDetailsCard/PlanPolicyDetails.esm.js +0 -48
  58. package/dist/components/PlanPolicyDetailsCard/PlanPolicyDetails.esm.js.map +0 -1
  59. package/dist-scalprum/static/2967.28e7c7f8.chunk.js.map +0 -1
  60. package/dist-scalprum/static/4447.222bb58d.chunk.js +0 -2
  61. package/dist-scalprum/static/4447.222bb58d.chunk.js.map +0 -1
  62. package/dist-scalprum/static/5203.e92a3353.chunk.js.map +0 -1
  63. package/dist-scalprum/static/6371.6ee1f11d.chunk.js.map +0 -1
  64. package/dist-scalprum/static/69.1decc5e8.chunk.js +0 -2
  65. package/dist-scalprum/static/69.1decc5e8.chunk.js.map +0 -1
  66. package/dist-scalprum/static/7005.a9f73153.chunk.js +0 -2
  67. package/dist-scalprum/static/7005.a9f73153.chunk.js.map +0 -1
  68. package/dist-scalprum/static/8789.84963cbc.chunk.js +0 -2
  69. package/dist-scalprum/static/8789.84963cbc.chunk.js.map +0 -1
  70. package/dist-scalprum/static/8804.63919453.chunk.js +0 -2
  71. package/dist-scalprum/static/8804.63919453.chunk.js.map +0 -1
  72. package/dist-scalprum/static/9051.db875198.chunk.js.map +0 -1
  73. package/dist-scalprum/static/9838.5df9b1de.chunk.js.map +0 -1
@@ -1,2 +1,2 @@
1
1
  "use strict";(self.webpackChunkinternal_plugin_kuadrant=self.webpackChunkinternal_plugin_kuadrant||[]).push([[7791],{57791:(e,s,i)=>{i.r(s),i.d(s,{ApiProductDetailPage:()=>K});var t=i(31085),l=i(95478),n=i(49634),a=i(22097),o=i(64047),r=i(35015),d=i(86687),c=i(42367),u=i(289),p=i(15831),h=i(37725),m=i(46681),v=i(94641),A=i(96040),x=i(41241),y=i(58837),j=i(10394),f=i(64947),P=i(71677),b=i(29365),g=i(72501),w=i(31653),k=i(38605),I=i(42899),S=i(67720),N=i(61009),C=i(47625),z=i(9719),T=i(54801),D=i(13677),R=i(55735),_=i(75625),B=i(39590),U=i(84441),O=i(26997),$=i(63221),E=i(89509),F=i(71255),L=i(46205),H=i(34955);const Y=(0,y.A)(e=>({label:{fontWeight:600,color:e.palette.text.secondary,marginBottom:e.spacing(.5),fontSize:"0.75rem",textTransform:"uppercase"},actionButtons:{display:"flex",gap:e.spacing(1),alignItems:"center"},cardHeader:{display:"flex",justifyContent:"space-between",alignItems:"flex-start",marginBottom:e.spacing(2)},cardActions:{display:"flex",gap:e.spacing(1),alignItems:"center"}})),K=()=>{var e,s,i,y,K,V,W,q,G,J,M,Q,X,Z,ee,se,ie,te,le,ne,ae,oe,re,de,ce,ue;const pe=Y(),{namespace:he,name:me}=(0,n.useParams)(),ve=(0,n.useNavigate)(),Ae=(0,a.useApi)(o.s),xe=(0,a.useApi)(a.alertApiRef),[ye,je]=(0,l.useState)(0),[fe,Pe]=(0,l.useState)(!1),[be,ge]=(0,l.useState)(!1),[we,ke]=(0,l.useState)(!1),[Ie,Se]=(0,l.useState)(0),{allowed:Ne}=(0,L.l)(H.v_),{allowed:Ce}=(0,L.l)(H.R_),ze=Ne,{value:Te,loading:De,error:Re}=(0,r.A)(async()=>await Ae.getApiProduct(he,me),[he,me,Ae,Ie]),{value:_e}=(0,r.A)(async()=>{var e,s,i;if(!(null==Te||null===(s=Te.spec)||void 0===s||null===(e=s.targetRef)||void 0===e?void 0:e.name))return null;const t=Te.spec.targetRef.name,l=await Ae.getHttpRoute(he,t);return(null==l||null===(i=l.spec)||void 0===i?void 0:i.hostnames)||null},[Te,he,Ae]),Be=async()=>{var e,s;if(!Te)return;const i="Published"===(null===(e=Te.spec)||void 0===e?void 0:e.publishStatus)?"Draft":"Published",t=null===(s=Te.metadata.labels)||void 0===s?void 0:s.lifecycle;if("Published"!==i||"retired"!==t)try{await Ae.updateApiProduct(he,me,{spec:{publishStatus:i}}),xe.post({message:`API Product ${"Published"===i?"published":"unpublished"} successfully`,severity:"success",display:"transient"}),Se(e=>e+1)}catch(e){const s=e instanceof Error?e.message:"unknown error occurred";xe.post({message:`Failed to update publish status. ${s}`,severity:"error",display:"transient"})}else xe.post({message:"Cannot publish a retired API product. Please change the lifecycle status first.",severity:"error",display:"transient"})};if(De)return(0,t.jsx)(d.k,{});if(Re||!Te)return(0,t.jsx)(c._,{error:Re||new Error("API product not found")});const Ue="Published"===(null===(e=Te.spec)||void 0===e?void 0:e.publishStatus),Oe=null===(i=Te.status)||void 0===i||null===(s=i.conditions)||void 0===s?void 0:s.find(e=>"PlanPolicyDiscovered"===e.type),$e=null===(K=Te.status)||void 0===K||null===(y=K.conditions)||void 0===y?void 0:y.find(e=>"AuthPolicyDiscovered"===e.type),Ee=(null===(V=Te.status)||void 0===V?void 0:V.discoveredPlans)||[],Fe=(null===(q=Te.status)||void 0===q||null===(W=q.discoveredAuthScheme)||void 0===W?void 0:W.authentication)||{},Le=Object.values(Fe),He=Le.some(e=>e.hasOwnProperty("jwt")),Ye=Le.find(e=>e.hasOwnProperty("jwt")),Ke=(null==Ye||null===(G=Ye.jwt)||void 0===G?void 0:G.issuerUrl)||"unknown",Ve=(null===(M=Te.status)||void 0===M||null===(J=M.oidcDiscovery)||void 0===J?void 0:J.tokenEndpoint)||"unknown",We=!(!(null===(X=Te.status)||void 0===X||null===(Q=X.openapi)||void 0===Q?void 0:Q.raw)&&!(null===(ee=Te.spec)||void 0===ee||null===(Z=ee.documentation)||void 0===Z?void 0:Z.openAPISpecURL)),qe=!!(Oe||$e||Ee.length>0);let Ge=1;const Je=We?Ge++:-1,Me=qe?Ge++:-1,Qe=He?Ge++:-1,Xe=e=>{if(!e)return"No limits";const s=[];return e.daily&&s.push(`${e.daily}/day`),e.weekly&&s.push(`${e.weekly}/week`),e.monthly&&s.push(`${e.monthly}/month`),e.yearly&&s.push(`${e.yearly}/year`),s.length>0?s.join(", "):"No limits"};return(0,t.jsxs)(u.Y,{themeId:"tool",children:[(0,t.jsx)(p.Y,{title:(null===(se=Te.spec)||void 0===se?void 0:se.displayName)||Te.metadata.name,subtitle:(null===(ie=Te.spec)||void 0===ie?void 0:ie.description)||"",children:(0,t.jsxs)(j.A,{className:pe.actionButtons,children:[(0,t.jsx)(h.N_,{to:"/kuadrant/api-products",children:(0,t.jsx)(f.A,{startIcon:(0,t.jsx)(R.A,{}),children:"Back"})}),ze&&(0,t.jsx)(f.A,{variant:"outlined",color:Ue?"default":"primary",onClick:Be,children:Ue?"Unpublish API product":"Publish API product"}),Ne&&(0,t.jsx)(P.Ay,{title:"Edit",children:(0,t.jsx)(b.A,{onClick:()=>Pe(!0),size:"small",children:(0,t.jsx)(_.A,{})})}),Ce&&(0,t.jsx)(P.Ay,{title:"Delete",children:(0,t.jsx)(b.A,{onClick:()=>ge(!0),size:"small",children:(0,t.jsx)(B.A,{})})})]})}),(0,t.jsxs)(m.U,{children:[(0,t.jsx)(j.A,{mb:2,children:(0,t.jsxs)(v.B,{"aria-label":"breadcrumb",children:[(0,t.jsx)(h.N_,{to:"/kuadrant/api-products",children:"API Products"}),(0,t.jsx)(g.A,{children:(null===(te=Te.spec)||void 0===te?void 0:te.displayName)||Te.metadata.name})]})}),"deprecated"===(null===(le=Te.metadata.labels)||void 0===le?void 0:le.lifecycle)&&(0,t.jsx)(j.A,{mb:2,children:(0,t.jsxs)(U.A,{severity:"warning",children:[(0,t.jsx)("strong",{children:"This API is deprecated."})," Please contact your administrator for more details."]})}),"retired"===(null===(ne=Te.metadata.labels)||void 0===ne?void 0:ne.lifecycle)&&(0,t.jsx)(j.A,{mb:2,children:(0,t.jsxs)(U.A,{severity:"error",children:[(0,t.jsx)("strong",{children:"This API has been retired and is no longer available."})," Please contact your administrator for alternatives."]})}),(0,t.jsx)(j.A,{mb:2,children:(0,t.jsxs)(w.A,{value:ye,onChange:(e,s)=>je(s),indicatorColor:"primary",textColor:"primary",children:[(0,t.jsx)(k.A,{label:"Overview"}),We&&(0,t.jsx)(k.A,{label:"Definition"}),qe&&(0,t.jsx)(k.A,{label:"Policies"}),He&&(0,t.jsx)(k.A,{label:"OIDC"})]})}),0===ye&&(0,t.jsxs)(A.n,{title:"API Product",children:[(0,t.jsxs)(j.A,{className:pe.cardHeader,children:[(0,t.jsxs)(j.A,{children:[(0,t.jsx)(g.A,{variant:"caption",className:pe.label,children:"Product Name"}),(0,t.jsx)(g.A,{variant:"h6",children:(null===(ae=Te.spec)||void 0===ae?void 0:ae.displayName)||Te.metadata.name})]}),(0,t.jsxs)(j.A,{className:pe.cardActions,children:[ze&&(0,t.jsx)(f.A,{variant:"outlined",color:Ue?"default":"primary",onClick:Be,size:"small",children:Ue?"Unpublish API product":"Publish API product"}),Ne&&(0,t.jsx)(b.A,{onClick:()=>Pe(!0),size:"small",children:(0,t.jsx)(_.A,{fontSize:"small"})}),Ce&&(0,t.jsx)(b.A,{onClick:()=>ge(!0),size:"small",children:(0,t.jsx)(B.A,{fontSize:"small"})})]})]}),(0,t.jsx)(E.O,{product:Te,showStatus:!0,showCatalogLink:!0,httpRouteHostnames:_e})]}),ye===Je&&We&&(0,t.jsx)(A.n,{title:"API Definition",children:(null===(re=Te.status)||void 0===re||null===(oe=re.openapi)||void 0===oe?void 0:oe.raw)?(0,t.jsx)(x.S,{definition:Te.status.openapi.raw}):(0,t.jsx)(g.A,{variant:"body2",color:"textSecondary",children:(null===(ce=Te.spec)||void 0===ce||null===(de=ce.documentation)||void 0===de?void 0:de.openAPISpecURL)?(0,t.jsxs)(t.Fragment,{children:["OpenAPI specification not yet synced. View at:"," ",(0,t.jsx)(h.N_,{to:Te.spec.documentation.openAPISpecURL,target:"_blank",children:Te.spec.documentation.openAPISpecURL})]}):"No OpenAPI specification available for this API product."})}),ye===Me&&qe&&(0,t.jsxs)(I.A,{container:!0,spacing:3,children:[(0,t.jsx)(I.A,{item:!0,xs:12,children:(0,t.jsx)(A.n,{title:"Discovered Policies",children:(0,t.jsxs)(I.A,{container:!0,spacing:2,children:[(0,t.jsxs)(I.A,{item:!0,xs:12,md:6,children:[(0,t.jsx)(g.A,{variant:"body2",color:"textSecondary",gutterBottom:!0,children:"Plan Policy"}),Oe?(0,t.jsxs)(j.A,{children:[(0,t.jsx)(S.A,{label:"True"===Oe.status?"Found":"Not Found",size:"small",style:{backgroundColor:"True"===Oe.status?"#4caf50":"#ff9800",color:"#fff",marginBottom:8}}),(0,t.jsx)(g.A,{variant:"body2",children:Oe.message||"No details available"})]}):(0,t.jsx)(g.A,{variant:"body2",children:"No plan policy information"})]}),(0,t.jsxs)(I.A,{item:!0,xs:12,md:6,children:[(0,t.jsx)(g.A,{variant:"body2",color:"textSecondary",gutterBottom:!0,children:"Auth Policy"}),$e?(0,t.jsxs)(j.A,{children:[(0,t.jsx)(S.A,{label:"True"===$e.status?"Found":"Not Found",size:"small",style:{backgroundColor:"True"===$e.status?"#4caf50":"#ff9800",color:"#fff",marginBottom:8}}),(0,t.jsx)(g.A,{variant:"body2",children:$e.message||"No details available"})]}):(0,t.jsx)(g.A,{variant:"body2",children:"No auth policy information"})]})]})})}),Ee.length>0&&(0,t.jsx)(I.A,{item:!0,xs:12,children:(0,t.jsxs)(A.n,{title:"Effective Plan Tiers",children:[(0,t.jsx)(g.A,{variant:"body2",color:"textSecondary",paragraph:!0,children:"These tiers are computed from all attached PlanPolicies (including gateway-level policies)."}),(0,t.jsxs)(N.A,{size:"small",children:[(0,t.jsx)(C.A,{children:(0,t.jsxs)(z.A,{children:[(0,t.jsx)(T.A,{children:"Tier"}),(0,t.jsx)(T.A,{children:"Rate Limits"})]})}),(0,t.jsx)(D.A,{children:Ee.map(e=>(0,t.jsxs)(z.A,{children:[(0,t.jsx)(T.A,{children:(0,t.jsx)(S.A,{label:e.tier,size:"small",color:"primary"})}),(0,t.jsx)(T.A,{children:Xe(e.limits)})]},e.tier))})]})]})})]}),ye===Qe&&He&&(0,t.jsx)(F.n,{issuerUrl:Ke,tokenEndpoint:Ve})]}),(0,t.jsx)(O.C,{open:fe,onClose:()=>Pe(!1),onSuccess:()=>{Pe(!1),Se(e=>e+1),xe.post({message:"API Product updated successfully",severity:"success",display:"transient"})},namespace:he||"",name:me||""}),(0,t.jsx)($.K,{open:be,title:"Delete API Product",description:`Are you sure you want to delete "${(null===(ue=Te.spec)||void 0===ue?void 0:ue.displayName)||Te.metadata.name}"? This action cannot be undone.`,severity:"high",confirmText:Te.metadata.name,deleting:we,onConfirm:async()=>{if(Te){ke(!0);try{await Ae.deleteApiProduct(he,me),ge(!1),xe.post({message:"API Product deleted successfully",severity:"success",display:"transient"}),ve("/kuadrant/api-products")}catch(e){const s=e instanceof Error?e.message:"unknown error occurred";xe.post({message:`Failed to delete API product. ${s}`,severity:"error",display:"transient"})}finally{ke(!1)}}},onCancel:()=>ge(!1)})]})}}}]);
2
- //# sourceMappingURL=7791.5c1eea8a.chunk.js.map
2
+ //# sourceMappingURL=7791.0a1d55bc.chunk.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"static/7791.5c1eea8a.chunk.js","mappings":"ymBAmDA,MAAMA,GAAYC,EAAAA,EAAAA,GAAYC,IAAW,CACvCC,MAAO,CACLC,WAAY,IACZC,MAAOH,EAAMI,QAAQC,KAAKC,UAC1BC,aAAcP,EAAMQ,QAAQ,IAC5BC,SAAU,UACVC,cAAe,aAEjBC,cAAe,CACbC,QAAS,OACTC,IAAKb,EAAMQ,QAAQ,GACnBM,WAAY,UAEdC,WAAY,CACVH,QAAS,OACTI,eAAgB,gBAChBF,WAAY,aACZP,aAAcP,EAAMQ,QAAQ,IAE9BS,YAAa,CACXL,QAAS,OACTC,IAAKb,EAAMQ,QAAQ,GACnBM,WAAY,aAIHI,EAAuB,K,IAmHdC,EAGQA,EAAAA,EAGAA,EAAAA,EAGJA,EAEJA,EAAAA,EAOF,EACOA,EAAAA,EAGGA,EAAAA,EAAgCA,EAAAA,GAqB/CA,GACGA,GAmCOA,GAKhBA,GAQAA,GA8BUA,GAsCNA,GAAAA,GAIIA,GAAAA,GAuHwCA,GA7YvD,MAAMC,GAAUtB,KACV,UAAEuB,GAAS,KAAEC,KAASC,EAAAA,EAAAA,aACtBC,IAAWC,EAAAA,EAAAA,eACXC,IAAcC,EAAAA,EAAAA,QAAOC,EAAAA,GACrBC,IAAWF,EAAAA,EAAAA,QAAOG,EAAAA,cAEjBC,GAAaC,KAAkBC,EAAAA,EAAAA,UAAS,IACxCC,GAAgBC,KAAqBF,EAAAA,EAAAA,WAAS,IAC9CG,GAAkBC,KAAuBJ,EAAAA,EAAAA,WAAS,IAClDK,GAAUC,KAAeN,EAAAA,EAAAA,WAAS,IAClCO,GAAYC,KAAiBR,EAAAA,EAAAA,UAAS,IAErCS,QAASC,KAAwBC,EAAAA,EAAAA,GAAsBC,EAAAA,KACvDH,QAASI,KAAwBF,EAAAA,EAAAA,GAAsBG,EAAAA,IACzDC,GAAuBL,IAG3BM,MAAO9B,GAAO,QACd+B,GAAO,MACPC,KACEC,EAAAA,EAAAA,GAASC,eACE3B,GAAY4B,cAAcjC,GAAYC,IAClD,CAACD,GAAWC,GAAMI,GAAac,MAG1BS,MAAOM,KAAuBH,EAAAA,EAAAA,GAASC,U,IACxClC,EAAAA,EAOEqC,EAPP,KAAKrC,UAAa,QAAbA,EAAAA,GAASsC,YAATtC,IAAAA,GAAwB,QAAxBA,EAAAA,EAAeuC,iBAAfvC,IAAAA,OAAAA,EAAAA,EAA0BG,MAC7B,OAAO,KAGT,MAAMqC,EAAYxC,GAAQsC,KAAKC,UAAUpC,KACnCkC,QAAc9B,GAAYkC,aAAavC,GAAYsC,GAEzD,OAAOH,SAAW,QAAXA,EAAAA,EAAOC,YAAPD,IAAAA,OAAAA,EAAAA,EAAaK,YAAa,MAChC,CAAC1C,GAASE,GAAWK,KAElBoC,GAAsBT,U,IAERlC,EACAA,EAFlB,IAAKA,GAAS,OACd,MAAM4C,EAA4C,eAApB,QAAZ5C,EAAAA,GAAQsC,YAARtC,IAAAA,OAAAA,EAAAA,EAAc6C,eAAgC,QAAU,YACpEC,EAAmC,QAAvB9C,EAAAA,GAAQ+C,SAASC,cAAjBhD,IAAAA,OAAAA,EAAAA,EAAyB8C,UAG3C,GAAkB,cAAdF,GAA2C,YAAdE,EASjC,UAEQvC,GAAY0C,iBAAiB/C,GAAYC,GAAO,CAAEmC,KAAM,CAAEO,cAAeD,KAC/ElC,GAASwC,KAAK,CACZC,QAAS,eAA6B,cAAdP,EAA4B,YAAc,6BAClEQ,SAAU,UACV3D,QAAS,cAEX6B,GAAe+B,GAAMA,EAAI,EAC3B,CAAE,MAAOC,GACP,MAAMC,EAAeD,aAAeE,MAAQF,EAAIH,QAAU,yBAC1DzC,GAASwC,KAAK,CACZC,QAAS,oCAAoCI,IAC7CH,SAAU,QACV3D,QAAS,aAEb,MAxBEiB,GAASwC,KAAK,CACZC,QAAS,kFACTC,SAAU,QACV3D,QAAS,eA0Df,GAAIsC,GACF,OAAO,SAAC0B,EAAAA,EAAQA,CAAAA,GAGlB,GAAIzB,KAAUhC,GACZ,OACE,SAAC0D,EAAAA,EAAkBA,CAAC1B,MAAOA,IAAS,IAAIwB,MAAM,2BAIlD,MAAMG,GAA8C,eAApB,QAAZ3D,EAAAA,GAAQsC,YAARtC,IAAAA,OAAAA,EAAAA,EAAc6C,eAG5Be,GAAoC,QAAd5D,EAAAA,GAAQ6D,cAAR7D,IAAAA,GAA0B,QAA1BA,EAAAA,EAAgB8D,kBAAhB9D,IAAAA,OAAAA,EAAAA,EAA4B+D,KACrDC,GAAiB,yBAAXA,EAAEC,MAELC,GAAoC,QAAdlE,EAAAA,GAAQ6D,cAAR7D,IAAAA,GAA0B,QAA1BA,EAAAA,EAAgB8D,kBAAhB9D,IAAAA,OAAAA,EAAAA,EAA4B+D,KACrDC,GAAiB,yBAAXA,EAAEC,MAELE,IAAgC,QAAdnE,EAAAA,GAAQ6D,cAAR7D,IAAAA,OAAAA,EAAAA,EAAgBmE,kBAAmB,GAErDC,IAA4B,QAAdpE,EAAAA,GAAQ6D,cAAR7D,IAAAA,GAAoC,QAApCA,EAAAA,EAAgBqE,4BAAhBrE,IAAAA,OAAAA,EAAAA,EAAsCsE,iBAAkB,CAAC,EACvEC,GAAgBC,OAAOC,OAAOL,IAC9BM,GAAaH,GAAcI,KAAMC,GACrCA,EAAOC,eAAe,QAGlBC,GAAYP,GAAcR,KAAMa,GAAgBA,EAAOC,eAAe,QACtEE,IAAaD,UAAsB,QAAvB,KAAoBE,WAApB,WAACF,EAAD,EAAyBG,YAAa,UAClDC,IAAiC,QAAdlF,EAAAA,GAAQ6D,cAAR7D,IAAAA,GAA6B,QAA7BA,EAAAA,EAAgBmF,qBAAhBnF,IAAAA,OAAAA,EAAAA,EAA+BoF,gBAAiB,UAGnEC,OAAoC,QAAdrF,EAAAA,GAAQ6D,cAAR7D,IAAAA,GAAuB,QAAvBA,EAAAA,EAAgBsF,eAAhBtF,IAAAA,OAAAA,EAAAA,EAAyBuF,QAAmB,QAAZvF,GAAAA,GAAQsC,YAARtC,IAAAA,IAA2B,QAA3BA,EAAAA,GAAcwF,qBAAdxF,IAAAA,OAAAA,EAAAA,EAA6ByF,iBACnFC,MAAoB9B,IAAuBM,IAAuBC,GAAgBwB,OAAS,GAEjG,IAAIC,GAAY,EAChB,MAAMC,GAAqBR,GAAmBO,MAAe,EACvDE,GAAmBJ,GAAiBE,MAAe,EACnDG,GAAerB,GAAakB,MAAe,EAE3CI,GAAgBC,IACpB,IAAKA,EAAQ,MAAO,YACpB,MAAMC,EAAkB,GAKxB,OAJID,EAAOE,OAAOD,EAAME,KAAK,GAAGH,EAAOE,aACnCF,EAAOI,QAAQH,EAAME,KAAK,GAAGH,EAAOI,eACpCJ,EAAOK,SAASJ,EAAME,KAAK,GAAGH,EAAOK,iBACrCL,EAAOM,QAAQL,EAAME,KAAK,GAAGH,EAAOM,eACjCL,EAAMP,OAAS,EAAIO,EAAMM,KAAK,MAAQ,aAG/C,OACE,UAACC,EAAAA,EAAIA,CAACC,QAAQ,O,WACZ,SAACC,EAAAA,EAAMA,CACLC,OAAmB,QAAZ5G,GAAAA,GAAQsC,YAARtC,IAAAA,QAAAA,EAAAA,GAAc6G,cAAe7G,GAAQ+C,SAAS5C,KACrD2G,UAAsB,QAAZ9G,GAAAA,GAAQsC,YAARtC,IAAAA,QAAAA,EAAAA,GAAc+G,cAAe,G,UAEvC,UAACC,EAAAA,EAAGA,CAACC,UAAWhH,GAAQT,c,WACtB,SAAC0H,EAAAA,GAAIA,CAACC,GAAG,yB,UACP,SAACC,EAAAA,EAAMA,CAACC,WAAW,SAACC,EAAAA,EAAaA,CAAAA,G,SAAK,WAEvCzF,KACC,SAACuF,EAAAA,EAAMA,CACLG,QAAQ,WACRvI,MAAO2E,GAAc,UAAY,UACjC6D,QAAS7E,G,SAERgB,GAAc,wBAA0B,wBAG5CnC,KACC,SAACiG,EAAAA,GAAOA,CAACb,MAAM,O,UACb,SAACc,EAAAA,EAAUA,CAACF,QAAS,IAAMxG,IAAkB,GAAO2G,KAAK,Q,UACvD,SAACC,EAAAA,EAAQA,CAAAA,OAIdjG,KACC,SAAC8F,EAAAA,GAAOA,CAACb,MAAM,S,UACb,SAACc,EAAAA,EAAUA,CAACF,QAAS,IAAMtG,IAAoB,GAAOyG,KAAK,Q,UACzD,SAACE,EAAAA,EAAUA,CAAAA,aAMrB,UAACC,EAAAA,EAAOA,C,WACN,SAACd,EAAAA,EAAGA,CAACe,GAAI,E,UACP,UAACC,EAAAA,EAAWA,CAACC,aAAW,a,WACtB,SAACf,EAAAA,GAAIA,CAACC,GAAG,yB,SAAyB,kBAClC,SAACe,EAAAA,EAAUA,C,UAAc,QAAZlI,GAAAA,GAAQsC,YAARtC,IAAAA,QAAAA,EAAAA,GAAc6G,cAAe7G,GAAQ+C,SAAS5C,YAKvB,gBAAhB,QAAvBH,GAAAA,GAAQ+C,SAASC,cAAjBhD,IAAAA,QAAAA,EAAAA,GAAyB8C,aACxB,SAACkE,EAAAA,EAAGA,CAACe,GAAI,E,UACP,UAACI,EAAAA,EAAKA,CAAC/E,SAAS,U,WACd,SAACgF,SAAAA,C,SAAO,4BAAgC,4DAKN,aAAhB,QAAvBpI,GAAAA,GAAQ+C,SAASC,cAAjBhD,IAAAA,QAAAA,EAAAA,GAAyB8C,aACxB,SAACkE,EAAAA,EAAGA,CAACe,GAAI,E,UACP,UAACI,EAAAA,EAAKA,CAAC/E,SAAS,Q,WACd,SAACgF,SAAAA,C,SAAO,0DAA8D,6DAK5E,SAACpB,EAAAA,EAAGA,CAACe,GAAI,E,UACP,UAACM,EAAAA,EAAIA,CACHvG,MAAOlB,GACP0H,SAAU,CAACC,EAAGC,IAAa3H,GAAe2H,GAC1CC,eAAe,UACfC,UAAU,U,WAEV,SAACC,EAAAA,EAAGA,CAAC7J,MAAM,aACVuG,KAAoB,SAACsD,EAAAA,EAAGA,CAAC7J,MAAM,eAC/B4G,KAAkB,SAACiD,EAAAA,EAAGA,CAAC7J,MAAM,aAC7B4F,KAAc,SAACiE,EAAAA,EAAGA,CAAC7J,MAAM,cAIb,IAAhB8B,KACC,UAACgI,EAAAA,EAAQA,CAAChC,MAAM,c,WACd,UAACI,EAAAA,EAAGA,CAACC,UAAWhH,GAAQL,W,WACtB,UAACoH,EAAAA,EAAGA,C,WACF,SAACkB,EAAAA,EAAUA,CAACX,QAAQ,UAAUN,UAAWhH,GAAQnB,M,SAAO,kBAGxD,SAACoJ,EAAAA,EAAUA,CAACX,QAAQ,K,UACL,QAAZvH,GAAAA,GAAQsC,YAARtC,IAAAA,QAAAA,EAAAA,GAAc6G,cAAe7G,GAAQ+C,SAAS5C,WAGnD,UAAC6G,EAAAA,EAAGA,CAACC,UAAWhH,GAAQH,Y,UACrB+B,KACC,SAACuF,EAAAA,EAAMA,CACLG,QAAQ,WACRvI,MAAO2E,GAAc,UAAY,UACjC6D,QAAS7E,GACTgF,KAAK,Q,SAEJhE,GAAc,wBAA0B,wBAG5CnC,KACC,SAACkG,EAAAA,EAAUA,CAACF,QAAS,IAAMxG,IAAkB,GAAO2G,KAAK,Q,UACvD,SAACC,EAAAA,EAAQA,CAACtI,SAAS,YAGtBqC,KACC,SAAC+F,EAAAA,EAAUA,CAACF,QAAS,IAAMtG,IAAoB,GAAOyG,KAAK,Q,UACzD,SAACE,EAAAA,EAAUA,CAACvI,SAAS,mBAM7B,SAACuJ,EAAAA,EAAiBA,CAChB7I,QAASA,GACT8I,YAAY,EACZC,iBAAiB,EACjB3G,mBAAoBA,QAKzBxB,KAAgBiF,IAAsBR,KACrC,SAACuD,EAAAA,EAAQA,CAAChC,MAAM,iB,UACC,QAAd5G,GAAAA,GAAQ6D,cAAR7D,IAAAA,IAAuB,QAAvBA,GAAAA,GAAgBsF,eAAhBtF,IAAAA,QAAAA,EAAAA,GAAyBuF,MACxB,SAACyD,EAAAA,EAAuBA,CAACC,WAAYjJ,GAAQ6D,OAAOyB,QAAQC,OAE5D,SAAC2C,EAAAA,EAAUA,CAACX,QAAQ,QAAQvI,MAAM,gB,UACnB,QAAZgB,GAAAA,GAAQsC,YAARtC,IAAAA,IAA2B,QAA3BA,GAAAA,GAAcwF,qBAAdxF,IAAAA,QAAAA,EAAAA,GAA6ByF,iBAC5B,sB,UAAE,iDAC+C,KAC/C,SAACyB,EAAAA,GAAIA,CAACC,GAAInH,GAAQsC,KAAKkD,cAAcC,eAAgByD,OAAO,S,SACzDlJ,GAAQsC,KAAKkD,cAAcC,oBAIhC,+DAOT7E,KAAgBkF,IAAoBJ,KACnC,UAACyD,EAAAA,EAAIA,CAACC,WAAS,EAAC/J,QAAS,E,WACvB,SAAC8J,EAAAA,EAAIA,CAACE,MAAI,EAACC,GAAI,G,UACb,SAACV,EAAAA,EAAQA,CAAChC,MAAM,sB,UACd,UAACuC,EAAAA,EAAIA,CAACC,WAAS,EAAC/J,QAAS,E,WACvB,UAAC8J,EAAAA,EAAIA,CAACE,MAAI,EAACC,GAAI,GAAIC,GAAI,E,WACrB,SAACrB,EAAAA,EAAUA,CAACX,QAAQ,QAAQvI,MAAM,gBAAgBwK,cAAY,E,SAAC,gBAG9D5F,IACC,UAACoD,EAAAA,EAAGA,C,WACF,SAACyC,EAAAA,EAAIA,CACH3K,MAAsC,SAA/B8E,GAAoBC,OAAoB,QAAU,YACzD8D,KAAK,QACL+B,MAAO,CACLC,gBAAgD,SAA/B/F,GAAoBC,OAAoB,UAAY,UACrE7E,MAAO,OACPI,aAAc,MAGlB,SAAC8I,EAAAA,EAAUA,CAACX,QAAQ,Q,SACjB3D,GAAoBT,SAAW,6BAIpC,SAAC+E,EAAAA,EAAUA,CAACX,QAAQ,Q,SAAQ,mCAGhC,UAAC4B,EAAAA,EAAIA,CAACE,MAAI,EAACC,GAAI,GAAIC,GAAI,E,WACrB,SAACrB,EAAAA,EAAUA,CAACX,QAAQ,QAAQvI,MAAM,gBAAgBwK,cAAY,E,SAAC,gBAG9DtF,IACC,UAAC8C,EAAAA,EAAGA,C,WACF,SAACyC,EAAAA,EAAIA,CACH3K,MAAsC,SAA/BoF,GAAoBL,OAAoB,QAAU,YACzD8D,KAAK,QACL+B,MAAO,CACLC,gBAAgD,SAA/BzF,GAAoBL,OAAoB,UAAY,UACrE7E,MAAO,OACPI,aAAc,MAGlB,SAAC8I,EAAAA,EAAUA,CAACX,QAAQ,Q,SACjBrD,GAAoBf,SAAW,6BAIpC,SAAC+E,EAAAA,EAAUA,CAACX,QAAQ,Q,SAAQ,yCAOrCpD,GAAgBwB,OAAS,IACxB,SAACwD,EAAAA,EAAIA,CAACE,MAAI,EAACC,GAAI,G,UACb,UAACV,EAAAA,EAAQA,CAAChC,MAAM,uB,WACd,SAACsB,EAAAA,EAAUA,CAACX,QAAQ,QAAQvI,MAAM,gBAAgB4K,WAAS,E,SAAC,iGAG5D,UAACC,EAAAA,EAAKA,CAAClC,KAAK,Q,WACV,SAACmC,EAAAA,EAASA,C,UACR,UAACC,EAAAA,EAAQA,C,WACP,SAACC,EAAAA,EAASA,C,SAAC,UACX,SAACA,EAAAA,EAASA,C,SAAC,sBAGf,SAACC,EAAAA,EAASA,C,SACP9F,GAAgB+F,IAAKC,IACpB,UAACJ,EAAAA,EAAQA,C,WACP,SAACC,EAAAA,EAASA,C,UACR,SAACP,EAAAA,EAAIA,CAAC3K,MAAOqL,EAAKC,KAAMzC,KAAK,QAAQ3I,MAAM,eAE7C,SAACgL,EAAAA,EAASA,C,SAAEhE,GAAamE,EAAKlE,YAJjBkE,EAAKC,oBAenCxJ,KAAgBmF,IAAgBrB,KAC/B,SAAC2F,EAAAA,EAAgBA,CACfpF,UAAWF,GACXK,cAAeF,SAKrB,SAACoF,EAAAA,EAAoBA,CACnBC,KAAMxJ,GACNyJ,QAAS,IAAMxJ,IAAkB,GACjCyJ,UA/ToB,KACxBzJ,IAAkB,GAClBM,GAAe+B,GAAMA,EAAI,GACzB3C,GAASwC,KAAK,CACZC,QAAS,mCACTC,SAAU,UACV3D,QAAS,eA0TPS,UAAWA,IAAa,GACxBC,KAAMA,IAAQ,MAGhB,SAACuK,EAAAA,EAAmBA,CAClBH,KAAMtJ,GACN2F,MAAM,qBACNG,YAAa,qCAAgD,QAAZ/G,GAAAA,GAAQsC,YAARtC,IAAAA,QAAAA,EAAAA,GAAc6G,cAAe7G,GAAQ+C,SAAS5C,uCAC/FiD,SAAS,OACTuH,YAAa3K,GAAQ+C,SAAS5C,KAC9BgB,SAAUA,GACVyJ,UAjUe1I,UACnB,GAAKlC,GAAL,CACAoB,IAAY,GACZ,UACQb,GAAYsK,iBAAiB3K,GAAYC,IAC/Ce,IAAoB,GACpBR,GAASwC,KAAK,CACZC,QAAS,mCACTC,SAAU,UACV3D,QAAS,cAEXY,GAAS,yBACX,CAAE,MAAOiD,GACP,MAAMC,EAAeD,aAAeE,MAAQF,EAAIH,QAAU,yBAC1DzC,GAASwC,KAAK,CACZC,QAAS,iCAAiCI,IAC1CH,SAAU,QACV3D,QAAS,aAEb,CAAE,QACA2B,IAAY,EACd,CApBoB,GAiUhB0J,SAAU,IAAM5J,IAAoB,Q","sources":["webpack://internal.plugin-kuadrant/./src/components/ApiProductDetailPage/ApiProductDetailPage.tsx"],"sourcesContent":["import React, { useState } from \"react\";\nimport { useParams, useNavigate } from \"react-router-dom\";\nimport {\n useApi,\n alertApiRef,\n} from \"@backstage/core-plugin-api\";\nimport { kuadrantApiRef } from '../../api';\nimport { useAsync } from \"react-use\";\nimport {\n Header,\n Page,\n Content,\n Progress,\n ResponseErrorPanel,\n InfoCard,\n Link,\n Breadcrumbs,\n} from \"@backstage/core-components\";\nimport { OpenApiDefinitionWidget } from \"@backstage/plugin-api-docs\";\nimport {\n Box,\n Typography,\n IconButton,\n Tooltip,\n Tabs,\n Tab,\n Button,\n makeStyles,\n Grid,\n Chip,\n Table,\n TableBody,\n TableCell,\n TableHead,\n TableRow,\n} from \"@material-ui/core\";\nimport ArrowBackIcon from \"@material-ui/icons/ArrowBack\";\nimport EditIcon from \"@material-ui/icons/Edit\";\nimport DeleteIcon from \"@material-ui/icons/Delete\";\nimport { Alert } from '@material-ui/lab';\nimport { APIProduct } from \"../../types/api-management\";\nimport { EditAPIProductDialog } from \"../EditAPIProductDialog\";\nimport { ConfirmDeleteDialog } from \"../ConfirmDeleteDialog\";\nimport { ApiProductDetails } from \"../ApiProductDetails\";\nimport { OidcProviderCard } from \"../OidcProviderCard\";\nimport { useKuadrantPermission } from \"../../utils/permissions\";\nimport {\n kuadrantApiProductUpdateAllPermission,\n kuadrantApiProductDeleteAllPermission,\n} from \"../../permissions\";\n\nconst useStyles = makeStyles((theme) => ({\n label: {\n fontWeight: 600,\n color: theme.palette.text.secondary,\n marginBottom: theme.spacing(0.5),\n fontSize: '0.75rem',\n textTransform: 'uppercase',\n },\n actionButtons: {\n display: \"flex\",\n gap: theme.spacing(1),\n alignItems: \"center\",\n },\n cardHeader: {\n display: 'flex',\n justifyContent: 'space-between',\n alignItems: 'flex-start',\n marginBottom: theme.spacing(2),\n },\n cardActions: {\n display: 'flex',\n gap: theme.spacing(1),\n alignItems: 'center',\n },\n}));\n\nexport const ApiProductDetailPage = () => {\n const classes = useStyles();\n const { namespace, name } = useParams<{ namespace: string; name: string }>();\n const navigate = useNavigate();\n const kuadrantApi = useApi(kuadrantApiRef);\n const alertApi = useApi(alertApiRef);\n\n const [selectedTab, setSelectedTab] = useState(0);\n const [editDialogOpen, setEditDialogOpen] = useState(false);\n const [deleteDialogOpen, setDeleteDialogOpen] = useState(false);\n const [deleting, setDeleting] = useState(false);\n const [refreshKey, setRefreshKey] = useState(0);\n\n const { allowed: canUpdateApiProduct } = useKuadrantPermission(kuadrantApiProductUpdateAllPermission);\n const { allowed: canDeleteApiProduct } = useKuadrantPermission(kuadrantApiProductDeleteAllPermission);\n const canPublishApiProduct = canUpdateApiProduct;\n\n const {\n value: product,\n loading,\n error,\n } = useAsync(async () => {\n return await kuadrantApi.getApiProduct(namespace!, name!) as APIProduct;\n }, [namespace, name, kuadrantApi, refreshKey]);\n\n // fetch httproute to get hostnames\n const { value: httpRouteHostnames } = useAsync(async () => {\n if (!product?.spec?.targetRef?.name) {\n return null;\n }\n\n const routeName = product.spec.targetRef.name;\n const route = await kuadrantApi.getHttpRoute(namespace!, routeName)\n\n return route?.spec?.hostnames || null;\n }, [product, namespace, kuadrantApi]);\n\n const handlePublishToggle = async () => {\n if (!product) return;\n const newStatus = product.spec?.publishStatus === \"Published\" ? \"Draft\" : \"Published\";\n const lifecycle = product.metadata.labels?.lifecycle;\n\n // prevent publishing retired APIs\n if (newStatus === \"Published\" && lifecycle === \"retired\") {\n alertApi.post({\n message: `Cannot publish a retired API product. Please change the lifecycle status first.`,\n severity: \"error\",\n display: \"transient\",\n });\n return;\n }\n\n try {\n // @ts-ignore since updating partially the obj\n await kuadrantApi.updateApiProduct(namespace!, name!, { spec: { publishStatus: newStatus } });\n alertApi.post({\n message: `API Product ${newStatus === \"Published\" ? \"published\" : \"unpublished\"} successfully`,\n severity: \"success\",\n display: \"transient\",\n });\n setRefreshKey((k) => k + 1);\n } catch (err) {\n const errorMessage = err instanceof Error ? err.message : \"unknown error occurred\";\n alertApi.post({\n message: `Failed to update publish status. ${errorMessage}`,\n severity: \"error\",\n display: \"transient\",\n });\n }\n };\n\n const handleEditSuccess = () => {\n setEditDialogOpen(false);\n setRefreshKey((k) => k + 1);\n alertApi.post({\n message: \"API Product updated successfully\",\n severity: \"success\",\n display: \"transient\",\n });\n };\n\n const handleDelete = async () => {\n if (!product) return;\n setDeleting(true);\n try {\n await kuadrantApi.deleteApiProduct(namespace!, name!);\n setDeleteDialogOpen(false);\n alertApi.post({\n message: \"API Product deleted successfully\",\n severity: \"success\",\n display: \"transient\",\n });\n navigate(\"/kuadrant/api-products\");\n } catch (err) {\n const errorMessage = err instanceof Error ? err.message : \"unknown error occurred\";\n alertApi.post({\n message: `Failed to delete API product. ${errorMessage}`,\n severity: \"error\",\n display: \"transient\",\n });\n } finally {\n setDeleting(false);\n }\n };\n\n if (loading) {\n return <Progress />;\n }\n\n if (error || !product) {\n return (\n <ResponseErrorPanel error={error || new Error(\"API product not found\")} />\n );\n }\n\n const isPublished = product.spec?.publishStatus === \"Published\";\n\n // get policy conditions from status\n const planPolicyCondition = product.status?.conditions?.find(\n (c) => c.type === \"PlanPolicyDiscovered\"\n );\n const authPolicyCondition = product.status?.conditions?.find(\n (c) => c.type === \"AuthPolicyDiscovered\"\n );\n const discoveredPlans = product.status?.discoveredPlans || [];\n\n const authSchemes = product.status?.discoveredAuthScheme?.authentication || {};\n const schemeObjects = Object.values(authSchemes);\n const hasOIDCTab = schemeObjects.some((scheme: any) =>\n scheme.hasOwnProperty(\"jwt\"),\n );\n // Extract JWT issuer from the first JWT scheme\n const jwtScheme = schemeObjects.find((scheme: any) => scheme.hasOwnProperty(\"jwt\"));\n const jwtIssuer = (jwtScheme as any)?.jwt?.issuerUrl || \"unknown\";\n const jwtTokenEndpoint = product.status?.oidcDiscovery?.tokenEndpoint || \"unknown\";\n\n // compute tab indices\n const hasDefinitionTab = !!(product.status?.openapi?.raw || product.spec?.documentation?.openAPISpecURL);\n const hasPoliciesTab = !!(planPolicyCondition || authPolicyCondition || discoveredPlans.length > 0);\n\n let nextIndex = 1; // Overview is always at index 0\n const definitionTabIndex = hasDefinitionTab ? nextIndex++ : -1;\n const policiesTabIndex = hasPoliciesTab ? nextIndex++ : -1;\n const oidcTabIndex = hasOIDCTab ? nextIndex++ : -1;\n\n const formatLimits = (limits: any): string => {\n if (!limits) return \"No limits\";\n const parts: string[] = [];\n if (limits.daily) parts.push(`${limits.daily}/day`);\n if (limits.weekly) parts.push(`${limits.weekly}/week`);\n if (limits.monthly) parts.push(`${limits.monthly}/month`);\n if (limits.yearly) parts.push(`${limits.yearly}/year`);\n return parts.length > 0 ? parts.join(\", \") : \"No limits\";\n };\n\n return (\n <Page themeId=\"tool\">\n <Header\n title={product.spec?.displayName || product.metadata.name}\n subtitle={product.spec?.description || \"\"}\n >\n <Box className={classes.actionButtons}>\n <Link to=\"/kuadrant/api-products\">\n <Button startIcon={<ArrowBackIcon />}>Back</Button>\n </Link>\n {canPublishApiProduct && (\n <Button\n variant=\"outlined\"\n color={isPublished ? \"default\" : \"primary\"}\n onClick={handlePublishToggle}\n >\n {isPublished ? \"Unpublish API product\" : \"Publish API product\"}\n </Button>\n )}\n {canUpdateApiProduct && (\n <Tooltip title=\"Edit\">\n <IconButton onClick={() => setEditDialogOpen(true)} size=\"small\">\n <EditIcon />\n </IconButton>\n </Tooltip>\n )}\n {canDeleteApiProduct && (\n <Tooltip title=\"Delete\">\n <IconButton onClick={() => setDeleteDialogOpen(true)} size=\"small\">\n <DeleteIcon />\n </IconButton>\n </Tooltip>\n )}\n </Box>\n </Header>\n <Content>\n <Box mb={2}>\n <Breadcrumbs aria-label=\"breadcrumb\">\n <Link to=\"/kuadrant/api-products\">API Products</Link>\n <Typography>{product.spec?.displayName || product.metadata.name}</Typography>\n </Breadcrumbs>\n </Box>\n\n\n {product.metadata.labels?.lifecycle === 'deprecated' && (\n <Box mb={2}>\n <Alert severity=\"warning\">\n <strong>This API is deprecated.</strong> Please contact your administrator for more details.\n </Alert>\n </Box>\n )}\n\n {product.metadata.labels?.lifecycle === 'retired' && (\n <Box mb={2}>\n <Alert severity=\"error\">\n <strong>This API has been retired and is no longer available.</strong> Please contact your administrator for alternatives.\n </Alert>\n </Box>\n )}\n\n <Box mb={2}>\n <Tabs\n value={selectedTab}\n onChange={(_, newValue) => setSelectedTab(newValue)}\n indicatorColor=\"primary\"\n textColor=\"primary\"\n >\n <Tab label=\"Overview\" />\n {hasDefinitionTab && <Tab label=\"Definition\" />}\n {hasPoliciesTab && <Tab label=\"Policies\" />}\n {hasOIDCTab && <Tab label=\"OIDC\" />}\n </Tabs>\n </Box>\n\n {selectedTab === 0 && (\n <InfoCard title=\"API Product\">\n <Box className={classes.cardHeader}>\n <Box>\n <Typography variant=\"caption\" className={classes.label}>\n Product Name\n </Typography>\n <Typography variant=\"h6\">\n {product.spec?.displayName || product.metadata.name}\n </Typography>\n </Box>\n <Box className={classes.cardActions}>\n {canPublishApiProduct && (\n <Button\n variant=\"outlined\"\n color={isPublished ? \"default\" : \"primary\"}\n onClick={handlePublishToggle}\n size=\"small\"\n >\n {isPublished ? \"Unpublish API product\" : \"Publish API product\"}\n </Button>\n )}\n {canUpdateApiProduct && (\n <IconButton onClick={() => setEditDialogOpen(true)} size=\"small\">\n <EditIcon fontSize=\"small\" />\n </IconButton>\n )}\n {canDeleteApiProduct && (\n <IconButton onClick={() => setDeleteDialogOpen(true)} size=\"small\">\n <DeleteIcon fontSize=\"small\" />\n </IconButton>\n )}\n </Box>\n </Box>\n\n <ApiProductDetails\n product={product}\n showStatus={true}\n showCatalogLink={true}\n httpRouteHostnames={httpRouteHostnames}\n />\n </InfoCard>\n )}\n\n {selectedTab === definitionTabIndex && hasDefinitionTab && (\n <InfoCard title=\"API Definition\">\n {product.status?.openapi?.raw ? (\n <OpenApiDefinitionWidget definition={product.status.openapi.raw} />\n ) : (\n <Typography variant=\"body2\" color=\"textSecondary\">\n {product.spec?.documentation?.openAPISpecURL ? (\n <>\n OpenAPI specification not yet synced. View at:{\" \"}\n <Link to={product.spec.documentation.openAPISpecURL} target=\"_blank\">\n {product.spec.documentation.openAPISpecURL}\n </Link>\n </>\n ) : (\n \"No OpenAPI specification available for this API product.\"\n )}\n </Typography>\n )}\n </InfoCard>\n )}\n\n {selectedTab === policiesTabIndex && hasPoliciesTab && (\n <Grid container spacing={3}>\n <Grid item xs={12}>\n <InfoCard title=\"Discovered Policies\">\n <Grid container spacing={2}>\n <Grid item xs={12} md={6}>\n <Typography variant=\"body2\" color=\"textSecondary\" gutterBottom>\n Plan Policy\n </Typography>\n {planPolicyCondition ? (\n <Box>\n <Chip\n label={planPolicyCondition.status === \"True\" ? \"Found\" : \"Not Found\"}\n size=\"small\"\n style={{\n backgroundColor: planPolicyCondition.status === \"True\" ? \"#4caf50\" : \"#ff9800\",\n color: \"#fff\",\n marginBottom: 8,\n }}\n />\n <Typography variant=\"body2\">\n {planPolicyCondition.message || \"No details available\"}\n </Typography>\n </Box>\n ) : (\n <Typography variant=\"body2\">No plan policy information</Typography>\n )}\n </Grid>\n <Grid item xs={12} md={6}>\n <Typography variant=\"body2\" color=\"textSecondary\" gutterBottom>\n Auth Policy\n </Typography>\n {authPolicyCondition ? (\n <Box>\n <Chip\n label={authPolicyCondition.status === \"True\" ? \"Found\" : \"Not Found\"}\n size=\"small\"\n style={{\n backgroundColor: authPolicyCondition.status === \"True\" ? \"#4caf50\" : \"#ff9800\",\n color: \"#fff\",\n marginBottom: 8,\n }}\n />\n <Typography variant=\"body2\">\n {authPolicyCondition.message || \"No details available\"}\n </Typography>\n </Box>\n ) : (\n <Typography variant=\"body2\">No auth policy information</Typography>\n )}\n </Grid>\n </Grid>\n </InfoCard>\n </Grid>\n\n {discoveredPlans.length > 0 && (\n <Grid item xs={12}>\n <InfoCard title=\"Effective Plan Tiers\">\n <Typography variant=\"body2\" color=\"textSecondary\" paragraph>\n These tiers are computed from all attached PlanPolicies (including gateway-level policies).\n </Typography>\n <Table size=\"small\">\n <TableHead>\n <TableRow>\n <TableCell>Tier</TableCell>\n <TableCell>Rate Limits</TableCell>\n </TableRow>\n </TableHead>\n <TableBody>\n {discoveredPlans.map((plan) => (\n <TableRow key={plan.tier}>\n <TableCell>\n <Chip label={plan.tier} size=\"small\" color=\"primary\" />\n </TableCell>\n <TableCell>{formatLimits(plan.limits)}</TableCell>\n </TableRow>\n ))}\n </TableBody>\n </Table>\n </InfoCard>\n </Grid>\n )}\n </Grid>\n )}\n\n {selectedTab === oidcTabIndex && hasOIDCTab && (\n <OidcProviderCard\n issuerUrl={jwtIssuer}\n tokenEndpoint={jwtTokenEndpoint}\n />\n )}\n </Content>\n\n <EditAPIProductDialog\n open={editDialogOpen}\n onClose={() => setEditDialogOpen(false)}\n onSuccess={handleEditSuccess}\n namespace={namespace || \"\"}\n name={name || \"\"}\n />\n\n <ConfirmDeleteDialog\n open={deleteDialogOpen}\n title=\"Delete API Product\"\n description={`Are you sure you want to delete \"${product.spec?.displayName || product.metadata.name}\"? This action cannot be undone.`}\n severity=\"high\"\n confirmText={product.metadata.name}\n deleting={deleting}\n onConfirm={handleDelete}\n onCancel={() => setDeleteDialogOpen(false)}\n />\n </Page>\n );\n};\n"],"names":["useStyles","makeStyles","theme","label","fontWeight","color","palette","text","secondary","marginBottom","spacing","fontSize","textTransform","actionButtons","display","gap","alignItems","cardHeader","justifyContent","cardActions","ApiProductDetailPage","product","classes","namespace","name","useParams","navigate","useNavigate","kuadrantApi","useApi","kuadrantApiRef","alertApi","alertApiRef","selectedTab","setSelectedTab","useState","editDialogOpen","setEditDialogOpen","deleteDialogOpen","setDeleteDialogOpen","deleting","setDeleting","refreshKey","setRefreshKey","allowed","canUpdateApiProduct","useKuadrantPermission","kuadrantApiProductUpdateAllPermission","canDeleteApiProduct","kuadrantApiProductDeleteAllPermission","canPublishApiProduct","value","loading","error","useAsync","async","getApiProduct","httpRouteHostnames","route","spec","targetRef","routeName","getHttpRoute","hostnames","handlePublishToggle","newStatus","publishStatus","lifecycle","metadata","labels","updateApiProduct","post","message","severity","k","err","errorMessage","Error","Progress","ResponseErrorPanel","isPublished","planPolicyCondition","status","conditions","find","c","type","authPolicyCondition","discoveredPlans","authSchemes","discoveredAuthScheme","authentication","schemeObjects","Object","values","hasOIDCTab","some","scheme","hasOwnProperty","jwtScheme","jwtIssuer","jwt","issuerUrl","jwtTokenEndpoint","oidcDiscovery","tokenEndpoint","hasDefinitionTab","openapi","raw","documentation","openAPISpecURL","hasPoliciesTab","length","nextIndex","definitionTabIndex","policiesTabIndex","oidcTabIndex","formatLimits","limits","parts","daily","push","weekly","monthly","yearly","join","Page","themeId","Header","title","displayName","subtitle","description","Box","className","Link","to","Button","startIcon","ArrowBackIcon","variant","onClick","Tooltip","IconButton","size","EditIcon","DeleteIcon","Content","mb","Breadcrumbs","aria-label","Typography","Alert","strong","Tabs","onChange","_","newValue","indicatorColor","textColor","Tab","InfoCard","ApiProductDetails","showStatus","showCatalogLink","OpenApiDefinitionWidget","definition","target","Grid","container","item","xs","md","gutterBottom","Chip","style","backgroundColor","paragraph","Table","TableHead","TableRow","TableCell","TableBody","map","plan","tier","OidcProviderCard","EditAPIProductDialog","open","onClose","onSuccess","ConfirmDeleteDialog","confirmText","onConfirm","deleteApiProduct","onCancel"],"sourceRoot":""}
1
+ {"version":3,"file":"static/7791.0a1d55bc.chunk.js","mappings":"ymBAmDA,MAAMA,GAAYC,EAAAA,EAAAA,GAAYC,IAAW,CACvCC,MAAO,CACLC,WAAY,IACZC,MAAOH,EAAMI,QAAQC,KAAKC,UAC1BC,aAAcP,EAAMQ,QAAQ,IAC5BC,SAAU,UACVC,cAAe,aAEjBC,cAAe,CACbC,QAAS,OACTC,IAAKb,EAAMQ,QAAQ,GACnBM,WAAY,UAEdC,WAAY,CACVH,QAAS,OACTI,eAAgB,gBAChBF,WAAY,aACZP,aAAcP,EAAMQ,QAAQ,IAE9BS,YAAa,CACXL,QAAS,OACTC,IAAKb,EAAMQ,QAAQ,GACnBM,WAAY,aAIHI,EAAuB,K,IAmHdC,EAGQA,EAAAA,EAGAA,EAAAA,EAGJA,EAEJA,EAAAA,EAOF,EACOA,EAAAA,EAGGA,EAAAA,EAAgCA,EAAAA,GAqB/CA,GACGA,GAmCOA,GAKhBA,GAQAA,GA8BUA,GAsCNA,GAAAA,GAIIA,GAAAA,GAuHwCA,GA7YvD,MAAMC,GAAUtB,KACV,UAAEuB,GAAS,KAAEC,KAASC,EAAAA,EAAAA,aACtBC,IAAWC,EAAAA,EAAAA,eACXC,IAAcC,EAAAA,EAAAA,QAAOC,EAAAA,GACrBC,IAAWF,EAAAA,EAAAA,QAAOG,EAAAA,cAEjBC,GAAaC,KAAkBC,EAAAA,EAAAA,UAAS,IACxCC,GAAgBC,KAAqBF,EAAAA,EAAAA,WAAS,IAC9CG,GAAkBC,KAAuBJ,EAAAA,EAAAA,WAAS,IAClDK,GAAUC,KAAeN,EAAAA,EAAAA,WAAS,IAClCO,GAAYC,KAAiBR,EAAAA,EAAAA,UAAS,IAErCS,QAASC,KAAwBC,EAAAA,EAAAA,GAAsBC,EAAAA,KACvDH,QAASI,KAAwBF,EAAAA,EAAAA,GAAsBG,EAAAA,IACzDC,GAAuBL,IAG3BM,MAAO9B,GAAO,QACd+B,GAAO,MACPC,KACEC,EAAAA,EAAAA,GAASC,eACE3B,GAAY4B,cAAcjC,GAAYC,IAClD,CAACD,GAAWC,GAAMI,GAAac,MAG1BS,MAAOM,KAAuBH,EAAAA,EAAAA,GAASC,U,IACxClC,EAAAA,EAOEqC,EAPP,KAAKrC,UAAa,QAAbA,EAAAA,GAASsC,YAATtC,IAAAA,GAAwB,QAAxBA,EAAAA,EAAeuC,iBAAfvC,IAAAA,OAAAA,EAAAA,EAA0BG,MAC7B,OAAO,KAGT,MAAMqC,EAAYxC,GAAQsC,KAAKC,UAAUpC,KACnCkC,QAAc9B,GAAYkC,aAAavC,GAAYsC,GAEzD,OAAOH,SAAW,QAAXA,EAAAA,EAAOC,YAAPD,IAAAA,OAAAA,EAAAA,EAAaK,YAAa,MAChC,CAAC1C,GAASE,GAAWK,KAElBoC,GAAsBT,U,IAERlC,EACAA,EAFlB,IAAKA,GAAS,OACd,MAAM4C,EAA4C,eAApB,QAAZ5C,EAAAA,GAAQsC,YAARtC,IAAAA,OAAAA,EAAAA,EAAc6C,eAAgC,QAAU,YACpEC,EAAmC,QAAvB9C,EAAAA,GAAQ+C,SAASC,cAAjBhD,IAAAA,OAAAA,EAAAA,EAAyB8C,UAG3C,GAAkB,cAAdF,GAA2C,YAAdE,EASjC,UAEQvC,GAAY0C,iBAAiB/C,GAAYC,GAAO,CAAEmC,KAAM,CAAEO,cAAeD,KAC/ElC,GAASwC,KAAK,CACZC,QAAS,eAA6B,cAAdP,EAA4B,YAAc,6BAClEQ,SAAU,UACV3D,QAAS,cAEX6B,GAAe+B,GAAMA,EAAI,EAC3B,CAAE,MAAOC,GACP,MAAMC,EAAeD,aAAeE,MAAQF,EAAIH,QAAU,yBAC1DzC,GAASwC,KAAK,CACZC,QAAS,oCAAoCI,IAC7CH,SAAU,QACV3D,QAAS,aAEb,MAxBEiB,GAASwC,KAAK,CACZC,QAAS,kFACTC,SAAU,QACV3D,QAAS,eA0Df,GAAIsC,GACF,OAAO,SAAC0B,EAAAA,EAAQA,CAAAA,GAGlB,GAAIzB,KAAUhC,GACZ,OACE,SAAC0D,EAAAA,EAAkBA,CAAC1B,MAAOA,IAAS,IAAIwB,MAAM,2BAIlD,MAAMG,GAA8C,eAApB,QAAZ3D,EAAAA,GAAQsC,YAARtC,IAAAA,OAAAA,EAAAA,EAAc6C,eAG5Be,GAAoC,QAAd5D,EAAAA,GAAQ6D,cAAR7D,IAAAA,GAA0B,QAA1BA,EAAAA,EAAgB8D,kBAAhB9D,IAAAA,OAAAA,EAAAA,EAA4B+D,KACrDC,GAAiB,yBAAXA,EAAEC,MAELC,GAAoC,QAAdlE,EAAAA,GAAQ6D,cAAR7D,IAAAA,GAA0B,QAA1BA,EAAAA,EAAgB8D,kBAAhB9D,IAAAA,OAAAA,EAAAA,EAA4B+D,KACrDC,GAAiB,yBAAXA,EAAEC,MAELE,IAAgC,QAAdnE,EAAAA,GAAQ6D,cAAR7D,IAAAA,OAAAA,EAAAA,EAAgBmE,kBAAmB,GAErDC,IAA4B,QAAdpE,EAAAA,GAAQ6D,cAAR7D,IAAAA,GAAoC,QAApCA,EAAAA,EAAgBqE,4BAAhBrE,IAAAA,OAAAA,EAAAA,EAAsCsE,iBAAkB,CAAC,EACvEC,GAAgBC,OAAOC,OAAOL,IAC9BM,GAAaH,GAAcI,KAAMC,GACrCA,EAAOC,eAAe,QAGlBC,GAAYP,GAAcR,KAAMa,GAAgBA,EAAOC,eAAe,QACtEE,IAAaD,UAAsB,QAAvB,KAAoBE,WAApB,WAACF,EAAD,EAAyBG,YAAa,UAClDC,IAAiC,QAAdlF,EAAAA,GAAQ6D,cAAR7D,IAAAA,GAA6B,QAA7BA,EAAAA,EAAgBmF,qBAAhBnF,IAAAA,OAAAA,EAAAA,EAA+BoF,gBAAiB,UAGnEC,OAAoC,QAAdrF,EAAAA,GAAQ6D,cAAR7D,IAAAA,GAAuB,QAAvBA,EAAAA,EAAgBsF,eAAhBtF,IAAAA,OAAAA,EAAAA,EAAyBuF,QAAmB,QAAZvF,GAAAA,GAAQsC,YAARtC,IAAAA,IAA2B,QAA3BA,EAAAA,GAAcwF,qBAAdxF,IAAAA,OAAAA,EAAAA,EAA6ByF,iBACnFC,MAAoB9B,IAAuBM,IAAuBC,GAAgBwB,OAAS,GAEjG,IAAIC,GAAY,EAChB,MAAMC,GAAqBR,GAAmBO,MAAe,EACvDE,GAAmBJ,GAAiBE,MAAe,EACnDG,GAAerB,GAAakB,MAAe,EAE3CI,GAAgBC,IACpB,IAAKA,EAAQ,MAAO,YACpB,MAAMC,EAAkB,GAKxB,OAJID,EAAOE,OAAOD,EAAME,KAAK,GAAGH,EAAOE,aACnCF,EAAOI,QAAQH,EAAME,KAAK,GAAGH,EAAOI,eACpCJ,EAAOK,SAASJ,EAAME,KAAK,GAAGH,EAAOK,iBACrCL,EAAOM,QAAQL,EAAME,KAAK,GAAGH,EAAOM,eACjCL,EAAMP,OAAS,EAAIO,EAAMM,KAAK,MAAQ,aAG/C,OACE,UAACC,EAAAA,EAAIA,CAACC,QAAQ,O,WACZ,SAACC,EAAAA,EAAMA,CACLC,OAAmB,QAAZ5G,GAAAA,GAAQsC,YAARtC,IAAAA,QAAAA,EAAAA,GAAc6G,cAAe7G,GAAQ+C,SAAS5C,KACrD2G,UAAsB,QAAZ9G,GAAAA,GAAQsC,YAARtC,IAAAA,QAAAA,EAAAA,GAAc+G,cAAe,G,UAEvC,UAACC,EAAAA,EAAGA,CAACC,UAAWhH,GAAQT,c,WACtB,SAAC0H,EAAAA,GAAIA,CAACC,GAAG,yB,UACP,SAACC,EAAAA,EAAMA,CAACC,WAAW,SAACC,EAAAA,EAAaA,CAAAA,G,SAAK,WAEvCzF,KACC,SAACuF,EAAAA,EAAMA,CACLG,QAAQ,WACRvI,MAAO2E,GAAc,UAAY,UACjC6D,QAAS7E,G,SAERgB,GAAc,wBAA0B,wBAG5CnC,KACC,SAACiG,EAAAA,GAAOA,CAACb,MAAM,O,UACb,SAACc,EAAAA,EAAUA,CAACF,QAAS,IAAMxG,IAAkB,GAAO2G,KAAK,Q,UACvD,SAACC,EAAAA,EAAQA,CAAAA,OAIdjG,KACC,SAAC8F,EAAAA,GAAOA,CAACb,MAAM,S,UACb,SAACc,EAAAA,EAAUA,CAACF,QAAS,IAAMtG,IAAoB,GAAOyG,KAAK,Q,UACzD,SAACE,EAAAA,EAAUA,CAAAA,aAMrB,UAACC,EAAAA,EAAOA,C,WACN,SAACd,EAAAA,EAAGA,CAACe,GAAI,E,UACP,UAACC,EAAAA,EAAWA,CAACC,aAAW,a,WACtB,SAACf,EAAAA,GAAIA,CAACC,GAAG,yB,SAAyB,kBAClC,SAACe,EAAAA,EAAUA,C,UAAc,QAAZlI,GAAAA,GAAQsC,YAARtC,IAAAA,QAAAA,EAAAA,GAAc6G,cAAe7G,GAAQ+C,SAAS5C,YAKvB,gBAAhB,QAAvBH,GAAAA,GAAQ+C,SAASC,cAAjBhD,IAAAA,QAAAA,EAAAA,GAAyB8C,aACxB,SAACkE,EAAAA,EAAGA,CAACe,GAAI,E,UACP,UAACI,EAAAA,EAAKA,CAAC/E,SAAS,U,WACd,SAACgF,SAAAA,C,SAAO,4BAAgC,4DAKN,aAAhB,QAAvBpI,GAAAA,GAAQ+C,SAASC,cAAjBhD,IAAAA,QAAAA,EAAAA,GAAyB8C,aACxB,SAACkE,EAAAA,EAAGA,CAACe,GAAI,E,UACP,UAACI,EAAAA,EAAKA,CAAC/E,SAAS,Q,WACd,SAACgF,SAAAA,C,SAAO,0DAA8D,6DAK5E,SAACpB,EAAAA,EAAGA,CAACe,GAAI,E,UACP,UAACM,EAAAA,EAAIA,CACHvG,MAAOlB,GACP0H,SAAU,CAACC,EAAGC,IAAa3H,GAAe2H,GAC1CC,eAAe,UACfC,UAAU,U,WAEV,SAACC,EAAAA,EAAGA,CAAC7J,MAAM,aACVuG,KAAoB,SAACsD,EAAAA,EAAGA,CAAC7J,MAAM,eAC/B4G,KAAkB,SAACiD,EAAAA,EAAGA,CAAC7J,MAAM,aAC7B4F,KAAc,SAACiE,EAAAA,EAAGA,CAAC7J,MAAM,cAIb,IAAhB8B,KACC,UAACgI,EAAAA,EAAQA,CAAChC,MAAM,c,WACd,UAACI,EAAAA,EAAGA,CAACC,UAAWhH,GAAQL,W,WACtB,UAACoH,EAAAA,EAAGA,C,WACF,SAACkB,EAAAA,EAAUA,CAACX,QAAQ,UAAUN,UAAWhH,GAAQnB,M,SAAO,kBAGxD,SAACoJ,EAAAA,EAAUA,CAACX,QAAQ,K,UACL,QAAZvH,GAAAA,GAAQsC,YAARtC,IAAAA,QAAAA,EAAAA,GAAc6G,cAAe7G,GAAQ+C,SAAS5C,WAGnD,UAAC6G,EAAAA,EAAGA,CAACC,UAAWhH,GAAQH,Y,UACrB+B,KACC,SAACuF,EAAAA,EAAMA,CACLG,QAAQ,WACRvI,MAAO2E,GAAc,UAAY,UACjC6D,QAAS7E,GACTgF,KAAK,Q,SAEJhE,GAAc,wBAA0B,wBAG5CnC,KACC,SAACkG,EAAAA,EAAUA,CAACF,QAAS,IAAMxG,IAAkB,GAAO2G,KAAK,Q,UACvD,SAACC,EAAAA,EAAQA,CAACtI,SAAS,YAGtBqC,KACC,SAAC+F,EAAAA,EAAUA,CAACF,QAAS,IAAMtG,IAAoB,GAAOyG,KAAK,Q,UACzD,SAACE,EAAAA,EAAUA,CAACvI,SAAS,mBAM7B,SAACuJ,EAAAA,EAAiBA,CAChB7I,QAASA,GACT8I,YAAY,EACZC,iBAAiB,EACjB3G,mBAAoBA,QAKzBxB,KAAgBiF,IAAsBR,KACrC,SAACuD,EAAAA,EAAQA,CAAChC,MAAM,iB,UACC,QAAd5G,GAAAA,GAAQ6D,cAAR7D,IAAAA,IAAuB,QAAvBA,GAAAA,GAAgBsF,eAAhBtF,IAAAA,QAAAA,EAAAA,GAAyBuF,MACxB,SAACyD,EAAAA,EAAuBA,CAACC,WAAYjJ,GAAQ6D,OAAOyB,QAAQC,OAE5D,SAAC2C,EAAAA,EAAUA,CAACX,QAAQ,QAAQvI,MAAM,gB,UACnB,QAAZgB,GAAAA,GAAQsC,YAARtC,IAAAA,IAA2B,QAA3BA,GAAAA,GAAcwF,qBAAdxF,IAAAA,QAAAA,EAAAA,GAA6ByF,iBAC5B,sB,UAAE,iDAC+C,KAC/C,SAACyB,EAAAA,GAAIA,CAACC,GAAInH,GAAQsC,KAAKkD,cAAcC,eAAgByD,OAAO,S,SACzDlJ,GAAQsC,KAAKkD,cAAcC,oBAIhC,+DAOT7E,KAAgBkF,IAAoBJ,KACnC,UAACyD,EAAAA,EAAIA,CAACC,WAAS,EAAC/J,QAAS,E,WACvB,SAAC8J,EAAAA,EAAIA,CAACE,MAAI,EAACC,GAAI,G,UACb,SAACV,EAAAA,EAAQA,CAAChC,MAAM,sB,UACd,UAACuC,EAAAA,EAAIA,CAACC,WAAS,EAAC/J,QAAS,E,WACvB,UAAC8J,EAAAA,EAAIA,CAACE,MAAI,EAACC,GAAI,GAAIC,GAAI,E,WACrB,SAACrB,EAAAA,EAAUA,CAACX,QAAQ,QAAQvI,MAAM,gBAAgBwK,cAAY,E,SAAC,gBAG9D5F,IACC,UAACoD,EAAAA,EAAGA,C,WACF,SAACyC,EAAAA,EAAIA,CACH3K,MAAsC,SAA/B8E,GAAoBC,OAAoB,QAAU,YACzD8D,KAAK,QACL+B,MAAO,CACLC,gBAAgD,SAA/B/F,GAAoBC,OAAoB,UAAY,UACrE7E,MAAO,OACPI,aAAc,MAGlB,SAAC8I,EAAAA,EAAUA,CAACX,QAAQ,Q,SACjB3D,GAAoBT,SAAW,6BAIpC,SAAC+E,EAAAA,EAAUA,CAACX,QAAQ,Q,SAAQ,mCAGhC,UAAC4B,EAAAA,EAAIA,CAACE,MAAI,EAACC,GAAI,GAAIC,GAAI,E,WACrB,SAACrB,EAAAA,EAAUA,CAACX,QAAQ,QAAQvI,MAAM,gBAAgBwK,cAAY,E,SAAC,gBAG9DtF,IACC,UAAC8C,EAAAA,EAAGA,C,WACF,SAACyC,EAAAA,EAAIA,CACH3K,MAAsC,SAA/BoF,GAAoBL,OAAoB,QAAU,YACzD8D,KAAK,QACL+B,MAAO,CACLC,gBAAgD,SAA/BzF,GAAoBL,OAAoB,UAAY,UACrE7E,MAAO,OACPI,aAAc,MAGlB,SAAC8I,EAAAA,EAAUA,CAACX,QAAQ,Q,SACjBrD,GAAoBf,SAAW,6BAIpC,SAAC+E,EAAAA,EAAUA,CAACX,QAAQ,Q,SAAQ,yCAOrCpD,GAAgBwB,OAAS,IACxB,SAACwD,EAAAA,EAAIA,CAACE,MAAI,EAACC,GAAI,G,UACb,UAACV,EAAAA,EAAQA,CAAChC,MAAM,uB,WACd,SAACsB,EAAAA,EAAUA,CAACX,QAAQ,QAAQvI,MAAM,gBAAgB4K,WAAS,E,SAAC,iGAG5D,UAACC,EAAAA,EAAKA,CAAClC,KAAK,Q,WACV,SAACmC,EAAAA,EAASA,C,UACR,UAACC,EAAAA,EAAQA,C,WACP,SAACC,EAAAA,EAASA,C,SAAC,UACX,SAACA,EAAAA,EAASA,C,SAAC,sBAGf,SAACC,EAAAA,EAASA,C,SACP9F,GAAgB+F,IAAKC,IACpB,UAACJ,EAAAA,EAAQA,C,WACP,SAACC,EAAAA,EAASA,C,UACR,SAACP,EAAAA,EAAIA,CAAC3K,MAAOqL,EAAKC,KAAMzC,KAAK,QAAQ3I,MAAM,eAE7C,SAACgL,EAAAA,EAASA,C,SAAEhE,GAAamE,EAAKlE,YAJjBkE,EAAKC,oBAenCxJ,KAAgBmF,IAAgBrB,KAC/B,SAAC2F,EAAAA,EAAgBA,CACfpF,UAAWF,GACXK,cAAeF,SAKrB,SAACoF,EAAAA,EAAoBA,CACnBC,KAAMxJ,GACNyJ,QAAS,IAAMxJ,IAAkB,GACjCyJ,UA/ToB,KACxBzJ,IAAkB,GAClBM,GAAe+B,GAAMA,EAAI,GACzB3C,GAASwC,KAAK,CACZC,QAAS,mCACTC,SAAU,UACV3D,QAAS,eA0TPS,UAAWA,IAAa,GACxBC,KAAMA,IAAQ,MAGhB,SAACuK,EAAAA,EAAmBA,CAClBH,KAAMtJ,GACN2F,MAAM,qBACNG,YAAa,qCAAgD,QAAZ/G,GAAAA,GAAQsC,YAARtC,IAAAA,QAAAA,EAAAA,GAAc6G,cAAe7G,GAAQ+C,SAAS5C,uCAC/FiD,SAAS,OACTuH,YAAa3K,GAAQ+C,SAAS5C,KAC9BgB,SAAUA,GACVyJ,UAjUe1I,UACnB,GAAKlC,GAAL,CACAoB,IAAY,GACZ,UACQb,GAAYsK,iBAAiB3K,GAAYC,IAC/Ce,IAAoB,GACpBR,GAASwC,KAAK,CACZC,QAAS,mCACTC,SAAU,UACV3D,QAAS,cAEXY,GAAS,yBACX,CAAE,MAAOiD,GACP,MAAMC,EAAeD,aAAeE,MAAQF,EAAIH,QAAU,yBAC1DzC,GAASwC,KAAK,CACZC,QAAS,iCAAiCI,IAC1CH,SAAU,QACV3D,QAAS,aAEb,CAAE,QACA2B,IAAY,EACd,CApBoB,GAiUhB0J,SAAU,IAAM5J,IAAoB,Q","sources":["webpack://internal.plugin-kuadrant/./src/components/ApiProductDetailPage/ApiProductDetailPage.tsx"],"sourcesContent":["import React, { useState } from \"react\";\nimport { useParams, useNavigate } from \"react-router-dom\";\nimport {\n useApi,\n alertApiRef,\n} from \"@backstage/core-plugin-api\";\nimport { kuadrantApiRef } from '../../api';\nimport { useAsync } from \"react-use\";\nimport {\n Header,\n Page,\n Content,\n Progress,\n ResponseErrorPanel,\n InfoCard,\n Link,\n Breadcrumbs,\n} from \"@backstage/core-components\";\nimport { OpenApiDefinitionWidget } from \"@backstage/plugin-api-docs\";\nimport {\n Box,\n Typography,\n IconButton,\n Tooltip,\n Tabs,\n Tab,\n Button,\n makeStyles,\n Grid,\n Chip,\n Table,\n TableBody,\n TableCell,\n TableHead,\n TableRow,\n} from \"@material-ui/core\";\nimport ArrowBackIcon from \"@material-ui/icons/ArrowBack\";\nimport EditIcon from \"@material-ui/icons/Edit\";\nimport DeleteIcon from \"@material-ui/icons/Delete\";\nimport { Alert } from '@material-ui/lab';\nimport { APIProduct } from \"../../types/api-management\";\nimport { EditAPIProductDialog } from \"../EditAPIProductDialog\";\nimport { ConfirmDeleteDialog } from \"../ConfirmDeleteDialog\";\nimport { ApiProductDetails } from \"../ApiProductDetails\";\nimport { OidcProviderCard } from \"../OidcProviderCard\";\nimport { useKuadrantPermission } from \"../../utils/permissions\";\nimport {\n kuadrantApiProductUpdateAllPermission,\n kuadrantApiProductDeleteAllPermission,\n} from \"../../permissions\";\n\nconst useStyles = makeStyles((theme) => ({\n label: {\n fontWeight: 600,\n color: theme.palette.text.secondary,\n marginBottom: theme.spacing(0.5),\n fontSize: '0.75rem',\n textTransform: 'uppercase',\n },\n actionButtons: {\n display: \"flex\",\n gap: theme.spacing(1),\n alignItems: \"center\",\n },\n cardHeader: {\n display: 'flex',\n justifyContent: 'space-between',\n alignItems: 'flex-start',\n marginBottom: theme.spacing(2),\n },\n cardActions: {\n display: 'flex',\n gap: theme.spacing(1),\n alignItems: 'center',\n },\n}));\n\nexport const ApiProductDetailPage = () => {\n const classes = useStyles();\n const { namespace, name } = useParams<{ namespace: string; name: string }>();\n const navigate = useNavigate();\n const kuadrantApi = useApi(kuadrantApiRef);\n const alertApi = useApi(alertApiRef);\n\n const [selectedTab, setSelectedTab] = useState(0);\n const [editDialogOpen, setEditDialogOpen] = useState(false);\n const [deleteDialogOpen, setDeleteDialogOpen] = useState(false);\n const [deleting, setDeleting] = useState(false);\n const [refreshKey, setRefreshKey] = useState(0);\n\n const { allowed: canUpdateApiProduct } = useKuadrantPermission(kuadrantApiProductUpdateAllPermission);\n const { allowed: canDeleteApiProduct } = useKuadrantPermission(kuadrantApiProductDeleteAllPermission);\n const canPublishApiProduct = canUpdateApiProduct;\n\n const {\n value: product,\n loading,\n error,\n } = useAsync(async () => {\n return await kuadrantApi.getApiProduct(namespace!, name!) as APIProduct;\n }, [namespace, name, kuadrantApi, refreshKey]);\n\n // fetch httproute to get hostnames\n const { value: httpRouteHostnames } = useAsync(async () => {\n if (!product?.spec?.targetRef?.name) {\n return null;\n }\n\n const routeName = product.spec.targetRef.name;\n const route = await kuadrantApi.getHttpRoute(namespace!, routeName)\n\n return route?.spec?.hostnames || null;\n }, [product, namespace, kuadrantApi]);\n\n const handlePublishToggle = async () => {\n if (!product) return;\n const newStatus = product.spec?.publishStatus === \"Published\" ? \"Draft\" : \"Published\";\n const lifecycle = product.metadata.labels?.lifecycle;\n\n // prevent publishing retired APIs\n if (newStatus === \"Published\" && lifecycle === \"retired\") {\n alertApi.post({\n message: `Cannot publish a retired API product. Please change the lifecycle status first.`,\n severity: \"error\",\n display: \"transient\",\n });\n return;\n }\n\n try {\n // @ts-ignore since updating partially the obj\n await kuadrantApi.updateApiProduct(namespace!, name!, { spec: { publishStatus: newStatus } });\n alertApi.post({\n message: `API Product ${newStatus === \"Published\" ? \"published\" : \"unpublished\"} successfully`,\n severity: \"success\",\n display: \"transient\",\n });\n setRefreshKey((k) => k + 1);\n } catch (err) {\n const errorMessage = err instanceof Error ? err.message : \"unknown error occurred\";\n alertApi.post({\n message: `Failed to update publish status. ${errorMessage}`,\n severity: \"error\",\n display: \"transient\",\n });\n }\n };\n\n const handleEditSuccess = () => {\n setEditDialogOpen(false);\n setRefreshKey((k) => k + 1);\n alertApi.post({\n message: \"API Product updated successfully\",\n severity: \"success\",\n display: \"transient\",\n });\n };\n\n const handleDelete = async () => {\n if (!product) return;\n setDeleting(true);\n try {\n await kuadrantApi.deleteApiProduct(namespace!, name!);\n setDeleteDialogOpen(false);\n alertApi.post({\n message: \"API Product deleted successfully\",\n severity: \"success\",\n display: \"transient\",\n });\n navigate(\"/kuadrant/api-products\");\n } catch (err) {\n const errorMessage = err instanceof Error ? err.message : \"unknown error occurred\";\n alertApi.post({\n message: `Failed to delete API product. ${errorMessage}`,\n severity: \"error\",\n display: \"transient\",\n });\n } finally {\n setDeleting(false);\n }\n };\n\n if (loading) {\n return <Progress />;\n }\n\n if (error || !product) {\n return (\n <ResponseErrorPanel error={error || new Error(\"API product not found\")} />\n );\n }\n\n const isPublished = product.spec?.publishStatus === \"Published\";\n\n // get policy conditions from status\n const planPolicyCondition = product.status?.conditions?.find(\n (c) => c.type === \"PlanPolicyDiscovered\"\n );\n const authPolicyCondition = product.status?.conditions?.find(\n (c) => c.type === \"AuthPolicyDiscovered\"\n );\n const discoveredPlans = product.status?.discoveredPlans || [];\n\n const authSchemes = product.status?.discoveredAuthScheme?.authentication || {};\n const schemeObjects = Object.values(authSchemes);\n const hasOIDCTab = schemeObjects.some((scheme: any) =>\n scheme.hasOwnProperty(\"jwt\"),\n );\n // Extract JWT issuer from the first JWT scheme\n const jwtScheme = schemeObjects.find((scheme: any) => scheme.hasOwnProperty(\"jwt\"));\n const jwtIssuer = (jwtScheme as any)?.jwt?.issuerUrl || \"unknown\";\n const jwtTokenEndpoint = product.status?.oidcDiscovery?.tokenEndpoint || \"unknown\";\n\n // compute tab indices\n const hasDefinitionTab = !!(product.status?.openapi?.raw || product.spec?.documentation?.openAPISpecURL);\n const hasPoliciesTab = !!(planPolicyCondition || authPolicyCondition || discoveredPlans.length > 0);\n\n let nextIndex = 1; // Overview is always at index 0\n const definitionTabIndex = hasDefinitionTab ? nextIndex++ : -1;\n const policiesTabIndex = hasPoliciesTab ? nextIndex++ : -1;\n const oidcTabIndex = hasOIDCTab ? nextIndex++ : -1;\n\n const formatLimits = (limits: any): string => {\n if (!limits) return \"No limits\";\n const parts: string[] = [];\n if (limits.daily) parts.push(`${limits.daily}/day`);\n if (limits.weekly) parts.push(`${limits.weekly}/week`);\n if (limits.monthly) parts.push(`${limits.monthly}/month`);\n if (limits.yearly) parts.push(`${limits.yearly}/year`);\n return parts.length > 0 ? parts.join(\", \") : \"No limits\";\n };\n\n return (\n <Page themeId=\"tool\">\n <Header\n title={product.spec?.displayName || product.metadata.name}\n subtitle={product.spec?.description || \"\"}\n >\n <Box className={classes.actionButtons}>\n <Link to=\"/kuadrant/api-products\">\n <Button startIcon={<ArrowBackIcon />}>Back</Button>\n </Link>\n {canPublishApiProduct && (\n <Button\n variant=\"outlined\"\n color={isPublished ? \"default\" : \"primary\"}\n onClick={handlePublishToggle}\n >\n {isPublished ? \"Unpublish API product\" : \"Publish API product\"}\n </Button>\n )}\n {canUpdateApiProduct && (\n <Tooltip title=\"Edit\">\n <IconButton onClick={() => setEditDialogOpen(true)} size=\"small\">\n <EditIcon />\n </IconButton>\n </Tooltip>\n )}\n {canDeleteApiProduct && (\n <Tooltip title=\"Delete\">\n <IconButton onClick={() => setDeleteDialogOpen(true)} size=\"small\">\n <DeleteIcon />\n </IconButton>\n </Tooltip>\n )}\n </Box>\n </Header>\n <Content>\n <Box mb={2}>\n <Breadcrumbs aria-label=\"breadcrumb\">\n <Link to=\"/kuadrant/api-products\">API Products</Link>\n <Typography>{product.spec?.displayName || product.metadata.name}</Typography>\n </Breadcrumbs>\n </Box>\n\n\n {product.metadata.labels?.lifecycle === 'deprecated' && (\n <Box mb={2}>\n <Alert severity=\"warning\">\n <strong>This API is deprecated.</strong> Please contact your administrator for more details.\n </Alert>\n </Box>\n )}\n\n {product.metadata.labels?.lifecycle === 'retired' && (\n <Box mb={2}>\n <Alert severity=\"error\">\n <strong>This API has been retired and is no longer available.</strong> Please contact your administrator for alternatives.\n </Alert>\n </Box>\n )}\n\n <Box mb={2}>\n <Tabs\n value={selectedTab}\n onChange={(_, newValue) => setSelectedTab(newValue)}\n indicatorColor=\"primary\"\n textColor=\"primary\"\n >\n <Tab label=\"Overview\" />\n {hasDefinitionTab && <Tab label=\"Definition\" />}\n {hasPoliciesTab && <Tab label=\"Policies\" />}\n {hasOIDCTab && <Tab label=\"OIDC\" />}\n </Tabs>\n </Box>\n\n {selectedTab === 0 && (\n <InfoCard title=\"API Product\">\n <Box className={classes.cardHeader}>\n <Box>\n <Typography variant=\"caption\" className={classes.label}>\n Product Name\n </Typography>\n <Typography variant=\"h6\">\n {product.spec?.displayName || product.metadata.name}\n </Typography>\n </Box>\n <Box className={classes.cardActions}>\n {canPublishApiProduct && (\n <Button\n variant=\"outlined\"\n color={isPublished ? \"default\" : \"primary\"}\n onClick={handlePublishToggle}\n size=\"small\"\n >\n {isPublished ? \"Unpublish API product\" : \"Publish API product\"}\n </Button>\n )}\n {canUpdateApiProduct && (\n <IconButton onClick={() => setEditDialogOpen(true)} size=\"small\">\n <EditIcon fontSize=\"small\" />\n </IconButton>\n )}\n {canDeleteApiProduct && (\n <IconButton onClick={() => setDeleteDialogOpen(true)} size=\"small\">\n <DeleteIcon fontSize=\"small\" />\n </IconButton>\n )}\n </Box>\n </Box>\n\n <ApiProductDetails\n product={product}\n showStatus={true}\n showCatalogLink={true}\n httpRouteHostnames={httpRouteHostnames}\n />\n </InfoCard>\n )}\n\n {selectedTab === definitionTabIndex && hasDefinitionTab && (\n <InfoCard title=\"API Definition\">\n {product.status?.openapi?.raw ? (\n <OpenApiDefinitionWidget definition={product.status.openapi.raw} />\n ) : (\n <Typography variant=\"body2\" color=\"textSecondary\">\n {product.spec?.documentation?.openAPISpecURL ? (\n <>\n OpenAPI specification not yet synced. View at:{\" \"}\n <Link to={product.spec.documentation.openAPISpecURL} target=\"_blank\">\n {product.spec.documentation.openAPISpecURL}\n </Link>\n </>\n ) : (\n \"No OpenAPI specification available for this API product.\"\n )}\n </Typography>\n )}\n </InfoCard>\n )}\n\n {selectedTab === policiesTabIndex && hasPoliciesTab && (\n <Grid container spacing={3}>\n <Grid item xs={12}>\n <InfoCard title=\"Discovered Policies\">\n <Grid container spacing={2}>\n <Grid item xs={12} md={6}>\n <Typography variant=\"body2\" color=\"textSecondary\" gutterBottom>\n Plan Policy\n </Typography>\n {planPolicyCondition ? (\n <Box>\n <Chip\n label={planPolicyCondition.status === \"True\" ? \"Found\" : \"Not Found\"}\n size=\"small\"\n style={{\n backgroundColor: planPolicyCondition.status === \"True\" ? \"#4caf50\" : \"#ff9800\",\n color: \"#fff\",\n marginBottom: 8,\n }}\n />\n <Typography variant=\"body2\">\n {planPolicyCondition.message || \"No details available\"}\n </Typography>\n </Box>\n ) : (\n <Typography variant=\"body2\">No plan policy information</Typography>\n )}\n </Grid>\n <Grid item xs={12} md={6}>\n <Typography variant=\"body2\" color=\"textSecondary\" gutterBottom>\n Auth Policy\n </Typography>\n {authPolicyCondition ? (\n <Box>\n <Chip\n label={authPolicyCondition.status === \"True\" ? \"Found\" : \"Not Found\"}\n size=\"small\"\n style={{\n backgroundColor: authPolicyCondition.status === \"True\" ? \"#4caf50\" : \"#ff9800\",\n color: \"#fff\",\n marginBottom: 8,\n }}\n />\n <Typography variant=\"body2\">\n {authPolicyCondition.message || \"No details available\"}\n </Typography>\n </Box>\n ) : (\n <Typography variant=\"body2\">No auth policy information</Typography>\n )}\n </Grid>\n </Grid>\n </InfoCard>\n </Grid>\n\n {discoveredPlans.length > 0 && (\n <Grid item xs={12}>\n <InfoCard title=\"Effective Plan Tiers\">\n <Typography variant=\"body2\" color=\"textSecondary\" paragraph>\n These tiers are computed from all attached PlanPolicies (including gateway-level policies).\n </Typography>\n <Table size=\"small\">\n <TableHead>\n <TableRow>\n <TableCell>Tier</TableCell>\n <TableCell>Rate Limits</TableCell>\n </TableRow>\n </TableHead>\n <TableBody>\n {discoveredPlans.map((plan) => (\n <TableRow key={plan.tier}>\n <TableCell>\n <Chip label={plan.tier} size=\"small\" color=\"primary\" />\n </TableCell>\n <TableCell>{formatLimits(plan.limits)}</TableCell>\n </TableRow>\n ))}\n </TableBody>\n </Table>\n </InfoCard>\n </Grid>\n )}\n </Grid>\n )}\n\n {selectedTab === oidcTabIndex && hasOIDCTab && (\n <OidcProviderCard\n issuerUrl={jwtIssuer}\n tokenEndpoint={jwtTokenEndpoint}\n />\n )}\n </Content>\n\n <EditAPIProductDialog\n open={editDialogOpen}\n onClose={() => setEditDialogOpen(false)}\n onSuccess={handleEditSuccess}\n namespace={namespace || \"\"}\n name={name || \"\"}\n />\n\n <ConfirmDeleteDialog\n open={deleteDialogOpen}\n title=\"Delete API Product\"\n description={`Are you sure you want to delete \"${product.spec?.displayName || product.metadata.name}\"? This action cannot be undone.`}\n severity=\"high\"\n confirmText={product.metadata.name}\n deleting={deleting}\n onConfirm={handleDelete}\n onCancel={() => setDeleteDialogOpen(false)}\n />\n </Page>\n );\n};\n"],"names":["useStyles","makeStyles","theme","label","fontWeight","color","palette","text","secondary","marginBottom","spacing","fontSize","textTransform","actionButtons","display","gap","alignItems","cardHeader","justifyContent","cardActions","ApiProductDetailPage","product","classes","namespace","name","useParams","navigate","useNavigate","kuadrantApi","useApi","kuadrantApiRef","alertApi","alertApiRef","selectedTab","setSelectedTab","useState","editDialogOpen","setEditDialogOpen","deleteDialogOpen","setDeleteDialogOpen","deleting","setDeleting","refreshKey","setRefreshKey","allowed","canUpdateApiProduct","useKuadrantPermission","kuadrantApiProductUpdateAllPermission","canDeleteApiProduct","kuadrantApiProductDeleteAllPermission","canPublishApiProduct","value","loading","error","useAsync","async","getApiProduct","httpRouteHostnames","route","spec","targetRef","routeName","getHttpRoute","hostnames","handlePublishToggle","newStatus","publishStatus","lifecycle","metadata","labels","updateApiProduct","post","message","severity","k","err","errorMessage","Error","Progress","ResponseErrorPanel","isPublished","planPolicyCondition","status","conditions","find","c","type","authPolicyCondition","discoveredPlans","authSchemes","discoveredAuthScheme","authentication","schemeObjects","Object","values","hasOIDCTab","some","scheme","hasOwnProperty","jwtScheme","jwtIssuer","jwt","issuerUrl","jwtTokenEndpoint","oidcDiscovery","tokenEndpoint","hasDefinitionTab","openapi","raw","documentation","openAPISpecURL","hasPoliciesTab","length","nextIndex","definitionTabIndex","policiesTabIndex","oidcTabIndex","formatLimits","limits","parts","daily","push","weekly","monthly","yearly","join","Page","themeId","Header","title","displayName","subtitle","description","Box","className","Link","to","Button","startIcon","ArrowBackIcon","variant","onClick","Tooltip","IconButton","size","EditIcon","DeleteIcon","Content","mb","Breadcrumbs","aria-label","Typography","Alert","strong","Tabs","onChange","_","newValue","indicatorColor","textColor","Tab","InfoCard","ApiProductDetails","showStatus","showCatalogLink","OpenApiDefinitionWidget","definition","target","Grid","container","item","xs","md","gutterBottom","Chip","style","backgroundColor","paragraph","Table","TableHead","TableRow","TableCell","TableBody","map","plan","tier","OidcProviderCard","EditAPIProductDialog","open","onClose","onSuccess","ConfirmDeleteDialog","confirmText","onConfirm","deleteApiProduct","onCancel"],"sourceRoot":""}
@@ -0,0 +1,2 @@
1
+ "use strict";(self.webpackChunkinternal_plugin_kuadrant=self.webpackChunkinternal_plugin_kuadrant||[]).push([[8789],{24170:(e,t,a)=>{a.d(t,{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 t=new URL(e);return["http:","https:"].includes(t.protocol)?null:"Must be a valid HTTP or HTTPS URL"}catch{return"Must be a valid URL"}}},26997:(e,t,a)=>{a.d(t,{C:()=>H});var i=a(31085),n=a(95478),l=a(76891),r=a(61477),s=a(46805),o=a(10394),d=a(72501),c=a(42899),u=a(16249),m=a(34839),p=a(29365),h=a(67720),x=a(71677),v=a(26343),A=a(95061),f=a(29635),j=a(86901),g=a(30285),b=a(93453),y=a(64947),k=a(78467),P=a(58837),C=a(89031),S=a(18466),T=a(22097),R=a(64047),I=a(84441),N=a(91638),w=a(86687),L=a(86197),B=a(24170),z=a(97859);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)}})),H=({open:e,onClose:t,onSuccess:a,namespace:P,name:H})=>{var D,U;const q=W(),F=(0,T.useApi)(R.s),[M,E]=(0,n.useState)(!1),[$,K]=(0,n.useState)(""),[_,V]=(0,n.useState)(""),[G,J]=(0,n.useState)("v1"),[O,Q]=(0,n.useState)("Draft"),[X,Y]=(0,n.useState)("production"),[Z,ee]=(0,n.useState)("manual"),[te,ae]=(0,n.useState)([]),[ie,ne]=(0,n.useState)(null),[le,re]=(0,n.useState)(""),[se,oe]=(0,n.useState)(""),[de,ce]=(0,n.useState)(""),[ue,me]=(0,n.useState)(""),[pe,he]=(0,n.useState)(""),[xe,ve]=(0,n.useState)(null),[Ae,fe]=(0,n.useState)(null),[je,ge]=(0,n.useState)(""),[be,ye]=(0,n.useState)(!1),[ke,Pe]=(0,n.useState)(null);(0,n.useEffect)(()=>{e&&P&&H&&(E(!0),ge(""),F.getApiProduct(P,H).then(e=>{var t,a,i,n,l,r,s,o,d,c;K(e.spec.displayName||""),V(e.spec.description||""),J(e.spec.version||"v1"),Q(e.spec.publishStatus||"Draft"),Y((null===(t=e.metadata.labels)||void 0===t?void 0:t.lifecycle)||"production"),ee(e.spec.approvalMode||"manual"),ae(e.spec.tags||[]),ne(e.spec.targetRef||null),oe((null===(a=e.spec.contact)||void 0===a?void 0:a.email)||""),ce((null===(i=e.spec.contact)||void 0===i?void 0:i.team)||""),me((null===(n=e.spec.documentation)||void 0===n?void 0:n.docsURL)||""),he((null===(l=e.spec.documentation)||void 0===l?void 0:l.openAPISpecURL)||"");const u=(null===(s=e.status)||void 0===s||null===(r=s.conditions)||void 0===r?void 0:r.find(e=>"PlanPolicyDiscovered"===e.type))||null;ve({statusCondition:u,discoveredPlans:(null===(o=e.status)||void 0===o?void 0:o.discoveredPlans)||null});const m=(null===(c=e.status)||void 0===c||null===(d=c.conditions)||void 0===d?void 0:d.find(e=>"AuthPolicyDiscovered"===e.type))||null;var p;const h="True"===(null==m?void 0:m.status)?(e=>{const t=e.split(" ");return t.length>=3?t[2]:""})(null!==(p=m.message)&&void 0!==p?p:""):"";fe({statusCondition:m,namespacedName:{name:h,namespace:e.metadata.namespace}}),Pe(null),E(!1)}).catch(e=>{ge(e.message||"Failed to load API product"),E(!1)}))},[e,P,H,F]),(0,n.useEffect)(()=>{e&&Pe(null)},[e]);const{value:Ce,error:Se}=(0,N.A)(async()=>await F.getRateLimitPolicies(),[F,e]),Te=ie?(0,z.d)(null==Ce?void 0:Ce.items,P,ie.name):null,Re=null==Te||null===(U=Te.status)||void 0===U||null===(D=U.conditions)||void 0===D?void 0:D.find(e=>"Accepted"===e.type),Ie={namespacedName:{namespace:null==Te?void 0:Te.metadata.namespace,name:null==Te?void 0:Te.metadata.name},statusCondition:Re},Ne=()=>{le.trim()&&!te.includes(le.trim())&&(ae([...te,le.trim()]),re(""))};return(0,i.jsxs)(l.A,{open:e,onClose:t,maxWidth:"md",fullWidth:!0,children:[(0,i.jsx)(r.A,{children:"Edit API Product"}),(0,i.jsxs)(s.A,{children:[je&&(0,i.jsx)(I.A,{severity:"error",style:{marginBottom:16},children:je}),Se&&(0,i.jsxs)(I.A,{severity:"error",style:{marginBottom:16},children:[(0,i.jsx)("strong",{children:"Failed to load RateLimitPolicies:"})," ",Se.message]}),M?(0,i.jsx)(w.k,{}):(0,i.jsxs)(i.Fragment,{children:[(0,i.jsx)(o.A,{className:q.sectionHeader,children:(0,i.jsx)(d.A,{variant:"subtitle1",children:(0,i.jsx)("strong",{children:"API product info"})})}),(0,i.jsxs)(c.A,{container:!0,spacing:2,children:[(0,i.jsx)(c.A,{item:!0,xs:6,children:(0,i.jsx)(u.A,{fullWidth:!0,label:"API product name",value:$,onChange:e=>K(e.target.value),placeholder:"My API",helperText:"Give a unique name for your API product",margin:"normal",required:!0,disabled:be,InputLabelProps:{classes:{asterisk:q.asterisk}}})}),(0,i.jsx)(c.A,{item:!0,xs:6,children:(0,i.jsx)(u.A,{fullWidth:!0,label:"Resource name",value:H,disabled:!0,helperText:"Kubernetes resource name (immutable)",margin:"normal"})}),(0,i.jsx)(c.A,{item:!0,xs:6,children:(0,i.jsx)(u.A,{fullWidth:!0,label:"Version",value:G,onChange:e=>J(e.target.value),placeholder:"v1",helperText:"Give a version to your API product",margin:"normal",disabled:be})}),(0,i.jsx)(c.A,{item:!0,xs:6,children:(0,i.jsx)(u.A,{fullWidth:!0,label:"Tag",value:le,onChange:e=>re(e.target.value),onKeyPress:e=>{"Enter"===e.key&&(e.preventDefault(),Ne())},placeholder:"Add tag",helperText:"Add a tag to your API product",margin:"normal",disabled:be,InputProps:{endAdornment:le?(0,i.jsx)(m.A,{position:"end",children:(0,i.jsx)(p.A,{size:"small",onClick:Ne,disabled:be,children:(0,i.jsx)(S.A,{fontSize:"small"})})}):void 0}})}),te.length>0&&(0,i.jsx)(c.A,{item:!0,xs:12,children:(0,i.jsx)(o.A,{display:"flex",flexWrap:"wrap",children:te.map(e=>(0,i.jsx)(h.A,{label:e,onDelete:be?void 0:()=>{return t=e,void ae(te.filter(e=>e!==t));var t},size:"small",className:q.tagChip,disabled:be},e))})}),(0,i.jsx)(c.A,{item:!0,xs:12,children:(0,i.jsx)(u.A,{fullWidth:!0,label:"Description",value:_,onChange:e=>V(e.target.value),placeholder:"API description",margin:"normal",multiline:!0,rows:2,required:!0,disabled:be,InputLabelProps:{classes:{asterisk:q.asterisk}}})})]}),(0,i.jsxs)(o.A,{className:q.sectionHeader,children:[(0,i.jsx)(d.A,{variant:"subtitle1",children:(0,i.jsx)("strong",{children:"Associated route"})}),(0,i.jsx)(x.Ay,{title:"The HTTPRoute this API product is associated with",children:(0,i.jsx)(C.A,{className:q.infoIcon})})]}),(0,i.jsxs)(c.A,{container:!0,spacing:2,children:[(0,i.jsx)(c.A,{item:!0,xs:12,children:(0,i.jsx)(u.A,{fullWidth:!0,label:"OpenAPI Spec URL",value:pe,onChange:e=>{return t=e.target.value,he(t),void Pe((0,B.q)(t));var t},placeholder:"https://api.example.com/openapi.json",helperText:ke||"Enter the full path to your API spec file",error:!!ke,margin:"normal",disabled:be})}),(0,i.jsx)(c.A,{item:!0,xs:12,children:(0,i.jsx)(u.A,{fullWidth:!0,label:"Documentation URL",value:ue,onChange:e=>me(e.target.value),placeholder:"https://docs.example.com/api",helperText:"Link to external documentation for this API",margin:"normal",disabled:be})}),ie&&(0,i.jsx)(c.A,{item:!0,xs:12,children:(0,i.jsx)(u.A,{fullWidth:!0,label:"HTTPRoute",value:`${ie.namespace||P}/${ie.name}`,disabled:!0,helperText:"Target HTTPRoute (immutable)",margin:"normal"})})]}),ie&&(0,i.jsxs)(i.Fragment,{children:[(0,i.jsxs)(o.A,{className:q.sectionHeader,children:[(0,i.jsx)(d.A,{variant:"subtitle1",children:(0,i.jsx)("strong",{children:"HTTPRoute policies"})}),(0,i.jsx)(x.Ay,{title:"Shows the associated policies and rate limit tiers for the HTTPRoute",children:(0,i.jsx)(C.A,{className:q.infoIcon})})]}),(0,i.jsx)(L.K,{planPolicy:xe,authPolicy:Ae,rateLimitPolicy:Ie,includeTopMargin:!1})]}),(0,i.jsxs)(o.A,{className:q.sectionHeader,children:[(0,i.jsx)(d.A,{variant:"subtitle1",children:(0,i.jsx)("strong",{children:"Lifecycle and Visibility"})}),(0,i.jsx)(x.Ay,{title:"Control the lifecycle state and catalog visibility of this API product",children:(0,i.jsx)(C.A,{className:q.infoIcon})})]}),(0,i.jsxs)(c.A,{container:!0,spacing:2,children:[(0,i.jsx)(c.A,{item:!0,xs:6,children:(0,i.jsxs)(u.A,{fullWidth:!0,select:!0,label:"Lifecycle",value:X,onChange:e=>Y(e.target.value),margin:"normal",helperText:"API lifecycle state",disabled:be,children:[(0,i.jsx)(v.A,{value:"experimental",children:"Experimental"}),(0,i.jsx)(v.A,{value:"production",children:"Production"}),(0,i.jsx)(v.A,{value:"deprecated",children:"Deprecated"}),(0,i.jsx)(v.A,{value:"retired",children:"Retired"})]})}),(0,i.jsx)(c.A,{item:!0,xs:6,children:(0,i.jsxs)(u.A,{fullWidth:!0,select:!0,label:"Publish Status",value:O,onChange:e=>Q(e.target.value),margin:"normal",helperText:"Controls catalog visibility",disabled:be,children:[(0,i.jsx)(v.A,{value:"Draft",children:"Draft"}),(0,i.jsx)(v.A,{value:"Published",children:"Published"})]})})]}),(0,i.jsxs)(o.A,{className:q.sectionHeader,children:[(0,i.jsx)(d.A,{variant:"subtitle1",children:(0,i.jsx)("strong",{children:"API Key approval"})}),(0,i.jsx)(x.Ay,{title:"Choose how API key requests are handled for this product",children:(0,i.jsx)(C.A,{className:q.infoIcon})})]}),(0,i.jsx)(A.A,{component:"fieldset",disabled:be,children:(0,i.jsxs)(f.A,{row:!0,value:Z,onChange:e=>ee(e.target.value),children:[(0,i.jsx)(j.A,{value:"manual",control:(0,i.jsx)(g.A,{color:"primary"}),label:(0,i.jsxs)(o.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)(g.A,{color:"primary"}),label:(0,i.jsxs)(o.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)(y.A,{onClick:t,disabled:be,children:"Cancel"}),(0,i.jsx)(y.A,{onClick:async()=>{if(ge(""),ye(!0),"retired"===X&&"Published"===O)return ge("Cannot publish a retired API product. Please set publish status to Draft."),void ye(!1);try{const e={metadata:{labels:{lifecycle:X}},spec:{displayName:$,description:_,version:G,publishStatus:O,approvalMode:Z,tags:te,targetRef:ie,...se||de?{contact:{...se&&{email:se},...de&&{team:de}}}:{},...ue||pe?{documentation:{...ue&&{docsURL:ue},...pe&&{openAPISpecURL:pe}}}:{}}};await F.updateApiProduct(P,H,e),a(),t()}catch(e){ge(e instanceof Error?e.message:String(e))}finally{ye(!1)}},color:"primary",variant:"contained",disabled:be||M||!$||!_||!!ke,startIcon:be?(0,i.jsx)(k.A,{size:16,color:"inherit"}):void 0,children:be?"Saving...":"Save"})]})]})}},34955:(e,t,a)=>{a.d(t,{Al:()=>r,EM:()=>d,FL:()=>l,J:()=>n,KV:()=>A,R_:()=>c,U3:()=>s,dp:()=>m,jH:()=>v,q0:()=>p,uL:()=>x,v_:()=>o,vs:()=>u,z4:()=>h});var i=a(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"}}),l=((0,i.i)({name:"kuadrant.authpolicy.list",attributes:{action:"read"}}),(0,i.i)({name:"kuadrant.ratelimitpolicy.list",attributes:{action:"read"}}),(0,i.i)({name:"kuadrant.apiproduct.create",attributes:{action:"create"}})),r=((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"}}),m=(0,i.i)({name:"kuadrant.apikey.create",attributes:{action:"create"},resourceType:"apiproduct"}),p=((0,i.i)({name:"kuadrant.apikey.read.own",attributes:{action:"read"}}),(0,i.i)({name:"kuadrant.apikey.read.all",attributes:{action:"read"}}),(0,i.i)({name:"kuadrant.apikey.update.own",attributes:{action:"update"}})),h=(0,i.i)({name:"kuadrant.apikey.update.all",attributes:{action:"update"}}),x=(0,i.i)({name:"kuadrant.apikey.delete.own",attributes:{action:"delete"}}),v=(0,i.i)({name:"kuadrant.apikey.delete.all",attributes:{action:"delete"}}),A=(0,i.i)({name:"kuadrant.apikey.approve",attributes:{action:"update"}})},46205:(e,t,a)=>{a.d(t,{W:()=>l,l:()=>n});var i=a(24217);function n(e,t){const a="resourceType"in e?{permission:e,resourceRef:t}:{permission:e},n=(0,i.J)(a);return{allowed:n.allowed,loading:n.loading,error:n.error}}function l(e,t,a,i){return!!i||!(!a||e!==t)}},46299:(e,t,a)=>{a.d(t,{ee:()=>l,g9:()=>i,uU:()=>n});const i=e=>{const t={border:"none"};switch(e){case"Approved":return{...t,backgroundColor:"#1976d2",color:"#fff"};case"Rejected":return{...t,backgroundColor:"#d32f2f",color:"#fff"};default:return{...t,backgroundColor:"#9c27b0",color:"#fff"}}},n=e=>{const t={border:"none"};switch(e){case"Approved":return{...t,backgroundColor:"#2e7d32",color:"#fff"};case"Rejected":return{...t,backgroundColor:"#d32f2f",color:"#fff"};default:return{...t,backgroundColor:"#ed6c02",color:"#fff"}}},l=e=>{switch(e){case"production":return{backgroundColor:"#1976d2",color:"#fff"};case"experimental":return{backgroundColor:"#9c27b0",color:"#fff"};case"deprecated":return{backgroundColor:"#ff9800",color:"#fff"};case"retired":return{backgroundColor:"#d32f2f",color:"#fff"};default:return{}}}},63221:(e,t,a)=>{a.d(t,{K:()=>v});var i=a(31085),n=a(95478),l=a(76891),r=a(61477),s=a(10394),o=a(46805),d=a(59461),c=a(72501),u=a(16249),m=a(93453),p=a(64947),h=a(78467),x=a(77225);const v=({open:e,title:t,description:a,confirmText:v,severity:A="normal",deleting:f=!1,onConfirm:j,onCancel:g})=>{const[b,y]=(0,n.useState)("");(0,n.useEffect)(()=>{e||y("")},[e]);const k="high"===A&&v,P=!k||b===v;return(0,i.jsxs)(l.A,{open:e,onClose:f?void 0:g,maxWidth:"sm",fullWidth:!0,children:[(0,i.jsxs)(r.A,{children:["high"===A&&(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:t})]}),"high"!==A&&t]}),(0,i.jsxs)(o.A,{children:[(0,i.jsx)(d.A,{style:{whiteSpace:"pre-line"},children:a}),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:v})," to confirm:"]}),(0,i.jsx)(u.A,{fullWidth:!0,variant:"outlined",size:"small",value:b,onChange:e=>y(e.target.value),disabled:f,autoFocus:!0,placeholder:v})]})]}),(0,i.jsxs)(m.A,{children:[(0,i.jsx)(p.A,{onClick:g,disabled:f,children:"Cancel"}),(0,i.jsx)(p.A,{onClick:()=>{P&&j()},color:"secondary",variant:"contained",disabled:f||!P,startIcon:f?(0,i.jsx)(h.A,{size:16,color:"inherit"}):void 0,children:f?"Deleting...":"Delete"})]})]})}},64047:(e,t,a)=>{a.d(t,{s:()=>n});var i=a(22097);a(54209);const n=(0,i.createApiRef)({id:"plugin.kuadrant.service"})},86197:(e,t,a)=>{a.d(t,{K:()=>d});var i=a(31085),n=(a(95478),a(10394)),l=a(42899),r=a(72501),s=a(67720),o=a(54917);const d=({planPolicy:e,authPolicy:t,rateLimitPolicy:a,includeTopMargin:d=!0})=>{var c,u,m,p;const h=(0,o.A)();return(0,i.jsx)(n.A,{mt:d?1:0,p:2,bgcolor:h.palette.background.default,borderRadius:1,border:`1px solid ${h.palette.divider}`,children:(0,i.jsxs)(l.A,{container:!0,spacing:2,children:[(null==e?void 0:e.statusCondition)&&"True"===e.statusCondition.status||!(null==a?void 0:a.statusCondition)?(0,i.jsxs)(l.A,{item:!0,xs:12,md:6,children:[(0,i.jsx)(r.A,{variant:"body2",color:"textSecondary",gutterBottom:!0,children:"Plan Policy"}),(null==e?void 0:e.statusCondition)?(0,i.jsxs)(n.A,{children:[(0,i.jsx)(s.A,{label:e.statusCondition.reason,size:"small",style:{backgroundColor:"True"===e.statusCondition.status?"#4caf50":"#ff9800",color:"#fff",marginBottom:8}}),e.discoveredPlans&&e.discoveredPlans.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 PlanPolicy Tiers:"}),(0,i.jsx)(n.A,{display:"flex",flexWrap:"wrap",mt:1,style:{gap:8},children:e.discoveredPlans.map((e,t)=>{var a,n,l;const r=(null===(a=e.limits)||void 0===a?void 0:a.daily)?`${e.limits.daily}/day`:(null===(n=e.limits)||void 0===n?void 0:n.monthly)?`${e.limits.monthly}/month`:(null===(l=e.limits)||void 0===l?void 0:l.yearly)?`${e.limits.yearly}/year`:"No limit";return(0,i.jsx)(s.A,{label:`${e.tier}: ${r}`,size:"small",variant:"outlined",color:"primary"},t)})})]})]}):(0,i.jsx)(s.A,{label:"NotFound",size:"small",style:{backgroundColor:"#ff9800",color:"#fff",marginBottom:8}})]}):(0,i.jsxs)(l.A,{item:!0,xs:12,md:6,children:[(0,i.jsx)(r.A,{variant:"body2",color:"textSecondary",gutterBottom:!0,children:"RateLimitPolicy"}),(null==a?void 0:a.statusCondition)?(0,i.jsxs)(n.A,{children:[(0,i.jsx)(s.A,{label:a.statusCondition.reason,size:"small",style:{backgroundColor:"True"===a.statusCondition.status?"#4caf50":"#ff9800",color:"#fff",marginBottom:8}}),"True"===a.statusCondition.status&&(0,i.jsxs)(l.A,{container:!0,spacing:2,children:[(0,i.jsx)(l.A,{item:!0,xs:12,md:6,children:(0,i.jsxs)(n.A,{children:[(0,i.jsx)(r.A,{variant:"body2",color:"textSecondary",gutterBottom:!0,children:"Resource Name"}),(0,i.jsx)(r.A,{variant:"body2",children:(null===(c=a.namespacedName)||void 0===c?void 0:c.name)||"No RateLimit Policy name available"})]})}),(0,i.jsx)(l.A,{item:!0,xs:12,md:6,children:(0,i.jsxs)(n.A,{children:[(0,i.jsx)(r.A,{variant:"body2",color:"textSecondary",gutterBottom:!0,children:"Resource Namespace"}),(0,i.jsx)(r.A,{variant:"body2",children:(null===(u=a.namespacedName)||void 0===u?void 0:u.namespace)||"No RateLimit Policy namespace available"})]})})]})]}):(0,i.jsx)(s.A,{label:"NotFound",size:"small",style:{backgroundColor:"#ff9800",color:"#fff",marginBottom:8}})]}),(0,i.jsxs)(l.A,{item:!0,xs:12,md:6,children:[(0,i.jsx)(r.A,{variant:"body2",color:"textSecondary",gutterBottom:!0,children:"Auth Policy"}),(null==t?void 0:t.statusCondition)?(0,i.jsxs)(n.A,{children:[(0,i.jsx)(s.A,{label:t.statusCondition.reason,size:"small",style:{backgroundColor:"True"===t.statusCondition.status?"#4caf50":"#ff9800",color:"#fff",marginBottom:8}}),"True"===t.statusCondition.status&&(0,i.jsxs)(l.A,{container:!0,spacing:2,children:[(0,i.jsx)(l.A,{item:!0,xs:12,md:6,children:(0,i.jsxs)(n.A,{children:[(0,i.jsx)(r.A,{variant:"body2",color:"textSecondary",gutterBottom:!0,children:"Resource Name"}),(0,i.jsx)(r.A,{variant:"body2",children:(null===(m=t.namespacedName)||void 0===m?void 0:m.name)||"No Auth Policy name available"})]})}),(0,i.jsx)(l.A,{item:!0,xs:12,md:6,children:(0,i.jsxs)(n.A,{children:[(0,i.jsx)(r.A,{variant:"body2",color:"textSecondary",gutterBottom:!0,children:"Resource Namespace"}),(0,i.jsx)(r.A,{variant:"body2",children:(null===(p=t.namespacedName)||void 0===p?void 0:p.namespace)||"No Auth Policy namespace available"})]})})]})]}):(0,i.jsx)(s.A,{label:"NotFound",size:"small",style:{backgroundColor:"#ff9800",color:"#fff",marginBottom:8}})]})]})})}},97859:(e,t,a)=>{a.d(t,{d:()=>i});const i=(e,t,a)=>e&&e.find(e=>{const i=e.spec.targetRef;var n;const l=null!==(n=null==i?void 0:i.namespace)&&void 0!==n?n:e.metadata.namespace;return"HTTPRoute"===(null==i?void 0:i.kind)&&(null==i?void 0:i.name)===a&&l===t})||null}}]);
2
+ //# sourceMappingURL=8789.2f437443.chunk.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"static/8789.2f437443.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,uYCIF,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,EAAGC,OAAMC,UAASC,YAAWC,YAAWC,W,IA8FjCC,EAAAA,EA7FzC,MAAMC,EAAU1B,IACV2B,GAAcC,EAAAA,EAAAA,QAAOC,EAAAA,IACpBC,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,EAAWC,IAAgBV,EAAAA,EAAAA,UAAoB,eAC/CW,EAAcC,KAAmBZ,EAAAA,EAAAA,UAAiC,WAClEa,GAAMC,KAAWd,EAAAA,EAAAA,UAAmB,KACpCe,GAAWC,KAAgBhB,EAAAA,EAAAA,UAAc,OACzCiB,GAAUC,KAAelB,EAAAA,EAAAA,UAAS,KAClCmB,GAAcC,KAAmBpB,EAAAA,EAAAA,UAAS,KAC1CqB,GAAaC,KAAkBtB,EAAAA,EAAAA,UAAS,KACxCuB,GAASC,KAAcxB,EAAAA,EAAAA,UAAS,KAChCyB,GAAaC,KAAkB1B,EAAAA,EAAAA,UAAS,KACxC2B,GAAiBC,KAAwB5B,EAAAA,EAAAA,UAAmC,OAC5E6B,GAAiBC,KAAsB9B,EAAAA,EAAAA,UAAmC,OAC1E+B,GAAOC,KAAYhC,EAAAA,EAAAA,UAAS,KAC5BiC,GAAQC,KAAalC,EAAAA,EAAAA,WAAS,IAC9BmC,GAAkBC,KAAuBpC,EAAAA,EAAAA,UAAwB,OAGxEqC,EAAAA,EAAAA,WAAU,KACJjD,GAAQG,GAAaC,IACvBO,GAAW,GACXiC,GAAS,IACTrC,EAAY2C,cAAc/C,EAAWC,GAClC+C,KAAKC,I,IAKSA,EAIGA,EACDA,EACJA,EACIA,EACaA,EAAAA,EAKTA,EAESA,EAAAA,EAnB5BtC,EAAesC,EAAKC,KAAKxC,aAAe,IACxCG,EAAeoC,EAAKC,KAAKtC,aAAe,IACxCG,EAAWkC,EAAKC,KAAKpC,SAAW,MAChCG,EAAiBgC,EAAKC,KAAKlC,eAAiB,SAC5CG,GAAiC,QAApB8B,EAAAA,EAAKE,SAASC,cAAdH,IAAAA,OAAAA,EAAAA,EAAsB/B,YAA0B,cAC7DG,GAAgB4B,EAAKC,KAAK9B,cAAgB,UAC1CG,GAAQ0B,EAAKC,KAAK5B,MAAQ,IAC1BG,GAAawB,EAAKC,KAAK1B,WAAa,MACpCK,IAAiC,QAAjBoB,EAAAA,EAAKC,KAAKG,eAAVJ,IAAAA,OAAAA,EAAAA,EAAmBK,QAAS,IAC5CvB,IAAgC,QAAjBkB,EAAAA,EAAKC,KAAKG,eAAVJ,IAAAA,OAAAA,EAAAA,EAAmBM,OAAQ,IAC1CtB,IAAkC,QAAvBgB,EAAAA,EAAKC,KAAKM,qBAAVP,IAAAA,OAAAA,EAAAA,EAAyBjB,UAAW,IAC/CG,IAAsC,QAAvBc,EAAAA,EAAKC,KAAKM,qBAAVP,IAAAA,OAAAA,EAAAA,EAAyBQ,iBAAkB,IAC1D,MAAMC,GAAiC,QAAXT,EAAAA,EAAKU,cAALV,IAAAA,GAAuB,QAAvBA,EAAAA,EAAaW,kBAAbX,IAAAA,OAAAA,EAAAA,EAAyBY,KAClDC,GAAsB,yBAAXA,EAAEC,QACX,KACL1B,GAAqB,CACnB2B,gBAAiBN,EACjBO,iBAA4B,QAAXhB,EAAAA,EAAKU,cAALV,IAAAA,OAAAA,EAAAA,EAAagB,kBAAmB,OAEnD,MAAMC,GAAiC,QAAXjB,EAAAA,EAAKU,cAALV,IAAAA,GAAuB,QAAvBA,EAAAA,EAAaW,kBAAbX,IAAAA,OAAAA,EAAAA,EAAyBY,KAClDC,GAAsB,yBAAXA,EAAEC,QACX,K,IAQgFG,EAArF,MAAMC,EAAiD,UAAhCD,aAAAA,EAAAA,EAAqBP,QAJf,CAAC3F,IAC5B,MAAMoG,EAAQpG,EAAMqG,MAAM,KAC1B,OAAOD,EAAMlG,QAAU,EAAIkG,EAAM,GAAK,IAEwBE,CAAgD,QAA3BJ,EAAAA,EAAoBK,eAApBL,IAAAA,EAAAA,EAA+B,IAAM,GAC1H3B,GAAmB,CACjByB,gBAAiBE,EACjBM,eAAgB,CACdvE,KAAMkE,EACNnE,UAAWiD,EAAKE,SAASnD,aAG7B6C,GAAoB,MACpBrC,GAAW,KAEZiE,MAAMC,IACLjC,GAASiC,EAAIH,SAAW,8BACxB/D,GAAW,OAGhB,CAACX,EAAMG,EAAWC,EAAMG,KAE3B0C,EAAAA,EAAAA,WAAU,KACJjD,GACFgD,GAAoB,OAErB,CAAChD,IAGJ,MACE7B,MAAO2G,GACPnC,MAAOoC,KACLC,EAAAA,EAAAA,GAASC,eACE1E,EAAY2E,uBACxB,CAAC3E,EAAaP,IAEXK,GAA0BsB,IAC5BwD,EAAAA,EAAAA,GAAkBL,cAAAA,EAAAA,GAAmBM,MAAOjF,EAAWwB,GAAUvB,MACjE,KACEiF,GAAmChF,UAA+B,QAA/BA,EAAAA,GAAyByD,cAAzBzD,IAAAA,GAA2C,QAA3CA,EAAAA,EAAiC0D,kBAAjC1D,IAAAA,OAAAA,EAAAA,EAA6C2D,KACnFC,GAAsB,aAAXA,EAAEC,MAEVoB,GAA+C,CACnDX,eAAgB,CACdxE,UAAWE,cAAAA,EAAAA,GAAyBiD,SAASnD,UAC7CC,KAAMC,cAAAA,EAAAA,GAAyBiD,SAASlD,MAE1C+D,gBAAiBkB,IAQbE,GAAe,KACf1D,GAASzD,SAAWqD,GAAK/C,SAASmD,GAASzD,UAC7CsD,GAAQ,IAAID,GAAMI,GAASzD,SAC3B0D,GAAY,MA4DhB,OACE,UAAC0D,EAAAA,EAAMA,CAACxF,KAAMA,EAAMC,QAASA,EAASwF,SAAS,KAAKC,WAAS,E,WAC3D,SAACC,EAAAA,EAAWA,C,SAAC,sBACb,UAACC,EAAAA,EAAaA,C,UACXjD,KACC,SAACkD,EAAAA,EAAKA,CAACC,SAAS,QAAQC,MAAO,CAAExG,aAAc,I,SAC5CoD,KAGJoC,KACC,UAACc,EAAAA,EAAKA,CAACC,SAAS,QAAQC,MAAO,CAAExG,aAAc,I,WAC7C,SAACyG,SAAAA,C,SAAO,sCAA0C,IAAEjB,GAAuBL,WAG9EhE,GACC,SAACuF,EAAAA,EAAQA,CAAAA,IAET,sB,WAEE,SAACC,EAAAA,EAAGA,CAACC,UAAW7F,EAAQrB,c,UACtB,SAACmH,EAAAA,EAAUA,CAACC,QAAQ,Y,UAAY,SAACL,SAAAA,C,SAAO,0BAE1C,UAACM,EAAAA,EAAIA,CAACC,WAAS,EAAClH,QAAS,E,WACvB,SAACiH,EAAAA,EAAIA,CAACE,MAAI,EAACC,GAAI,E,UACb,SAACC,EAAAA,EAASA,CACRhB,WAAS,EACTiB,MAAM,mBACNxI,MAAO0C,EACP+F,SAAUC,GAAK/F,EAAe+F,EAAEC,OAAO3I,OACvC4I,YAAY,SACZC,WAAW,0CACXC,OAAO,SACPC,UAAQ,EACRC,SAAUtE,GACVuE,gBAAiB,CACf9G,QAAS,CACPvB,SAAUuB,EAAQvB,gBAK1B,SAACuH,EAAAA,EAAIA,CAACE,MAAI,EAACC,GAAI,E,UACb,SAACC,EAAAA,EAASA,CACRhB,WAAS,EACTiB,MAAM,gBACNxI,MAAOiC,EACP+G,UAAQ,EACRH,WAAW,uCACXC,OAAO,cAGX,SAACX,EAAAA,EAAIA,CAACE,MAAI,EAACC,GAAI,E,UACb,SAACC,EAAAA,EAASA,CACRhB,WAAS,EACTiB,MAAM,UACNxI,MAAO8C,EACP2F,SAAUC,GAAK3F,EAAW2F,EAAEC,OAAO3I,OACnC4I,YAAY,KACZC,WAAW,qCACXC,OAAO,SACPE,SAAUtE,QAGd,SAACyD,EAAAA,EAAIA,CAACE,MAAI,EAACC,GAAI,E,UACb,SAACC,EAAAA,EAASA,CACRhB,WAAS,EACTiB,MAAM,MACNxI,MAAO0D,GACP+E,SAAUC,GAAK/E,GAAY+E,EAAEC,OAAO3I,OACpCkJ,WAAYR,IACI,UAAVA,EAAES,MACJT,EAAEU,iBACFhC,OAGJwB,YAAY,UACZC,WAAW,gCACXC,OAAO,SACPE,SAAUtE,GACV2E,WAAY,CACVC,aAAc5F,IACZ,SAAC6F,EAAAA,EAAcA,CAACC,SAAS,M,UACvB,SAACC,EAAAA,EAAUA,CAACC,KAAK,QAAQC,QAASvC,GAAc4B,SAAUtE,G,UACxD,SAACkF,EAAAA,EAAOA,CAACtI,SAAS,mBAGpBuI,OAITvG,GAAKpD,OAAS,IACb,SAACiI,EAAAA,EAAIA,CAACE,MAAI,EAACC,GAAI,G,UACb,SAACP,EAAAA,EAAGA,CAAChH,QAAQ,OAAO+I,SAAS,O,SAC1BxG,GAAKyG,IAAIC,IACR,SAACC,EAAAA,EAAIA,CAEHzB,MAAOwB,EACPE,SAAUxF,QAASmF,EAAY,KAAMM,OAzJlCC,EAyJkDJ,OAxJzEzG,GAAQD,GAAK+G,OAAOL,GAAOA,IAAQI,IADb,IAACA,GA0JHV,KAAK,QACL1B,UAAW7F,EAAQT,QACnBsH,SAAUtE,IALLsF,SAWf,SAAC7B,EAAAA,EAAIA,CAACE,MAAI,EAACC,GAAI,G,UACb,SAACC,EAAAA,EAASA,CACRhB,WAAS,EACTiB,MAAM,cACNxI,MAAO4C,EACP6F,SAAUC,GAAK7F,EAAe6F,EAAEC,OAAO3I,OACvC4I,YAAY,kBACZE,OAAO,SACPwB,WAAS,EACTC,KAAM,EACNxB,UAAQ,EACRC,SAAUtE,GACVuE,gBAAiB,CACf9G,QAAS,CACPvB,SAAUuB,EAAQvB,mBAQ5B,UAACmH,EAAAA,EAAGA,CAACC,UAAW7F,EAAQrB,c,WACtB,SAACmH,EAAAA,EAAUA,CAACC,QAAQ,Y,UAAY,SAACL,SAAAA,C,SAAO,wBACxC,SAAC2C,EAAAA,GAAOA,CAACC,MAAM,oD,UACb,SAACC,EAAAA,EAAgBA,CAAC1C,UAAW7F,EAAQd,iBAGzC,UAAC8G,EAAAA,EAAIA,CAACC,WAAS,EAAClH,QAAS,E,WACvB,SAACiH,EAAAA,EAAIA,CAACE,MAAI,EAACC,GAAI,G,UACb,SAACC,EAAAA,EAASA,CACRhB,WAAS,EACTiB,MAAM,mBACNxI,MAAOkE,GACPuE,SAAUC,IAAKiC,OAhNE3K,EAgNsB0I,EAAEC,OAAO3I,MA/M9DmE,GAAenE,QACf6E,IAAoBzE,EAAAA,EAAAA,GAAYJ,IAFF,IAACA,GAiNjB4I,YAAY,uCACZC,WAAYjE,IAAoB,4CAChCJ,QAASI,GACTkE,OAAO,SACPE,SAAUtE,QAGd,SAACyD,EAAAA,EAAIA,CAACE,MAAI,EAACC,GAAI,G,UACb,SAACC,EAAAA,EAASA,CACRhB,WAAS,EACTiB,MAAM,oBACNxI,MAAOgE,GACPyE,SAAUC,GAAKzE,GAAWyE,EAAEC,OAAO3I,OACnC4I,YAAY,+BACZC,WAAW,8CACXC,OAAO,SACPE,SAAUtE,OAGblB,KACC,SAAC2E,EAAAA,EAAIA,CAACE,MAAI,EAACC,GAAI,G,UACb,SAACC,EAAAA,EAASA,CACRhB,WAAS,EACTiB,MAAM,YACNxI,MAAO,GAAGwD,GAAUxB,WAAaA,KAAawB,GAAUvB,OACxD+G,UAAQ,EACRH,WAAW,+BACXC,OAAO,gBAOdtF,KACC,sB,WACE,UAACuE,EAAAA,EAAGA,CAACC,UAAW7F,EAAQrB,c,WACtB,SAACmH,EAAAA,EAAUA,CAACC,QAAQ,Y,UAAY,SAACL,SAAAA,C,SAAO,0BACxC,SAAC2C,EAAAA,GAAOA,CAACC,MAAM,uE,UACb,SAACC,EAAAA,EAAgBA,CAAC1C,UAAW7F,EAAQd,iBAGzC,SAACuJ,EAAAA,EAAkBA,CACjBC,WAAYzG,GACZ0G,WAAYxG,GACZyG,gBAAiB5D,GACjB6D,kBAAkB,QAMxB,UAACjD,EAAAA,EAAGA,CAACC,UAAW7F,EAAQrB,c,WACtB,SAACmH,EAAAA,EAAUA,CAACC,QAAQ,Y,UAAY,SAACL,SAAAA,C,SAAO,gCACxC,SAAC2C,EAAAA,GAAOA,CAACC,MAAM,yE,UACb,SAACC,EAAAA,EAAgBA,CAAC1C,UAAW7F,EAAQd,iBAGzC,UAAC8G,EAAAA,EAAIA,CAACC,WAAS,EAAClH,QAAS,E,WACvB,SAACiH,EAAAA,EAAIA,CAACE,MAAI,EAACC,GAAI,E,UACb,UAACC,EAAAA,EAASA,CACRhB,WAAS,EACT0D,QAAM,EACNzC,MAAM,YACNxI,MAAOkD,EACPuF,SAAUC,GAAKvF,EAAauF,EAAEC,OAAO3I,OACrC8I,OAAO,SACPD,WAAW,sBACXG,SAAUtE,G,WAEV,SAACwG,EAAAA,EAAQA,CAAClL,MAAM,e,SAAe,kBAC/B,SAACkL,EAAAA,EAAQA,CAAClL,MAAM,a,SAAa,gBAC7B,SAACkL,EAAAA,EAAQA,CAAClL,MAAM,a,SAAa,gBAC7B,SAACkL,EAAAA,EAAQA,CAAClL,MAAM,U,SAAU,kBAG9B,SAACmI,EAAAA,EAAIA,CAACE,MAAI,EAACC,GAAI,E,UACb,UAACC,EAAAA,EAASA,CACRhB,WAAS,EACT0D,QAAM,EACNzC,MAAM,iBACNxI,MAAOgD,EACPyF,SAAUC,GAAKzF,EAAiByF,EAAEC,OAAO3I,OACzC8I,OAAO,SACPD,WAAW,8BACXG,SAAUtE,G,WAEV,SAACwG,EAAAA,EAAQA,CAAClL,MAAM,Q,SAAQ,WACxB,SAACkL,EAAAA,EAAQA,CAAClL,MAAM,Y,SAAY,uBAMlC,UAAC+H,EAAAA,EAAGA,CAACC,UAAW7F,EAAQrB,c,WACtB,SAACmH,EAAAA,EAAUA,CAACC,QAAQ,Y,UAAY,SAACL,SAAAA,C,SAAO,wBACxC,SAAC2C,EAAAA,GAAOA,CAACC,MAAM,2D,UACb,SAACC,EAAAA,EAAgBA,CAAC1C,UAAW7F,EAAQd,iBAGzC,SAAC8J,EAAAA,EAAWA,CAACC,UAAU,WAAWpC,SAAUtE,G,UAC1C,UAAC2G,EAAAA,EAAUA,CACTC,KAAG,EACHtL,MAAOoD,EACPqF,SAAUC,GAAKrF,GAAgBqF,EAAEC,OAAO3I,O,WAExC,SAACuL,EAAAA,EAAgBA,CACfvL,MAAM,SACNwL,SAAS,SAACC,EAAAA,EAAKA,CAAC5K,MAAM,YACtB2H,OACE,UAACT,EAAAA,EAAGA,C,WACF,SAACE,EAAAA,EAAUA,CAACC,QAAQ,Q,SAAQ,0BAC5B,SAACD,EAAAA,EAAUA,CAACC,QAAQ,UAAUrH,MAAM,gB,SAAgB,oDAM1D,SAAC0K,EAAAA,EAAgBA,CACfvL,MAAM,YACNwL,SAAS,SAACC,EAAAA,EAAKA,CAAC5K,MAAM,YACtB2H,OACE,UAACT,EAAAA,EAAGA,C,WACF,SAACE,EAAAA,EAAUA,CAACC,QAAQ,Q,SAAQ,eAC5B,SAACD,EAAAA,EAAUA,CAACC,QAAQ,UAAUrH,MAAM,gB,SAAgB,kEAWpE,UAAC6K,EAAAA,EAAaA,C,WACZ,SAACC,EAAAA,EAAMA,CAAChC,QAAS7H,EAASkH,SAAUtE,G,SAAQ,YAC5C,SAACiH,EAAAA,EAAMA,CACLhC,QA3UW7C,UAKjB,GAJArC,GAAS,IACTE,IAAU,GAGQ,YAAdzB,GAA6C,cAAlBF,EAG7B,OAFAyB,GAAS,kFACTE,IAAU,GAIZ,IACE,MAAMiH,EAA6B,CAEjCzG,SAAU,CACRC,OAAQ,CACNlC,cAGJgC,KAAM,CACJxC,cACAE,cACAE,UACAE,gBACAI,eACAE,QACAE,gBACII,IAAgBE,GAAc,CAChCuB,QAAS,IACHzB,IAAgB,CAAE0B,MAAO1B,OACzBE,IAAe,CAAEyB,KAAMzB,MAE3B,CAAC,KACDE,IAAWE,GAAc,CAC3BsB,cAAe,IACTxB,IAAW,CAAEA,eACbE,IAAe,CAAEuB,eAAgBvB,MAErC,CAAC,UAIH9B,EAAYyJ,iBAAiB7J,EAAWC,EAAM2J,GACpD7J,IACAD,GACF,CAAE,MAAO4E,GACPjC,GAASiC,aAAeoF,MAAQpF,EAAIH,QAAUwF,OAAOrF,GACvD,CAAE,QACA/B,IAAU,EACZ,GA2RM9D,MAAM,UACNqH,QAAQ,YACRc,SAAUtE,IAAUnC,IAAYG,IAAgBE,KAAiBgC,GACjEoH,UAAWtH,IAAS,SAACuH,EAAAA,EAAgBA,CAACvC,KAAM,GAAI7I,MAAM,iBAAegJ,E,SAEpEnF,GAAS,YAAc,e,wKClfgBwH,EAAAA,EAAAA,GAAiB,CACjEjK,KAAM,6BACNkK,WAAY,CAAEC,OAAQ,aAGwBF,EAAAA,EAAAA,GAAiB,CAC/DjK,KAAM,2BACNkK,WAAY,CAAEC,OAAQ,WAG0BF,EAAAA,EAAAA,GAAiB,CACjEjK,KAAM,6BACNkK,WAAY,CAAEC,OAAQ,aAG0BF,EAAAA,EAAAA,GAAiB,CACjEjK,KAAM,6BACNkK,WAAY,CAAEC,OAAQ,YAjBjB,MAoBMC,GAAmCH,EAAAA,EAAAA,GAAiB,CAC/DjK,KAAM,2BACNkK,WAAY,CAAEC,OAAQ,UAqBXE,IAjBmCJ,EAAAA,EAAAA,GAAiB,CAC/DjK,KAAM,2BACNkK,WAAY,CAAEC,OAAQ,WAI6BF,EAAAA,EAAAA,GAAiB,CACpEjK,KAAM,gCACNkK,WAAY,CAAEC,OAAQ,WAS0BF,EAAAA,EAAAA,GAAiB,CACjEjK,KAAM,6BACNkK,WAAY,CAAEC,OAAQ,aAgBXG,IATsCL,EAAAA,EAAAA,GAAiB,CAClEjK,KAAM,+BACNkK,WAAY,CAAEC,OAAQ,WAO2BF,EAAAA,EAAAA,GAAiB,CAClEjK,KAAM,+BACNkK,WAAY,CAAEC,OAAQ,WAOXI,GAAwCN,EAAAA,EAAAA,GAAiB,CACpEjK,KAAM,iCACNkK,WAAY,CAAEC,OAAQ,YAOXK,GAAwCP,EAAAA,EAAAA,GAAiB,CACpEjK,KAAM,iCACNkK,WAAY,CAAEC,OAAQ,YAOXM,GAAwCR,EAAAA,EAAAA,GAAiB,CACpEjK,KAAM,iCACNkK,WAAY,CAAEC,OAAQ,YAOXO,GAAwCT,EAAAA,EAAAA,GAAiB,CACpEjK,KAAM,iCACNkK,WAAY,CAAEC,OAAQ,YAOXQ,GAAmCV,EAAAA,EAAAA,GAAiB,CAC/DjK,KAAM,2BACNkK,WAAY,CAAEC,OAAQ,UAcXS,GAAiCX,EAAAA,EAAAA,GAAiB,CAC7DjK,KAAM,yBACNkK,WAAY,CAAEC,OAAQ,UACtBU,aAAc,eAyBHC,IAlBkCb,EAAAA,EAAAA,GAAiB,CAC9DjK,KAAM,2BACNkK,WAAY,CAAEC,OAAQ,WAOuBF,EAAAA,EAAAA,GAAiB,CAC9DjK,KAAM,2BACNkK,WAAY,CAAEC,OAAQ,WAOyBF,EAAAA,EAAAA,GAAiB,CAChEjK,KAAM,6BACNkK,WAAY,CAAEC,OAAQ,aAOXY,GAAoCd,EAAAA,EAAAA,GAAiB,CAChEjK,KAAM,6BACNkK,WAAY,CAAEC,OAAQ,YAOXa,GAAoCf,EAAAA,EAAAA,GAAiB,CAChEjK,KAAM,6BACNkK,WAAY,CAAEC,OAAQ,YAOXc,GAAoChB,EAAAA,EAAAA,GAAiB,CAChEjK,KAAM,6BACNkK,WAAY,CAAEC,OAAQ,YAQXe,GAAkCjB,EAAAA,EAAAA,GAAiB,CAC9DjK,KAAM,0BACNkK,WAAY,CAAEC,OAAQ,W,0DCrLjB,SAASgB,EACdC,EACAC,GAGA,MAAMC,EAAoB,iBAAkBF,EACxC,CAAEA,WAAYA,EAAkCC,eAChD,CAAED,cAEAG,GAASC,EAAAA,EAAAA,GAAcF,GAE7B,MAAO,CACLG,QAASF,EAAOE,QAChBnL,QAASiL,EAAOjL,QAChBiC,MAAOgJ,EAAOhJ,MAElB,CAWO,SAASmJ,EACdC,EACAC,EACAC,EACAC,GAEA,QAAIA,MACAD,GAAgBF,IAAYC,EAElC,C,sDChEO,MAAMG,EAA+BC,IAC1C,MAAMC,EAAO,CAAEC,OAAQ,QACvB,OAAQF,GACN,IAAK,WACH,MAAO,IAAKC,EAAME,gBAAiB,UAAWvN,MAAO,QACvD,IAAK,WACH,MAAO,IAAKqN,EAAME,gBAAiB,UAAWvN,MAAO,QAGvD,QACE,MAAO,IAAKqN,EAAME,gBAAiB,UAAWvN,MAAO,UAQ9CwN,EAAmCJ,IAC9C,MAAMC,EAAO,CAAEC,OAAQ,QACvB,OAAQF,GACN,IAAK,WACH,MAAO,IAAKC,EAAME,gBAAiB,UAAWvN,MAAO,QACvD,IAAK,WACH,MAAO,IAAKqN,EAAME,gBAAiB,UAAWvN,MAAO,QAGvD,QACE,MAAO,IAAKqN,EAAME,gBAAiB,UAAWvN,MAAO,UAQ9CyN,EAAyBpL,IACpC,OAAQA,GACN,IAAK,aACH,MAAO,CAAEkL,gBAAiB,UAAWvN,MAAO,QAC9C,IAAK,eACH,MAAO,CAAEuN,gBAAiB,UAAWvN,MAAO,QAC9C,IAAK,aACH,MAAO,CAAEuN,gBAAiB,UAAWvN,MAAO,QAC9C,IAAK,UACH,MAAO,CAAEuN,gBAAiB,UAAWvN,MAAO,QAC9C,QACE,MAAO,CAAC,G,sLCxBP,MAAM0N,EAAsB,EACjC1M,OACA4I,QACA7H,cACA4L,cACA7G,WAAW,SACX8G,YAAW,EACXC,YACAC,eAEA,MAAOC,EAAYC,IAAiBpM,EAAAA,EAAAA,UAAS,KAG7CqC,EAAAA,EAAAA,WAAU,KACHjD,GACHgN,EAAc,KAEf,CAAChN,IAEJ,MAAMiN,EAAwC,SAAbnH,GAAuB6G,EAClDO,GAAaD,GAA2BF,IAAeJ,EAQ7D,OACE,UAACnH,EAAAA,EAAMA,CACLxF,KAAMA,EACNC,QAAS2M,OAAW5E,EAAY8E,EAChCrH,SAAS,KACTC,WAAS,E,WAET,UAACC,EAAAA,EAAWA,C,UACI,SAAbG,IACC,UAACI,EAAAA,EAAGA,CAAChH,QAAQ,OAAOC,WAAW,SAAS4G,MAAO,CAAE3G,IAAK,G,WACpD,SAAC+N,EAAAA,EAAWA,CAACnO,MAAM,WACnB,SAACoO,OAAAA,C,SAAMxE,OAGG,SAAb9C,GAAuB8C,MAE1B,UAAChD,EAAAA,EAAaA,C,WACZ,SAACyH,EAAAA,EAAiBA,CAACtH,MAAO,CAAEuH,WAAY,Y,SACrCvM,IAEFkM,IACC,UAAC/G,EAAAA,EAAGA,CAACqH,GAAI,E,WACP,UAACnH,EAAAA,EAAUA,CAACC,QAAQ,QAAQrH,MAAM,gBAAgBwO,cAAY,E,UAAC,SACxD,SAACxH,SAAAA,C,SAAQ2G,IAAqB,mBAErC,SAACjG,EAAAA,EAASA,CACRhB,WAAS,EACTW,QAAQ,WACRwB,KAAK,QACL1J,MAAO4O,EACPnG,SAAUC,GAAKmG,EAAcnG,EAAEC,OAAO3I,OACtCgJ,SAAUyF,EACVa,WAAS,EACT1G,YAAa4F,WAKrB,UAAC9C,EAAAA,EAAaA,C,WACZ,SAACC,EAAAA,EAAMA,CAAChC,QAASgF,EAAU3F,SAAUyF,E,SAAU,YAG/C,SAAC9C,EAAAA,EAAMA,CACLhC,QAjDc,KAChBoF,GACFL,KAgDI7N,MAAM,YACNqH,QAAQ,YACRc,SAAUyF,IAAaM,EACvB/C,UAAWyC,GAAW,SAACxC,EAAAA,EAAgBA,CAACvC,KAAM,GAAI7I,MAAM,iBAAegJ,E,SAEtE4E,EAAW,cAAgB,iB,2DCnEtC,MAsNanM,GAAiBiN,EAAAA,EAAAA,cAA0B,CACtDC,GAAI,2B,oHCtNC,MAAM5E,EAAwD,EACnEC,aACAC,aACAC,kBACAC,oBAAmB,M,IAoGMD,EAUAA,EA2CFD,EAUAA,EAjKvB,MAAMnK,GAAQ8O,EAAAA,EAAAA,KAEd,OACE,SAAC1H,EAAAA,EAAGA,CACFqH,GAAIpE,EAAmB,EAAI,EAC3B0E,EAAG,EACHC,QAAShP,EAAMY,QAAQqO,WAAWC,QAClCC,aAAc,EACd3B,OAAQ,aAAaxN,EAAMY,QAAQwO,U,UAEnC,UAAC5H,EAAAA,EAAIA,CAACC,WAAS,EAAClH,QAAS,E,WAEpB2J,aAAAA,EAAAA,EAAY7E,kBAAyD,SAAtC6E,EAAW7E,gBAAgBL,UAAuBoF,aAAAA,EAAAA,EAAiB/E,kBACnG,UAACmC,EAAAA,EAAIA,CAACE,MAAI,EAACC,GAAI,GAAI0H,GAAI,E,WACrB,SAAC/H,EAAAA,EAAUA,CAACC,QAAQ,QAAQrH,MAAM,gBAAgBwO,cAAY,E,SAAC,iBAG9DxE,aAAAA,EAAAA,EAAY7E,kBACX,UAAC+B,EAAAA,EAAGA,C,WACF,SAACkC,EAAAA,EAAIA,CACHzB,MAAOqC,EAAW7E,gBAAgBiK,OAClCvG,KAAK,QACL9B,MAAO,CACLwG,gBAAuD,SAAtCvD,EAAW7E,gBAAgBL,OAAoB,UAAY,UAC5E9E,MAAO,OACPO,aAAc,KAGjByJ,EAAW5E,iBAAmB4E,EAAW5E,gBAAgB/F,OAAS,IACjE,sB,WACE,SAAC+H,EAAAA,EAAUA,CACTC,QAAQ,UACRnH,QAAQ,QACRsO,cAAY,EACZxO,MAAM,gBACN+G,MAAO,CAAEzG,UAAW,G,SACrB,iCAGD,SAAC4G,EAAAA,EAAGA,CAAChH,QAAQ,OAAO+I,SAAS,OAAOsF,GAAI,EAAGxH,MAAO,CAAE3G,IAAK,G,SACtD4J,EAAW5E,gBAAgB8D,IAAI,CAACmG,EAAWC,K,IACxBD,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,WACR,OACE,SAACvG,EAAAA,EAAIA,CAEHzB,MAAO,GAAG0H,EAAKO,SAASL,IACxB1G,KAAK,QACLxB,QAAQ,WACRrH,MAAM,WAJDsP,cAanB,SAAClG,EAAAA,EAAIA,CACHzB,MAAM,WACNkB,KAAK,QACL9B,MAAO,CACLwG,gBAAiB,UACjBvN,MAAO,OACPO,aAAc,SAMtB,UAAC+G,EAAAA,EAAIA,CAACE,MAAI,EAACC,GAAI,GAAI0H,GAAI,E,WACrB,SAAC/H,EAAAA,EAAUA,CAACC,QAAQ,QAAQrH,MAAM,gBAAgBwO,cAAY,E,SAAC,qBAG9DtE,aAAAA,EAAAA,EAAiB/E,kBAChB,UAAC+B,EAAAA,EAAGA,C,WACF,SAACkC,EAAAA,EAAIA,CACHzB,MAAOuC,EAAgB/E,gBAAgBiK,OACvCvG,KAAK,QACL9B,MAAO,CACLwG,gBAA4D,SAA3CrD,EAAgB/E,gBAAgBL,OAAoB,UAAY,UACjF9E,MAAO,OACPO,aAAc,KAG0B,SAA3C2J,EAAgB/E,gBAAgBL,SAC/B,UAACwC,EAAAA,EAAIA,CAACC,WAAS,EAAClH,QAAS,E,WACvB,SAACiH,EAAAA,EAAIA,CAACE,MAAI,EAACC,GAAI,GAAI0H,GAAI,E,UACrB,UAACjI,EAAAA,EAAGA,C,WACF,SAACE,EAAAA,EAAUA,CAACC,QAAQ,QAAQrH,MAAM,gBAAgBwO,cAAY,E,SAAC,mBAG/D,SAACpH,EAAAA,EAAUA,CAACC,QAAQ,Q,UACa,QAA9B6C,EAAAA,EAAgBvE,sBAAhBuE,IAAAA,OAAAA,EAAAA,EAAgC9I,OAAQ,6CAI/C,SAACkG,EAAAA,EAAIA,CAACE,MAAI,EAACC,GAAI,GAAI0H,GAAI,E,UACrB,UAACjI,EAAAA,EAAGA,C,WACF,SAACE,EAAAA,EAAUA,CAACC,QAAQ,QAAQrH,MAAM,gBAAgBwO,cAAY,E,SAAC,wBAG/D,SAACpH,EAAAA,EAAUA,CAACC,QAAQ,Q,UACa,QAA9B6C,EAAAA,EAAgBvE,sBAAhBuE,IAAAA,OAAAA,EAAAA,EAAgC/I,YAAa,wDAQ1D,SAACiI,EAAAA,EAAIA,CACHzB,MAAM,WACNkB,KAAK,QACL9B,MAAO,CACLwG,gBAAiB,UACjBvN,MAAO,OACPO,aAAc,SAMxB,UAAC+G,EAAAA,EAAIA,CAACE,MAAI,EAACC,GAAI,GAAI0H,GAAI,E,WACrB,SAAC/H,EAAAA,EAAUA,CAACC,QAAQ,QAAQrH,MAAM,gBAAgBwO,cAAY,E,SAAC,iBAG9DvE,aAAAA,EAAAA,EAAY9E,kBACX,UAAC+B,EAAAA,EAAGA,C,WACF,SAACkC,EAAAA,EAAIA,CACHzB,MAAOsC,EAAW9E,gBAAgBiK,OAClCvG,KAAK,QACL9B,MAAO,CACLwG,gBAAuD,SAAtCtD,EAAW9E,gBAAgBL,OAAoB,UAAY,UAC5E9E,MAAO,OACPO,aAAc,KAGqB,SAAtC0J,EAAW9E,gBAAgBL,SAC1B,UAACwC,EAAAA,EAAIA,CAACC,WAAS,EAAClH,QAAS,E,WACvB,SAACiH,EAAAA,EAAIA,CAACE,MAAI,EAACC,GAAI,GAAI0H,GAAI,E,UACrB,UAACjI,EAAAA,EAAGA,C,WACF,SAACE,EAAAA,EAAUA,CAACC,QAAQ,QAAQrH,MAAM,gBAAgBwO,cAAY,E,SAAC,mBAG/D,SAACpH,EAAAA,EAAUA,CAACC,QAAQ,Q,UACQ,QAAzB4C,EAAAA,EAAWtE,sBAAXsE,IAAAA,OAAAA,EAAAA,EAA2B7I,OAAQ,wCAI1C,SAACkG,EAAAA,EAAIA,CAACE,MAAI,EAACC,GAAI,GAAI0H,GAAI,E,UACrB,UAACjI,EAAAA,EAAGA,C,WACF,SAACE,EAAAA,EAAUA,CAACC,QAAQ,QAAQrH,MAAM,gBAAgBwO,cAAY,E,SAAC,wBAG/D,SAACpH,EAAAA,EAAUA,CAACC,QAAQ,Q,UACQ,QAAzB4C,EAAAA,EAAWtE,sBAAXsE,IAAAA,OAAAA,EAAAA,EAA2B9I,YAAa,mDAQrD,SAACiI,EAAAA,EAAIA,CACHzB,MAAM,WACNkB,KAAK,QACL9B,MAAO,CACLwG,gBAAiB,UACjBvN,MAAO,OACPO,aAAc,a,mCC9MvB,MAAM4F,EAAoB,CAC/B0J,EACAC,EACAC,IAEKF,GAEEA,EAAS7K,KAAMgL,IACpB,MAAMC,EAAMD,EAAG3L,KAAK1B,U,IACIsN,EAAxB,MAAMC,EAAgC,QAAdD,EAAAA,aAAAA,EAAAA,EAAK9O,iBAAL8O,IAAAA,EAAAA,EAAkBD,EAAG1L,SAASnD,UAEtD,MACgB,eAAd8O,aAAAA,EAAAA,EAAKE,QACLF,aAAAA,EAAAA,EAAK7O,QAAS2O,GACdG,IAAoBJ,KATF,I","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/utils/styles.ts","webpack://internal.plugin-kuadrant/./src/components/ConfirmDeleteDialog/ConfirmDeleteDialog.tsx","webpack://internal.plugin-kuadrant/./src/api.ts","webpack://internal.plugin-kuadrant/./src/components/ApiProductPolicies/ApiProductPolicies.tsx","webpack://internal.plugin-kuadrant/./src/utils/policies.ts"],"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 MenuItem,\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 } from '@backstage/core-plugin-api';\nimport { kuadrantApiRef } from '../../api';\nimport { Alert } from '@material-ui/lab';\nimport useAsync from 'react-use/lib/useAsync';\nimport { Progress } from '@backstage/core-components';\nimport { ApiProductPolicies, PlanPoliciesProps, AuthPoliciesProps, RateLimitPoliciesProps } from '../ApiProductPolicies';\nimport { validateURL } from '../../utils/validation';\nimport { APIProduct } from \"../../types/api-management.ts\";\nimport { Lifecycle } from '../../types/api-management';\nimport { getPolicyForRoute } from '../../utils/policies';\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 kuadrantApi = useApi(kuadrantApiRef);\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 [lifecycle, setLifecycle] = useState<Lifecycle>('production');\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 [planPolicyProps, setPlanPoliciesProps] = useState<PlanPoliciesProps | null>(null);\n const [authPolicyProps, setAuthPolicyProps] = useState<AuthPoliciesProps | null>(null);\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 kuadrantApi.getApiProduct(namespace, name)\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 setLifecycle(data.metadata.labels?.lifecycle as Lifecycle || 'production' as Lifecycle);\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 const planPolicyCondition = data.status?.conditions?.find(\n (c: any) => c.type === \"PlanPolicyDiscovered\"\n ) || null;\n setPlanPoliciesProps({\n statusCondition: planPolicyCondition,\n discoveredPlans: data.status?.discoveredPlans || null,\n });\n const authPolicyCondition = data.status?.conditions?.find(\n (c: any) => c.type === \"AuthPolicyDiscovered\"\n ) || null;\n // Parse the auth policy name from the AuthPolicyDiscovered condition message\n // It's the only place to get the policy name without fetching all the policies.\n // Consider enhancing the developer portal controller to provide name in an structured way\n const parseNameFromMessage = (value: string) => {\n const parts = value.split(' ');\n return parts.length >= 3 ? parts[2] : '';\n };\n const authPolicyName = authPolicyCondition?.status === \"True\" ? parseNameFromMessage(authPolicyCondition.message ?? \"\") : \"\";\n setAuthPolicyProps({\n statusCondition: authPolicyCondition,\n namespacedName: {\n name: authPolicyName,\n namespace: data.metadata.namespace,\n },\n });\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, kuadrantApi]);\n\n useEffect(() => {\n if (open) {\n setOpenAPISpecError(null);\n }\n }, [open]);\n\n // load ratelimitpolicies\n const {\n value: rateLimitPolicies,\n error: rateLimitPoliciesError\n } = useAsync(async () => {\n return await kuadrantApi.getRateLimitPolicies();\n }, [kuadrantApi, open]);\n\n const selectedRateLimitPolicy = targetRef\n ? getPolicyForRoute(rateLimitPolicies?.items, namespace, targetRef.name)\n : null;\n const rateLimitPolicyAcceptedCondition = selectedRateLimitPolicy?.status?.conditions?.find(\n (c: any) => c.type === \"Accepted\"\n );\n const rateLimitPolicyProps: RateLimitPoliciesProps = {\n namespacedName: {\n namespace: selectedRateLimitPolicy?.metadata.namespace,\n name: selectedRateLimitPolicy?.metadata.name,\n },\n statusCondition: rateLimitPolicyAcceptedCondition,\n }\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 // validate retired APIs cannot be published\n if (lifecycle === 'retired' && publishStatus === 'Published') {\n setError('Cannot publish a retired API product. Please set publish status to Draft.');\n setSaving(false);\n return;\n }\n\n try {\n const patch: Partial<APIProduct> = {\n // @ts-ignore We are applying a partial obj, not full\n metadata: {\n labels: {\n lifecycle,\n },\n },\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 await kuadrantApi.updateApiProduct(namespace, name, patch);\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 {rateLimitPoliciesError && (\n <Alert severity=\"error\" style={{ marginBottom: 16 }}>\n <strong>Failed to load RateLimitPolicies:</strong> {rateLimitPoliciesError.message}\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 <ApiProductPolicies\n planPolicy={planPolicyProps}\n authPolicy={authPolicyProps}\n rateLimitPolicy={rateLimitPolicyProps}\n includeTopMargin={false}\n />\n </>\n )}\n\n {/* Lifecycle and Visibility section */}\n <Box className={classes.sectionHeader}>\n <Typography variant=\"subtitle1\"><strong>Lifecycle and Visibility</strong></Typography>\n <Tooltip title=\"Control the lifecycle state and catalog visibility of this API product\">\n <InfoOutlinedIcon className={classes.infoIcon} />\n </Tooltip>\n </Box>\n <Grid container spacing={2}>\n <Grid item xs={6}>\n <TextField\n fullWidth\n select\n label=\"Lifecycle\"\n value={lifecycle}\n onChange={e => setLifecycle(e.target.value as Lifecycle)}\n margin=\"normal\"\n helperText=\"API lifecycle state\"\n disabled={saving}\n >\n <MenuItem value=\"experimental\">Experimental</MenuItem>\n <MenuItem value=\"production\">Production</MenuItem>\n <MenuItem value=\"deprecated\">Deprecated</MenuItem>\n <MenuItem value=\"retired\">Retired</MenuItem>\n </TextField>\n </Grid>\n <Grid item xs={6}>\n <TextField\n fullWidth\n select\n label=\"Publish Status\"\n value={publishStatus}\n onChange={e => setPublishStatus(e.target.value as 'Draft' | 'Published')}\n margin=\"normal\"\n helperText=\"Controls catalog visibility\"\n disabled={saving}\n >\n <MenuItem value=\"Draft\">Draft</MenuItem>\n <MenuItem value=\"Published\">Published</MenuItem>\n </TextField>\n </Grid>\n </Grid>\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// authpolicy permissions\nexport const kuadrantAuthPolicyListPermission = createPermission({\n name: 'kuadrant.authpolicy.list',\n attributes: { action: 'read' },\n});\n\n// ratelimitpolicy permissions\nexport const kuadrantRateLimitPolicyListPermission = createPermission({\n name: 'kuadrant.ratelimitpolicy.list',\n attributes: { action: 'read' },\n});\n\n// apiproduct permissions\n\n/**\n * permission to create new API products\n * granted to api owners and admins\n */\nexport const kuadrantApiProductCreatePermission = createPermission({\n name: 'kuadrant.apiproduct.create',\n attributes: { action: 'create' },\n});\n\n/**\n * permission to read API products owned by the current user\n * for api owners to view their own products\n */\nexport const kuadrantApiProductReadOwnPermission = createPermission({\n name: 'kuadrant.apiproduct.read.own',\n attributes: { action: 'read' },\n});\n\n/**\n * permission to read all API products regardless of ownership\n * for platform engineers/admins who need to view all products\n */\nexport const kuadrantApiProductReadAllPermission = createPermission({\n name: 'kuadrant.apiproduct.read.all',\n attributes: { action: 'read' },\n});\n\n/**\n * permission to update API products owned by the current user\n * for api owners to modify their own products\n */\nexport const kuadrantApiProductUpdateOwnPermission = createPermission({\n name: 'kuadrant.apiproduct.update.own',\n attributes: { action: 'update' },\n});\n\n/**\n * permission to update any API product regardless of ownership\n * for platform engineers/admins\n */\nexport const kuadrantApiProductUpdateAllPermission = createPermission({\n name: 'kuadrant.apiproduct.update.all',\n attributes: { action: 'update' },\n});\n\n/**\n * permission to delete API products owned by the current user\n * for api owners to remove their own products\n */\nexport const kuadrantApiProductDeleteOwnPermission = createPermission({\n name: 'kuadrant.apiproduct.delete.own',\n attributes: { action: 'delete' },\n});\n\n/**\n * permission to delete any API product regardless of ownership\n * for platform engineers/admins\n */\nexport const kuadrantApiProductDeleteAllPermission = createPermission({\n name: 'kuadrant.apiproduct.delete.all',\n attributes: { action: 'delete' },\n});\n\n/**\n * permission to list API products\n * backend filters results based on .own vs .all read permissions\n */\nexport const kuadrantApiProductListPermission = createPermission({\n name: 'kuadrant.apiproduct.list',\n attributes: { action: 'read' },\n});\n\n// apikey permissions\n\n/**\n * permission to create API keys (request API access)\n *\n * this is a ResourcePermission scoped to 'apiproduct', allowing\n * fine-grained control over which API products users can request access to.\n *\n * use in frontend: useKuadrantPermission(kuadrantApiKeyCreatePermission)\n * use in backend with resource: { permission, resourceRef: 'apiproduct:namespace/name' }\n */\nexport const kuadrantApiKeyCreatePermission = createPermission({\n name: 'kuadrant.apikey.create',\n attributes: { action: 'create' },\n resourceType: 'apiproduct',\n});\n\n/**\n * permission to read API keys owned by the current user\n * allows users to view their own API keys and request history\n */\nexport const kuadrantApiKeyReadOwnPermission = createPermission({\n name: 'kuadrant.apikey.read.own',\n attributes: { action: 'read' },\n});\n\n/**\n * permission to read all API keys regardless of ownership\n * for platform engineers/admins who need to view the approval queue and audit keys\n */\nexport const kuadrantApiKeyReadAllPermission = createPermission({\n name: 'kuadrant.apikey.read.all',\n attributes: { action: 'read' },\n});\n\n/**\n * permission to update API keys owned by the current user\n * allows users to edit their own pending requests (change plan tier, use case)\n */\nexport const kuadrantApiKeyUpdateOwnPermission = createPermission({\n name: 'kuadrant.apikey.update.own',\n attributes: { action: 'update' },\n});\n\n/**\n * permission to update any API key regardless of ownership\n * typically granted to API owners and platform engineers for approving/rejecting requests\n */\nexport const kuadrantApiKeyUpdateAllPermission = createPermission({\n name: 'kuadrant.apikey.update.all',\n attributes: { action: 'update' },\n});\n\n/**\n * permission to delete API keys owned by the current user\n * allows users to cancel their own requests or revoke their own access\n */\nexport const kuadrantApiKeyDeleteOwnPermission = createPermission({\n name: 'kuadrant.apikey.delete.own',\n attributes: { action: 'delete' },\n});\n\n/**\n * permission to delete any API key regardless of ownership\n * for platform engineers/admins who need to revoke access\n */\nexport const kuadrantApiKeyDeleteAllPermission = createPermission({\n name: 'kuadrant.apikey.delete.all',\n attributes: { action: 'delete' },\n});\n\n/**\n * permission to approve/reject API key requests\n * grants access to the approval queue - for API owners and admins only\n * separate from update.own which consumers use to edit their pending requests\n */\nexport const kuadrantApiKeyApprovePermission = createPermission({\n name: 'kuadrant.apikey.approve',\n attributes: { action: 'update' },\n});\n\nexport const kuadrantPermissions = [\n kuadrantPlanPolicyCreatePermission,\n kuadrantPlanPolicyReadPermission,\n kuadrantPlanPolicyUpdatePermission,\n kuadrantPlanPolicyDeletePermission,\n kuadrantPlanPolicyListPermission,\n kuadrantApiProductCreatePermission,\n kuadrantApiProductReadOwnPermission,\n kuadrantApiProductReadAllPermission,\n kuadrantApiProductUpdateOwnPermission,\n kuadrantApiProductUpdateAllPermission,\n kuadrantApiProductDeleteOwnPermission,\n kuadrantApiProductDeleteAllPermission,\n kuadrantApiProductListPermission,\n kuadrantApiKeyCreatePermission,\n kuadrantApiKeyReadOwnPermission,\n kuadrantApiKeyReadAllPermission,\n kuadrantApiKeyUpdateOwnPermission,\n kuadrantApiKeyUpdateAllPermission,\n kuadrantApiKeyDeleteOwnPermission,\n kuadrantApiKeyDeleteAllPermission,\n kuadrantApiKeyApprovePermission,\n kuadrantAuthPolicyListPermission,\n kuadrantRateLimitPolicyListPermission,\n];\n","import { usePermission } from '@backstage/plugin-permission-react';\nimport { Permission, ResourcePermission } from '@backstage/plugin-permission-common';\n\n/**\n * result of a permission check including error state\n */\nexport interface PermissionCheckResult {\n allowed: boolean;\n loading: boolean;\n error?: Error;\n}\n\n/**\n * custom hook for checking kuadrant permissions that handles both\n * BasicPermission and ResourcePermission types without type bypasses\n *\n * @param permission - the permission to check\n * @param resourceRef - optional resource reference for ResourcePermissions\n * @returns permission check result with error handling\n *\n * @example\n * // basic permission\n * const { allowed, loading, error } = useKuadrantPermission(\n * kuadrantApiProductListPermission\n * );\n *\n * @example\n * // resource permission\n * const { allowed, loading, error } = useKuadrantPermission(\n * kuadrantApiKeyCreatePermission,\n * 'apiproduct:namespace/name'\n * );\n */\nexport function useKuadrantPermission(\n permission: Permission,\n resourceRef?: string,\n): PermissionCheckResult {\n // construct the permission request based on whether it's a ResourcePermission\n const permissionRequest = 'resourceType' in permission\n ? { permission: permission as ResourcePermission, resourceRef }\n : { permission };\n\n const result = usePermission(permissionRequest as any);\n\n return {\n allowed: result.allowed,\n loading: result.loading,\n error: result.error,\n };\n}\n\n/**\n * helper to determine if a user can delete a specific API key or request\n *\n * @param ownerId - the user id who owns the key/request\n * @param currentUserId - the current user's id\n * @param canDeleteOwn - whether user has permission to delete their own keys\n * @param canDeleteAll - whether user has permission to delete all keys\n * @returns true if user can delete this specific key/request\n */\nexport function canDeleteResource(\n ownerId: string,\n currentUserId: string,\n canDeleteOwn: boolean,\n canDeleteAll: boolean,\n): boolean {\n if (canDeleteAll) return true;\n if (canDeleteOwn && ownerId === currentUserId) return true;\n return false;\n}\n","import { CSSProperties } from \"react\";\n\n/**\n * Returns inline styles for API key status chips on the My API Keys page.\n */\nexport const getMyApiKeysStatusChipStyle = (phase: string): CSSProperties => {\n const base = { border: \"none\" };\n switch (phase) {\n case \"Approved\":\n return { ...base, backgroundColor: \"#1976d2\", color: \"#fff\" }; // Blue\n case \"Rejected\":\n return { ...base, backgroundColor: \"#d32f2f\", color: \"#fff\" }; // Red\n case \"Pending\":\n return { ...base, backgroundColor: \"#9c27b0\", color: \"#fff\" }; // Purple\n default:\n return { ...base, backgroundColor: \"#9c27b0\", color: \"#fff\" }; // Purple (fallback for Pending)\n }\n};\n\n/**\n * Returns inline styles for API key status chips on the Approval Queue page.\n * Uses inline styles to ensure proper specificity with Material-UI Chip.\n */\nexport const getApprovalQueueStatusChipStyle = (phase: string): CSSProperties => {\n const base = { border: \"none\" };\n switch (phase) {\n case \"Approved\":\n return { ...base, backgroundColor: \"#2e7d32\", color: \"#fff\" }; // Green\n case \"Rejected\":\n return { ...base, backgroundColor: \"#d32f2f\", color: \"#fff\" }; // Red\n case \"Pending\":\n return { ...base, backgroundColor: \"#ed6c02\", color: \"#fff\" }; // Orange\n default:\n return { ...base, backgroundColor: \"#ed6c02\", color: \"#fff\" }; // Orange (fallback for Pending)\n }\n};\n\n/**\n * Returns inline styles for lifecycle chips.\n * Uses inline styles to ensure proper specificity with Material-UI Chip.\n */\nexport const getLifecycleChipStyle = (lifecycle: string): CSSProperties => {\n switch (lifecycle) {\n case \"production\":\n return { backgroundColor: \"#1976d2\", color: \"#fff\" }; // Blue\n case \"experimental\":\n return { backgroundColor: \"#9c27b0\", color: \"#fff\" }; // Purple\n case \"deprecated\":\n return { backgroundColor: \"#ff9800\", color: \"#fff\" }; // Orange\n case \"retired\":\n return { backgroundColor: \"#d32f2f\", color: \"#fff\" }; // Red\n default:\n return {};\n }\n};\n","import React, { useState, useEffect } from 'react';\nimport {\n Dialog,\n DialogTitle,\n DialogContent,\n DialogContentText,\n DialogActions,\n Button,\n TextField,\n Typography,\n Box,\n CircularProgress,\n} from '@material-ui/core';\nimport WarningIcon from '@material-ui/icons/Warning';\n\nexport interface ConfirmDeleteDialogProps {\n open: boolean;\n title: string;\n description: string;\n // for dangerous deletes, require typing this text to confirm\n confirmText?: string;\n // severity affects styling - 'high' shows warning icon and requires text confirmation\n severity?: 'normal' | 'high';\n deleting?: boolean;\n onConfirm: () => void;\n onCancel: () => void;\n}\n\nexport const ConfirmDeleteDialog = ({\n open,\n title,\n description,\n confirmText,\n severity = 'normal',\n deleting = false,\n onConfirm,\n onCancel,\n}: ConfirmDeleteDialogProps) => {\n const [inputValue, setInputValue] = useState('');\n\n // reset input when dialog opens/closes\n useEffect(() => {\n if (!open) {\n setInputValue('');\n }\n }, [open]);\n\n const requiresTextConfirmation = severity === 'high' && confirmText;\n const canConfirm = requiresTextConfirmation ? inputValue === confirmText : true;\n\n const handleConfirm = () => {\n if (canConfirm) {\n onConfirm();\n }\n };\n\n return (\n <Dialog\n open={open}\n onClose={deleting ? undefined : onCancel}\n maxWidth=\"sm\"\n fullWidth\n >\n <DialogTitle>\n {severity === 'high' && (\n <Box display=\"flex\" alignItems=\"center\" style={{ gap: 8 }}>\n <WarningIcon color=\"error\" />\n <span>{title}</span>\n </Box>\n )}\n {severity !== 'high' && title}\n </DialogTitle>\n <DialogContent>\n <DialogContentText style={{ whiteSpace: 'pre-line' }}>\n {description}\n </DialogContentText>\n {requiresTextConfirmation && (\n <Box mt={2}>\n <Typography variant=\"body2\" color=\"textSecondary\" gutterBottom>\n Type <strong>{confirmText}</strong> to confirm:\n </Typography>\n <TextField\n fullWidth\n variant=\"outlined\"\n size=\"small\"\n value={inputValue}\n onChange={e => setInputValue(e.target.value)}\n disabled={deleting}\n autoFocus\n placeholder={confirmText}\n />\n </Box>\n )}\n </DialogContent>\n <DialogActions>\n <Button onClick={onCancel} disabled={deleting}>\n Cancel\n </Button>\n <Button\n onClick={handleConfirm}\n color=\"secondary\"\n variant=\"contained\"\n disabled={deleting || !canConfirm}\n startIcon={deleting ? <CircularProgress size={16} color=\"inherit\" /> : undefined}\n >\n {deleting ? 'Deleting...' : 'Delete'}\n </Button>\n </DialogActions>\n </Dialog>\n );\n};\n","import {\n createApiRef,\n DiscoveryApi,\n FetchApi,\n IdentityApi,\n} from '@backstage/core-plugin-api';\nimport retry from 'async-retry';\nimport { handleFetchError } from './utils/errors';\nimport {\n APIKey, APIKeyRequest,\n APIKeySpec,\n APIProduct,\n BulkOperationResult, ExtractedSecret, K8sList, K8sResource,\n PlanPolicy,\n AuthPolicy,\n RateLimitPolicy,\n} from './types/api-management';\n\n/**\n * Generic Kuadrant list type for API responses\n */\nexport interface KuadrantList<T = any> {\n items: T[];\n}\n\n/**\n * Options for constructing the KuadrantApiClient\n */\nexport type Options = {\n discoveryApi: DiscoveryApi;\n fetchApi: FetchApi;\n identityApi: IdentityApi;\n};\n\n/**\n * Retry configuration for read operations (GET requests only)\n * Conservative strategy: 3 retries with exponential backoff\n */\nconst RETRY_OPTIONS: retry.Options = {\n retries: 3,\n factor: 2,\n minTimeout: 300, // 300ms, 600ms, 1200ms\n maxTimeout: 3000,\n randomize: true,\n};\n\n/**\n * Kuadrant API interface defining all operations for managing\n * API products, API keys, and related resources\n */\nexport interface KuadrantAPI {\n // ===== APIKey Requests =====\n\n /**\n * Fetch all API key requests per user\n * @returns Promise with list of all API key requests\n */\n getRequests(): Promise<KuadrantList<APIKey>>;\n\n /**\n * Fetch all API key requests\n * @returns Promise with list of all API key requests\n */\n getAllRequests(): Promise<KuadrantList<APIKey>>;\n\n /**\n * Fetch API key requests for a specific namespace\n * @param namespace - Kubernetes namespace\n * @returns Promise with list of requests in the namespace\n */\n getRequestsByNamespace(namespace: string): Promise<KuadrantList<APIKey>>;\n\n /**\n * Fetch a single API key request\n * @param namespace - API key request name\n * @param name - Kubernetes namespace\n * @returns Promise with the API key request\n */\n getRequest(namespace: string, name: string): Promise<APIKey>;\n\n /**\n * Create a new API key request\n * @param request - APIKeyRequest specification\n * @returns Promise with the created API key\n */\n createRequest(request: APIKeyRequest): Promise<APIKey>;\n\n /**\n * Update an existing API key request\n * @param namespace - Kubernetes namespace\n * @param name - API key request name\n * @param patch - Partial API key spec with fields to update\n * @returns Promise with the updated API key\n */\n updateRequest(\n namespace: string,\n name: string,\n patch: Partial<APIKeySpec>,\n ): Promise<APIKey>;\n\n /**\n * Delete an API key request\n * @param namespace - Kubernetes namespace\n * @param name - API key request name\n * @returns Promise that resolves when deletion completes\n */\n deleteRequest(namespace: string, name: string): Promise<void>;\n\n /**\n * Approve an API key request\n * @param namespace - Kubernetes namespace\n * @param name - API key request name\n * @param reviewedBy - Reviewed By User / System\n * @returns Promise with the approved API key\n */\n approveRequest(namespace: string, name: string, reviewedBy: string): Promise<APIKey>;\n\n /**\n * Reject an API key request\n * @param namespace - Kubernetes namespace\n * @param name - API key request name\n * @param reviewedBy - Reviewed By User / System\n * @returns Promise with the rejected API key\n */\n rejectRequest(namespace: string, name: string, reviewedBy: string): Promise<APIKey>;\n\n /**\n * Bulk approve multiple API key requests\n * @param requests - Array of namespace/name pairs to approve\n * @param reviewedBy - Reviewed By User / System\n * @returns Promise that resolves when all approvals complete\n */\n bulkApproveRequests(\n requests: Array<{ namespace: string; name: string }>,\n reviewedBy: string): Promise<Array<BulkOperationResult>>;\n\n /**\n * Bulk reject multiple API key requests\n * @param requests - Array of namespace/name pairs to reject\n * @param reviewedBy - Reviewed By User / System\n * @returns Promise that resolves when all rejections complete\n */\n bulkRejectRequests(\n requests: Array<{ namespace: string; name: string }>,\n reviewedBy: string): Promise<Array<BulkOperationResult>>;\n\n // ===== API Keys/Secrets =====\n\n /**\n * Fetch an API key resource\n * @param namespace - Kubernetes namespace\n * @param name - API key name\n * @returns Promise with the API key\n */\n getApiKey(namespace: string, name: string): Promise<APIKey>;\n\n /**\n * Retrieve the secret value for an API key (one-time operation)\n * @param namespace - Kubernetes namespace\n * @param name - API key name\n * @returns Promise with the secret value\n */\n getApiKeySecret(namespace: string, name: string): Promise<ExtractedSecret>;\n\n // ===== API Products =====\n\n /**\n * Fetch all API products\n * @returns Promise with list of all API products\n */\n getApiProducts(): Promise<KuadrantList<APIProduct>>;\n\n /**\n * Fetch a single API product\n * @param namespace - Kubernetes namespace\n * @param name - API product name\n * @returns Promise with the API product\n */\n getApiProduct(namespace: string, name: string): Promise<APIProduct>;\n\n /**\n * Create a new API product\n * @param product - API product specification\n * @returns Promise with the created API product\n */\n createApiProduct(product: APIProduct): Promise<APIProduct>;\n\n /**\n * Update an existing API product\n * @param namespace - Kubernetes namespace\n * @param name - API product name\n * @param patch - Partial API product spec with fields to update\n * @returns Promise with the updated API product\n */\n updateApiProduct(\n namespace: string,\n name: string,\n patch: Partial<APIProduct>,\n ): Promise<APIProduct>;\n\n /**\n * Delete an API product\n * @param namespace - Kubernetes namespace\n * @param name - API product name\n * @returns Promise that resolves when deletion completes\n */\n deleteApiProduct(namespace: string, name: string): Promise<void>;\n\n // ===== HTTP Routes & Policies =====\n\n /**\n * Fetch all HTTPRoute(s)\n * @returns Promise with list of all HTTP routes\n */\n getHttpRoutes(): Promise<K8sList>;\n\n /**\n * Fetch a specific HTTPRoute\n * @param namespace - Kubernetes namespace\n * @param name - HTTPRoute name\n * @returns Promise with an HTTPRoute\n */\n getHttpRoute(namespace: string, name: string): Promise<K8sResource>;\n\n // ===== Plan Policies =====\n\n /**\n * Fetch all plan policies\n * @returns Promise with list of all plan policies\n */\n getPlanPolicies(): Promise<KuadrantList<PlanPolicy>>;\n\n // ===== Auth Policies =====\n\n /**\n * Fetch all auth policies\n * @returns Promise with list of all auth policies\n */\n getAuthPolicies(): Promise<KuadrantList<AuthPolicy>>;\n\n // ===== RateLimitPolicies =====\n\n /**\n * Fetch all ratelimitpolicies\n * @returns Promise with list of all ratelimitpolicies\n */\n getRateLimitPolicies(): Promise<KuadrantList<RateLimitPolicy>>;\n}\n\n/**\n * API reference for the Kuadrant API\n */\nexport const kuadrantApiRef = createApiRef<KuadrantAPI>({\n id: 'plugin.kuadrant.service',\n});\n\n/**\n * Implementation of the Kuadrant API client\n */\nexport class KuadrantApiClient implements KuadrantAPI {\n private readonly discoveryApi: DiscoveryApi;\n private readonly fetchApi: FetchApi;\n\n constructor(options: Options) {\n this.discoveryApi = options.discoveryApi;\n this.fetchApi = options.fetchApi;\n }\n\n /**\n * Get the base URL for the backend API\n */\n private async getBaseUrl(): Promise<string> {\n return await this.discoveryApi.getBaseUrl('');\n }\n\n /**\n * Wrapper for GET requests with automatic retry logic\n * Retries on network failures or 5xx errors with exponential backoff\n */\n private async fetchWithRetry<T>(url: string, errorMsg: string = \"\"): Promise<T> {\n return retry(\n async (bail) => {\n const response = await this.fetchApi.fetch(url);\n if (response.status === 401 || response.status === 403) { // Don't retry on Unauthenticated/Unauthorized\n const error = await handleFetchError(response);\n bail(new Error(error));\n }\n else if (!response.ok) {\n const error = await handleFetchError(response);\n throw new Error(`${errorMsg} ${error}`);\n }\n return await response.json();\n },\n RETRY_OPTIONS,\n );\n }\n\n /**\n * Wrapper for mutations (POST, PATCH, DELETE) without retry\n * These operations are not retried to avoid duplicate side effects\n */\n private async fetchWithoutRetry<T>(\n url: string,\n options: RequestInit,\n errorMsg: string = \"\"): Promise<T> {\n const response = await this.fetchApi.fetch(url, {\n headers: {\n \"Content-Type\": \"application/json\",\n },\n ...options,\n });\n\n if (!response.ok) {\n const error = await handleFetchError(response);\n throw new Error(`${errorMsg} ${error}`);\n }\n\n // DELETE operations don't return a body\n if (options.method === 'DELETE') {\n return undefined as T;\n }\n\n return await response.json();\n }\n\n // ===== API Requests Implementation =====\n\n async getRequests(): Promise<KuadrantList<APIKey>> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithRetry(\n `${baseUrl}kuadrant/requests/my`,\n \"Failed to fetch API Key requests.\"\n );\n }\n\n async getAllRequests(): Promise<KuadrantList<APIKey>> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithRetry(\n `${baseUrl}kuadrant/requests`,\n \"Failed to fetch API Key requests.\"\n );\n }\n\n async getRequestsByNamespace(namespace: string): Promise<KuadrantList<APIKey>> {\n const baseUrl = await this.getBaseUrl();\n const url = namespace\n ? `${baseUrl}kuadrant/requests/my?namespace=${namespace}`\n : `${baseUrl}kuadrant/requests/my`;\n return this.fetchWithRetry(url, \"Failed to fetch API Key requests by namespace.\");\n }\n\n async getRequest(namespace: string, name: string): Promise<APIKey> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithRetry(\n `${baseUrl}kuadrant/requests/${namespace}/${name}`,\n \"Failed to fetch API Key request.\"\n );\n }\n\n async createRequest(request: APIKeyRequest): Promise<APIKey> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithoutRetry(`${baseUrl}kuadrant/requests`, {\n method: 'POST',\n body: JSON.stringify(request),\n }, \"Failed to create APIKey request.\");\n }\n\n async updateRequest(\n namespace: string,\n name: string,\n patch: Partial<APIKeySpec>,\n ): Promise<APIKey> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithoutRetry(`${baseUrl}kuadrant/requests/${namespace}/${name}`, {\n method: 'PATCH',\n body: JSON.stringify(patch),\n }, \"Failed to update APIKey request.\");\n }\n\n async deleteRequest(namespace: string, name: string): Promise<void> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithoutRetry(`${baseUrl}kuadrant/requests/${namespace}/${name}`, {\n method: 'DELETE',\n }, \"Failed to delete APIKey request.\");\n }\n\n async approveRequest(namespace: string, name: string, reviewedBy: string = \"system\"): Promise<APIKey> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithoutRetry(`${baseUrl}kuadrant/requests/${namespace}/${name}/approve`, {\n method: 'POST',\n body: JSON.stringify({ reviewedBy }),\n }, \"Failed to approve APIKey request.\");\n }\n\n async rejectRequest(namespace: string, name: string, reviewedBy: string = \"system\"): Promise<APIKey> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithoutRetry(`${baseUrl}kuadrant/requests/${namespace}/${name}/reject`, {\n method: 'POST',\n body: JSON.stringify({ reviewedBy }),\n }, \"Failed to reject APIKey request.\");\n }\n\n async bulkApproveRequests(\n requests: Array<{ namespace: string; name: string }>,\n reviewedBy: string): Promise<Array<BulkOperationResult>> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithoutRetry(`${baseUrl}kuadrant/requests/bulk-approve`, {\n method: 'POST',\n body: JSON.stringify({ requests, reviewedBy }),\n }, \"Failed to bulk approve APIKey requests.\");\n }\n\n async bulkRejectRequests(\n requests: Array<{ namespace: string; name: string }>,\n reviewedBy: string): Promise<Array<BulkOperationResult>> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithoutRetry(`${baseUrl}kuadrant/requests/bulk-reject`, {\n method: 'POST',\n body: JSON.stringify({ requests, reviewedBy }),\n }, \"Failed to bulk reject APIKey requests\");\n }\n\n // ===== API Keys/Secrets Implementation =====\n\n async getApiKey(namespace: string, name: string): Promise<APIKey> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithRetry(\n `${baseUrl}kuadrant/apikeys/${namespace}/${name}`,\n \"Failed to fetch API Key.\"\n );\n }\n\n async getApiKeySecret(\n namespace: string,\n name: string,\n ): Promise<ExtractedSecret> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithRetry(\n `${baseUrl}kuadrant/apikeys/${namespace}/${name}/secret`,\n \"Failed to fetch API Key Secret.\"\n );\n }\n\n // ===== API Products Implementation =====\n\n async getApiProducts(): Promise<KuadrantList<APIProduct>> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithRetry(\n `${baseUrl}kuadrant/apiproducts`,\n \"Failed to fetch API Products.\"\n );\n }\n\n async getApiProduct(namespace: string, name: string): Promise<APIProduct> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithRetry(\n `${baseUrl}kuadrant/apiproducts/${namespace}/${name}`,\n \"Failed to fetch API Product.\"\n );\n }\n\n async createApiProduct(product: APIProduct): Promise<APIProduct> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithoutRetry(`${baseUrl}kuadrant/apiproducts`, {\n method: 'POST',\n body: JSON.stringify(product),\n }, \"Failed to create API Product.\");\n }\n\n async updateApiProduct(\n namespace: string,\n name: string,\n patch: Partial<APIProduct>,\n ): Promise<APIProduct> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithoutRetry(`${baseUrl}kuadrant/apiproducts/${namespace}/${name}`, {\n method: 'PATCH',\n body: JSON.stringify(patch),\n }, \"Failed to update API Product.\");\n }\n\n async deleteApiProduct(namespace: string, name: string): Promise<void> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithoutRetry(`${baseUrl}kuadrant/apiproducts/${namespace}/${name}`, {\n method: 'DELETE',\n }, \"Failed to delete API Product.\");\n }\n\n // ===== HTTP Routes =====\n\n async getHttpRoutes(): Promise<K8sList> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithRetry(\n `${baseUrl}kuadrant/httproutes`,\n \"Failed to fetch HTTPRoutes.\"\n );\n }\n\n async getHttpRoute(namespace: string, name: string): Promise<K8sResource> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithRetry(\n `${baseUrl}kuadrant/httproutes/${namespace}/${name}`,\n `Failed to fetc HTTPRoute ${namespace}/${name}.`\n );\n }\n\n // ===== Plan Policies Implementation =====\n\n async getPlanPolicies(): Promise<KuadrantList<PlanPolicy>> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithRetry(\n `${baseUrl}kuadrant/planpolicies`,\n \"Failed to fetch PlanPolicies.\"\n );\n }\n\n // ===== AuthPolicies Implementation =====\n\n async getAuthPolicies(): Promise<KuadrantList<AuthPolicy>> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithRetry(\n `${baseUrl}kuadrant/authpolicies`,\n \"Failed to fetch AuthPolicies.\"\n );\n }\n\n // ===== RateLimitPolicies Implementation =====\n\n async getRateLimitPolicies(): Promise<KuadrantList<RateLimitPolicy>> {\n const baseUrl = await this.getBaseUrl();\n return this.fetchWithRetry(\n `${baseUrl}kuadrant/ratelimitpolicies`,\n \"Failed to fetch RateLimitPolicies.\"\n );\n }\n}\n","import React from 'react';\nimport { Box, Grid, Typography, Chip, useTheme } from '@material-ui/core';\nimport { StatusCondition } from \"../../types/api-management\";\n\nexport type PlanPoliciesProps = {\n statusCondition: StatusCondition | null;\n discoveredPlans: Array<{\n tier: string;\n limits?: {\n daily?: number;\n monthly?: number;\n yearly?: number;\n };\n }> | null;\n};\n\nexport type AuthPoliciesProps = {\n namespacedName: {\n name: string | null;\n namespace: string | null;\n } | null;\n statusCondition: StatusCondition | null;\n};\n\nexport type RateLimitPoliciesProps = {\n namespacedName: {\n name: string | null;\n namespace: string | null;\n } | null;\n statusCondition: StatusCondition | null;\n};\n\ninterface ApiProductPoliciesProps {\n planPolicy: PlanPoliciesProps | null;\n authPolicy: AuthPoliciesProps | null;\n rateLimitPolicy: RateLimitPoliciesProps | null;\n includeTopMargin?: boolean;\n}\n// Displays APIProduct policies\nexport const ApiProductPolicies: React.FC<ApiProductPoliciesProps> = ({\n planPolicy,\n authPolicy,\n rateLimitPolicy,\n includeTopMargin = true,\n}) => {\n const theme = useTheme();\n\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 <Grid container spacing={2}>\n {/* planpolicy chip shown when either plan exist or ratelimitpolicy does not exist*/}\n {((planPolicy?.statusCondition && planPolicy.statusCondition.status === \"True\") || !rateLimitPolicy?.statusCondition) ? (\n <Grid item xs={12} md={6}>\n <Typography variant=\"body2\" color=\"textSecondary\" gutterBottom>\n Plan Policy\n </Typography>\n {planPolicy?.statusCondition ? (\n <Box>\n <Chip\n label={planPolicy.statusCondition.reason}\n size=\"small\"\n style={{\n backgroundColor: planPolicy.statusCondition.status === \"True\" ? \"#4caf50\" : \"#ff9800\",\n color: \"#fff\",\n marginBottom: 8,\n }}\n />\n {planPolicy.discoveredPlans && planPolicy.discoveredPlans.length > 0 && (\n <>\n <Typography\n variant=\"caption\"\n display=\"block\"\n gutterBottom\n color=\"textSecondary\"\n style={{ marginTop: 8 }}\n >\n Available PlanPolicy Tiers:\n </Typography>\n <Box display=\"flex\" flexWrap=\"wrap\" mt={1} style={{ gap: 8 }}>\n {planPolicy.discoveredPlans.map((plan: any, idx: number) => {\n const limitText = plan.limits?.daily\n ? `${plan.limits.daily}/day`\n : plan.limits?.monthly\n ? `${plan.limits.monthly}/month`\n : plan.limits?.yearly\n ? `${plan.limits.yearly}/year`\n : 'No limit';\n 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 </Box>\n ) : (\n <Chip\n label=\"NotFound\"\n size=\"small\"\n style={{\n backgroundColor: \"#ff9800\",\n color: \"#fff\",\n marginBottom: 8,\n }}\n />\n )}\n </Grid>\n ) : (\n <Grid item xs={12} md={6}>\n <Typography variant=\"body2\" color=\"textSecondary\" gutterBottom>\n RateLimitPolicy\n </Typography>\n {rateLimitPolicy?.statusCondition ? (\n <Box>\n <Chip\n label={rateLimitPolicy.statusCondition.reason}\n size=\"small\"\n style={{\n backgroundColor: rateLimitPolicy.statusCondition.status === \"True\" ? \"#4caf50\" : \"#ff9800\",\n color: \"#fff\",\n marginBottom: 8,\n }}\n />\n {rateLimitPolicy.statusCondition.status === \"True\" && (\n <Grid container spacing={2}>\n <Grid item xs={12} md={6}>\n <Box>\n <Typography variant=\"body2\" color=\"textSecondary\" gutterBottom>\n Resource Name\n </Typography>\n <Typography variant=\"body2\">\n {rateLimitPolicy.namespacedName?.name || \"No RateLimit Policy name available\"}\n </Typography>\n </Box>\n </Grid>\n <Grid item xs={12} md={6}>\n <Box>\n <Typography variant=\"body2\" color=\"textSecondary\" gutterBottom>\n Resource Namespace\n </Typography>\n <Typography variant=\"body2\">\n {rateLimitPolicy.namespacedName?.namespace || \"No RateLimit Policy namespace available\"}\n </Typography>\n </Box>\n </Grid>\n </Grid>\n )}\n </Box>\n ) : (\n <Chip\n label=\"NotFound\"\n size=\"small\"\n style={{\n backgroundColor: \"#ff9800\",\n color: \"#fff\",\n marginBottom: 8,\n }}\n />\n )}\n </Grid>\n )}\n <Grid item xs={12} md={6}>\n <Typography variant=\"body2\" color=\"textSecondary\" gutterBottom>\n Auth Policy\n </Typography>\n {authPolicy?.statusCondition ? (\n <Box>\n <Chip\n label={authPolicy.statusCondition.reason}\n size=\"small\"\n style={{\n backgroundColor: authPolicy.statusCondition.status === \"True\" ? \"#4caf50\" : \"#ff9800\",\n color: \"#fff\",\n marginBottom: 8,\n }}\n />\n {authPolicy.statusCondition.status === \"True\" && (\n <Grid container spacing={2}>\n <Grid item xs={12} md={6}>\n <Box>\n <Typography variant=\"body2\" color=\"textSecondary\" gutterBottom>\n Resource Name\n </Typography>\n <Typography variant=\"body2\">\n {authPolicy.namespacedName?.name || \"No Auth Policy name available\"}\n </Typography>\n </Box>\n </Grid>\n <Grid item xs={12} md={6}>\n <Box>\n <Typography variant=\"body2\" color=\"textSecondary\" gutterBottom>\n Resource Namespace\n </Typography>\n <Typography variant=\"body2\">\n {authPolicy.namespacedName?.namespace || \"No Auth Policy namespace available\"}\n </Typography>\n </Box>\n </Grid>\n </Grid>\n )}\n </Box>\n ) : (\n <Chip\n label=\"NotFound\"\n size=\"small\"\n style={{\n backgroundColor: \"#ff9800\",\n color: \"#fff\",\n marginBottom: 8,\n }}\n />\n )}\n </Grid>\n </Grid>\n </Box>\n );\n}\n","/**\n * Find a policy that targets a specific APIProduct\n *\n * @param policies - Array of policies with targetRef\n * @param routeNamespace - Namespace of the target HTTPRoute\n * @param routeName - Name of the target HTTPRoute\n * @returns The matching policy or null if not found\n *\n * @remarks\n * The function matches a policy if:\n * - The targetRef.kind is 'HTTPRoute'\n * - The targetRef.name matches the routeName parameter\n * - The targetRef.namespace (or policy's metadata.namespace if not specified) matches the routeNamespace parameter\n */\nexport const getPolicyForRoute = (\n policies: any[] | undefined,\n routeNamespace: string,\n routeName: string,\n) => {\n if (!policies) return null;\n\n return policies.find((pp: any) => {\n const ref = pp.spec.targetRef;\n const targetNamespace = ref?.namespace ?? pp.metadata.namespace;\n\n return (\n ref?.kind === 'HTTPRoute' &&\n ref?.name === routeName &&\n targetNamespace === routeNamespace\n );\n }) || null;\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","selectedRateLimitPolicy","classes","kuadrantApi","useApi","kuadrantApiRef","loading","setLoading","useState","displayName","setDisplayName","description","setDescription","version","setVersion","publishStatus","setPublishStatus","lifecycle","setLifecycle","approvalMode","setApprovalMode","tags","setTags","targetRef","setTargetRef","tagInput","setTagInput","contactEmail","setContactEmail","contactTeam","setContactTeam","docsURL","setDocsURL","openAPISpec","setOpenAPISpec","planPolicyProps","setPlanPoliciesProps","authPolicyProps","setAuthPolicyProps","error","setError","saving","setSaving","openAPISpecError","setOpenAPISpecError","useEffect","getApiProduct","then","data","spec","metadata","labels","contact","email","team","documentation","openAPISpecURL","planPolicyCondition","status","conditions","find","c","type","statusCondition","discoveredPlans","authPolicyCondition","authPolicyName","parts","split","parseNameFromMessage","message","namespacedName","catch","err","rateLimitPolicies","rateLimitPoliciesError","useAsync","async","getRateLimitPolicies","getPolicyForRoute","items","rateLimitPolicyAcceptedCondition","rateLimitPolicyProps","handleAddTag","Dialog","maxWidth","fullWidth","DialogTitle","DialogContent","Alert","severity","style","strong","Progress","Box","className","Typography","variant","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","ApiProductPolicies","planPolicy","authPolicy","rateLimitPolicy","includeTopMargin","select","MenuItem","FormControl","component","RadioGroup","row","FormControlLabel","control","Radio","DialogActions","Button","patch","updateApiProduct","Error","String","startIcon","CircularProgress","createPermission","attributes","action","kuadrantPlanPolicyListPermission","kuadrantApiProductCreatePermission","kuadrantApiProductReadAllPermission","kuadrantApiProductUpdateOwnPermission","kuadrantApiProductUpdateAllPermission","kuadrantApiProductDeleteOwnPermission","kuadrantApiProductDeleteAllPermission","kuadrantApiProductListPermission","kuadrantApiKeyCreatePermission","resourceType","kuadrantApiKeyUpdateOwnPermission","kuadrantApiKeyUpdateAllPermission","kuadrantApiKeyDeleteOwnPermission","kuadrantApiKeyDeleteAllPermission","kuadrantApiKeyApprovePermission","useKuadrantPermission","permission","resourceRef","permissionRequest","result","usePermission","allowed","canDeleteResource","ownerId","currentUserId","canDeleteOwn","canDeleteAll","getMyApiKeysStatusChipStyle","phase","base","border","backgroundColor","getApprovalQueueStatusChipStyle","getLifecycleChipStyle","ConfirmDeleteDialog","confirmText","deleting","onConfirm","onCancel","inputValue","setInputValue","requiresTextConfirmation","canConfirm","WarningIcon","span","DialogContentText","whiteSpace","mt","gutterBottom","autoFocus","createApiRef","id","useTheme","p","bgcolor","background","default","borderRadius","divider","md","reason","plan","idx","limitText","limits","daily","monthly","yearly","tier","policies","routeNamespace","routeName","pp","ref","targetNamespace","kind"],"sourceRoot":""}
@@ -1,2 +1,2 @@
1
- "use strict";(self.webpackChunkinternal_plugin_kuadrant=self.webpackChunkinternal_plugin_kuadrant||[]).push([[9051],{14933:(e,a,t)=>{t.d(a,{c:()=>g});var i=t(31085),r=t(95478),n=t(76891),s=t(61477),l=t(46805),d=t(10394),o=t(72501),u=t(95061),c=t(48543),p=t(81215),m=t(26343),b=t(16249),k=t(93453),y=t(64947),h=t(78467),A=t(39053),x=t(22097),f=t(64047);const g=({open:e,onClose:a,onSuccess:t,apiProductName:g,namespace:v,userEmail:j,plans:w})=>{const C=(0,x.useApi)(f.s),I=(0,x.useApi)(x.alertApiRef),[R,S]=(0,r.useState)(""),[P,q]=(0,r.useState)(""),[T,_]=(0,r.useState)(!1),[E,W]=(0,r.useState)(null),$=()=>{S(""),q(""),W(null),a()};return(0,i.jsxs)(n.A,{open:e,onClose:$,maxWidth:"sm",fullWidth:!0,children:[(0,i.jsx)(s.A,{children:"Request API Access"}),(0,i.jsxs)(l.A,{children:[(0,i.jsxs)(d.A,{mb:2,p:1.5,bgcolor:"info.light",borderRadius:1,display:"flex",alignItems:"flex-start",style:{gap:8},children:[(0,i.jsx)(A.A,{color:"primary",fontSize:"small",style:{marginTop:2}}),(0,i.jsx)(o.A,{variant:"body2",children:"Your request will be reviewed by an API owner before access is granted."})]}),E&&(0,i.jsx)(d.A,{mb:2,p:2,bgcolor:"error.main",color:"error.contrastText",borderRadius:1,children:(0,i.jsx)(o.A,{variant:"body2",children:E})}),(0,i.jsxs)(u.A,{fullWidth:!0,margin:"normal",disabled:T,"data-testid":"tier-select-form",children:[(0,i.jsx)(c.A,{id:"tier-select-label",children:"Select Tier"}),(0,i.jsx)(p.A,{labelId:"tier-select-label","data-testid":"tier-select",value:R,onChange:e=>S(e.target.value),disabled:T,children:w.map(e=>{const a=Object.entries(e.limits||{}).map(([e,a])=>`${a} per ${e}`).join(", ");return(0,i.jsxs)(m.A,{value:e.tier,"data-testid":`tier-option-${e.tier}`,children:[e.tier," ",a?`(${a})`:""]},e.tier)})})]}),(0,i.jsx)(b.A,{label:"Use Case (optional)",placeholder:"Describe how you plan to use this API",multiline:!0,rows:3,fullWidth:!0,margin:"normal",value:P,onChange:e=>q(e.target.value),helperText:"Explain your intended use of this API for admin review",disabled:T})]}),(0,i.jsxs)(k.A,{children:[(0,i.jsx)(y.A,{onClick:$,disabled:T,children:"Cancel"}),(0,i.jsx)(y.A,{onClick:async()=>{if(R){_(!0),W(null);try{await C.createRequest({apiProductName:g,namespace:v,planTier:R,useCase:P.trim()||"",userEmail:j}),I.post({message:"API key requested successfully",severity:"success",display:"transient"}),S(""),q(""),t()}catch(e){const a=e instanceof Error?e.message:"unknown error occurred";I.post({message:`Failed to request API key: ${a}`,severity:"error",display:"transient"}),W(a)}finally{_(!1)}}},color:"primary",variant:"contained",disabled:!R||T,startIcon:T?(0,i.jsx)(h.A,{size:16,color:"inherit"}):void 0,children:T?"Submitting...":"Submit Request"})]})]})}},34955:(e,a,t)=>{t.d(a,{Al:()=>s,EM:()=>o,FL:()=>n,J:()=>r,KV:()=>h,R_:()=>u,U3:()=>l,dp:()=>p,jH:()=>y,q0:()=>m,uL:()=>k,v_:()=>d,vs:()=>c,z4:()=>b});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 r=(0,i.i)({name:"kuadrant.planpolicy.list",attributes:{action:"read"}}),n=(0,i.i)({name:"kuadrant.apiproduct.create",attributes:{action:"create"}}),s=((0,i.i)({name:"kuadrant.apiproduct.read.own",attributes:{action:"read"}}),(0,i.i)({name:"kuadrant.apiproduct.read.all",attributes:{action:"read"}})),l=(0,i.i)({name:"kuadrant.apiproduct.update.own",attributes:{action:"update"}}),d=(0,i.i)({name:"kuadrant.apiproduct.update.all",attributes:{action:"update"}}),o=(0,i.i)({name:"kuadrant.apiproduct.delete.own",attributes:{action:"delete"}}),u=(0,i.i)({name:"kuadrant.apiproduct.delete.all",attributes:{action:"delete"}}),c=(0,i.i)({name:"kuadrant.apiproduct.list",attributes:{action:"read"}}),p=(0,i.i)({name:"kuadrant.apikey.create",attributes:{action:"create"},resourceType:"apiproduct"}),m=((0,i.i)({name:"kuadrant.apikey.read.own",attributes:{action:"read"}}),(0,i.i)({name:"kuadrant.apikey.read.all",attributes:{action:"read"}}),(0,i.i)({name:"kuadrant.apikey.update.own",attributes:{action:"update"}})),b=(0,i.i)({name:"kuadrant.apikey.update.all",attributes:{action:"update"}}),k=(0,i.i)({name:"kuadrant.apikey.delete.own",attributes:{action:"delete"}}),y=(0,i.i)({name:"kuadrant.apikey.delete.all",attributes:{action:"delete"}}),h=(0,i.i)({name:"kuadrant.apikey.approve",attributes:{action:"update"}})},46205:(e,a,t)=>{t.d(a,{W:()=>n,l:()=>r});var i=t(24217);function r(e,a){const t="resourceType"in e?{permission:e,resourceRef:a}:{permission:e},r=(0,i.J)(t);return{allowed:r.allowed,loading:r.loading,error:r.error}}function n(e,a,t,i){return!!i||!(!t||e!==a)}},64047:(e,a,t)=>{t.d(a,{s:()=>r});var i=t(22097);t(54209);const r=(0,i.createApiRef)({id:"plugin.kuadrant.service"})}}]);
2
- //# sourceMappingURL=9051.db875198.chunk.js.map
1
+ "use strict";(self.webpackChunkinternal_plugin_kuadrant=self.webpackChunkinternal_plugin_kuadrant||[]).push([[9051],{14933:(e,a,t)=>{t.d(a,{c:()=>g});var i=t(31085),r=t(95478),n=t(76891),s=t(61477),l=t(46805),d=t(10394),o=t(72501),u=t(95061),c=t(48543),p=t(81215),m=t(26343),b=t(16249),k=t(93453),y=t(64947),h=t(78467),A=t(39053),x=t(22097),f=t(64047);const g=({open:e,onClose:a,onSuccess:t,apiProductName:g,namespace:v,userEmail:j,plans:w})=>{const C=(0,x.useApi)(f.s),I=(0,x.useApi)(x.alertApiRef),[R,S]=(0,r.useState)(""),[P,q]=(0,r.useState)(""),[T,_]=(0,r.useState)(!1),[E,W]=(0,r.useState)(null),$=()=>{S(""),q(""),W(null),a()};return(0,i.jsxs)(n.A,{open:e,onClose:$,maxWidth:"sm",fullWidth:!0,children:[(0,i.jsx)(s.A,{children:"Request API Access"}),(0,i.jsxs)(l.A,{children:[(0,i.jsxs)(d.A,{mb:2,p:1.5,bgcolor:"info.light",borderRadius:1,display:"flex",alignItems:"flex-start",style:{gap:8},children:[(0,i.jsx)(A.A,{color:"primary",fontSize:"small",style:{marginTop:2}}),(0,i.jsx)(o.A,{variant:"body2",children:"Your request will be reviewed by an API owner before access is granted."})]}),E&&(0,i.jsx)(d.A,{mb:2,p:2,bgcolor:"error.main",color:"error.contrastText",borderRadius:1,children:(0,i.jsx)(o.A,{variant:"body2",children:E})}),(0,i.jsxs)(u.A,{fullWidth:!0,margin:"normal",disabled:T,"data-testid":"tier-select-form",children:[(0,i.jsx)(c.A,{id:"tier-select-label",children:"Select Tier"}),(0,i.jsx)(p.A,{labelId:"tier-select-label","data-testid":"tier-select",value:R,onChange:e=>S(e.target.value),disabled:T,children:w.map(e=>{const a=Object.entries(e.limits||{}).map(([e,a])=>`${a} per ${e}`).join(", ");return(0,i.jsxs)(m.A,{value:e.tier,"data-testid":`tier-option-${e.tier}`,children:[e.tier," ",a?`(${a})`:""]},e.tier)})})]}),(0,i.jsx)(b.A,{label:"Use Case (optional)",placeholder:"Describe how you plan to use this API",multiline:!0,rows:3,fullWidth:!0,margin:"normal",value:P,onChange:e=>q(e.target.value),helperText:"Explain your intended use of this API for admin review",disabled:T})]}),(0,i.jsxs)(k.A,{children:[(0,i.jsx)(y.A,{onClick:$,disabled:T,children:"Cancel"}),(0,i.jsx)(y.A,{onClick:async()=>{if(R){_(!0),W(null);try{await C.createRequest({apiProductName:g,namespace:v,planTier:R,useCase:P.trim()||"",userEmail:j}),I.post({message:"API key requested successfully",severity:"success",display:"transient"}),S(""),q(""),t()}catch(e){const a=e instanceof Error?e.message:"unknown error occurred";I.post({message:`Failed to request API key: ${a}`,severity:"error",display:"transient"}),W(a)}finally{_(!1)}}},color:"primary",variant:"contained",disabled:!R||T,startIcon:T?(0,i.jsx)(h.A,{size:16,color:"inherit"}):void 0,children:T?"Submitting...":"Submit Request"})]})]})}},34955:(e,a,t)=>{t.d(a,{Al:()=>s,EM:()=>o,FL:()=>n,J:()=>r,KV:()=>h,R_:()=>u,U3:()=>l,dp:()=>p,jH:()=>y,q0:()=>m,uL:()=>k,v_:()=>d,vs:()=>c,z4:()=>b});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 r=(0,i.i)({name:"kuadrant.planpolicy.list",attributes:{action:"read"}}),n=((0,i.i)({name:"kuadrant.authpolicy.list",attributes:{action:"read"}}),(0,i.i)({name:"kuadrant.ratelimitpolicy.list",attributes:{action:"read"}}),(0,i.i)({name:"kuadrant.apiproduct.create",attributes:{action:"create"}})),s=((0,i.i)({name:"kuadrant.apiproduct.read.own",attributes:{action:"read"}}),(0,i.i)({name:"kuadrant.apiproduct.read.all",attributes:{action:"read"}})),l=(0,i.i)({name:"kuadrant.apiproduct.update.own",attributes:{action:"update"}}),d=(0,i.i)({name:"kuadrant.apiproduct.update.all",attributes:{action:"update"}}),o=(0,i.i)({name:"kuadrant.apiproduct.delete.own",attributes:{action:"delete"}}),u=(0,i.i)({name:"kuadrant.apiproduct.delete.all",attributes:{action:"delete"}}),c=(0,i.i)({name:"kuadrant.apiproduct.list",attributes:{action:"read"}}),p=(0,i.i)({name:"kuadrant.apikey.create",attributes:{action:"create"},resourceType:"apiproduct"}),m=((0,i.i)({name:"kuadrant.apikey.read.own",attributes:{action:"read"}}),(0,i.i)({name:"kuadrant.apikey.read.all",attributes:{action:"read"}}),(0,i.i)({name:"kuadrant.apikey.update.own",attributes:{action:"update"}})),b=(0,i.i)({name:"kuadrant.apikey.update.all",attributes:{action:"update"}}),k=(0,i.i)({name:"kuadrant.apikey.delete.own",attributes:{action:"delete"}}),y=(0,i.i)({name:"kuadrant.apikey.delete.all",attributes:{action:"delete"}}),h=(0,i.i)({name:"kuadrant.apikey.approve",attributes:{action:"update"}})},46205:(e,a,t)=>{t.d(a,{W:()=>n,l:()=>r});var i=t(24217);function r(e,a){const t="resourceType"in e?{permission:e,resourceRef:a}:{permission:e},r=(0,i.J)(t);return{allowed:r.allowed,loading:r.loading,error:r.error}}function n(e,a,t,i){return!!i||!(!t||e!==a)}},64047:(e,a,t)=>{t.d(a,{s:()=>r});var i=t(22097);t(54209);const r=(0,i.createApiRef)({id:"plugin.kuadrant.service"})}}]);
2
+ //# sourceMappingURL=9051.d45ac154.chunk.js.map