@kuadrant/kuadrant-backstage-plugin-frontend 0.0.2-dev-90b0d1b → 0.0.2-dev-21b2773
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.
- package/dist/components/ConfirmDeleteDialog/ConfirmDeleteDialog.esm.js +1 -1
- package/dist/components/ConfirmDeleteDialog/ConfirmDeleteDialog.esm.js.map +1 -1
- package/dist/components/KuadrantPage/KuadrantPage.esm.js +23 -2
- package/dist/components/KuadrantPage/KuadrantPage.esm.js.map +1 -1
- package/dist-scalprum/{internal.plugin-kuadrant.160fafb604b5892d378c.js → internal.plugin-kuadrant.f6c080742dae4db9f38e.js} +2 -2
- package/dist-scalprum/{internal.plugin-kuadrant.160fafb604b5892d378c.js.map → internal.plugin-kuadrant.f6c080742dae4db9f38e.js.map} +1 -1
- package/dist-scalprum/plugin-manifest.json +2 -2
- package/dist-scalprum/static/{4306.f3bb29b1.chunk.js → 4306.e5d97c39.chunk.js} +2 -2
- package/dist-scalprum/static/4306.e5d97c39.chunk.js.map +1 -0
- package/dist-scalprum/static/7632.d54b3f92.chunk.js +2 -0
- package/dist-scalprum/static/7632.d54b3f92.chunk.js.map +1 -0
- package/package.json +1 -1
- package/dist-scalprum/static/4306.f3bb29b1.chunk.js.map +0 -1
- package/dist-scalprum/static/7632.e4c19350.chunk.js +0 -2
- package/dist-scalprum/static/7632.e4c19350.chunk.js.map +0 -1
|
@@ -1,2 +0,0 @@
|
|
|
1
|
-
"use strict";(self.webpackChunkinternal_plugin_kuadrant=self.webpackChunkinternal_plugin_kuadrant||[]).push([[7632],{63221:(e,a,t)=>{t.d(a,{K:()=>v});var s=t(31085),l=t(95478),n=t(76891),r=t(61477),i=t(10394),o=t(46805),d=t(59461),c=t(72501),u=t(16249),p=t(93453),m=t(64947),h=t(78467),x=t(77225);const v=({open:e,title:a,description:t,confirmText:v,severity:A="normal",deleting:g=!1,onConfirm:j,onCancel:f})=>{const[y,b]=(0,l.useState)("");(0,l.useEffect)(()=>{e||b("")},[e]);const P="high"===A&&v,S=!P||y===v;return(0,s.jsxs)(n.A,{open:e,onClose:g?void 0:f,maxWidth:"sm",fullWidth:!0,children:[(0,s.jsxs)(r.A,{children:["high"===A&&(0,s.jsxs)(i.A,{display:"flex",alignItems:"center",style:{gap:8},children:[(0,s.jsx)(x.A,{color:"error"}),(0,s.jsx)("span",{children:a})]}),"high"!==A&&a]}),(0,s.jsxs)(o.A,{children:[(0,s.jsx)(d.A,{children:t}),P&&(0,s.jsxs)(i.A,{mt:2,children:[(0,s.jsxs)(c.A,{variant:"body2",color:"textSecondary",gutterBottom:!0,children:["Type ",(0,s.jsx)("strong",{children:v})," to confirm:"]}),(0,s.jsx)(u.A,{fullWidth:!0,variant:"outlined",size:"small",value:y,onChange:e=>b(e.target.value),disabled:g,autoFocus:!0,placeholder:v})]})]}),(0,s.jsxs)(p.A,{children:[(0,s.jsx)(m.A,{onClick:f,disabled:g,children:"Cancel"}),(0,s.jsx)(m.A,{onClick:()=>{S&&j()},color:"secondary",variant:"contained",disabled:g||!S,startIcon:g?(0,s.jsx)(h.A,{size:16,color:"inherit"}):void 0,children:g?"Deleting...":"Delete"})]})]})}},84189:(e,a,t)=>{t.d(a,{n:()=>j});var s=t(31085),l=t(95478),n=t(76891),r=t(61477),i=t(46805),o=t(10394),d=t(72501),c=t(95061),u=t(48543),p=t(81215),m=t(26343),h=t(16249),x=t(93453),v=t(64947),A=t(78467),g=t(22097);const j=({open:e,onClose:a,onSuccess:t,request:j,availablePlans:f})=>{const y=(0,g.useApi)(g.configApiRef),b=(0,g.useApi)(g.fetchApiRef),P=y.getString("backend.baseUrl"),[S,k]=(0,l.useState)(""),[w,C]=(0,l.useState)(""),[T,R]=(0,l.useState)(!1),[I,$]=(0,l.useState)("");(0,l.useEffect)(()=>{e&&j&&(k(j.spec.planTier||""),C(j.spec.useCase||""),$(""))},[e,j]);const q=()=>{T||($(""),a())};return(0,s.jsxs)(n.A,{open:e,onClose:q,maxWidth:"sm",fullWidth:!0,children:[(0,s.jsx)(r.A,{children:"Edit API Access Request"}),(0,s.jsxs)(i.A,{children:[I&&(0,s.jsx)(o.A,{mb:2,p:2,bgcolor:"error.main",color:"error.contrastText",borderRadius:1,children:(0,s.jsx)(d.A,{variant:"body2",children:I})}),(0,s.jsxs)(c.A,{fullWidth:!0,margin:"normal",children:[(0,s.jsx)(u.A,{children:"Tier"}),(0,s.jsx)(p.A,{value:S,onChange:e=>k(e.target.value),disabled:T,children:f.map(e=>{const a=Object.entries(e.limits||{}).map(([e,a])=>`${a} per ${e}`).join(", ");return(0,s.jsxs)(m.A,{value:e.tier,children:[e.tier," ",a?`(${a})`:""]},e.tier)})})]}),(0,s.jsx)(h.A,{label:"Use Case",placeholder:"Describe how you plan to use this API",multiline:!0,rows:3,fullWidth:!0,margin:"normal",value:w,onChange:e=>C(e.target.value),disabled:T,helperText:"Explain your intended use of this API for admin review"})]}),(0,s.jsxs)(x.A,{children:[(0,s.jsx)(v.A,{onClick:q,disabled:T,children:"Cancel"}),(0,s.jsx)(v.A,{onClick:async()=>{if(S){$(""),R(!0);try{const e={spec:{planTier:S,useCase:w.trim()}},s=await b.fetch(`${P}/api/kuadrant/requests/${j.metadata.namespace}/${j.metadata.name}`,{method:"PATCH",headers:{"Content-Type":"application/json"},body:JSON.stringify(e)});if(!s.ok){const e=await s.json().catch(()=>({}));throw new Error(e.error||`Failed to update request: ${s.status}`)}t(),a()}catch(e){console.error("Error updating API key request:",e),$(e instanceof Error?e.message:"Unknown error occurred")}finally{R(!1)}}else $("Please select a tier")},color:"primary",variant:"contained",disabled:!S||T,startIcon:T?(0,s.jsx)(A.A,{size:16,color:"inherit"}):void 0,children:T?"Saving...":"Save Changes"})]})]})}},95251:(e,a,t)=>{t.r(a),t.d(a,{KuadrantPage:()=>ae});var s=t(31085),l=t(95478),n=t.n(l),r=t(67720),i=t(10394),o=t(29365),d=t(72501),c=t(42899),u=t(64947),p=t(18466),m=t(39590),h=t(75625),x=t(37725),v=t(25010),A=t(289),g=t(55639),j=t(45210),f=t(46681),y=t(86687),b=t(42367),P=t(96040),S=t(91638),k=t(22097),w=t(16281),C=t(71677),T=t(78467),R=t(31653),I=t(38605),$=t(37757),q=t(26343),W=t(32269),E=t(61524),N=t(71407),D=t(84189),z=t(63221);const M=()=>{var e,a;const t=(0,k.useApi)(k.configApiRef),n=(0,k.useApi)(k.fetchApiRef),c=(0,k.useApi)(k.identityApiRef),u=(0,k.useApi)(k.alertApiRef),p=t.getString("backend.baseUrl"),[m,h]=(0,l.useState)(0),[,A]=(0,l.useState)(""),[g,j]=(0,l.useState)(new Set),[f,b]=(0,l.useState)(null),[w,M]=(0,l.useState)(null),[K,L]=(0,l.useState)({open:!1,request:null,plans:[]}),[B,U]=(0,l.useState)(0),[H,F]=(0,l.useState)(null),[O,_]=(0,l.useState)({open:!1,request:null});(0,S.A)(async()=>{const e=await c.getBackstageIdentity(),a=e.userEntityRef.split("/")[1]||"guest";console.log(`MyApiKeysCard: setting userId from userEntityRef: ${e.userEntityRef} -> "${a}"`),A(a)},[c]);const[V,J]=(0,l.useState)(new Set),{value:Y,loading:X,error:G}=(0,S.A)(async()=>{const e=await n.fetch(`${p}/api/kuadrant/requests/my`);if(!e.ok)throw new Error("failed to fetch requests");return(await e.json()).items||[]},[p,n,B]);if(X)return(0,s.jsx)(P.n,{title:"My API Keys",children:(0,s.jsx)(y.k,{})});if(G)return(0,s.jsx)(P.n,{title:"My API Keys",children:(0,s.jsxs)(d.A,{color:"error",children:["Error loading API keys: ",G.message]})});const Q=(Y||[]).filter(e=>!V.has(e.metadata.name)),Z=Q.filter(e=>{var a;return"Approved"===(null===(a=e.status)||void 0===a?void 0:a.phase)}),ee=Q.filter(e=>{var a;return!(null===(a=e.status)||void 0===a?void 0:a.phase)||"Pending"===e.status.phase}),ae=Q.filter(e=>{var a;return"Rejected"===(null===(a=e.status)||void 0===a?void 0:a.phase)}),te=()=>{b(null),M(null)},se=async()=>{if(!w)return;const e=w;te();try{var a;const s=null===(a=e.spec.apiProductRef)||void 0===a?void 0:a.name,l=e.metadata.namespace,r=await n.fetch(`${p}/api/kuadrant/apiproducts/${l}/${s}`);if(r.ok){var t;const a=(null===(t=(await r.json()).spec)||void 0===t?void 0:t.plans)||[];L({open:!0,request:e,plans:a})}else console.error("Failed to fetch API product"),L({open:!0,request:e,plans:[]})}catch(a){console.error("Error fetching plans:",a),L({open:!0,request:e,plans:[]})}},le=()=>{if(!w)return;const e=w;te(),_({open:!0,request:e})},ne=[{title:"API Product",field:"spec.apiProductRef.name",render:e=>{var a;const t=(null===(a=e.spec.apiProductRef)||void 0===a?void 0:a.name)||"unknown";return(0,s.jsx)(x.N_,{to:`/catalog/default/api/${t}/api-keys`,children:(0,s.jsx)("strong",{children:t})})}},{title:"Tier",field:"spec.planTier",render:e=>{const a="gold"===e.spec.planTier?"primary":"silver"===e.spec.planTier?"default":"secondary";return(0,s.jsx)(r.A,{label:e.spec.planTier,color:a,size:"small"})}},{title:"Use Case",field:"spec.useCase",render:e=>e.spec.useCase?(0,s.jsx)(C.Ay,{title:e.spec.useCase,placement:"top",children:(0,s.jsx)(d.A,{variant:"body2",style:{maxWidth:"200px",overflow:"hidden",textOverflow:"ellipsis",whiteSpace:"nowrap"},children:e.spec.useCase})}):(0,s.jsx)(d.A,{variant:"body2",children:"-"})},{title:"Status",field:"status.phase",render:e=>{var a;const t=(null===(a=e.status)||void 0===a?void 0:a.phase)||"Pending",l="Approved"===t?"primary":"Rejected"===t?"secondary":"default";return(0,s.jsx)(r.A,{label:t,color:l,size:"small"})}},{title:"Reason",field:"status.reason",render:e=>{var a;if(null===(a=e.status)||void 0===a?void 0:a.reason){const a="Rejected"===e.status.phase?"error":"textPrimary";return(0,s.jsx)(C.Ay,{title:e.status.reason,placement:"top",children:(0,s.jsx)(d.A,{variant:"body2",color:a,style:{maxWidth:"200px",overflow:"hidden",textOverflow:"ellipsis",whiteSpace:"nowrap"},children:e.status.reason})})}return(0,s.jsx)(d.A,{variant:"body2",color:"textSecondary",children:"-"})}},{title:"Reviewed By",field:"status.reviewedBy",render:e=>{var a,t;if(("Approved"===(null===(a=e.status)||void 0===a?void 0:a.phase)||"Rejected"===(null===(t=e.status)||void 0===t?void 0:t.phase))&&e.status.reviewedBy){const a=e.status.reviewedAt?new Date(e.status.reviewedAt).toLocaleDateString():"";return(0,s.jsxs)(i.A,{children:[(0,s.jsx)(d.A,{variant:"body2",children:e.status.reviewedBy}),a&&(0,s.jsx)(d.A,{variant:"caption",color:"textSecondary",children:a})]})}return(0,s.jsx)(d.A,{variant:"body2",color:"textSecondary",children:"-"})}},{title:"API Key",field:"status.apiKey",filtering:!1,render:e=>{var a;if("Approved"===(null===(a=e.status)||void 0===a?void 0:a.phase)&&e.status.apiKey){const a=g.has(e.metadata.name);return(0,s.jsxs)(i.A,{display:"flex",alignItems:"center",style:{gap:8},children:[(0,s.jsx)(i.A,{fontFamily:"monospace",fontSize:"0.875rem",children:a?e.status.apiKey:"•".repeat(20)+"..."}),(0,s.jsx)(C.Ay,{title:a?"hide key":"show key",children:(0,s.jsx)(o.A,{size:"small",onClick:()=>{return a=e.metadata.name,void j(e=>{const t=new Set(e);return t.has(a)?t.delete(a):t.add(a),t});var a},children:a?(0,s.jsx)(E.A,{fontSize:"small"}):(0,s.jsx)(W.A,{fontSize:"small"})})})]})}return(0,s.jsx)(d.A,{variant:"body2",color:"textSecondary",children:"-"})}},{title:"Requested",field:"metadata.creationTimestamp",render:e=>{if(!e.metadata.creationTimestamp)return(0,s.jsx)(d.A,{variant:"body2",children:"-"});const a=new Date(e.metadata.creationTimestamp);return(0,s.jsx)(d.A,{variant:"body2",children:a.toLocaleDateString()})}},{title:"",filtering:!1,render:e=>H===e.metadata.name?(0,s.jsx)(T.A,{size:20}):(0,s.jsx)(o.A,{size:"small",onClick:a=>{a.stopPropagation();const t=a.currentTarget.getBoundingClientRect();b({top:t.bottom,left:t.left}),M(e)},"aria-controls":f?"myapikeys-menu":void 0,"aria-haspopup":"true",children:(0,s.jsx)(N.A,{})})}],re=(()=>{switch(m){case 0:return Z;case 1:return ee;case 2:return ae;default:return Q}})(),ie=(()=>{switch(m){case 0:return ne.filter(e=>"Reason"!==e.title);case 1:return ne.filter(e=>"Reason"!==e.title&&"Reviewed By"!==e.title&&"API Key"!==e.title);case 2:return ne.filter(e=>"API Key"!==e.title);default:return ne}})();return(0,s.jsxs)(s.Fragment,{children:[(0,s.jsxs)(P.n,{title:"My API Keys",subheader:`${Z.length} active, ${ee.length} pending`,children:[(0,s.jsx)(i.A,{mb:2,children:(0,s.jsxs)(R.A,{value:m,onChange:(e,a)=>h(a),indicatorColor:"primary",textColor:"primary",children:[(0,s.jsx)(I.A,{label:`Active (${Z.length})`}),(0,s.jsx)(I.A,{label:`Pending (${ee.length})`}),(0,s.jsx)(I.A,{label:`Rejected (${ae.length})`})]})}),0===re.length?(0,s.jsx)(i.A,{p:3,textAlign:"center",children:(0,s.jsxs)(d.A,{variant:"body1",color:"textSecondary",children:[0===m&&"No active API keys. Request access to an API to get started.",1===m&&"No pending requests.",2===m&&"No rejected requests."]})}):(0,s.jsx)(v.X,{options:{paging:re.length>5,pageSize:20,search:!0,filtering:!0,debounceInterval:300,toolbar:!0,emptyRowsWhenPaging:!1},columns:ie,data:re.map(e=>({...e,id:e.metadata.name}))})]}),(0,s.jsx)($.A,{id:"myapikeys-menu",open:Boolean(f),onClose:te,anchorReference:"anchorPosition",anchorPosition:f||{top:0,left:0},children:w&&(()=>{const e=[];var a;return(a=w).status&&"Pending"!==a.status.phase||e.push((0,s.jsx)(q.A,{onClick:se,children:"Edit"},"edit")),e.push((0,s.jsx)(q.A,{onClick:le,children:"Delete"},"delete")),e})()}),K.request&&(0,s.jsx)(D.n,{open:K.open,request:K.request,availablePlans:K.plans,onClose:()=>L({open:!1,request:null,plans:[]}),onSuccess:()=>{L({open:!1,request:null,plans:[]}),U(e=>e+1)}}),(0,s.jsx)(z.K,{open:O.open,title:"Delete API Key Request",description:`Are you sure you want to delete the API key request for ${(null===(a=O.request)||void 0===a||null===(e=a.spec.apiProductRef)||void 0===e?void 0:e.name)||"this API"}?`,deleting:null!==H,onConfirm:async()=>{if(!O.request)return;const e=O.request,a=e.metadata.name;J(e=>new Set(e).add(a)),F(a);try{if(!(await n.fetch(`${p}/api/kuadrant/requests/${e.metadata.namespace}/${e.metadata.name}`,{method:"DELETE"})).ok)throw new Error("Failed to delete request");U(e=>e+1),u.post({message:"Request deleted",severity:"success",display:"transient"}),_({open:!1,request:null})}catch(e){console.error("Error deleting request:",e),J(e=>{const t=new Set(e);return t.delete(a),t}),u.post({message:"Failed to delete request",severity:"error",display:"transient"})}finally{F(null)}},onCancel:()=>{_({open:!1,request:null})}})]})};var K=t(46205);const L=({children:e,permission:a,fallback:t,errorMessage:l})=>{const{allowed:n,loading:r,error:o}=(0,K.l)(a);return r?(0,s.jsx)(y.k,{}):o?(0,s.jsxs)(i.A,{p:4,children:[(0,s.jsxs)(d.A,{color:"error",children:["Unable to check permissions: ",o.message]}),(0,s.jsx)(d.A,{variant:"body2",color:"textSecondary",children:"Please try again or contact your administrator"})]}):n?(0,s.jsx)(s.Fragment,{children:e}):t?(0,s.jsx)(s.Fragment,{children:t}):(0,s.jsxs)(i.A,{p:4,children:[(0,s.jsx)(d.A,{color:"textSecondary",children:l||"You don't have permission to view this page"}),(0,s.jsx)(i.A,{mt:1,children:(0,s.jsxs)(d.A,{variant:"caption",color:"textSecondary",children:["Required permission: ",a.name]})})]})};var B=t(58837),U=t(76891),H=t(61477),F=t(46805),O=t(16249),_=t(93453),V=t(84441);const J=({selectedPolicy:e,alertSeverity:a="warning",alertMessage:t="No PlanPolicy found for this HTTPRoute. API keys and rate limiting may not be available.",includeTopMargin:l=!0})=>(0,s.jsx)(i.A,{mt:l?1:0,p:2,bgcolor:"#f5f5f5",borderRadius:1,border:"1px solid #e0e0e0",children:e?(0,s.jsxs)(s.Fragment,{children:[(0,s.jsxs)(d.A,{variant:"subtitle2",gutterBottom:!0,style:{fontWeight:600},children:["Associated PlanPolicy: ",(0,s.jsx)("strong",{children:e.metadata.name})]}),e.plans&&e.plans.length>0?(0,s.jsxs)(s.Fragment,{children:[(0,s.jsx)(d.A,{variant:"caption",display:"block",gutterBottom:!0,color:"textSecondary",style:{marginTop:8},children:"Available Plans:"}),(0,s.jsx)(i.A,{display:"flex",flexWrap:"wrap",mt:1,style:{gap:8},children:e.plans.map((e,a)=>{var t,l,n;const i=(null===(t=e.limits)||void 0===t?void 0:t.daily)?`${e.limits.daily}/day`:(null===(l=e.limits)||void 0===l?void 0:l.monthly)?`${e.limits.monthly}/month`:(null===(n=e.limits)||void 0===n?void 0:n.yearly)?`${e.limits.yearly}/year`:"No limit";return(0,s.jsx)(r.A,{label:`${e.tier}: ${i}`,size:"small",variant:"outlined",color:"primary"},a)})}),e.plans.some(e=>e.description)&&(0,s.jsx)(i.A,{mt:1,children:e.plans.filter(e=>e.description).map((e,a)=>(0,s.jsxs)(d.A,{variant:"caption",display:"block",color:"textSecondary",children:["• ",(0,s.jsxs)("strong",{children:[e.tier,":"]})," ",e.description]},a))})]}):(0,s.jsx)(d.A,{variant:"caption",color:"textSecondary",children:"No plans defined in this PlanPolicy"})]}):(0,s.jsx)(V.A,{severity:a,children:t})}),Y=(0,B.A)({asterisk:{color:"#f44336"}}),X=({open:e,onClose:a,onSuccess:t})=>{const n=Y(),o=(0,k.useApi)(k.configApiRef),p=(0,k.useApi)(k.fetchApiRef),m=o.getString("backend.baseUrl"),[h,x]=(0,l.useState)(""),[v,A]=(0,l.useState)(""),[g,j]=(0,l.useState)(""),[f,y]=(0,l.useState)("v1"),[b,P]=(0,l.useState)("manual"),[w,C]=(0,l.useState)("Published"),[R,I]=(0,l.useState)([]),[$,W]=(0,l.useState)(""),[E,N]=(0,l.useState)(""),[D,z]=(0,l.useState)(""),[M,K]=(0,l.useState)(""),[L,B]=(0,l.useState)(""),[X,G]=(0,l.useState)(""),[Q,Z]=(0,l.useState)(""),[ee,ae]=(0,l.useState)(!1),{value:te,loading:se}=(0,S.A)(async()=>{const e=await p.fetch(`${m}/api/kuadrant/httproutes`);return((await e.json()).items||[]).filter(e=>{var a;return"true"===(null===(a=e.metadata.annotations)||void 0===a?void 0:a["backstage.io/expose"])})},[m,p,e]),{value:le}=(0,S.A)(async()=>{const e=await p.fetch(`${m}/api/kuadrant/planpolicies`);return await e.json()},[m,p,e]),ne=E?E.split("/"):null,re=ne?(ie=ne[0],oe=ne[1],(null==le?void 0:le.items)?le.items.find(e=>{const a=e.targetRef;return"HTTPRoute"===(null==a?void 0:a.kind)&&(null==a?void 0:a.name)===oe&&(!(null==a?void 0:a.namespace)||(null==a?void 0:a.namespace)===ie)}):null):null;var ie,oe;const de=()=>{$.trim()&&!R.includes($.trim())&&(I([...R,$.trim()]),W(""))},ce=()=>{x(""),A(""),j(""),y("v1"),P("manual"),C("Published"),I([]),W(""),N(""),z(""),K(""),B(""),G(""),Z(""),a()};return(0,s.jsxs)(U.A,{open:e,onClose:ce,maxWidth:"md",fullWidth:!0,children:[(0,s.jsx)(H.A,{children:"Create API Product"}),(0,s.jsxs)(F.A,{children:[Q&&(0,s.jsx)(V.A,{severity:"error",style:{marginBottom:16},children:Q}),(0,s.jsxs)(c.A,{container:!0,spacing:2,children:[(0,s.jsx)(c.A,{item:!0,xs:6,children:(0,s.jsx)(O.A,{fullWidth:!0,label:"Name",value:h,onChange:e=>x(e.target.value),placeholder:"my-api",helperText:"Kubernetes resource name (lowercase, hyphens)",margin:"normal",required:!0,disabled:ee,InputLabelProps:{classes:{asterisk:n.asterisk}}})}),(0,s.jsx)(c.A,{item:!0,xs:6,children:(0,s.jsx)(O.A,{fullWidth:!0,label:"Display Name",value:v,onChange:e=>A(e.target.value),placeholder:"My API",margin:"normal",required:!0,disabled:ee,InputLabelProps:{classes:{asterisk:n.asterisk}}})}),(0,s.jsx)(c.A,{item:!0,xs:6,children:(0,s.jsx)(O.A,{fullWidth:!0,label:"Version",value:f,onChange:e=>y(e.target.value),placeholder:"v1",margin:"normal",disabled:ee})}),(0,s.jsx)(c.A,{item:!0,xs:12,children:(0,s.jsxs)(O.A,{fullWidth:!0,select:!0,label:"Approval Mode",value:b,onChange:e=>P(e.target.value),margin:"normal",helperText:"Automatic: keys are created immediately. Manual: requires approval.",disabled:ee,children:[(0,s.jsx)(q.A,{value:"manual",children:"Manual"}),(0,s.jsx)(q.A,{value:"automatic",children:"Automatic"})]})}),(0,s.jsx)(c.A,{item:!0,xs:12,children:(0,s.jsxs)(O.A,{fullWidth:!0,select:!0,label:"Publish Status",value:w,onChange:e=>C(e.target.value),margin:"normal",helperText:"Draft: hidden from catalog. Published: visible to consumers.",disabled:ee,children:[(0,s.jsx)(q.A,{value:"Draft",children:"Draft"}),(0,s.jsx)(q.A,{value:"Published",children:"Published"})]})}),(0,s.jsx)(c.A,{item:!0,xs:12,children:(0,s.jsx)(O.A,{fullWidth:!0,label:"Description",value:g,onChange:e=>j(e.target.value),placeholder:"API description",margin:"normal",multiline:!0,rows:2,required:!0,disabled:ee,InputLabelProps:{classes:{asterisk:n.asterisk}}})}),(0,s.jsxs)(c.A,{item:!0,xs:12,children:[(0,s.jsx)(d.A,{variant:"subtitle2",gutterBottom:!0,style:{marginTop:16},children:"Tags"}),(0,s.jsx)(i.A,{display:"flex",flexWrap:"wrap",marginBottom:1,style:{gap:8},children:R.map(e=>(0,s.jsx)(r.A,{label:e,onDelete:ee?void 0:()=>{return a=e,void I(R.filter(e=>e!==a));var a},size:"small",disabled:ee},e))}),(0,s.jsxs)(i.A,{display:"flex",style:{gap:8},children:[(0,s.jsx)(O.A,{fullWidth:!0,size:"small",value:$,onChange:e=>W(e.target.value),onKeyPress:e=>"Enter"===e.key&&de(),placeholder:"Add tag",disabled:ee}),(0,s.jsx)(u.A,{onClick:de,variant:"outlined",size:"small",disabled:ee,children:"Add"})]})]}),(0,s.jsx)(c.A,{item:!0,xs:12,children:(0,s.jsxs)(O.A,{fullWidth:!0,select:!0,label:"HTTPRoute",value:E,onChange:e=>N(e.target.value),margin:"normal",required:!0,helperText:"Select an HTTPRoute (backstage.io/expose: true). APIProduct will be created in the same namespace.",disabled:se||ee,InputLabelProps:{classes:{asterisk:n.asterisk}},children:[se&&(0,s.jsx)(q.A,{value:"",children:"Loading..."}),!se&&te&&0===te.length&&(0,s.jsx)(q.A,{value:"",children:"No HTTPRoutes available"}),!se&&te&&te.map(e=>(0,s.jsxs)(q.A,{value:`${e.metadata.namespace}/${e.metadata.name}`,children:[e.metadata.name," (",e.metadata.namespace,")"]},`${e.metadata.namespace}/${e.metadata.name}`))]})}),E&&(0,s.jsx)(c.A,{item:!0,xs:12,children:(0,s.jsx)(J,{selectedPolicy:re,alertSeverity:"warning",alertMessage:"No PlanPolicy found for this HTTPRoute. API keys and rate limiting may not be available.",includeTopMargin:!0})}),(0,s.jsx)(c.A,{item:!0,xs:6,children:(0,s.jsx)(O.A,{fullWidth:!0,label:"Contact Email",value:D,onChange:e=>z(e.target.value),placeholder:"api-team@example.com",margin:"normal",disabled:ee})}),(0,s.jsx)(c.A,{item:!0,xs:6,children:(0,s.jsx)(O.A,{fullWidth:!0,label:"Contact Team",value:M,onChange:e=>K(e.target.value),placeholder:"platform-team",margin:"normal",disabled:ee})}),(0,s.jsx)(c.A,{item:!0,xs:6,children:(0,s.jsx)(O.A,{fullWidth:!0,label:"Docs URL",value:L,onChange:e=>B(e.target.value),placeholder:"https://api.example.com/docs",margin:"normal",disabled:ee})}),(0,s.jsx)(c.A,{item:!0,xs:6,children:(0,s.jsx)(O.A,{fullWidth:!0,label:"OpenAPI Spec URL",value:X,onChange:e=>G(e.target.value),placeholder:"https://api.example.com/openapi.json",margin:"normal",disabled:ee})})]})]}),(0,s.jsxs)(_.A,{children:[(0,s.jsx)(u.A,{onClick:ce,disabled:ee,children:"Cancel"}),(0,s.jsx)(u.A,{onClick:async()=>{Z(""),ae(!0);try{if(!E)throw new Error("Please select an HTTPRoute");const[e,a]=E.split("/"),s={apiVersion:"devportal.kuadrant.io/v1alpha1",kind:"APIProduct",metadata:{name:h,namespace:e},spec:{displayName:v,description:g,version:f,approvalMode:b,publishStatus:w,tags:R,targetRef:{group:"gateway.networking.k8s.io",kind:"HTTPRoute",name:a,namespace:e},...D||M?{contact:{...D&&{email:D},...M&&{team:M}}}:{},...L||X?{documentation:{...L&&{docsURL:L},...X&&{openAPISpec:X}}}:{}}},l=await p.fetch(`${m}/api/kuadrant/apiproducts`,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify(s)});if(!l.ok){const e=await l.json();throw new Error(e.error||"failed to create apiproduct")}t(),ce()}catch(e){Z(e instanceof Error?e.message:String(e))}finally{ae(!1)}},color:"primary",variant:"contained",disabled:ee||!h||!v||!g||!E,startIcon:ee?(0,s.jsx)(T.A,{size:16,color:"inherit"}):void 0,children:ee?"Creating...":"Create"})]})]})};var G=t(34955);const Q=(0,B.A)({asterisk:{color:"#f44336"}}),Z=({open:e,onClose:a,onSuccess:t,namespace:o,name:p})=>{const m=Q(),h=(0,k.useApi)(k.configApiRef),x=(0,k.useApi)(k.fetchApiRef),v=h.getString("backend.baseUrl"),[A,g]=(0,l.useState)(!1),[j,f]=(0,l.useState)(""),[b,P]=(0,l.useState)(""),[w,C]=(0,l.useState)("v1"),[R,I]=(0,l.useState)("Draft"),[$,W]=(0,l.useState)("manual"),[E,N]=(0,l.useState)([]),[D,z]=(0,l.useState)(null),[M,K]=(0,l.useState)(""),[L,B]=(0,l.useState)(""),[Y,X]=(0,l.useState)(""),[G,Z]=(0,l.useState)(""),[ee,ae]=(0,l.useState)(""),[te,se]=(0,l.useState)(""),[le,ne]=(0,l.useState)(!1);(0,l.useEffect)(()=>{e&&o&&p&&(g(!0),se(""),x.fetch(`${v}/api/kuadrant/apiproducts/${o}/${p}`).then(async e=>{if(!e.ok){const a=await e.json();throw new Error(a.error||`Failed to fetch API product: ${e.status}`)}return e.json()}).then(e=>{var a,t,s,l;f(e.spec.displayName||""),P(e.spec.description||""),C(e.spec.version||"v1"),I(e.spec.publishStatus||"Draft"),W(e.spec.approvalMode||"manual"),N(e.spec.tags||[]),z(e.spec.targetRef||null),B((null===(a=e.spec.contact)||void 0===a?void 0:a.email)||""),X((null===(t=e.spec.contact)||void 0===t?void 0:t.team)||""),Z((null===(s=e.spec.documentation)||void 0===s?void 0:s.docsURL)||""),ae((null===(l=e.spec.documentation)||void 0===l?void 0:l.openAPISpec)||""),g(!1)}).catch(e=>{se(e.message||"Failed to load API product"),g(!1)}))},[e,o,p,v,x]);const{value:re}=(0,S.A)(async()=>{if(!e)return null;const a=await x.fetch(`${v}/api/kuadrant/planpolicies`);return await a.json()},[v,x,e]),ie=n().useMemo(()=>(null==re?void 0:re.items)&&D?re.items.find(e=>{const a=e.targetRef;return"HTTPRoute"===(null==a?void 0:a.kind)&&(null==a?void 0:a.name)===D.name&&(!(null==a?void 0:a.namespace)||(null==a?void 0:a.namespace)===(D.namespace||o))}):null,[re,D,o]),oe=()=>{M.trim()&&!E.includes(M.trim())&&(N([...E,M.trim()]),K(""))};return(0,s.jsxs)(U.A,{open:e,onClose:a,maxWidth:"md",fullWidth:!0,children:[(0,s.jsx)(H.A,{children:"Edit API Product"}),(0,s.jsxs)(F.A,{children:[te&&(0,s.jsx)(V.A,{severity:"error",style:{marginBottom:16},children:te}),A?(0,s.jsx)(y.k,{}):(0,s.jsxs)(c.A,{container:!0,spacing:2,children:[(0,s.jsx)(c.A,{item:!0,xs:6,children:(0,s.jsx)(O.A,{fullWidth:!0,label:"Name",value:p,disabled:!0,helperText:"Kubernetes resource name (immutable)",margin:"normal"})}),(0,s.jsx)(c.A,{item:!0,xs:6,children:(0,s.jsx)(O.A,{fullWidth:!0,label:"Namespace",value:o,disabled:!0,helperText:"Derived from HTTPRoute (immutable)",margin:"normal"})}),(0,s.jsx)(c.A,{item:!0,xs:6,children:(0,s.jsx)(O.A,{fullWidth:!0,label:"Display Name",value:j,onChange:e=>f(e.target.value),placeholder:"My API",margin:"normal",required:!0,disabled:le,InputLabelProps:{classes:{asterisk:m.asterisk}}})}),(0,s.jsx)(c.A,{item:!0,xs:6,children:(0,s.jsx)(O.A,{fullWidth:!0,label:"Version",value:w,onChange:e=>C(e.target.value),placeholder:"v1",margin:"normal",disabled:le})}),(0,s.jsx)(c.A,{item:!0,xs:6,children:(0,s.jsxs)(O.A,{fullWidth:!0,select:!0,label:"Publish Status",value:R,onChange:e=>I(e.target.value),margin:"normal",helperText:"Draft: Hidden from catalog. Published: Visible in catalog.",disabled:le,children:[(0,s.jsx)(q.A,{value:"Draft",children:"Draft (Hidden)"}),(0,s.jsx)(q.A,{value:"Published",children:"Published (Visible)"})]})}),(0,s.jsx)(c.A,{item:!0,xs:6,children:(0,s.jsxs)(O.A,{fullWidth:!0,select:!0,label:"Approval Mode",value:$,onChange:e=>W(e.target.value),margin:"normal",helperText:"Automatic: keys created immediately. Manual: requires approval.",disabled:le,children:[(0,s.jsx)(q.A,{value:"manual",children:"Manual"}),(0,s.jsx)(q.A,{value:"automatic",children:"Automatic"})]})}),(0,s.jsx)(c.A,{item:!0,xs:12,children:(0,s.jsx)(O.A,{fullWidth:!0,label:"Description",value:b,onChange:e=>P(e.target.value),placeholder:"API description",margin:"normal",multiline:!0,rows:2,required:!0,disabled:le,InputLabelProps:{classes:{asterisk:m.asterisk}}})}),(0,s.jsxs)(c.A,{item:!0,xs:12,children:[(0,s.jsx)(d.A,{variant:"subtitle2",gutterBottom:!0,style:{marginTop:16},children:"Tags"}),(0,s.jsx)(i.A,{display:"flex",flexWrap:"wrap",marginBottom:1,style:{gap:8},children:E.map(e=>(0,s.jsx)(r.A,{label:e,onDelete:le?void 0:()=>{return a=e,void N(E.filter(e=>e!==a));var a},size:"small",disabled:le},e))}),(0,s.jsxs)(i.A,{display:"flex",style:{gap:8},children:[(0,s.jsx)(O.A,{fullWidth:!0,size:"small",value:M,onChange:e=>K(e.target.value),onKeyPress:e=>"Enter"===e.key&&oe(),placeholder:"Add tag",disabled:le}),(0,s.jsx)(u.A,{onClick:oe,variant:"outlined",size:"small",disabled:le,children:"Add"})]})]}),D&&(0,s.jsxs)(s.Fragment,{children:[(0,s.jsx)(c.A,{item:!0,xs:12,children:(0,s.jsx)(O.A,{fullWidth:!0,label:"HTTPRoute",value:`${D.namespace||o}/${D.name}`,disabled:!0,helperText:"Target HTTPRoute (immutable)",margin:"normal"})}),(0,s.jsx)(c.A,{item:!0,xs:12,children:(0,s.jsx)(J,{selectedPolicy:ie,alertSeverity:"info",alertMessage:"No PlanPolicy found for this HTTPRoute.",includeTopMargin:!1})})]}),(0,s.jsx)(c.A,{item:!0,xs:6,children:(0,s.jsx)(O.A,{fullWidth:!0,label:"Contact Email",value:L,onChange:e=>B(e.target.value),placeholder:"api-team@example.com",margin:"normal",disabled:le})}),(0,s.jsx)(c.A,{item:!0,xs:6,children:(0,s.jsx)(O.A,{fullWidth:!0,label:"Contact Team",value:Y,onChange:e=>X(e.target.value),placeholder:"platform-team",margin:"normal",disabled:le})}),(0,s.jsx)(c.A,{item:!0,xs:6,children:(0,s.jsx)(O.A,{fullWidth:!0,label:"Docs URL",value:G,onChange:e=>Z(e.target.value),placeholder:"https://api.example.com/docs",margin:"normal",disabled:le})}),(0,s.jsx)(c.A,{item:!0,xs:6,children:(0,s.jsx)(O.A,{fullWidth:!0,label:"OpenAPI Spec URL",value:ee,onChange:e=>ae(e.target.value),placeholder:"https://api.example.com/openapi.json",margin:"normal",disabled:le})})]})]}),(0,s.jsxs)(_.A,{children:[(0,s.jsx)(u.A,{onClick:a,disabled:le,children:"Cancel"}),(0,s.jsx)(u.A,{onClick:async()=>{se(""),ne(!0);try{const e={spec:{displayName:j,description:b,version:w,publishStatus:R,approvalMode:$,tags:E,targetRef:D,...L||Y?{contact:{...L&&{email:L},...Y&&{team:Y}}}:{},...G||ee?{documentation:{...G&&{docsURL:G},...ee&&{openAPISpec:ee}}}:{}}},s=await x.fetch(`${v}/api/kuadrant/apiproducts/${o}/${p}`,{method:"PATCH",headers:{"Content-Type":"application/json"},body:JSON.stringify(e)});if(!s.ok){const e=await s.json();throw new Error(e.error||"failed to update apiproduct")}t(),a()}catch(e){se(e instanceof Error?e.message:String(e))}finally{ne(!1)}},color:"primary",variant:"contained",disabled:le||A||!j||!b,startIcon:le?(0,s.jsx)(T.A,{size:16,color:"inherit"}):void 0,children:le?"Saving...":"Save"})]})]})},ee=()=>{const e=(0,k.useApi)(k.configApiRef),a=(0,k.useApi)(k.fetchApiRef),t=(0,k.useApi)(k.alertApiRef),n=(0,k.useApi)(k.identityApiRef),C=e.getString("backend.baseUrl"),[T,R]=(0,l.useState)(""),[I,$]=(0,l.useState)(!1),[q,W]=(0,l.useState)(!1),[E,N]=(0,l.useState)(0),[D,L]=(0,l.useState)(!1),[B,U]=(0,l.useState)(null),[H,F]=(0,l.useState)(null),[O,_]=(0,l.useState)(!1),{allowed:V,loading:J,error:Y}=(0,K.l)(G.FL),{allowed:Q,loading:ee}=(0,K.l)(G.TE),{allowed:ae,loading:te,error:se}=(0,K.l)(G.SU),le=Q||ae,ne=ee||te,{allowed:re,loading:ie}=(0,K.l)(G.EM),{allowed:oe,loading:de,error:ce}=(0,K.l)(G.R_),{allowed:ue}=(0,K.l)(G.U3),{allowed:pe}=(0,K.l)(G.v_),me=ie||de,{allowed:he,loading:xe,error:ve}=(0,K.l)(G.J);(0,S.A)(async()=>{const e=await n.getBackstageIdentity();R(e.userEntityRef)},[n]);const{value:Ae,loading:ge,error:je}=(0,S.A)(async()=>{const e=await a.fetch(`${C}/api/kuadrant/apiproducts`);return await e.json()},[C,a,E]),{value:fe,loading:ye,error:be}=(0,S.A)(async()=>{const e=await a.fetch(`${C}/api/kuadrant/planpolicies`);return await e.json()},[C,a,E]),Pe=ge||ye||J||ne||me||xe,Se=je||be,ke=Y||se||ce||ve,we=[{title:"Name",field:"spec.displayName",render:e=>{var a,t;const l="Published"===(null===(a=e.spec)||void 0===a?void 0:a.publishStatus);var n;const r=null!==(n=null===(t=e.spec)||void 0===t?void 0:t.displayName)&&void 0!==n?n:e.metadata.name;return l?(0,s.jsx)(x.N_,{to:`/catalog/default/api/${e.metadata.name}/api-product`,children:(0,s.jsx)("strong",{children:r})}):(0,s.jsx)("span",{className:"text-muted",children:(0,s.jsx)("strong",{children:r})})},customFilterAndSearch:(e,a)=>{var t;return((null===(t=a.spec)||void 0===t?void 0:t.displayName)||a.metadata.name||"").toLowerCase().includes(e.toLowerCase())}},{title:"Resource Name",field:"metadata.name"},{title:"Version",field:"spec.version",render:e=>{var a;return(null===(a=e.spec)||void 0===a?void 0:a.version)||"-"}},{title:"HTTPRoute",field:"spec.targetRef.name",render:e=>{var a,t;return(null===(t=e.spec)||void 0===t||null===(a=t.targetRef)||void 0===a?void 0:a.name)||"-"}},{title:"Publish Status",field:"spec.publishStatus",render:e=>{var a;const t=(null===(a=e.spec)||void 0===a?void 0:a.publishStatus)||"Draft";return(0,s.jsx)(r.A,{label:t,size:"small",color:"Published"===t?"primary":"default"})}},{title:"Approval Mode",field:"spec.approvalMode",render:e=>{var a;const t=(null===(a=e.spec)||void 0===a?void 0:a.approvalMode)||"manual";return(0,s.jsx)(r.A,{label:t,size:"small",color:"automatic"===t?"secondary":"default"})}},{title:"Namespace",field:"metadata.namespace"},{title:"Created",field:"metadata.creationTimestamp",render:e=>{return a=e.metadata.creationTimestamp,new Date(a).toLocaleDateString("en-GB",{year:"numeric",month:"short",day:"numeric"});var a}},{title:"Actions",field:"actions",filtering:!1,render:e=>{var a,t;const l=(null===(t=e.metadata)||void 0===t||null===(a=t.annotations)||void 0===a?void 0:a["backstage.io/owner"])===T,n=pe||ue&&l,r=oe||re&&l;return n||r?(0,s.jsxs)(i.A,{display:"flex",style:{gap:4},children:[n&&(0,s.jsx)(o.A,{size:"small",onClick:()=>{return a=e.metadata.namespace,t=e.metadata.name,F({namespace:a,name:t}),void W(!0);var a,t},title:"Edit API Product",children:(0,s.jsx)(h.A,{fontSize:"small"})}),r&&(0,s.jsx)(o.A,{size:"small",onClick:()=>{return a=e.metadata.namespace,t=e.metadata.name,U({namespace:a,name:t}),void L(!0);var a,t},title:"Delete API Product",children:(0,s.jsx)(m.A,{fontSize:"small"})})]}):null}}],Ce=[{title:"Name",field:"metadata.name",render:e=>(0,s.jsx)(x.N_,{to:`/kuadrant/planpolicy/${e.metadata.namespace}/${e.metadata.name}`,children:(0,s.jsx)("strong",{children:e.metadata.name})})},{title:"Namespace",field:"metadata.namespace"}];return(0,s.jsxs)(A.Y,{themeId:"tool",children:[(0,s.jsx)(g.Y,{title:"Kuadrant",subtitle:"API management for Kubernetes",children:(0,s.jsx)(j.Y,{children:"Manage API products and access requests"})}),(0,s.jsxs)(f.U,{children:[Pe&&(0,s.jsx)(y.k,{}),Se&&(0,s.jsx)(b._,{error:Se}),ke&&(0,s.jsxs)(i.A,{p:2,children:[(0,s.jsxs)(d.A,{color:"error",children:["unable to check permissions: ",ke.message]}),(0,s.jsxs)(d.A,{variant:"body2",color:"textSecondary",children:["permission: ",Y?"kuadrant.apiproduct.create":ce?"kuadrant.apiproduct.delete":se?"kuadrant.apikeyrequest.read.all":ve?"kuadrant.planpolicy.list":"unknown"]}),(0,s.jsx)(d.A,{variant:"body2",color:"textSecondary",children:"please try again or contact your administrator"})]}),!Pe&&!Se&&!ke&&(0,s.jsxs)(c.A,{container:!0,spacing:3,direction:"column",children:[(0,s.jsx)(c.A,{item:!0,children:(0,s.jsx)(M,{})}),(0,s.jsx)(c.A,{item:!0,children:(0,s.jsx)(P.n,{title:"API Products",action:V?(0,s.jsx)(i.A,{display:"flex",alignItems:"center",height:"100%",mt:1,children:(0,s.jsx)(u.A,{variant:"contained",color:"primary",size:"small",startIcon:(0,s.jsx)(p.A,{}),onClick:()=>$(!0),children:"Create API Product"})}):void 0,children:(Te=null==Ae?void 0:Ae.items,Te&&0!==Te.length?(0,s.jsx)(v.X,{options:{paging:Te.length>5,pageSize:20,search:!0,filtering:!0,debounceInterval:300,toolbar:!0,emptyRowsWhenPaging:!1},columns:we,data:Te}):(0,s.jsx)(d.A,{variant:"body2",color:"textSecondary",children:"No API products found"}))})}),he&&(0,s.jsx)(c.A,{item:!0,children:(0,s.jsx)(P.n,{title:"Plan Policies",children:(e=>e&&0!==e.length?(0,s.jsx)(v.X,{options:{paging:!1,search:!1,toolbar:!1},columns:Ce,data:e}):(0,s.jsx)(d.A,{variant:"body2",color:"textSecondary",children:"No plan policies found"}))(null==fe?void 0:fe.items)})}),le&&(0,s.jsx)(c.A,{item:!0,children:(0,s.jsx)(w.d,{})})]}),(0,s.jsx)(X,{open:I,onClose:()=>$(!1),onSuccess:()=>{N(e=>e+1),t.post({message:"API Product created",severity:"success",display:"transient"})}}),(0,s.jsx)(Z,{open:q,onClose:()=>W(!1),onSuccess:()=>{N(e=>e+1),t.post({message:"API Product updated",severity:"success",display:"transient"})},namespace:(null==H?void 0:H.namespace)||"",name:(null==H?void 0:H.name)||""}),(0,s.jsx)(z.K,{open:D,title:"Delete API Product",description:`This will permanently delete "${null==B?void 0:B.name}" from namespace "${null==B?void 0:B.namespace}" and remove it from Kubernetes. Any associated API keys will stop working.`,confirmText:null==B?void 0:B.name,severity:"high",deleting:O,onConfirm:async()=>{if(B){_(!0);try{if(!(await a.fetch(`${C}/api/kuadrant/apiproducts/${B.namespace}/${B.name}`,{method:"DELETE"})).ok)throw new Error("failed to delete apiproduct");N(e=>e+1),t.post({message:"API Product deleted",severity:"success",display:"transient"})}catch(e){console.error("error deleting apiproduct:",e),t.post({message:"Failed to delete API Product",severity:"error",display:"transient"})}finally{_(!1),L(!1),U(null)}}},onCancel:()=>{L(!1),U(null)}})]})]});var Te},ae=()=>(0,s.jsx)(L,{permission:G.OP,errorMessage:"you don't have permission to view the Kuadrant page",children:(0,s.jsx)(ee,{})})}}]);
|
|
2
|
-
//# sourceMappingURL=7632.e4c19350.chunk.js.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"static/7632.e4c19350.chunk.js","mappings":"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","sources":["webpack://internal.plugin-kuadrant/./src/components/ConfirmDeleteDialog/ConfirmDeleteDialog.tsx","webpack://internal.plugin-kuadrant/./src/components/EditAPIKeyRequestDialog/EditAPIKeyRequestDialog.tsx","webpack://internal.plugin-kuadrant/./src/components/MyApiKeysCard/MyApiKeysCard.tsx","webpack://internal.plugin-kuadrant/./src/components/PermissionGate/PermissionGate.tsx","webpack://internal.plugin-kuadrant/./src/components/PlanPolicyDetailsCard/PlanPolicyDetails.tsx","webpack://internal.plugin-kuadrant/./src/components/CreateAPIProductDialog/CreateAPIProductDialog.tsx","webpack://internal.plugin-kuadrant/./src/components/EditAPIProductDialog/EditAPIProductDialog.tsx","webpack://internal.plugin-kuadrant/./src/components/KuadrantPage/KuadrantPage.tsx"],"sourcesContent":["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>{description}</DialogContentText>\n {requiresTextConfirmation && (\n <Box mt={2}>\n <Typography variant=\"body2\" color=\"textSecondary\" gutterBottom>\n Type <strong>{confirmText}</strong> to confirm:\n </Typography>\n <TextField\n fullWidth\n variant=\"outlined\"\n size=\"small\"\n value={inputValue}\n onChange={e => setInputValue(e.target.value)}\n disabled={deleting}\n autoFocus\n placeholder={confirmText}\n />\n </Box>\n )}\n </DialogContent>\n <DialogActions>\n <Button onClick={onCancel} disabled={deleting}>\n Cancel\n </Button>\n <Button\n onClick={handleConfirm}\n color=\"secondary\"\n variant=\"contained\"\n disabled={deleting || !canConfirm}\n startIcon={deleting ? <CircularProgress size={16} color=\"inherit\" /> : undefined}\n >\n {deleting ? 'Deleting...' : 'Delete'}\n </Button>\n </DialogActions>\n </Dialog>\n );\n};\n","import React, { useState, useEffect } from 'react';\nimport {\n Dialog,\n DialogTitle,\n DialogContent,\n DialogActions,\n Button,\n TextField,\n Box,\n Typography,\n FormControl,\n InputLabel,\n Select,\n MenuItem,\n CircularProgress,\n} from '@material-ui/core';\nimport { useApi, configApiRef, fetchApiRef } from '@backstage/core-plugin-api';\nimport { APIKey } from '../../types/api-management';\n\ninterface EditAPIKeyDialogProps {\n open: boolean;\n onClose: () => void;\n onSuccess: () => void;\n request: APIKey;\n availablePlans: Array<{\n tier: string;\n description?: string;\n limits?: any;\n }>;\n}\n\nexport const EditAPIKeyRequestDialog = ({\n open,\n onClose,\n onSuccess,\n request,\n availablePlans,\n}: EditAPIKeyDialogProps) => {\n const config = useApi(configApiRef);\n const fetchApi = useApi(fetchApiRef);\n const backendUrl = config.getString('backend.baseUrl');\n\n const [planTier, setPlanTier] = useState('');\n const [useCase, setUseCase] = useState('');\n const [saving, setSaving] = useState(false);\n const [error, setError] = useState('');\n\n useEffect(() => {\n if (open && request) {\n setPlanTier(request.spec.planTier || '');\n setUseCase(request.spec.useCase || '');\n setError('');\n }\n }, [open, request]);\n\n const handleSave = async () => {\n if (!planTier) {\n setError('Please select a tier');\n return;\n }\n\n setError('');\n setSaving(true);\n\n try {\n const patch = {\n spec: {\n planTier,\n useCase: useCase.trim(),\n },\n };\n\n const response = await fetchApi.fetch(\n `${backendUrl}/api/kuadrant/requests/${request.metadata.namespace}/${request.metadata.name}`,\n {\n method: 'PATCH',\n headers: {\n 'Content-Type': 'application/json',\n },\n body: JSON.stringify(patch),\n }\n );\n\n if (!response.ok) {\n const errorData = await response.json().catch(() => ({}));\n throw new Error(errorData.error || `Failed to update request: ${response.status}`);\n }\n\n onSuccess();\n onClose();\n } catch (err) {\n console.error('Error updating API key request:', err);\n setError(err instanceof Error ? err.message : 'Unknown error occurred');\n } finally {\n setSaving(false);\n }\n };\n\n const handleClose = () => {\n if (!saving) {\n setError('');\n onClose();\n }\n };\n\n return (\n <Dialog open={open} onClose={handleClose} maxWidth=\"sm\" fullWidth>\n <DialogTitle>Edit API Access Request</DialogTitle>\n <DialogContent>\n {error && (\n <Box mb={2} p={2} bgcolor=\"error.main\" color=\"error.contrastText\" borderRadius={1}>\n <Typography variant=\"body2\">{error}</Typography>\n </Box>\n )}\n\n <FormControl fullWidth margin=\"normal\">\n <InputLabel>Tier</InputLabel>\n <Select\n value={planTier}\n onChange={(e) => setPlanTier(e.target.value as string)}\n disabled={saving}\n >\n {availablePlans.map((plan) => {\n const limitDesc = Object.entries(plan.limits || {})\n .map(([key, val]) => `${val} per ${key}`)\n .join(', ');\n return (\n <MenuItem key={plan.tier} value={plan.tier}>\n {plan.tier} {limitDesc ? `(${limitDesc})` : ''}\n </MenuItem>\n );\n })}\n </Select>\n </FormControl>\n\n <TextField\n label=\"Use Case\"\n placeholder=\"Describe how you plan to use this API\"\n multiline\n rows={3}\n fullWidth\n margin=\"normal\"\n value={useCase}\n onChange={(e) => setUseCase(e.target.value)}\n disabled={saving}\n helperText=\"Explain your intended use of this API for admin review\"\n />\n </DialogContent>\n <DialogActions>\n <Button onClick={handleClose} disabled={saving}>\n Cancel\n </Button>\n <Button\n onClick={handleSave}\n color=\"primary\"\n variant=\"contained\"\n disabled={!planTier || saving}\n startIcon={saving ? <CircularProgress size={16} color=\"inherit\" /> : undefined}\n >\n {saving ? 'Saving...' : 'Save Changes'}\n </Button>\n </DialogActions>\n </Dialog>\n );\n};\n","import React, { useState } from 'react';\nimport { InfoCard, Table, TableColumn, Link, Progress } from '@backstage/core-components';\nimport { useApi, configApiRef, fetchApiRef, identityApiRef, alertApiRef } from '@backstage/core-plugin-api';\nimport useAsync from 'react-use/lib/useAsync';\nimport { Box, Chip, Typography, Tabs, Tab, IconButton, Tooltip, Menu, MenuItem, CircularProgress } from '@material-ui/core';\nimport VisibilityIcon from '@material-ui/icons/Visibility';\nimport VisibilityOffIcon from '@material-ui/icons/VisibilityOff';\nimport MoreVertIcon from '@material-ui/icons/MoreVert';\nimport { EditAPIKeyRequestDialog } from '../EditAPIKeyRequestDialog';\nimport { ConfirmDeleteDialog } from '../ConfirmDeleteDialog';\nimport { APIKey } from '../../types/api-management';\n\nexport const MyApiKeysCard = () => {\n const config = useApi(configApiRef);\n const fetchApi = useApi(fetchApiRef);\n const identityApi = useApi(identityApiRef);\n const alertApi = useApi(alertApiRef);\n const backendUrl = config.getString('backend.baseUrl');\n const [selectedTab, setSelectedTab] = useState(0);\n const [, setUserId] = useState<string>('');\n const [visibleKeys, setVisibleKeys] = useState<Set<string>>(new Set());\n const [menuAnchor, setMenuAnchor] = useState<{ top: number; left: number } | null>(null);\n const [menuRequest, setMenuRequest] = useState<APIKey | null>(null);\n const [editDialogState, setEditDialogState] = useState<{ open: boolean; request: APIKey | null; plans: any[] }>({\n open: false,\n request: null,\n plans: [],\n });\n const [refresh, setRefresh] = useState(0);\n const [deleting, setDeleting] = useState<string | null>(null);\n const [deleteDialogState, setDeleteDialogState] = useState<{\n open: boolean;\n request: APIKey | null;\n }>({ open: false, request: null });\n\n useAsync(async () => {\n const identity = await identityApi.getBackstageIdentity();\n const extractedUserId = identity.userEntityRef.split('/')[1] || 'guest';\n console.log(`MyApiKeysCard: setting userId from userEntityRef: ${identity.userEntityRef} -> \"${extractedUserId}\"`);\n setUserId(extractedUserId);\n }, [identityApi]);\n\n const [optimisticallyDeleted, setOptimisticallyDeleted] = useState<Set<string>>(new Set());\n\n const { value: requests, loading, error } = useAsync(async () => {\n const response = await fetchApi.fetch(\n `${backendUrl}/api/kuadrant/requests/my`\n );\n if (!response.ok) {\n throw new Error('failed to fetch requests');\n }\n const data = await response.json();\n return data.items || [];\n }, [backendUrl, fetchApi, refresh]);\n\n if (loading) {\n return (\n <InfoCard title=\"My API Keys\">\n <Progress />\n </InfoCard>\n );\n }\n\n if (error) {\n return (\n <InfoCard title=\"My API Keys\">\n <Typography color=\"error\">Error loading API keys: {error.message}</Typography>\n </InfoCard>\n );\n }\n\n const allRequests = (requests || []).filter(\n (r: APIKey) => !optimisticallyDeleted.has(r.metadata.name)\n );\n const approvedRequests = allRequests.filter((r: APIKey) => r.status?.phase === 'Approved');\n const pendingRequests = allRequests.filter((r: APIKey) => !r.status?.phase || r.status.phase === 'Pending');\n const rejectedRequests = allRequests.filter((r: APIKey) => r.status?.phase === 'Rejected');\n\n const toggleKeyVisibility = (keyName: string) => {\n setVisibleKeys(prev => {\n const newSet = new Set(prev);\n if (newSet.has(keyName)) {\n newSet.delete(keyName);\n } else {\n newSet.add(keyName);\n }\n return newSet;\n });\n };\n\n const handleMenuClose = () => {\n setMenuAnchor(null);\n setMenuRequest(null);\n };\n\n const handleEdit = async () => {\n if (!menuRequest) return;\n\n const request = menuRequest;\n handleMenuClose();\n\n // Fetch available plans for this API\n try {\n const apiProductName = request.spec.apiProductRef?.name;\n const apiProductNamespace = request.metadata.namespace;\n const apiProductResponse = await fetchApi.fetch(\n `${backendUrl}/api/kuadrant/apiproducts/${apiProductNamespace}/${apiProductName}`\n );\n\n if (apiProductResponse.ok) {\n const apiProduct = await apiProductResponse.json();\n const plans = apiProduct.spec?.plans || [];\n setEditDialogState({ open: true, request, plans });\n } else {\n console.error('Failed to fetch API product');\n setEditDialogState({ open: true, request, plans: [] });\n }\n } catch (err) {\n console.error('Error fetching plans:', err);\n setEditDialogState({ open: true, request, plans: [] });\n }\n };\n\n const handleDeleteClick = () => {\n if (!menuRequest) return;\n const request = menuRequest;\n handleMenuClose();\n setDeleteDialogState({ open: true, request });\n };\n\n const handleDeleteConfirm = async () => {\n if (!deleteDialogState.request) return;\n\n const request = deleteDialogState.request;\n const requestName = request.metadata.name;\n\n // optimistic update - remove from UI immediately\n setOptimisticallyDeleted(prev => new Set(prev).add(requestName));\n setDeleting(requestName);\n\n try {\n const response = await fetchApi.fetch(\n `${backendUrl}/api/kuadrant/requests/${request.metadata.namespace}/${request.metadata.name}`,\n { method: 'DELETE' }\n );\n\n if (!response.ok) {\n throw new Error('Failed to delete request');\n }\n\n setRefresh(r => r + 1);\n alertApi.post({ message: 'Request deleted', severity: 'success', display: 'transient' });\n setDeleteDialogState({ open: false, request: null });\n } catch (err) {\n console.error('Error deleting request:', err);\n // rollback optimistic update on error\n setOptimisticallyDeleted(prev => {\n const next = new Set(prev);\n next.delete(requestName);\n return next;\n });\n alertApi.post({ message: 'Failed to delete request', severity: 'error', display: 'transient' });\n } finally {\n setDeleting(null);\n }\n };\n\n const handleDeleteCancel = () => {\n setDeleteDialogState({ open: false, request: null });\n };\n\n const columns: TableColumn<APIKey>[] = [\n {\n title: 'API Product',\n field: 'spec.apiProductRef.name',\n render: (row: APIKey) => {\n const apiProductName = row.spec.apiProductRef?.name || 'unknown';\n return (\n <Link to={`/catalog/default/api/${apiProductName}/api-keys`}>\n <strong>{apiProductName}</strong>\n </Link>\n );\n },\n },\n {\n title: 'Tier',\n field: 'spec.planTier',\n render: (row: APIKey) => {\n const color = row.spec.planTier === 'gold' ? 'primary' :\n row.spec.planTier === 'silver' ? 'default' : 'secondary';\n return <Chip label={row.spec.planTier} color={color} size=\"small\" />;\n },\n },\n {\n title: 'Use Case',\n field: 'spec.useCase',\n render: (row: APIKey) => {\n if (!row.spec.useCase) {\n return <Typography variant=\"body2\">-</Typography>;\n }\n return (\n <Tooltip title={row.spec.useCase} placement=\"top\">\n <Typography\n variant=\"body2\"\n style={{\n maxWidth: '200px',\n overflow: 'hidden',\n textOverflow: 'ellipsis',\n whiteSpace: 'nowrap',\n }}\n >\n {row.spec.useCase}\n </Typography>\n </Tooltip>\n );\n },\n },\n {\n title: 'Status',\n field: 'status.phase',\n render: (row: APIKey) => {\n const phase = row.status?.phase || 'Pending';\n const color = phase === 'Approved' ? 'primary' :\n phase === 'Rejected' ? 'secondary' : 'default';\n return <Chip label={phase} color={color} size=\"small\" />;\n },\n },\n {\n title: 'Reason',\n field: 'status.reason',\n render: (row: APIKey) => {\n if (row.status?.reason) {\n const color = row.status.phase === 'Rejected' ? 'error' : 'textPrimary';\n return (\n <Tooltip title={row.status.reason} placement=\"top\">\n <Typography\n variant=\"body2\"\n color={color}\n style={{\n maxWidth: '200px',\n overflow: 'hidden',\n textOverflow: 'ellipsis',\n whiteSpace: 'nowrap',\n }}\n >\n {row.status.reason}\n </Typography>\n </Tooltip>\n );\n }\n return <Typography variant=\"body2\" color=\"textSecondary\">-</Typography>;\n },\n },\n {\n title: 'Reviewed By',\n field: 'status.reviewedBy',\n render: (row: APIKey) => {\n if ((row.status?.phase === 'Approved' || row.status?.phase === 'Rejected') && row.status.reviewedBy) {\n const reviewedDate = row.status.reviewedAt ? new Date(row.status.reviewedAt).toLocaleDateString() : '';\n return (\n <Box>\n <Typography variant=\"body2\">{row.status.reviewedBy}</Typography>\n {reviewedDate && (\n <Typography variant=\"caption\" color=\"textSecondary\">\n {reviewedDate}\n </Typography>\n )}\n </Box>\n );\n }\n return <Typography variant=\"body2\" color=\"textSecondary\">-</Typography>;\n },\n },\n {\n title: 'API Key',\n field: 'status.apiKey',\n filtering: false,\n render: (row: APIKey) => {\n if (row.status?.phase === 'Approved' && row.status.apiKey) {\n const isVisible = visibleKeys.has(row.metadata.name);\n return (\n <Box display=\"flex\" alignItems=\"center\" style={{ gap: 8 }}>\n <Box fontFamily=\"monospace\" fontSize=\"0.875rem\">\n {isVisible ? row.status.apiKey : '•'.repeat(20) + '...'}\n </Box>\n <Tooltip title={isVisible ? 'hide key' : 'show key'}>\n <IconButton\n size=\"small\"\n onClick={() => toggleKeyVisibility(row.metadata.name)}\n >\n {isVisible ? <VisibilityOffIcon fontSize=\"small\" /> : <VisibilityIcon fontSize=\"small\" />}\n </IconButton>\n </Tooltip>\n </Box>\n );\n }\n return <Typography variant=\"body2\" color=\"textSecondary\">-</Typography>;\n },\n },\n {\n title: 'Requested',\n field: 'metadata.creationTimestamp',\n render: (row: APIKey) => {\n if (!row.metadata.creationTimestamp) {\n return <Typography variant=\"body2\">-</Typography>;\n }\n const date = new Date(row.metadata.creationTimestamp);\n return <Typography variant=\"body2\">{date.toLocaleDateString()}</Typography>;\n },\n },\n {\n title: '',\n filtering: false,\n render: (row: APIKey) => {\n const isDeleting = deleting === row.metadata.name;\n if (isDeleting) {\n return <CircularProgress size={20} />;\n }\n return (\n <IconButton\n size=\"small\"\n onClick={(e) => {\n e.stopPropagation();\n const rect = e.currentTarget.getBoundingClientRect();\n setMenuAnchor({ top: rect.bottom, left: rect.left });\n setMenuRequest(row);\n }}\n aria-controls={menuAnchor ? 'myapikeys-menu' : undefined}\n aria-haspopup=\"true\"\n >\n <MoreVertIcon />\n </IconButton>\n );\n },\n },\n ];\n\n const getTabData = () => {\n switch (selectedTab) {\n case 0:\n return approvedRequests;\n case 1:\n return pendingRequests;\n case 2:\n return rejectedRequests;\n default:\n return allRequests;\n }\n };\n\n const getTabColumns = () => {\n switch (selectedTab) {\n case 0: // Active - no Reason\n return columns.filter(col => col.title !== 'Reason');\n case 1: // Pending - no Reason, Reviewed By, API Key\n return columns.filter(col =>\n col.title !== 'Reason' &&\n col.title !== 'Reviewed By' &&\n col.title !== 'API Key'\n );\n case 2: // Rejected - no API Key\n return columns.filter(col => col.title !== 'API Key');\n default:\n return columns;\n }\n };\n\n const tabData = getTabData();\n const tabColumns = getTabColumns();\n const isPending = (row: APIKey) => !row.status || row.status.phase === 'Pending';\n\n return (\n <>\n <InfoCard\n title=\"My API Keys\"\n subheader={`${approvedRequests.length} active, ${pendingRequests.length} pending`}\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={`Active (${approvedRequests.length})`} />\n <Tab label={`Pending (${pendingRequests.length})`} />\n <Tab label={`Rejected (${rejectedRequests.length})`} />\n </Tabs>\n </Box>\n {tabData.length === 0 ? (\n <Box p={3} textAlign=\"center\">\n <Typography variant=\"body1\" color=\"textSecondary\">\n {selectedTab === 0 && 'No active API keys. Request access to an API to get started.'}\n {selectedTab === 1 && 'No pending requests.'}\n {selectedTab === 2 && 'No rejected requests.'}\n </Typography>\n </Box>\n ) : (\n <Table\n options={{\n paging: tabData.length > 5,\n pageSize: 20,\n search: true,\n filtering: true,\n debounceInterval: 300,\n toolbar: true,\n emptyRowsWhenPaging: false,\n }}\n columns={tabColumns}\n data={tabData.map((item: APIKey) => ({\n ...item,\n id: item.metadata.name,\n }))}\n />\n )}\n </InfoCard>\n\n <Menu\n id=\"myapikeys-menu\"\n open={Boolean(menuAnchor)}\n onClose={handleMenuClose}\n anchorReference=\"anchorPosition\"\n anchorPosition={menuAnchor || { top: 0, left: 0 }}\n >\n {menuRequest && (() => {\n const items = [];\n if (isPending(menuRequest)) {\n items.push(<MenuItem key=\"edit\" onClick={handleEdit}>Edit</MenuItem>);\n }\n items.push(<MenuItem key=\"delete\" onClick={handleDeleteClick}>Delete</MenuItem>);\n return items;\n })()}\n </Menu>\n\n {editDialogState.request && (\n <EditAPIKeyRequestDialog\n open={editDialogState.open}\n request={editDialogState.request}\n availablePlans={editDialogState.plans}\n onClose={() => setEditDialogState({ open: false, request: null, plans: [] })}\n onSuccess={() => {\n setEditDialogState({ open: false, request: null, plans: [] });\n setRefresh(r => r + 1);\n }}\n />\n )}\n\n <ConfirmDeleteDialog\n open={deleteDialogState.open}\n title=\"Delete API Key Request\"\n description={`Are you sure you want to delete the API key request for ${deleteDialogState.request?.spec.apiProductRef?.name || 'this API'}?`}\n deleting={deleting !== null}\n onConfirm={handleDeleteConfirm}\n onCancel={handleDeleteCancel}\n />\n </>\n );\n};\n","import React from 'react';\nimport { Typography, Box } from '@material-ui/core';\nimport { Progress } from '@backstage/core-components';\nimport { Permission } from '@backstage/plugin-permission-common';\nimport { useKuadrantPermission } from '../../utils/permissions';\n\ninterface PermissionGateProps {\n children: React.ReactNode;\n permission: Permission;\n fallback?: React.ReactNode;\n errorMessage?: string;\n}\n\nexport const PermissionGate = ({ children, permission, fallback, errorMessage }: PermissionGateProps) => {\n const { allowed, loading, error } = useKuadrantPermission(permission);\n\n if (loading) {\n return <Progress />;\n }\n\n if (error) {\n return (\n <Box p={4}>\n <Typography color=\"error\">\n Unable to check permissions: {error.message}\n </Typography>\n <Typography variant=\"body2\" color=\"textSecondary\">\n Please try again or contact your administrator\n </Typography>\n </Box>\n );\n }\n\n if (!allowed) {\n if (fallback) {\n return <>{fallback}</>;\n }\n return (\n <Box p={4}>\n <Typography color=\"textSecondary\">\n {errorMessage || 'You don\\'t have permission to view this page'}\n </Typography>\n <Box mt={1}>\n <Typography variant=\"caption\" color=\"textSecondary\">\n Required permission: {permission.name}\n </Typography>\n </Box>\n </Box>\n );\n }\n\n return <>{children}</>;\n};\n","import React from 'react';\nimport { Box, Typography, Chip } from '@material-ui/core';\nimport { Alert } from '@material-ui/lab';\n\ninterface PlanPolicyDetailsProps {\n selectedPolicy: {\n metadata: {\n name: string;\n };\n plans?: Array<{\n tier: string;\n description?: string;\n limits?: {\n daily?: number;\n monthly?: number;\n yearly?: number;\n };\n }>;\n } | null;\n alertSeverity?: 'warning' | 'info';\n alertMessage?: string;\n includeTopMargin?: boolean;\n}\n\nexport const PlanPolicyDetails: React.FC<PlanPolicyDetailsProps> = ({\n selectedPolicy,\n alertSeverity = 'warning',\n alertMessage = 'No PlanPolicy found for this HTTPRoute. API keys and rate limiting may not be available.',\n includeTopMargin = true,\n}) => {\n return (\n <Box\n mt={includeTopMargin ? 1 : 0}\n p={2}\n bgcolor=\"#f5f5f5\"\n borderRadius={1}\n border=\"1px solid #e0e0e0\"\n >\n {selectedPolicy ? (\n <>\n <Typography variant=\"subtitle2\" gutterBottom style={{ fontWeight: 600 }}>\n Associated PlanPolicy: <strong>{selectedPolicy.metadata.name}</strong>\n </Typography>\n\n {selectedPolicy.plans && selectedPolicy.plans.length > 0 ? (\n <>\n <Typography\n variant=\"caption\"\n display=\"block\"\n gutterBottom\n color=\"textSecondary\"\n style={{ marginTop: 8 }}\n >\n Available Plans:\n </Typography>\n <Box display=\"flex\" flexWrap=\"wrap\" mt={1} style={{ gap: 8 }}>\n {selectedPolicy.plans.map((plan: any, idx: number) => {\n const limitText = plan.limits?.daily\n ? `${plan.limits.daily}/day`\n : plan.limits?.monthly\n ? `${plan.limits.monthly}/month`\n : plan.limits?.yearly\n ? `${plan.limits.yearly}/year`\n : 'No limit';\n\n return (\n <Chip\n key={idx}\n label={`${plan.tier}: ${limitText}`}\n size=\"small\"\n variant=\"outlined\"\n color=\"primary\"\n />\n );\n })}\n </Box>\n {selectedPolicy.plans.some((p: any) => p.description) && (\n <Box mt={1}>\n {selectedPolicy.plans.filter((p: any) => p.description).map((plan: any, idx: number) => (\n <Typography key={idx} variant=\"caption\" display=\"block\" color=\"textSecondary\">\n • <strong>{plan.tier}:</strong> {plan.description}\n </Typography>\n ))}\n </Box>\n )}\n </>\n ) : (\n <Typography variant=\"caption\" color=\"textSecondary\">\n No plans defined in this PlanPolicy\n </Typography>\n )}\n </>\n ) : (\n <Alert severity={alertSeverity}>{alertMessage}</Alert>\n )}\n </Box>\n );\n};\n\n","import React, { useState } 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} from '@material-ui/core';\nimport { useApi, configApiRef, fetchApiRef } from '@backstage/core-plugin-api';\nimport { Alert } from '@material-ui/lab';\nimport useAsync from 'react-use/lib/useAsync';\nimport { PlanPolicyDetails } from '../PlanPolicyDetailsCard';\n\nconst useStyles = makeStyles({\n asterisk: {\n color: '#f44336',\n },\n});\n\ninterface CreateAPIProductDialogProps {\n open: boolean;\n onClose: () => void;\n onSuccess: () => void;\n}\n\nexport const CreateAPIProductDialog = ({ open, onClose, onSuccess }: CreateAPIProductDialogProps) => {\n const classes = useStyles();\n const config = useApi(configApiRef);\n const fetchApi = useApi(fetchApiRef);\n const backendUrl = config.getString('backend.baseUrl');\n\n const [name, setName] = useState('');\n const [displayName, setDisplayName] = useState('');\n const [description, setDescription] = useState('');\n const [version, setVersion] = useState('v1');\n const [approvalMode, setApprovalMode] = useState<'automatic' | 'manual'>('manual');\n const [publishStatus, setPublishStatus] = useState<'Draft' | 'Published'>('Published');\n const [tags, setTags] = useState<string[]>([]);\n const [tagInput, setTagInput] = useState('');\n const [selectedHTTPRoute, setSelectedHTTPRoute] = useState('');\n const [contactEmail, setContactEmail] = useState('');\n const [contactTeam, setContactTeam] = useState('');\n const [docsURL, setDocsURL] = useState('');\n const [openAPISpec, setOpenAPISpec] = useState('');\n const [error, setError] = useState('');\n const [creating, setCreating] = useState(false);\n\n const { value: httpRoutes, loading: httpRoutesLoading } = useAsync(async () => {\n const response = await fetchApi.fetch(`${backendUrl}/api/kuadrant/httproutes`);\n const data = await response.json();\n // filter to only show httproutes annotated for backstage exposure\n return (data.items || []).filter((route: any) =>\n route.metadata.annotations?.['backstage.io/expose'] === 'true'\n );\n }, [backendUrl, fetchApi, open]);\n\n // load planpolicies with full details to show associated plans\n const { value: planPolicies } = useAsync(async () => {\n const response = await fetchApi.fetch(`${backendUrl}/api/kuadrant/planpolicies`);\n return await response.json();\n }, [backendUrl, fetchApi, open]);\n\n // find planpolicy associated with selected httproute\n const getPlanPolicyForRoute = (routeNamespace: string, routeName: string) => {\n if (!planPolicies?.items) return null;\n\n return planPolicies.items.find((pp: any) => {\n const ref = pp.targetRef;\n return (\n ref?.kind === 'HTTPRoute' &&\n ref?.name === routeName &&\n (!ref?.namespace || ref?.namespace === routeNamespace)\n );\n });\n };\n\n const selectedRouteInfo = selectedHTTPRoute ? selectedHTTPRoute.split('/') : null;\n const selectedPolicy = selectedRouteInfo\n ? getPlanPolicyForRoute(selectedRouteInfo[0], selectedRouteInfo[1])\n : null;\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 handleCreate = async () => {\n setError('');\n setCreating(true);\n\n try {\n if (!selectedHTTPRoute) {\n throw new Error('Please select an HTTPRoute');\n }\n\n const [selectedRouteNamespace, selectedRouteName] = selectedHTTPRoute.split('/');\n\n // derive namespace from selected httproute\n const namespace = selectedRouteNamespace;\n\n const apiProduct = {\n apiVersion: 'devportal.kuadrant.io/v1alpha1',\n kind: 'APIProduct',\n metadata: {\n name,\n namespace,\n },\n spec: {\n displayName,\n description,\n version,\n approvalMode,\n publishStatus,\n tags,\n targetRef: {\n group: 'gateway.networking.k8s.io',\n kind: 'HTTPRoute',\n name: selectedRouteName,\n namespace: selectedRouteNamespace,\n },\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 && { openAPISpec }),\n },\n } : {}),\n },\n };\n\n const response = await fetchApi.fetch(`${backendUrl}/api/kuadrant/apiproducts`, {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n },\n body: JSON.stringify(apiProduct),\n });\n\n if (!response.ok) {\n const errorData = await response.json();\n throw new Error(errorData.error || 'failed to create apiproduct');\n }\n\n onSuccess();\n handleClose();\n } catch (err) {\n setError(err instanceof Error ? err.message : String(err));\n } finally {\n setCreating(false);\n }\n };\n\n const handleClose = () => {\n setName('');\n setDisplayName('');\n setDescription('');\n setVersion('v1');\n setApprovalMode('manual');\n setPublishStatus('Published');\n setTags([]);\n setTagInput('');\n setSelectedHTTPRoute('');\n setContactEmail('');\n setContactTeam('');\n setDocsURL('');\n setOpenAPISpec('');\n setError('');\n onClose();\n };\n\n return (\n <Dialog open={open} onClose={handleClose} maxWidth=\"md\" fullWidth>\n <DialogTitle>Create API Product</DialogTitle>\n <DialogContent>\n {error && (\n <Alert severity=\"error\" style={{ marginBottom: 16 }}>\n {error}\n </Alert>\n )}\n\n <Grid container spacing={2}>\n <Grid item xs={6}>\n <TextField\n fullWidth\n label=\"Name\"\n value={name}\n onChange={e => setName(e.target.value)}\n placeholder=\"my-api\"\n helperText=\"Kubernetes resource name (lowercase, hyphens)\"\n margin=\"normal\"\n required\n disabled={creating}\n InputLabelProps={{\n classes: {\n asterisk: classes.asterisk,\n },\n }}\n />\n </Grid>\n <Grid item xs={6}>\n <TextField\n fullWidth\n label=\"Display Name\"\n value={displayName}\n onChange={e => setDisplayName(e.target.value)}\n placeholder=\"My API\"\n margin=\"normal\"\n required\n disabled={creating}\n InputLabelProps={{\n classes: {\n asterisk: classes.asterisk,\n },\n }}\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 margin=\"normal\"\n disabled={creating}\n />\n </Grid>\n <Grid item xs={12}>\n <TextField\n fullWidth\n select\n label=\"Approval Mode\"\n value={approvalMode}\n onChange={e => setApprovalMode(e.target.value as 'automatic' | 'manual')}\n margin=\"normal\"\n helperText=\"Automatic: keys are created immediately. Manual: requires approval.\"\n disabled={creating}\n >\n <MenuItem value=\"manual\">Manual</MenuItem>\n <MenuItem value=\"automatic\">Automatic</MenuItem>\n </TextField>\n </Grid>\n <Grid item xs={12}>\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=\"Draft: hidden from catalog. Published: visible to consumers.\"\n disabled={creating}\n >\n <MenuItem value=\"Draft\">Draft</MenuItem>\n <MenuItem value=\"Published\">Published</MenuItem>\n </TextField>\n </Grid>\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={creating}\n InputLabelProps={{\n classes: {\n asterisk: classes.asterisk,\n },\n }}\n />\n </Grid>\n\n <Grid item xs={12}>\n <Typography variant=\"subtitle2\" gutterBottom style={{ marginTop: 16 }}>\n Tags\n </Typography>\n <Box display=\"flex\" flexWrap=\"wrap\" marginBottom={1} style={{ gap: 8 }}>\n {tags.map(tag => (\n <Chip\n key={tag}\n label={tag}\n onDelete={creating ? undefined : () => handleDeleteTag(tag)}\n size=\"small\"\n disabled={creating}\n />\n ))}\n </Box>\n <Box display=\"flex\" style={{ gap: 8 }}>\n <TextField\n fullWidth\n size=\"small\"\n value={tagInput}\n onChange={e => setTagInput(e.target.value)}\n onKeyPress={e => e.key === 'Enter' && handleAddTag()}\n placeholder=\"Add tag\"\n disabled={creating}\n />\n <Button onClick={handleAddTag} variant=\"outlined\" size=\"small\" disabled={creating}>\n Add\n </Button>\n </Box>\n </Grid>\n\n <Grid item xs={12}>\n <TextField\n fullWidth\n select\n label=\"HTTPRoute\"\n value={selectedHTTPRoute}\n onChange={e => setSelectedHTTPRoute(e.target.value)}\n margin=\"normal\"\n required\n helperText=\"Select an HTTPRoute (backstage.io/expose: true). APIProduct will be created in the same namespace.\"\n disabled={httpRoutesLoading || creating}\n InputLabelProps={{\n classes: {\n asterisk: classes.asterisk,\n },\n }}\n >\n {httpRoutesLoading && (\n <MenuItem value=\"\">Loading...</MenuItem>\n )}\n {!httpRoutesLoading && httpRoutes && httpRoutes.length === 0 && (\n <MenuItem value=\"\">No HTTPRoutes available</MenuItem>\n )}\n {!httpRoutesLoading && httpRoutes && httpRoutes.map((route: any) => (\n <MenuItem\n key={`${route.metadata.namespace}/${route.metadata.name}`}\n value={`${route.metadata.namespace}/${route.metadata.name}`}\n >\n {route.metadata.name} ({route.metadata.namespace})\n </MenuItem>\n ))}\n </TextField>\n </Grid>\n {selectedHTTPRoute && (\n <Grid item xs={12}>\n <PlanPolicyDetails\n selectedPolicy={selectedPolicy}\n alertSeverity=\"warning\"\n alertMessage=\"No PlanPolicy found for this HTTPRoute. API keys and rate limiting may not be available.\"\n includeTopMargin={true}\n />\n </Grid>\n )}\n\n <Grid item xs={6}>\n <TextField\n fullWidth\n label=\"Contact Email\"\n value={contactEmail}\n onChange={e => setContactEmail(e.target.value)}\n placeholder=\"api-team@example.com\"\n margin=\"normal\"\n disabled={creating}\n />\n </Grid>\n <Grid item xs={6}>\n <TextField\n fullWidth\n label=\"Contact Team\"\n value={contactTeam}\n onChange={e => setContactTeam(e.target.value)}\n placeholder=\"platform-team\"\n margin=\"normal\"\n disabled={creating}\n />\n </Grid>\n <Grid item xs={6}>\n <TextField\n fullWidth\n label=\"Docs URL\"\n value={docsURL}\n onChange={e => setDocsURL(e.target.value)}\n placeholder=\"https://api.example.com/docs\"\n margin=\"normal\"\n disabled={creating}\n />\n </Grid>\n <Grid item xs={6}>\n <TextField\n fullWidth\n label=\"OpenAPI Spec URL\"\n value={openAPISpec}\n onChange={e => setOpenAPISpec(e.target.value)}\n placeholder=\"https://api.example.com/openapi.json\"\n margin=\"normal\"\n disabled={creating}\n />\n </Grid>\n </Grid>\n </DialogContent>\n <DialogActions>\n <Button onClick={handleClose} disabled={creating}>Cancel</Button>\n <Button\n onClick={handleCreate}\n color=\"primary\"\n variant=\"contained\"\n disabled={creating || !name || !displayName || !description || !selectedHTTPRoute}\n startIcon={creating ? <CircularProgress size={16} color=\"inherit\" /> : undefined}\n >\n {creating ? 'Creating...' : 'Create'}\n </Button>\n </DialogActions>\n </Dialog>\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} from '@material-ui/core';\nimport { useApi, configApiRef, fetchApiRef } from '@backstage/core-plugin-api';\nimport { Alert } from '@material-ui/lab';\nimport { Progress } from '@backstage/core-components';\nimport useAsync from 'react-use/lib/useAsync';\nimport { PlanPolicyDetails } from '../PlanPolicyDetailsCard';\n\nconst useStyles = makeStyles({\n asterisk: {\n color: '#f44336',\n },\n});\n\ninterface EditAPIProductDialogProps {\n open: boolean;\n onClose: () => void;\n onSuccess: () => void;\n namespace: string;\n name: string;\n}\n\nexport const EditAPIProductDialog = ({open, onClose, onSuccess, namespace, name}: EditAPIProductDialogProps) => {\n const classes = useStyles();\n const config = useApi(configApiRef);\n const fetchApi = useApi(fetchApiRef);\n const backendUrl = config.getString('backend.baseUrl');\n const [loading, setLoading] = useState(false);\n const [displayName, setDisplayName] = useState('');\n const [description, setDescription] = useState('');\n const [version, setVersion] = useState('v1');\n const [publishStatus, setPublishStatus] = useState<'Draft' | 'Published'>('Draft');\n const [approvalMode, setApprovalMode] = useState<'automatic' | 'manual'>('manual');\n const [tags, setTags] = useState<string[]>([]);\n const [targetRef, setTargetRef] = useState<any>(null);\n const [tagInput, setTagInput] = useState('');\n const [contactEmail, setContactEmail] = useState('');\n const [contactTeam, setContactTeam] = useState('');\n const [docsURL, setDocsURL] = useState('');\n const [openAPISpec, setOpenAPISpec] = useState('');\n const [error, setError] = useState('');\n const [saving, setSaving] = useState(false);\n\n // Load APIProduct data when dialog opens\n useEffect(() => {\n if (open && namespace && name) {\n setLoading(true);\n setError('');\n\n fetchApi.fetch(`${backendUrl}/api/kuadrant/apiproducts/${namespace}/${name}`)\n .then(async res => {\n if (!res.ok) {\n const errorData = await res.json();\n throw new Error(errorData.error || `Failed to fetch API product: ${res.status}`);\n }\n return res.json();\n })\n .then(data => {\n setDisplayName(data.spec.displayName || '');\n setDescription(data.spec.description || '');\n setVersion(data.spec.version || 'v1');\n setPublishStatus(data.spec.publishStatus || 'Draft');\n setApprovalMode(data.spec.approvalMode || 'manual');\n setTags(data.spec.tags || []);\n setTargetRef(data.spec.targetRef || null);\n setContactEmail(data.spec.contact?.email || '');\n setContactTeam(data.spec.contact?.team || '');\n setDocsURL(data.spec.documentation?.docsURL || '');\n setOpenAPISpec(data.spec.documentation?.openAPISpec || '');\n setLoading(false);\n })\n .catch(err => {\n setError(err.message || 'Failed to load API product');\n setLoading(false);\n });\n }\n }, [open, namespace, name, backendUrl, fetchApi]);\n\n // load planpolicies with full details to show associated plans\n const { value: planPolicies } = useAsync(async () => {\n if (!open) return null;\n const response = await fetchApi.fetch(`${backendUrl}/api/kuadrant/planpolicies`);\n return await response.json();\n }, [backendUrl, fetchApi, open]);\n\n // find planpolicy associated with targetRef\n const selectedPolicy = React.useMemo(() => {\n if (!planPolicies?.items || !targetRef) return null;\n\n return planPolicies.items.find((pp: any) => {\n const ref = pp.targetRef;\n return (\n ref?.kind === 'HTTPRoute' &&\n ref?.name === targetRef.name &&\n (!ref?.namespace || ref?.namespace === (targetRef.namespace || namespace))\n );\n });\n }, [planPolicies, targetRef, namespace]);\n\n const handleAddTag = () => {\n if (tagInput.trim() && !tags.includes(tagInput.trim())) {\n setTags([...tags, tagInput.trim()]);\n setTagInput('');\n }\n };\n\n const handleDeleteTag = (tagToDelete: string) => {\n setTags(tags.filter(tag => tag !== tagToDelete));\n };\n\n const handleSave = async () => {\n setError('');\n setSaving(true);\n\n try {\n const patch = {\n spec: {\n displayName,\n description,\n version,\n publishStatus,\n approvalMode,\n tags,\n targetRef,\n ...(contactEmail || contactTeam ? {\n contact: {\n ...(contactEmail && { email: contactEmail }),\n ...(contactTeam && { team: contactTeam }),\n },\n } : {}),\n ...(docsURL || openAPISpec ? {\n documentation: {\n ...(docsURL && { docsURL }),\n ...(openAPISpec && { openAPISpec }),\n },\n } : {}),\n },\n };\n\n const response = await fetchApi.fetch(\n `${backendUrl}/api/kuadrant/apiproducts/${namespace}/${name}`,\n {\n method: 'PATCH',\n headers: {\n 'Content-Type': 'application/json',\n },\n body: JSON.stringify(patch),\n }\n );\n\n if (!response.ok) {\n const errorData = await response.json();\n throw new Error(errorData.error || 'failed to update apiproduct');\n }\n\n onSuccess();\n onClose();\n } catch (err) {\n setError(err instanceof Error ? err.message : String(err));\n } finally {\n setSaving(false);\n }\n };\n\n return (\n <Dialog open={open} onClose={onClose} maxWidth=\"md\" fullWidth>\n <DialogTitle>Edit API Product</DialogTitle>\n <DialogContent>\n {error && (\n <Alert severity=\"error\" style={{ marginBottom: 16 }}>\n {error}\n </Alert>\n )}\n\n {loading ? (\n <Progress />\n ) : (\n <Grid container spacing={2}>\n <Grid item xs={6}>\n <TextField\n fullWidth\n label=\"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=\"Namespace\"\n value={namespace}\n disabled\n helperText=\"Derived from HTTPRoute (immutable)\"\n margin=\"normal\"\n />\n </Grid>\n <Grid item xs={6}>\n <TextField\n fullWidth\n label=\"Display Name\"\n value={displayName}\n onChange={e => setDisplayName(e.target.value)}\n placeholder=\"My API\"\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=\"Version\"\n value={version}\n onChange={e => setVersion(e.target.value)}\n placeholder=\"v1\"\n margin=\"normal\"\n disabled={saving}\n />\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=\"Draft: Hidden from catalog. Published: Visible in catalog.\"\n disabled={saving}\n >\n <MenuItem value=\"Draft\">Draft (Hidden)</MenuItem>\n <MenuItem value=\"Published\">Published (Visible)</MenuItem>\n </TextField>\n </Grid>\n <Grid item xs={6}>\n <TextField\n fullWidth\n select\n label=\"Approval Mode\"\n value={approvalMode}\n onChange={e => setApprovalMode(e.target.value as 'automatic' | 'manual')}\n margin=\"normal\"\n helperText=\"Automatic: keys created immediately. Manual: requires approval.\"\n disabled={saving}\n >\n <MenuItem value=\"manual\">Manual</MenuItem>\n <MenuItem value=\"automatic\">Automatic</MenuItem>\n </TextField>\n </Grid>\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\n <Grid item xs={12}>\n <Typography variant=\"subtitle2\" gutterBottom style={{ marginTop: 16 }}>\n Tags\n </Typography>\n <Box display=\"flex\" flexWrap=\"wrap\" marginBottom={1} style={{ gap: 8 }}>\n {tags.map(tag => (\n <Chip\n key={tag}\n label={tag}\n onDelete={saving ? undefined : () => handleDeleteTag(tag)}\n size=\"small\"\n disabled={saving}\n />\n ))}\n </Box>\n <Box display=\"flex\" style={{ gap: 8 }}>\n <TextField\n fullWidth\n size=\"small\"\n value={tagInput}\n onChange={e => setTagInput(e.target.value)}\n onKeyPress={e => e.key === 'Enter' && handleAddTag()}\n placeholder=\"Add tag\"\n disabled={saving}\n />\n <Button onClick={handleAddTag} variant=\"outlined\" size=\"small\" disabled={saving}>\n Add\n </Button>\n </Box>\n </Grid>\n {targetRef && (\n <>\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 item xs={12}>\n <PlanPolicyDetails\n selectedPolicy={selectedPolicy}\n alertSeverity=\"info\"\n alertMessage=\"No PlanPolicy found for this HTTPRoute.\"\n includeTopMargin={false}\n />\n </Grid>\n </>\n )}\n\n <Grid item xs={6}>\n <TextField\n fullWidth\n label=\"Contact Email\"\n value={contactEmail}\n onChange={e => setContactEmail(e.target.value)}\n placeholder=\"api-team@example.com\"\n margin=\"normal\"\n disabled={saving}\n />\n </Grid>\n <Grid item xs={6}>\n <TextField\n fullWidth\n label=\"Contact Team\"\n value={contactTeam}\n onChange={e => setContactTeam(e.target.value)}\n placeholder=\"platform-team\"\n margin=\"normal\"\n disabled={saving}\n />\n </Grid>\n <Grid item xs={6}>\n <TextField\n fullWidth\n label=\"Docs URL\"\n value={docsURL}\n onChange={e => setDocsURL(e.target.value)}\n placeholder=\"https://api.example.com/docs\"\n margin=\"normal\"\n disabled={saving}\n />\n </Grid>\n <Grid item xs={6}>\n <TextField\n fullWidth\n label=\"OpenAPI Spec URL\"\n value={openAPISpec}\n onChange={e => setOpenAPISpec(e.target.value)}\n placeholder=\"https://api.example.com/openapi.json\"\n margin=\"normal\"\n disabled={saving}\n />\n </Grid>\n </Grid>\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}\n startIcon={saving ? <CircularProgress size={16} color=\"inherit\" /> : undefined}\n >\n {saving ? 'Saving...' : 'Save'}\n </Button>\n </DialogActions>\n </Dialog>\n );\n};\n","import React, { useState } from 'react';\nimport { Typography, Grid, Box, Chip, Button, IconButton } from '@material-ui/core';\nimport AddIcon from '@material-ui/icons/Add';\nimport DeleteIcon from '@material-ui/icons/Delete';\nimport EditIcon from '@material-ui/icons/Edit';\nimport {\n InfoCard,\n Header,\n Page,\n Content,\n SupportButton,\n Progress,\n ResponseErrorPanel,\n Link,\n Table,\n TableColumn,\n} from '@backstage/core-components';\nimport useAsync from 'react-use/lib/useAsync';\nimport { useApi, configApiRef, fetchApiRef, alertApiRef, identityApiRef } from '@backstage/core-plugin-api';\nimport { ApprovalQueueCard } from '../ApprovalQueueCard';\nimport { MyApiKeysCard } from '../MyApiKeysCard';\nimport { PermissionGate } from '../PermissionGate';\nimport { CreateAPIProductDialog } from '../CreateAPIProductDialog';\nimport {\n kuadrantApiProductCreatePermission,\n kuadrantApiProductDeleteOwnPermission,\n kuadrantApiProductDeleteAllPermission,\n kuadrantApiProductUpdateOwnPermission,\n kuadrantApiProductUpdateAllPermission,\n kuadrantApiProductListPermission,\n kuadrantApiKeyRequestReadAllPermission,\n kuadrantApiKeyRequestReadOwnPermission,\n kuadrantPlanPolicyListPermission,\n} from '../../permissions';\nimport { useKuadrantPermission } from '../../utils/permissions';\nimport { EditAPIProductDialog } from '../EditAPIProductDialog';\nimport { ConfirmDeleteDialog } from '../ConfirmDeleteDialog';\n\ntype KuadrantResource = {\n metadata: {\n name: string;\n namespace: string;\n creationTimestamp: string;\n annotations?: Record<string, string>;\n };\n spec?: any;\n};\n\ntype KuadrantList = {\n items: KuadrantResource[];\n};\n\nexport const ResourceList = () => {\n const config = useApi(configApiRef);\n const fetchApi = useApi(fetchApiRef);\n const alertApi = useApi(alertApiRef);\n const identityApi = useApi(identityApiRef);\n const backendUrl = config.getString('backend.baseUrl');\n const [userEntityRef, setUserEntityRef] = useState<string>('');\n const [createDialogOpen, setCreateDialogOpen] = useState(false);\n const [editDialogOpen, setEditDialogOpen] = useState(false);\n const [refreshTrigger, setRefreshTrigger] = useState(0);\n const [deleteDialogOpen, setDeleteDialogOpen] = useState(false);\n const [apiProductToDelete, setApiProductToDelete] = useState<{ namespace: string; name: string } | null>(null);\n const [apiProductToEdit, setApiProductToEdit] = useState<{ namespace: string; name: string } | null>(null);\n const [deleting, setDeleting] = useState(false);\n\n const {\n allowed: canCreateApiProduct,\n loading: createPermissionLoading,\n error: createPermissionError,\n } = useKuadrantPermission(kuadrantApiProductCreatePermission);\n\n const {\n allowed: canViewAllRequests,\n loading: approvalQueueAllPermissionLoading,\n } = useKuadrantPermission(kuadrantApiKeyRequestReadAllPermission);\n\n const {\n allowed: canViewOwnRequests,\n loading: approvalQueueOwnPermissionLoading,\n error: approvalQueuePermissionError,\n } = useKuadrantPermission(kuadrantApiKeyRequestReadOwnPermission);\n\n const canViewApprovalQueue = canViewAllRequests || canViewOwnRequests;\n const approvalQueuePermissionLoading = approvalQueueAllPermissionLoading || approvalQueueOwnPermissionLoading;\n\n const {\n allowed: canDeleteOwnApiProduct,\n loading: deleteOwnPermissionLoading,\n } = useKuadrantPermission(kuadrantApiProductDeleteOwnPermission);\n\n const {\n allowed: canDeleteAllApiProducts,\n loading: deleteAllPermissionLoading,\n error: deletePermissionError,\n } = useKuadrantPermission(kuadrantApiProductDeleteAllPermission);\n\n const {\n allowed: canUpdateOwnApiProduct,\n } = useKuadrantPermission(kuadrantApiProductUpdateOwnPermission);\n\n const {\n allowed: canUpdateAllApiProducts,\n } = useKuadrantPermission(kuadrantApiProductUpdateAllPermission);\n\n const deletePermissionLoading = deleteOwnPermissionLoading || deleteAllPermissionLoading;\n\n const {\n allowed: canListPlanPolicies,\n loading: planPolicyPermissionLoading,\n error: planPolicyPermissionError,\n } = useKuadrantPermission(kuadrantPlanPolicyListPermission);\n\n useAsync(async () => {\n const identity = await identityApi.getBackstageIdentity();\n setUserEntityRef(identity.userEntityRef);\n }, [identityApi]);\n\n const { value: apiProducts, loading: apiProductsLoading, error: apiProductsError } = useAsync(async (): Promise<KuadrantList> => {\n const response = await fetchApi.fetch(`${backendUrl}/api/kuadrant/apiproducts`);\n return await response.json();\n }, [backendUrl, fetchApi, refreshTrigger]);\n\n const { value: planPolicies, loading: planPoliciesLoading, error: planPoliciesError } = useAsync(async (): Promise<KuadrantList> => {\n const response = await fetchApi.fetch(`${backendUrl}/api/kuadrant/planpolicies`);\n return await response.json();\n }, [backendUrl, fetchApi, refreshTrigger]);\n\n const loading = apiProductsLoading || planPoliciesLoading || createPermissionLoading || approvalQueuePermissionLoading || deletePermissionLoading || planPolicyPermissionLoading;\n const error = apiProductsError || planPoliciesError;\n const permissionError = createPermissionError || approvalQueuePermissionError || deletePermissionError || planPolicyPermissionError;\n\n const handleCreateSuccess = () => {\n setRefreshTrigger(prev => prev + 1);\n alertApi.post({ message: 'API Product created', severity: 'success', display: 'transient' });\n };\n\n const handleEditClick = (namespace: string, name: string) => {\n setApiProductToEdit({ namespace, name });\n setEditDialogOpen(true);\n };\n\n const handleEditSuccess = () => {\n setRefreshTrigger(prev => prev + 1);\n alertApi.post({ message: 'API Product updated', severity: 'success', display: 'transient' });\n };\n\n const handleDeleteClick = (namespace: string, name: string) => {\n setApiProductToDelete({ namespace, name });\n setDeleteDialogOpen(true);\n };\n\n const handleDeleteConfirm = async () => {\n if (!apiProductToDelete) return;\n\n setDeleting(true);\n try {\n const response = await fetchApi.fetch(\n `${backendUrl}/api/kuadrant/apiproducts/${apiProductToDelete.namespace}/${apiProductToDelete.name}`,\n { method: 'DELETE' }\n );\n\n if (!response.ok) {\n throw new Error('failed to delete apiproduct');\n }\n\n setRefreshTrigger(prev => prev + 1);\n alertApi.post({ message: 'API Product deleted', severity: 'success', display: 'transient' });\n } catch (err) {\n console.error('error deleting apiproduct:', err);\n alertApi.post({ message: 'Failed to delete API Product', severity: 'error', display: 'transient' });\n } finally {\n setDeleting(false);\n setDeleteDialogOpen(false);\n setApiProductToDelete(null);\n }\n };\n\n const handleDeleteCancel = () => {\n setDeleteDialogOpen(false);\n setApiProductToDelete(null);\n };\n\n const formatDate = (dateString: string) => {\n const date = new Date(dateString);\n return date.toLocaleDateString('en-GB', {\n year: 'numeric',\n month: 'short',\n day: 'numeric',\n });\n };\n\n const columns: TableColumn[] = [\n {\n title: 'Name',\n field: 'spec.displayName',\n render: (row: any) => {\n const publishStatus = row.spec?.publishStatus;\n const isPublished = publishStatus === 'Published';\n const displayName = row.spec?.displayName ?? row.metadata.name;\n\n if (isPublished) {\n return (\n <Link to={`/catalog/default/api/${row.metadata.name}/api-product`}>\n <strong>{displayName}</strong>\n </Link>\n );\n }\n\n return (\n <span className=\"text-muted\">\n <strong>{displayName}</strong>\n </span>\n );\n },\n customFilterAndSearch: (term, row: any) => {\n const displayName = row.spec?.displayName || row.metadata.name || '';\n return displayName.toLowerCase().includes(term.toLowerCase());\n },\n },\n {\n title: 'Resource Name',\n field: 'metadata.name',\n },\n {\n title: 'Version',\n field: 'spec.version',\n render: (row: any) => row.spec?.version || '-',\n },\n {\n title: 'HTTPRoute',\n field: 'spec.targetRef.name',\n render: (row: any) => row.spec?.targetRef?.name || '-',\n },\n {\n title: 'Publish Status',\n field: 'spec.publishStatus',\n render: (row: any) => {\n const status = row.spec?.publishStatus || 'Draft';\n return (\n <Chip\n label={status}\n size=\"small\"\n color={status === 'Published' ? 'primary' : 'default'}\n />\n );\n },\n },\n {\n title: 'Approval Mode',\n field: 'spec.approvalMode',\n render: (row: any) => {\n const mode = row.spec?.approvalMode || 'manual';\n return (\n <Chip\n label={mode}\n size=\"small\"\n color={mode === 'automatic' ? 'secondary' : 'default'}\n />\n );\n },\n },\n {\n title: 'Namespace',\n field: 'metadata.namespace',\n },\n {\n title: 'Created',\n field: 'metadata.creationTimestamp',\n render: (row: any) => formatDate(row.metadata.creationTimestamp),\n },\n {\n title: 'Actions',\n field: 'actions',\n filtering: false,\n render: (row: any) => {\n const owner = row.metadata?.annotations?.['backstage.io/owner'];\n const isOwner = owner === userEntityRef;\n const canEdit = canUpdateAllApiProducts || (canUpdateOwnApiProduct && isOwner);\n const canDelete = canDeleteAllApiProducts || (canDeleteOwnApiProduct && isOwner);\n\n if (!canEdit && !canDelete) return null;\n\n return (\n <Box display=\"flex\" style={{ gap: 4 }}>\n {canEdit && (\n <IconButton\n size=\"small\"\n onClick={() => handleEditClick(row.metadata.namespace, row.metadata.name)}\n title=\"Edit API Product\"\n >\n <EditIcon fontSize=\"small\" />\n </IconButton>\n )}\n\n {canDelete && (\n <IconButton\n size=\"small\"\n onClick={() => handleDeleteClick(row.metadata.namespace, row.metadata.name)}\n title=\"Delete API Product\"\n >\n <DeleteIcon fontSize=\"small\" />\n </IconButton>\n )}\n </Box>\n );\n },\n },\n ];\n\n const planPolicyColumns: TableColumn[] = [\n {\n title: 'Name',\n field: 'metadata.name',\n render: (row: any) => (\n <Link to={`/kuadrant/planpolicy/${row.metadata.namespace}/${row.metadata.name}`}>\n <strong>{row.metadata.name}</strong>\n </Link>\n ),\n },\n {\n title: 'Namespace',\n field: 'metadata.namespace',\n },\n ];\n\n const renderResources = (resources: KuadrantResource[] | undefined) => {\n if (!resources || resources.length === 0) {\n return <Typography variant=\"body2\" color=\"textSecondary\">No API products found</Typography>;\n }\n return (\n <Table\n options={{\n paging: resources.length > 5,\n pageSize: 20,\n search: true,\n filtering: true,\n debounceInterval: 300,\n toolbar: true,\n emptyRowsWhenPaging: false,\n }}\n columns={columns}\n data={resources}\n />\n );\n };\n\n const renderPlanPolicies = (resources: KuadrantResource[] | undefined) => {\n if (!resources || resources.length === 0) {\n return <Typography variant=\"body2\" color=\"textSecondary\">No plan policies found</Typography>;\n }\n return (\n <Table\n options={{ paging: false, search: false, toolbar: false }}\n columns={planPolicyColumns}\n data={resources}\n />\n );\n };\n\n return (\n <Page themeId=\"tool\">\n <Header title=\"Kuadrant\" subtitle=\"API management for Kubernetes\">\n <SupportButton>Manage API products and access requests</SupportButton>\n </Header>\n <Content>\n {loading && <Progress />}\n {error && <ResponseErrorPanel error={error} />}\n {permissionError && (\n <Box p={2}>\n <Typography color=\"error\">\n unable to check permissions: {permissionError.message}\n </Typography>\n <Typography variant=\"body2\" color=\"textSecondary\">\n permission: {createPermissionError ? 'kuadrant.apiproduct.create' :\n deletePermissionError ? 'kuadrant.apiproduct.delete' :\n approvalQueuePermissionError ? 'kuadrant.apikeyrequest.read.all' :\n planPolicyPermissionError ? 'kuadrant.planpolicy.list' : 'unknown'}\n </Typography>\n <Typography variant=\"body2\" color=\"textSecondary\">\n please try again or contact your administrator\n </Typography>\n </Box>\n )}\n {!loading && !error && !permissionError && (\n <Grid container spacing={3} direction=\"column\">\n <Grid item>\n <MyApiKeysCard />\n </Grid>\n\n <Grid item>\n <InfoCard\n title=\"API Products\"\n action={\n canCreateApiProduct ? (\n <Box display=\"flex\" alignItems=\"center\" height=\"100%\" mt={1}>\n <Button\n variant=\"contained\"\n color=\"primary\"\n size=\"small\"\n startIcon={<AddIcon />}\n onClick={() => setCreateDialogOpen(true)}\n >\n Create API Product\n </Button>\n </Box>\n ) : undefined\n }\n >\n {renderResources(apiProducts?.items)}\n </InfoCard>\n </Grid>\n\n {canListPlanPolicies && (\n <Grid item>\n <InfoCard title=\"Plan Policies\">\n {renderPlanPolicies(planPolicies?.items)}\n </InfoCard>\n </Grid>\n )}\n\n {canViewApprovalQueue && (\n <Grid item>\n <ApprovalQueueCard />\n </Grid>\n )}\n </Grid>\n )}\n <CreateAPIProductDialog\n open={createDialogOpen}\n onClose={() => setCreateDialogOpen(false)}\n onSuccess={handleCreateSuccess}\n />\n <EditAPIProductDialog\n open={editDialogOpen}\n onClose={() => setEditDialogOpen(false)}\n onSuccess={handleEditSuccess}\n namespace={apiProductToEdit?.namespace || ''}\n name={apiProductToEdit?.name || ''}\n />\n <ConfirmDeleteDialog\n open={deleteDialogOpen}\n title=\"Delete API Product\"\n description={`This will permanently delete \"${apiProductToDelete?.name}\" from namespace \"${apiProductToDelete?.namespace}\" and remove it from Kubernetes. Any associated API keys will stop working.`}\n confirmText={apiProductToDelete?.name}\n severity=\"high\"\n deleting={deleting}\n onConfirm={handleDeleteConfirm}\n onCancel={handleDeleteCancel}\n />\n </Content>\n </Page>\n );\n};\n\nexport const KuadrantPage = () => {\n return (\n <PermissionGate\n permission={kuadrantApiProductListPermission}\n errorMessage=\"you don't have permission to view the Kuadrant page\"\n >\n <ResourceList />\n </PermissionGate>\n );\n};\n"],"names":["ConfirmDeleteDialog","open","title","description","confirmText","severity","deleting","onConfirm","onCancel","inputValue","setInputValue","useState","useEffect","requiresTextConfirmation","canConfirm","Dialog","onClose","undefined","maxWidth","fullWidth","DialogTitle","Box","display","alignItems","style","gap","WarningIcon","color","span","DialogContent","DialogContentText","mt","Typography","variant","gutterBottom","strong","TextField","size","value","onChange","e","target","disabled","autoFocus","placeholder","DialogActions","Button","onClick","startIcon","CircularProgress","EditAPIKeyRequestDialog","onSuccess","request","availablePlans","config","useApi","configApiRef","fetchApi","fetchApiRef","backendUrl","getString","planTier","setPlanTier","useCase","setUseCase","saving","setSaving","error","setError","spec","handleClose","mb","p","bgcolor","borderRadius","FormControl","margin","InputLabel","Select","map","plan","limitDesc","Object","entries","limits","key","val","join","MenuItem","tier","label","multiline","rows","helperText","async","patch","trim","response","fetch","metadata","namespace","name","method","headers","body","JSON","stringify","ok","errorData","json","catch","Error","status","err","console","message","MyApiKeysCard","deleteDialogState","identityApi","identityApiRef","alertApi","alertApiRef","selectedTab","setSelectedTab","setUserId","visibleKeys","setVisibleKeys","Set","menuAnchor","setMenuAnchor","menuRequest","setMenuRequest","editDialogState","setEditDialogState","plans","refresh","setRefresh","setDeleting","setDeleteDialogState","useAsync","identity","getBackstageIdentity","extractedUserId","userEntityRef","split","log","optimisticallyDeleted","setOptimisticallyDeleted","requests","loading","items","InfoCard","Progress","allRequests","filter","r","has","approvedRequests","phase","pendingRequests","rejectedRequests","handleMenuClose","handleEdit","apiProductName","apiProductRef","apiProductNamespace","apiProductResponse","apiProduct","handleDeleteClick","columns","field","render","row","Link","to","Chip","Tooltip","placement","overflow","textOverflow","whiteSpace","reason","reviewedBy","reviewedDate","reviewedAt","Date","toLocaleDateString","filtering","apiKey","isVisible","fontFamily","fontSize","repeat","IconButton","toggleKeyVisibility","keyName","prev","newSet","delete","add","VisibilityOffIcon","VisibilityIcon","creationTimestamp","date","stopPropagation","rect","currentTarget","getBoundingClientRect","top","bottom","left","aria-controls","aria-haspopup","MoreVertIcon","tabData","getTabData","tabColumns","col","getTabColumns","subheader","length","Tabs","_","newValue","indicatorColor","textColor","Tab","textAlign","Table","options","paging","pageSize","search","debounceInterval","toolbar","emptyRowsWhenPaging","data","item","id","Menu","Boolean","anchorReference","anchorPosition","push","requestName","post","next","PermissionGate","children","permission","fallback","errorMessage","allowed","useKuadrantPermission","PlanPolicyDetails","selectedPolicy","alertSeverity","alertMessage","includeTopMargin","border","fontWeight","marginTop","flexWrap","idx","limitText","daily","monthly","yearly","some","Alert","useStyles","makeStyles","asterisk","CreateAPIProductDialog","classes","setName","displayName","setDisplayName","setDescription","version","setVersion","approvalMode","setApprovalMode","publishStatus","setPublishStatus","tags","setTags","tagInput","setTagInput","selectedHTTPRoute","setSelectedHTTPRoute","contactEmail","setContactEmail","contactTeam","setContactTeam","docsURL","setDocsURL","openAPISpec","setOpenAPISpec","creating","setCreating","httpRoutes","httpRoutesLoading","route","annotations","planPolicies","selectedRouteInfo","routeNamespace","routeName","find","pp","ref","targetRef","kind","handleAddTag","includes","marginBottom","Grid","container","spacing","xs","required","InputLabelProps","select","tag","onDelete","handleDeleteTag","tagToDelete","onKeyPress","selectedRouteNamespace","selectedRouteName","apiVersion","group","contact","email","team","documentation","String","EditAPIProductDialog","setLoading","setTargetRef","then","res","React","ResourceList","setUserEntityRef","createDialogOpen","setCreateDialogOpen","editDialogOpen","setEditDialogOpen","refreshTrigger","setRefreshTrigger","deleteDialogOpen","setDeleteDialogOpen","apiProductToDelete","setApiProductToDelete","apiProductToEdit","setApiProductToEdit","canCreateApiProduct","createPermissionLoading","createPermissionError","kuadrantApiProductCreatePermission","canViewAllRequests","approvalQueueAllPermissionLoading","kuadrantApiKeyRequestReadAllPermission","canViewOwnRequests","approvalQueueOwnPermissionLoading","approvalQueuePermissionError","kuadrantApiKeyRequestReadOwnPermission","canViewApprovalQueue","approvalQueuePermissionLoading","canDeleteOwnApiProduct","deleteOwnPermissionLoading","kuadrantApiProductDeleteOwnPermission","canDeleteAllApiProducts","deleteAllPermissionLoading","deletePermissionError","kuadrantApiProductDeleteAllPermission","canUpdateOwnApiProduct","kuadrantApiProductUpdateOwnPermission","canUpdateAllApiProducts","kuadrantApiProductUpdateAllPermission","deletePermissionLoading","canListPlanPolicies","planPolicyPermissionLoading","planPolicyPermissionError","kuadrantPlanPolicyListPermission","apiProducts","apiProductsLoading","apiProductsError","planPoliciesLoading","planPoliciesError","permissionError","isPublished","className","customFilterAndSearch","term","toLowerCase","mode","formatDate","dateString","year","month","day","isOwner","canEdit","canDelete","handleEditClick","EditIcon","DeleteIcon","planPolicyColumns","Page","themeId","Header","subtitle","SupportButton","Content","ResponseErrorPanel","direction","action","height","AddIcon","resources","renderPlanPolicies","ApprovalQueueCard","KuadrantPage","kuadrantApiProductListPermission"],"sourceRoot":""}
|