@kuadrant/kuadrant-backstage-plugin-frontend 0.0.2-dev-3d7caf4 → 0.0.2-dev-24d0757
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/assets/empty-state-illustration.png +0 -0
- package/dist/components/ApiKeyManagementTab/ApiKeyManagementTab.esm.js +6 -1
- package/dist/components/ApiKeyManagementTab/ApiKeyManagementTab.esm.js.map +1 -1
- package/dist/components/ApiProductDetailPage/ApiProductDetailPage.esm.js +248 -0
- package/dist/components/ApiProductDetailPage/ApiProductDetailPage.esm.js.map +1 -0
- package/dist/components/ApiProductDetailPage/index.esm.js +2 -0
- package/dist/components/ApiProductDetailPage/index.esm.js.map +1 -0
- package/dist/components/ApiProductDetails/ApiProductDetails.esm.js +86 -0
- package/dist/components/ApiProductDetails/ApiProductDetails.esm.js.map +1 -0
- package/dist/components/ApiProductInfoCard/ApiProductInfoCard.esm.js +24 -43
- package/dist/components/ApiProductInfoCard/ApiProductInfoCard.esm.js.map +1 -1
- package/dist/components/ApprovalQueueTable/ApprovalQueueTable.esm.js +2 -1
- package/dist/components/ApprovalQueueTable/ApprovalQueueTable.esm.js.map +1 -1
- package/dist/components/CreateAPIProductDialog/CreateAPIProductDialog.esm.js +148 -134
- package/dist/components/CreateAPIProductDialog/CreateAPIProductDialog.esm.js.map +1 -1
- package/dist/components/EditAPIProductDialog/EditAPIProductDialog.esm.js +104 -128
- package/dist/components/EditAPIProductDialog/EditAPIProductDialog.esm.js.map +1 -1
- package/dist/components/KuadrantPage/KuadrantPage.esm.js +328 -125
- package/dist/components/KuadrantPage/KuadrantPage.esm.js.map +1 -1
- package/dist/components/MyApiKeysTable/MyApiKeysTable.esm.js +8 -2
- package/dist/components/MyApiKeysTable/MyApiKeysTable.esm.js.map +1 -1
- package/dist/index.d.ts +2 -1
- package/dist/index.esm.js +1 -1
- package/dist/plugin.esm.js +8 -1
- package/dist/plugin.esm.js.map +1 -1
- package/dist/utils/validation.esm.js +1 -11
- package/dist/utils/validation.esm.js.map +1 -1
- package/dist-scalprum/{internal.plugin-kuadrant.793ab10dddb55e70abe2.js → internal.plugin-kuadrant.95817f34e88db81b5e8f.js} +2 -2
- package/dist-scalprum/internal.plugin-kuadrant.95817f34e88db81b5e8f.js.map +1 -0
- package/dist-scalprum/plugin-manifest.json +2 -2
- package/dist-scalprum/static/2759.fceb317f.chunk.js +2 -0
- package/dist-scalprum/static/2759.fceb317f.chunk.js.map +1 -0
- package/dist-scalprum/static/2928.4303c12e.chunk.js +3 -0
- package/dist-scalprum/static/2928.4303c12e.chunk.js.map +1 -0
- package/dist-scalprum/static/2967.ac3a4bee.chunk.js +2 -0
- package/dist-scalprum/static/2967.ac3a4bee.chunk.js.map +1 -0
- package/dist-scalprum/static/{6979.9699b350.chunk.js → 2987.1da15560.chunk.js} +2 -2
- package/dist-scalprum/static/2987.1da15560.chunk.js.map +1 -0
- package/dist-scalprum/static/3459.5c90b5a3.chunk.js +2 -0
- package/dist-scalprum/static/3459.5c90b5a3.chunk.js.map +1 -0
- package/dist-scalprum/static/3503.66b6e510.chunk.js +2 -0
- package/dist-scalprum/static/3503.66b6e510.chunk.js.map +1 -0
- package/dist-scalprum/static/{3650.aa8552f3.chunk.js → 3650.515c743a.chunk.js} +2 -2
- package/dist-scalprum/static/3650.515c743a.chunk.js.map +1 -0
- package/dist-scalprum/static/4682.6959fcd1.chunk.js +2 -0
- package/dist-scalprum/static/4682.6959fcd1.chunk.js.map +1 -0
- package/dist-scalprum/static/5010.a4aa0f8e.chunk.js +3 -0
- package/dist-scalprum/static/5010.a4aa0f8e.chunk.js.map +1 -0
- package/dist-scalprum/static/5453.280127dd.chunk.js +2 -0
- package/dist-scalprum/static/5453.280127dd.chunk.js.map +1 -0
- package/dist-scalprum/static/6422.97baf774.chunk.js +2 -0
- package/dist-scalprum/static/6422.97baf774.chunk.js.map +1 -0
- package/dist-scalprum/static/7791.12162a71.chunk.js +2 -0
- package/dist-scalprum/static/7791.12162a71.chunk.js.map +1 -0
- package/dist-scalprum/static/8799.7c749838.chunk.js +2 -0
- package/dist-scalprum/static/8799.7c749838.chunk.js.map +1 -0
- package/dist-scalprum/static/empty-state-illustration.7e3ad5a9..png +0 -0
- package/dist-scalprum/static/exposed-PluginRoot.a5792fb2.chunk.js +2 -0
- package/dist-scalprum/static/exposed-PluginRoot.a5792fb2.chunk.js.map +1 -0
- package/package.json +1 -1
- package/dist-scalprum/internal.plugin-kuadrant.793ab10dddb55e70abe2.js.map +0 -1
- package/dist-scalprum/static/2120.44884438.chunk.js +0 -3
- package/dist-scalprum/static/2120.44884438.chunk.js.map +0 -1
- package/dist-scalprum/static/2967.c684efaf.chunk.js +0 -2
- package/dist-scalprum/static/2967.c684efaf.chunk.js.map +0 -1
- package/dist-scalprum/static/3650.aa8552f3.chunk.js.map +0 -1
- package/dist-scalprum/static/5010.acf9a415.chunk.js +0 -3
- package/dist-scalprum/static/5010.acf9a415.chunk.js.map +0 -1
- package/dist-scalprum/static/5453.c1f90bdf.chunk.js +0 -2
- package/dist-scalprum/static/5453.c1f90bdf.chunk.js.map +0 -1
- package/dist-scalprum/static/6813.036a322f.chunk.js +0 -2
- package/dist-scalprum/static/6813.036a322f.chunk.js.map +0 -1
- package/dist-scalprum/static/6979.9699b350.chunk.js.map +0 -1
- package/dist-scalprum/static/7684.3d1fc1a1.chunk.js +0 -2
- package/dist-scalprum/static/7684.3d1fc1a1.chunk.js.map +0 -1
- package/dist-scalprum/static/8416.3604a311.chunk.js +0 -2
- package/dist-scalprum/static/8416.3604a311.chunk.js.map +0 -1
- package/dist-scalprum/static/exposed-PluginRoot.16bf7897.chunk.js +0 -2
- package/dist-scalprum/static/exposed-PluginRoot.16bf7897.chunk.js.map +0 -1
- /package/dist-scalprum/static/{2120.44884438.chunk.js.LICENSE.txt → 2928.4303c12e.chunk.js.LICENSE.txt} +0 -0
- /package/dist-scalprum/static/{5010.acf9a415.chunk.js.LICENSE.txt → 5010.a4aa0f8e.chunk.js.LICENSE.txt} +0 -0
|
@@ -1,2 +0,0 @@
|
|
|
1
|
-
"use strict";(self.webpackChunkinternal_plugin_kuadrant=self.webpackChunkinternal_plugin_kuadrant||[]).push([[7684],{27684:(e,a,t)=>{t.r(a),t.d(a,{ApiProductsPage:()=>X,KuadrantPage:()=>X});var l=t(31085),i=t(95478),n=t.n(i),r=t(67720),s=t(72501),o=t(10394),d=t(29365),c=t(42899),u=t(64947),p=t(18466),m=t(39590),h=t(75625),x=t(21702),v=t(85142),A=t(37725),g=t(25010),j=t(289),y=t(15831),b=t(45210),f=t(46681),P=t(86687),k=t(42367),S=t(96040),w=t(91638),T=t(22097),C=t(38599),I=t(58837),R=t(76891),$=t(61477),W=t(46805),M=t(16249),N=t(26343),D=t(93453),L=t(78467),z=t(84441),E=t(54917);const H=({selectedPolicy:e,alertSeverity:a="warning",alertMessage:t="No PlanPolicy found for this HTTPRoute. API keys and rate limiting may not be available.",includeTopMargin:i=!0})=>{const n=(0,E.A)();return(0,l.jsx)(o.A,{mt:i?1:0,p:2,bgcolor:n.palette.background.default,borderRadius:1,border:`1px solid ${n.palette.divider}`,children:e?(0,l.jsxs)(l.Fragment,{children:[(0,l.jsxs)(s.A,{variant:"subtitle2",gutterBottom:!0,style:{fontWeight:600},children:["Associated PlanPolicy: ",(0,l.jsx)("strong",{children:e.metadata.name})]}),e.plans&&e.plans.length>0?(0,l.jsxs)(l.Fragment,{children:[(0,l.jsx)(s.A,{variant:"caption",display:"block",gutterBottom:!0,color:"textSecondary",style:{marginTop:8},children:"Available Tiers:"}),(0,l.jsx)(o.A,{display:"flex",flexWrap:"wrap",mt:1,style:{gap:8},children:e.plans.map((e,a)=>{var t,i,n;const s=(null===(t=e.limits)||void 0===t?void 0:t.daily)?`${e.limits.daily}/day`:(null===(i=e.limits)||void 0===i?void 0:i.monthly)?`${e.limits.monthly}/month`:(null===(n=e.limits)||void 0===n?void 0:n.yearly)?`${e.limits.yearly}/year`:"No limit";return(0,l.jsx)(r.A,{label:`${e.tier}: ${s}`,size:"small",variant:"outlined",color:"primary"},a)})})]}):(0,l.jsx)(s.A,{variant:"caption",color:"textSecondary",children:"No plans defined in this PlanPolicy"})]}):(0,l.jsx)(z.A,{severity:a,children:t})})},U=e=>e?/^[^\s@]+@[^\s@]+\.[^\s@]+$/.test(e)?null:"Must be a valid email address":null,B=e=>{if(!e)return null;try{const a=new URL(e);return["http:","https:"].includes(a.protocol)?null:"Must be a valid HTTP or HTTPS URL"}catch{return"Must be a valid URL"}},F=(0,I.A)({asterisk:{color:"#f44336"}}),K=({open:e,onClose:a,onSuccess:t})=>{const n=F(),d=(0,T.useApi)(T.configApiRef),p=(0,T.useApi)(T.fetchApiRef),m=d.getString("backend.baseUrl"),[h,x]=(0,i.useState)(""),[v,A]=(0,i.useState)(""),[g,j]=(0,i.useState)(""),[y,b]=(0,i.useState)("v1"),[f,P]=(0,i.useState)("manual"),[k,S]=(0,i.useState)("Published"),[C,I]=(0,i.useState)([]),[E,K]=(0,i.useState)(""),[q,O]=(0,i.useState)(""),[_,V]=(0,i.useState)(""),[J,Y]=(0,i.useState)(""),[X,G]=(0,i.useState)(""),[Q,Z]=(0,i.useState)(""),[ee,ae]=(0,i.useState)(""),[te,le]=(0,i.useState)(!1),[ie,ne]=(0,i.useState)(0),[re,se]=(0,i.useState)(null),[oe,de]=(0,i.useState)(null),[ce,ue]=(0,i.useState)(null),[pe,me]=(0,i.useState)(null),{value:he,loading:xe,error:ve}=(0,w.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,ie]),{value:Ae,error:ge}=(0,w.A)(async()=>{const e=await p.fetch(`${m}/api/kuadrant/planpolicies`);return await e.json()},[m,p,e]),je=q?q.split("/"):null,ye=je?(be=je[0],fe=je[1],(null==Ae?void 0:Ae.items)?Ae.items.find(e=>{const a=e.targetRef;return"HTTPRoute"===(null==a?void 0:a.kind)&&(null==a?void 0:a.name)===fe&&(!(null==a?void 0:a.namespace)||(null==a?void 0:a.namespace)===be)}):null):null;var be,fe;(0,i.useEffect)(()=>{e&&(se(null),de(null),ue(null),me(null))},[e]);const Pe=()=>{E.trim()&&!C.includes(E.trim())&&(I([...C,E.trim()]),K(""))},ke=()=>{x(""),A(""),j(""),b("v1"),P("manual"),S("Published"),I([]),K(""),O(""),V(""),Y(""),G(""),Z(""),ae(""),se(null),de(null),ue(null),me(null),a()},Se=!!(re||oe||ce||pe);return(0,l.jsxs)(R.A,{open:e,onClose:ke,maxWidth:"md",fullWidth:!0,children:[(0,l.jsx)($.A,{children:"Create API Product"}),(0,l.jsxs)(W.A,{children:[ee&&(0,l.jsx)(z.A,{severity:"error",style:{marginBottom:16},children:ee}),ve&&(0,l.jsxs)(z.A,{severity:"error",style:{marginBottom:16},children:[(0,l.jsx)("strong",{children:"Failed to load HTTPRoutes:"})," ",ve.message,(0,l.jsx)(o.A,{mt:1,children:(0,l.jsx)(u.A,{size:"small",variant:"outlined",onClick:()=>ne(e=>e+1),children:"Retry"})})]}),ge&&(0,l.jsxs)(z.A,{severity:"warning",style:{marginBottom:16},children:[(0,l.jsx)("strong",{children:"Failed to load PlanPolicies:"})," ",ge.message,(0,l.jsx)(s.A,{variant:"body2",style:{marginTop:8},children:"You can still create the API Product, but plan information may be incomplete."})]}),(0,l.jsxs)(c.A,{container:!0,spacing:2,children:[(0,l.jsx)(c.A,{item:!0,xs:6,children:(0,l.jsx)(M.A,{fullWidth:!0,label:"Name",value:h,onChange:e=>{return a=e.target.value,x(a),void se((e=>e&&e.trim()?e.length>253?"Must be 253 characters or less":/^[a-z0-9]([-a-z0-9]*[a-z0-9])?$/.test(e)?null:"Must be lowercase alphanumeric with hyphens, start and end with alphanumeric":"Name is required")(a));var a},placeholder:"my-api",helperText:re||"Kubernetes resource name (lowercase, hyphens)",error:!!re,margin:"normal",required:!0,disabled:te,InputLabelProps:{classes:{asterisk:n.asterisk}}})}),(0,l.jsx)(c.A,{item:!0,xs:6,children:(0,l.jsx)(M.A,{fullWidth:!0,label:"Display Name",value:v,onChange:e=>A(e.target.value),placeholder:"My API",margin:"normal",required:!0,disabled:te,InputLabelProps:{classes:{asterisk:n.asterisk}}})}),(0,l.jsx)(c.A,{item:!0,xs:6,children:(0,l.jsx)(M.A,{fullWidth:!0,label:"Version",value:y,onChange:e=>b(e.target.value),placeholder:"v1",margin:"normal",disabled:te})}),(0,l.jsx)(c.A,{item:!0,xs:12,children:(0,l.jsxs)(M.A,{fullWidth:!0,select:!0,label:"Approval Mode",value:f,onChange:e=>P(e.target.value),margin:"normal",helperText:"Automatic: keys are created immediately. Manual: requires approval.",disabled:te,children:[(0,l.jsx)(N.A,{value:"manual",children:"Manual"}),(0,l.jsx)(N.A,{value:"automatic",children:"Automatic"})]})}),(0,l.jsx)(c.A,{item:!0,xs:12,children:(0,l.jsxs)(M.A,{fullWidth:!0,select:!0,label:"Publish Status",value:k,onChange:e=>S(e.target.value),margin:"normal",helperText:"Draft: hidden from catalog. Published: visible to consumers.",disabled:te,children:[(0,l.jsx)(N.A,{value:"Draft",children:"Draft"}),(0,l.jsx)(N.A,{value:"Published",children:"Published"})]})}),(0,l.jsx)(c.A,{item:!0,xs:12,children:(0,l.jsx)(M.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:te,InputLabelProps:{classes:{asterisk:n.asterisk}}})}),(0,l.jsxs)(c.A,{item:!0,xs:12,children:[(0,l.jsx)(s.A,{variant:"subtitle2",gutterBottom:!0,style:{marginTop:16},children:"Tags"}),(0,l.jsx)(o.A,{display:"flex",flexWrap:"wrap",marginBottom:1,style:{gap:8},children:C.map(e=>(0,l.jsx)(r.A,{label:e,onDelete:te?void 0:()=>{return a=e,void I(C.filter(e=>e!==a));var a},size:"small",disabled:te},e))}),(0,l.jsxs)(o.A,{display:"flex",style:{gap:8},children:[(0,l.jsx)(M.A,{fullWidth:!0,size:"small",value:E,onChange:e=>K(e.target.value),onKeyPress:e=>"Enter"===e.key&&Pe(),placeholder:"Add tag",disabled:te}),(0,l.jsx)(u.A,{onClick:Pe,variant:"outlined",size:"small",disabled:te,children:"Add"})]})]}),(0,l.jsx)(c.A,{item:!0,xs:12,children:(0,l.jsxs)(M.A,{fullWidth:!0,select:!0,label:"HTTPRoute",value:q,onChange:e=>O(e.target.value),margin:"normal",required:!0,helperText:ve?"Unable to load HTTPRoutes. Please retry.":"Select an HTTPRoute (backstage.io/expose: true). APIProduct will be created in the same namespace.",error:!!ve,disabled:xe||te||!!ve,InputLabelProps:{classes:{asterisk:n.asterisk}},SelectProps:{"data-testid":"httproute-select"},children:[xe&&(0,l.jsx)(N.A,{value:"",children:"Loading..."}),ve&&(0,l.jsx)(N.A,{value:"",children:"Error loading routes"}),!xe&&!ve&&he&&0===he.length&&(0,l.jsx)(N.A,{value:"",children:"No HTTPRoutes available"}),!xe&&!ve&&he&&he.map(e=>(0,l.jsxs)(N.A,{value:`${e.metadata.namespace}/${e.metadata.name}`,children:[e.metadata.name," (",e.metadata.namespace,")"]},`${e.metadata.namespace}/${e.metadata.name}`))]})}),q&&(0,l.jsx)(c.A,{item:!0,xs:12,children:(0,l.jsx)(H,{selectedPolicy:ye,alertSeverity:"warning",alertMessage:"No PlanPolicy found for this HTTPRoute. API keys and rate limiting may not be available.",includeTopMargin:!0})}),(0,l.jsx)(c.A,{item:!0,xs:6,children:(0,l.jsx)(M.A,{fullWidth:!0,label:"Contact Email",value:_,onChange:e=>{return a=e.target.value,V(a),void de(U(a));var a},placeholder:"api-team@example.com",helperText:oe||"Contact email for API support",error:!!oe,margin:"normal",disabled:te})}),(0,l.jsx)(c.A,{item:!0,xs:6,children:(0,l.jsx)(M.A,{fullWidth:!0,label:"Contact Team",value:J,onChange:e=>Y(e.target.value),placeholder:"platform-team",margin:"normal",disabled:te})}),(0,l.jsx)(c.A,{item:!0,xs:6,children:(0,l.jsx)(M.A,{fullWidth:!0,label:"Docs URL",value:X,onChange:e=>{return a=e.target.value,G(a),void ue(B(a));var a},placeholder:"https://api.example.com/docs",helperText:ce||"Link to API documentation",error:!!ce,margin:"normal",disabled:te})}),(0,l.jsx)(c.A,{item:!0,xs:6,children:(0,l.jsx)(M.A,{fullWidth:!0,label:"OpenAPI Spec URL",value:Q,onChange:e=>{return a=e.target.value,Z(a),void me(B(a));var a},placeholder:"https://api.example.com/openapi.json",helperText:pe||"Link to OpenAPI specification",error:!!pe,margin:"normal",disabled:te})})]})]}),(0,l.jsxs)(D.A,{children:[(0,l.jsx)(u.A,{onClick:ke,disabled:te,children:"Cancel"}),(0,l.jsx)(u.A,{onClick:async()=>{ae(""),le(!0);try{if(!q)throw new Error("Please select an HTTPRoute");const[e,a]=q.split("/"),l={apiVersion:"devportal.kuadrant.io/v1alpha1",kind:"APIProduct",metadata:{name:h,namespace:e},spec:{displayName:v,description:g,version:y,approvalMode:f,publishStatus:k,tags:C,targetRef:{group:"gateway.networking.k8s.io",kind:"HTTPRoute",name:a,namespace:e},..._||J?{contact:{..._&&{email:_},...J&&{team:J}}}:{},...X||Q?{documentation:{...X&&{docsURL:X},...Q&&{openAPISpec:Q}}}:{}}},i=await p.fetch(`${m}/api/kuadrant/apiproducts`,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify(l)});if(!i.ok){const e=await i.json();throw new Error(e.error||"failed to create apiproduct")}t(),ke()}catch(e){ae(e instanceof Error?e.message:String(e))}finally{le(!1)}},color:"primary",variant:"contained",disabled:te||!h||!v||!g||!q||Se,startIcon:te?(0,l.jsx)(L.A,{size:16,color:"inherit"}):void 0,children:te?"Creating...":"Create"})]})]})};var q=t(34955),O=t(46205);const _=(0,I.A)({asterisk:{color:"#f44336"}}),V=({open:e,onClose:a,onSuccess:t,namespace:d,name:p})=>{const m=_(),h=(0,T.useApi)(T.configApiRef),x=(0,T.useApi)(T.fetchApiRef),v=h.getString("backend.baseUrl"),[A,g]=(0,i.useState)(!1),[j,y]=(0,i.useState)(""),[b,f]=(0,i.useState)(""),[k,S]=(0,i.useState)("v1"),[C,I]=(0,i.useState)("Draft"),[E,F]=(0,i.useState)("manual"),[K,q]=(0,i.useState)([]),[O,V]=(0,i.useState)(null),[J,Y]=(0,i.useState)(""),[X,G]=(0,i.useState)(""),[Q,Z]=(0,i.useState)(""),[ee,ae]=(0,i.useState)(""),[te,le]=(0,i.useState)(""),[ie,ne]=(0,i.useState)(""),[re,se]=(0,i.useState)(!1),[oe,de]=(0,i.useState)(null),[ce,ue]=(0,i.useState)(null),[pe,me]=(0,i.useState)(null);(0,i.useEffect)(()=>{e&&d&&p&&(g(!0),ne(""),x.fetch(`${v}/api/kuadrant/apiproducts/${d}/${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,l,i;y(e.spec.displayName||""),f(e.spec.description||""),S(e.spec.version||"v1"),I(e.spec.publishStatus||"Draft"),F(e.spec.approvalMode||"manual"),q(e.spec.tags||[]),V(e.spec.targetRef||null),G((null===(a=e.spec.contact)||void 0===a?void 0:a.email)||""),Z((null===(t=e.spec.contact)||void 0===t?void 0:t.team)||""),ae((null===(l=e.spec.documentation)||void 0===l?void 0:l.docsURL)||""),le((null===(i=e.spec.documentation)||void 0===i?void 0:i.openAPISpec)||""),de(null),ue(null),me(null),g(!1)}).catch(e=>{ne(e.message||"Failed to load API product"),g(!1)}))},[e,d,p,v,x]);const{value:he,error:xe}=(0,w.A)(async()=>{if(!e)return null;const a=await x.fetch(`${v}/api/kuadrant/planpolicies`);return await a.json()},[v,x,e]),ve=n().useMemo(()=>(null==he?void 0:he.items)&&O?he.items.find(e=>{const a=e.targetRef;return"HTTPRoute"===(null==a?void 0:a.kind)&&(null==a?void 0:a.name)===O.name&&(!(null==a?void 0:a.namespace)||(null==a?void 0:a.namespace)===(O.namespace||d))}):null,[he,O,d]);(0,i.useEffect)(()=>{e&&(de(null),ue(null),me(null))},[e]);const Ae=()=>{J.trim()&&!K.includes(J.trim())&&(q([...K,J.trim()]),Y(""))};return(0,l.jsxs)(R.A,{open:e,onClose:a,maxWidth:"md",fullWidth:!0,children:[(0,l.jsx)($.A,{children:"Edit API Product"}),(0,l.jsxs)(W.A,{children:[ie&&(0,l.jsx)(z.A,{severity:"error",style:{marginBottom:16},children:ie}),xe&&(0,l.jsxs)(z.A,{severity:"warning",style:{marginBottom:16},children:[(0,l.jsx)("strong",{children:"Failed to load PlanPolicies:"})," ",xe.message,(0,l.jsx)(s.A,{variant:"body2",style:{marginTop:8},children:"Plan information may be incomplete."})]}),A?(0,l.jsx)(P.k,{}):(0,l.jsxs)(c.A,{container:!0,spacing:2,children:[(0,l.jsx)(c.A,{item:!0,xs:6,children:(0,l.jsx)(M.A,{fullWidth:!0,label:"Name",value:p,disabled:!0,helperText:"Kubernetes resource name (immutable)",margin:"normal"})}),(0,l.jsx)(c.A,{item:!0,xs:6,children:(0,l.jsx)(M.A,{fullWidth:!0,label:"Namespace",value:d,disabled:!0,helperText:"Derived from HTTPRoute (immutable)",margin:"normal"})}),(0,l.jsx)(c.A,{item:!0,xs:6,children:(0,l.jsx)(M.A,{fullWidth:!0,label:"Display Name",value:j,onChange:e=>y(e.target.value),placeholder:"My API",margin:"normal",required:!0,disabled:re,InputLabelProps:{classes:{asterisk:m.asterisk}}})}),(0,l.jsx)(c.A,{item:!0,xs:6,children:(0,l.jsx)(M.A,{fullWidth:!0,label:"Version",value:k,onChange:e=>S(e.target.value),placeholder:"v1",margin:"normal",disabled:re})}),(0,l.jsx)(c.A,{item:!0,xs:6,children:(0,l.jsxs)(M.A,{fullWidth:!0,select:!0,label:"Publish Status",value:C,onChange:e=>I(e.target.value),margin:"normal",helperText:"Draft: Hidden from catalog. Published: Visible in catalog.",disabled:re,children:[(0,l.jsx)(N.A,{value:"Draft",children:"Draft (Hidden)"}),(0,l.jsx)(N.A,{value:"Published",children:"Published (Visible)"})]})}),(0,l.jsx)(c.A,{item:!0,xs:6,children:(0,l.jsxs)(M.A,{fullWidth:!0,select:!0,label:"Approval Mode",value:E,onChange:e=>F(e.target.value),margin:"normal",helperText:"Automatic: keys created immediately. Manual: requires approval.",disabled:re,children:[(0,l.jsx)(N.A,{value:"manual",children:"Manual"}),(0,l.jsx)(N.A,{value:"automatic",children:"Automatic"})]})}),(0,l.jsx)(c.A,{item:!0,xs:12,children:(0,l.jsx)(M.A,{fullWidth:!0,label:"Description",value:b,onChange:e=>f(e.target.value),placeholder:"API description",margin:"normal",multiline:!0,rows:2,required:!0,disabled:re,InputLabelProps:{classes:{asterisk:m.asterisk}}})}),(0,l.jsxs)(c.A,{item:!0,xs:12,children:[(0,l.jsx)(s.A,{variant:"subtitle2",gutterBottom:!0,style:{marginTop:16},children:"Tags"}),(0,l.jsx)(o.A,{display:"flex",flexWrap:"wrap",marginBottom:1,style:{gap:8},children:K.map(e=>(0,l.jsx)(r.A,{label:e,onDelete:re?void 0:()=>{return a=e,void q(K.filter(e=>e!==a));var a},size:"small",disabled:re},e))}),(0,l.jsxs)(o.A,{display:"flex",style:{gap:8},children:[(0,l.jsx)(M.A,{fullWidth:!0,size:"small",value:J,onChange:e=>Y(e.target.value),onKeyPress:e=>"Enter"===e.key&&Ae(),placeholder:"Add tag",disabled:re}),(0,l.jsx)(u.A,{onClick:Ae,variant:"outlined",size:"small",disabled:re,children:"Add"})]})]}),O&&(0,l.jsxs)(l.Fragment,{children:[(0,l.jsx)(c.A,{item:!0,xs:12,children:(0,l.jsx)(M.A,{fullWidth:!0,label:"HTTPRoute",value:`${O.namespace||d}/${O.name}`,disabled:!0,helperText:"Target HTTPRoute (immutable)",margin:"normal"})}),(0,l.jsx)(c.A,{item:!0,xs:12,children:(0,l.jsx)(H,{selectedPolicy:ve,alertSeverity:"info",alertMessage:"No PlanPolicy found for this HTTPRoute.",includeTopMargin:!1})})]}),(0,l.jsx)(c.A,{item:!0,xs:6,children:(0,l.jsx)(M.A,{fullWidth:!0,label:"Contact Email",value:X,onChange:e=>{return a=e.target.value,G(a),void de(U(a));var a},placeholder:"api-team@example.com",helperText:oe||"Contact email for API support",error:!!oe,margin:"normal",disabled:re})}),(0,l.jsx)(c.A,{item:!0,xs:6,children:(0,l.jsx)(M.A,{fullWidth:!0,label:"Contact Team",value:Q,onChange:e=>Z(e.target.value),placeholder:"platform-team",margin:"normal",disabled:re})}),(0,l.jsx)(c.A,{item:!0,xs:6,children:(0,l.jsx)(M.A,{fullWidth:!0,label:"Docs URL",value:ee,onChange:e=>{return a=e.target.value,ae(a),void ue(B(a));var a},placeholder:"https://api.example.com/docs",helperText:ce||"Link to API documentation",error:!!ce,margin:"normal",disabled:re})}),(0,l.jsx)(c.A,{item:!0,xs:6,children:(0,l.jsx)(M.A,{fullWidth:!0,label:"OpenAPI Spec URL",value:te,onChange:e=>{return a=e.target.value,le(a),void me(B(a));var a},placeholder:"https://api.example.com/openapi.json",helperText:pe||"Link to OpenAPI specification",error:!!pe,margin:"normal",disabled:re})})]})]}),(0,l.jsxs)(D.A,{children:[(0,l.jsx)(u.A,{onClick:a,disabled:re,children:"Cancel"}),(0,l.jsx)(u.A,{onClick:async()=>{ne(""),se(!0);try{const e={spec:{displayName:j,description:b,version:k,publishStatus:C,approvalMode:E,tags:K,targetRef:O,...X||Q?{contact:{...X&&{email:X},...Q&&{team:Q}}}:{},...ee||te?{documentation:{...ee&&{docsURL:ee},...te&&{openAPISpec:te}}}:{}}},l=await x.fetch(`${v}/api/kuadrant/apiproducts/${d}/${p}`,{method:"PATCH",headers:{"Content-Type":"application/json"},body:JSON.stringify(e)});if(!l.ok){const e=await l.json();throw new Error(e.error||"failed to update apiproduct")}t(),a()}catch(e){ne(e instanceof Error?e.message:String(e))}finally{se(!1)}},color:"primary",variant:"contained",disabled:re||A||!j||!b||!!oe||!!ce||!!pe,startIcon:re?(0,l.jsx)(L.A,{size:16,color:"inherit"}):void 0,children:re?"Saving...":"Save"})]})]})};var J=t(63221);const Y=()=>{const e=(0,T.useApi)(T.configApiRef),a=(0,T.useApi)(T.fetchApiRef),t=(0,T.useApi)(T.alertApiRef),n=(0,T.useApi)(T.identityApiRef),C=e.getString("backend.baseUrl"),[I,R]=(0,i.useState)(""),[$,W]=(0,i.useState)(!1),[M,N]=(0,i.useState)(!1),[D,L]=(0,i.useState)(0),[z,E]=(0,i.useState)(!1),[H,U]=(0,i.useState)(null),[B,F]=(0,i.useState)(null),[_,Y]=(0,i.useState)(!1),[X,G]=(0,i.useState)(null),{allowed:Q,loading:Z,error:ee}=(0,O.l)(q.FL),{allowed:ae,loading:te}=(0,O.l)(q.EM),{allowed:le,loading:ie,error:ne}=(0,O.l)(q.R_),{allowed:re}=(0,O.l)(q.U3),{allowed:se}=(0,O.l)(q.v_),oe=te||ie,{allowed:de,loading:ce,error:ue}=(0,O.l)(q.J);(0,w.A)(async()=>{const e=await n.getBackstageIdentity();R(e.userEntityRef)},[n]);const{value:pe,loading:me,error:he}=(0,w.A)(async()=>{const e=await a.fetch(`${C}/api/kuadrant/apiproducts`);return await e.json()},[C,a,D]),{value:xe,loading:ve,error:Ae}=(0,w.A)(async()=>{const e=await a.fetch(`${C}/api/kuadrant/planpolicies`);return await e.json()},[C,a,D]),ge=me||ve||Z||oe||ce,je=he||Ae,ye=ee||ne||ue,be=[{title:"Name",field:"spec.displayName",render:e=>{var a,t;const i="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 i?(0,l.jsx)(A.N_,{to:`/catalog/default/api/${e.metadata.name}/api-product`,children:(0,l.jsx)("strong",{children:r})}):(0,l.jsx)("span",{className:"text-muted",children:(0,l.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,l.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,l.jsx)(r.A,{label:t,size:"small",color:"automatic"===t?"secondary":"default"})}},{title:"Authentication",field:"status.discoveredAuthScheme",render:e=>{var a,t;const i=(null===(t=e.status)||void 0===t||null===(a=t.discoveredAuthScheme)||void 0===a?void 0:a.authentication)||{},n=Object.values(i),d=n.some(e=>e.hasOwnProperty("apiKey")),c=n.some(e=>e.hasOwnProperty("jwt"));return d||c?(0,l.jsxs)(o.A,{display:"flex",style:{gap:4},children:[d&&(0,l.jsx)(r.A,{icon:(0,l.jsx)(x.A,{}),label:"API Key",size:"small",color:"primary"}),c&&(0,l.jsx)(r.A,{icon:(0,l.jsx)(v.A,{}),label:"OIDC",size:"small",color:"secondary"})]}):(0,l.jsx)(s.A,{variant:"body2",style:{fontStyle:"italic"},children:"unknown"})}},{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 t,i;const n=(null===(i=e.metadata)||void 0===i||null===(t=i.annotations)||void 0===t?void 0:t["backstage.io/owner"])===I,r=se||re&&n,s=le||ae&&n;return r||s?(0,l.jsxs)(o.A,{display:"flex",style:{gap:4},children:[r&&(0,l.jsx)(d.A,{size:"small",onClick:()=>{return a=e.metadata.namespace,t=e.metadata.name,F({namespace:a,name:t}),void N(!0);var a,t},title:"Edit API Product",children:(0,l.jsx)(h.A,{fontSize:"small"})}),s&&(0,l.jsx)(d.A,{size:"small",onClick:()=>(async(e,t)=>{U({namespace:e,name:t}),G(null);try{const l=await a.fetch(`${C}/api/kuadrant/requests?namespace=${e}`);if(l.ok){const a=((await l.json()).items||[]).filter(a=>a.spec.apiName===t&&a.spec.apiNamespace===e),i=a.filter(e=>{var a;return"Approved"===(null===(a=e.status)||void 0===a?void 0:a.phase)}).length;G({requests:a.length,secrets:i})}}catch(e){console.warn("Failed to fetch delete stats:",e)}E(!0)})(e.metadata.namespace,e.metadata.name),title:"Delete API Product",children:(0,l.jsx)(m.A,{fontSize:"small"})})]}):null}}],fe=[{title:"Name",field:"metadata.name",render:e=>(0,l.jsx)(A.N_,{to:`/kuadrant/planpolicy/${e.metadata.namespace}/${e.metadata.name}`,children:(0,l.jsx)("strong",{children:e.metadata.name})})},{title:"Namespace",field:"metadata.namespace"}];return(0,l.jsxs)(j.Y,{themeId:"tool",children:[(0,l.jsx)(y.Y,{title:"API Products",subtitle:"Manage API products for Kubernetes",children:(0,l.jsx)(b.Y,{children:"Manage API products and plan policies"})}),(0,l.jsxs)(f.U,{children:[ge&&(0,l.jsx)(P.k,{}),je&&(0,l.jsx)(k._,{error:je}),ye&&(0,l.jsxs)(o.A,{p:2,children:[(0,l.jsxs)(s.A,{color:"error",children:["unable to check permissions: ",ye.message]}),(0,l.jsxs)(s.A,{variant:"body2",color:"textSecondary",children:["permission:"," ",ee?"kuadrant.apiproduct.create":ne?"kuadrant.apiproduct.delete":ue?"kuadrant.planpolicy.list":"unknown"]}),(0,l.jsx)(s.A,{variant:"body2",color:"textSecondary",children:"please try again or contact your administrator"})]}),!ge&&!je&&!ye&&(0,l.jsxs)(c.A,{container:!0,spacing:3,direction:"column",children:[(0,l.jsx)(c.A,{item:!0,children:(0,l.jsx)(S.n,{title:"API Products",action:Q?(0,l.jsx)(o.A,{display:"flex",alignItems:"center",height:"100%",mt:1,children:(0,l.jsx)(u.A,{variant:"contained",color:"primary",size:"small",startIcon:(0,l.jsx)(p.A,{}),onClick:()=>W(!0),children:"Create API Product"})}):void 0,children:(Pe=null==pe?void 0:pe.items,Pe&&0!==Pe.length?(0,l.jsx)(g.X,{options:{paging:Pe.length>5,pageSize:20,search:!0,filtering:!0,debounceInterval:300,toolbar:!0,emptyRowsWhenPaging:!1},columns:be,data:Pe}):(0,l.jsx)(s.A,{variant:"body2",color:"textSecondary",children:"No API products found"}))})}),de&&(0,l.jsx)(c.A,{item:!0,children:(0,l.jsx)(S.n,{title:"Plan Policies",children:(e=>e&&0!==e.length?(0,l.jsx)(g.X,{options:{paging:!1,search:!1,toolbar:!1},columns:fe,data:e}):(0,l.jsx)(s.A,{variant:"body2",color:"textSecondary",children:"No plan policies found"}))(null==xe?void 0:xe.items)})})]}),(0,l.jsx)(K,{open:$,onClose:()=>W(!1),onSuccess:()=>{L(e=>e+1),t.post({message:"API Product created",severity:"success",display:"transient"})}}),(0,l.jsx)(V,{open:M,onClose:()=>N(!1),onSuccess:()=>{L(e=>e+1),t.post({message:"API Product updated",severity:"success",display:"transient"})},namespace:(null==B?void 0:B.namespace)||"",name:(null==B?void 0:B.name)||""}),(0,l.jsx)(J.K,{open:z,title:"Delete API Product",description:X?`Deleting "${null==H?void 0:H.name}" will also remove:\n\n• ${X.requests} API Key(s)\n• ${X.secrets} API Key Secret(s)\n\nThis action cannot be undone.`:`Deleting "${null==H?void 0:H.name}" will also remove all associated API Keys and Secrets.\nThis action cannot be undone.`,confirmText:null==H?void 0:H.name,severity:"high",deleting:_,onConfirm:async()=>{if(H){Y(!0);try{if(!(await a.fetch(`${C}/api/kuadrant/apiproducts/${H.namespace}/${H.name}`,{method:"DELETE"})).ok)throw new Error("failed to delete apiproduct");L(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{Y(!1),E(!1),U(null)}}},onCancel:()=>{E(!1),U(null)}})]})]});var Pe},X=()=>(0,l.jsx)(C.B,{permission:q.vs,errorMessage:"you don't have permission to view the Kuadrant page",children:(0,l.jsx)(Y,{})})},34955:(e,a,t)=>{t.d(a,{Al:()=>r,DS:()=>m,EM:()=>d,FL:()=>n,J:()=>i,KV:()=>g,R_:()=>c,U3:()=>s,dp:()=>p,jH:()=>A,q0:()=>h,uL:()=>v,v_:()=>o,vs:()=>u,z4:()=>x});var l=t(83572);(0,l.i)({name:"kuadrant.planpolicy.create",attributes:{action:"create"}}),(0,l.i)({name:"kuadrant.planpolicy.read",attributes:{action:"read"}}),(0,l.i)({name:"kuadrant.planpolicy.update",attributes:{action:"update"}}),(0,l.i)({name:"kuadrant.planpolicy.delete",attributes:{action:"delete"}});const i=(0,l.i)({name:"kuadrant.planpolicy.list",attributes:{action:"read"}}),n=(0,l.i)({name:"kuadrant.apiproduct.create",attributes:{action:"create"}}),r=((0,l.i)({name:"kuadrant.apiproduct.read.own",attributes:{action:"read"}}),(0,l.i)({name:"kuadrant.apiproduct.read.all",attributes:{action:"read"}})),s=(0,l.i)({name:"kuadrant.apiproduct.update.own",attributes:{action:"update"}}),o=(0,l.i)({name:"kuadrant.apiproduct.update.all",attributes:{action:"update"}}),d=(0,l.i)({name:"kuadrant.apiproduct.delete.own",attributes:{action:"delete"}}),c=(0,l.i)({name:"kuadrant.apiproduct.delete.all",attributes:{action:"delete"}}),u=(0,l.i)({name:"kuadrant.apiproduct.list",attributes:{action:"read"}}),p=(0,l.i)({name:"kuadrant.apikey.create",attributes:{action:"create"},resourceType:"apiproduct"}),m=(0,l.i)({name:"kuadrant.apikey.read.own",attributes:{action:"read"}}),h=((0,l.i)({name:"kuadrant.apikey.read.all",attributes:{action:"read"}}),(0,l.i)({name:"kuadrant.apikey.update.own",attributes:{action:"update"}})),x=(0,l.i)({name:"kuadrant.apikey.update.all",attributes:{action:"update"}}),v=(0,l.i)({name:"kuadrant.apikey.delete.own",attributes:{action:"delete"}}),A=(0,l.i)({name:"kuadrant.apikey.delete.all",attributes:{action:"delete"}}),g=(0,l.i)({name:"kuadrant.apikey.approve",attributes:{action:"update"}})},38599:(e,a,t)=>{t.d(a,{B:()=>o});var l=t(31085),i=(t(95478),t(10394)),n=t(72501),r=t(86687),s=t(46205);const o=({children:e,permission:a,fallback:t,errorMessage:o})=>{const{allowed:d,loading:c,error:u}=(0,s.l)(a);return c?(0,l.jsx)(r.k,{}):u?(0,l.jsxs)(i.A,{p:4,children:[(0,l.jsxs)(n.A,{color:"error",children:["Unable to check permissions: ",u.message]}),(0,l.jsx)(n.A,{variant:"body2",color:"textSecondary",children:"Please try again or contact your administrator"})]}):d?(0,l.jsx)(l.Fragment,{children:e}):t?(0,l.jsx)(l.Fragment,{children:t}):(0,l.jsxs)(i.A,{p:4,children:[(0,l.jsx)(n.A,{color:"textSecondary",children:o||"You don't have permission to view this page"}),(0,l.jsx)(i.A,{mt:1,children:(0,l.jsxs)(n.A,{variant:"caption",color:"textSecondary",children:["Required permission: ",a.name]})})]})}},46205:(e,a,t)=>{t.d(a,{W:()=>n,l:()=>i});var l=t(87421);function i(e,a){const t="resourceType"in e?{permission:e,resourceRef:a}:{permission:e},i=(0,l.J)(t);return{allowed:i.allowed,loading:i.loading,error:i.error}}function n(e,a,t,l){return!!l||!(!t||e!==a)}},63221:(e,a,t)=>{t.d(a,{K:()=>v});var l=t(31085),i=t(95478),n=t(76891),r=t(61477),s=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:y})=>{const[b,f]=(0,i.useState)("");(0,i.useEffect)(()=>{e||f("")},[e]);const P="high"===A&&v,k=!P||b===v;return(0,l.jsxs)(n.A,{open:e,onClose:g?void 0:y,maxWidth:"sm",fullWidth:!0,children:[(0,l.jsxs)(r.A,{children:["high"===A&&(0,l.jsxs)(s.A,{display:"flex",alignItems:"center",style:{gap:8},children:[(0,l.jsx)(x.A,{color:"error"}),(0,l.jsx)("span",{children:a})]}),"high"!==A&&a]}),(0,l.jsxs)(o.A,{children:[(0,l.jsx)(d.A,{style:{whiteSpace:"pre-line"},children:t}),P&&(0,l.jsxs)(s.A,{mt:2,children:[(0,l.jsxs)(c.A,{variant:"body2",color:"textSecondary",gutterBottom:!0,children:["Type ",(0,l.jsx)("strong",{children:v})," to confirm:"]}),(0,l.jsx)(u.A,{fullWidth:!0,variant:"outlined",size:"small",value:b,onChange:e=>f(e.target.value),disabled:g,autoFocus:!0,placeholder:v})]})]}),(0,l.jsxs)(p.A,{children:[(0,l.jsx)(m.A,{onClick:y,disabled:g,children:"Cancel"}),(0,l.jsx)(m.A,{onClick:()=>{k&&j()},color:"secondary",variant:"contained",disabled:g||!k,startIcon:g?(0,l.jsx)(h.A,{size:16,color:"inherit"}):void 0,children:g?"Deleting...":"Delete"})]})]})}}}]);
|
|
2
|
-
//# sourceMappingURL=7684.3d1fc1a1.chunk.js.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"static/7684.3d1fc1a1.chunk.js","mappings":"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","sources":["webpack://internal.plugin-kuadrant/./src/components/PlanPolicyDetailsCard/PlanPolicyDetails.tsx","webpack://internal.plugin-kuadrant/./src/utils/validation.ts","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","webpack://internal.plugin-kuadrant/./src/permissions.ts","webpack://internal.plugin-kuadrant/./src/components/PermissionGate/PermissionGate.tsx","webpack://internal.plugin-kuadrant/./src/utils/permissions.ts","webpack://internal.plugin-kuadrant/./src/components/ConfirmDeleteDialog/ConfirmDeleteDialog.tsx"],"sourcesContent":["import React from 'react';\nimport { Box, Typography, Chip, useTheme } from '@material-ui/core';\nimport { Alert } from '@material-ui/lab';\n\ninterface PlanPolicyDetailsProps {\n selectedPolicy: {\n metadata: {\n name: string;\n };\n plans?: Array<{\n tier: string;\n description?: string;\n limits?: {\n daily?: number;\n monthly?: number;\n yearly?: number;\n };\n }>;\n } | null;\n alertSeverity?: 'warning' | 'info';\n alertMessage?: string;\n includeTopMargin?: boolean;\n}\n\nexport const PlanPolicyDetails: React.FC<PlanPolicyDetailsProps> = ({\n selectedPolicy,\n alertSeverity = 'warning',\n alertMessage = 'No PlanPolicy found for this HTTPRoute. API keys and rate limiting may not be available.',\n includeTopMargin = true,\n}) => {\n const theme = useTheme();\n return (\n <Box\n mt={includeTopMargin ? 1 : 0}\n p={2}\n bgcolor={theme.palette.background.default}\n borderRadius={1}\n border={`1px solid ${theme.palette.divider}`}\n >\n {selectedPolicy ? (\n <>\n <Typography variant=\"subtitle2\" gutterBottom style={{ fontWeight: 600 }}>\n Associated PlanPolicy: <strong>{selectedPolicy.metadata.name}</strong>\n </Typography>\n\n {selectedPolicy.plans && selectedPolicy.plans.length > 0 ? (\n <>\n <Typography\n variant=\"caption\"\n display=\"block\"\n gutterBottom\n color=\"textSecondary\"\n style={{ marginTop: 8 }}\n >\n Available Tiers:\n </Typography>\n <Box display=\"flex\" flexWrap=\"wrap\" mt={1} style={{ gap: 8 }}>\n {selectedPolicy.plans.map((plan: any, idx: number) => {\n const limitText = plan.limits?.daily\n ? `${plan.limits.daily}/day`\n : plan.limits?.monthly\n ? `${plan.limits.monthly}/month`\n : plan.limits?.yearly\n ? `${plan.limits.yearly}/year`\n : 'No limit';\n\n return (\n <Chip\n key={idx}\n label={`${plan.tier}: ${limitText}`}\n size=\"small\"\n variant=\"outlined\"\n color=\"primary\"\n />\n );\n })}\n </Box>\n </>\n ) : (\n <Typography variant=\"caption\" color=\"textSecondary\">\n No plans defined in this PlanPolicy\n </Typography>\n )}\n </>\n ) : (\n <Alert severity={alertSeverity}>{alertMessage}</Alert>\n )}\n </Box>\n );\n};\n","// Kubernetes name validation\nexport const validateKubernetesName = (value: string): string | null => {\n if (!value || !value.trim()) {\n return 'Name is required';\n }\n if (value.length > 253) {\n return 'Must be 253 characters or less';\n }\n\n const dns1123Regex = /^[a-z0-9]([-a-z0-9]*[a-z0-9])?$/;\n\n if (!dns1123Regex.test(value)) {\n return 'Must be lowercase alphanumeric with hyphens, start and end with alphanumeric';\n }\n\n return null;\n};\n\n// email validation\nexport const validateEmail = (value: string): string | null => {\n if (!value) {\n return null;\n }\n\n const emailRegex = /^[^\\s@]+@[^\\s@]+\\.[^\\s@]+$/;\n\n if (!emailRegex.test(value)) {\n return 'Must be a valid email address';\n }\n\n return null;\n};\n\n// URL validation\nexport const validateURL = (value: string): string | null => {\n if (!value) {\n return null;\n }\n\n try {\n const url = new URL(value);\n if (!['http:', 'https:'].includes(url.protocol)) {\n return 'Must be a valid HTTP or HTTPS URL';\n }\n return null;\n } catch {\n return 'Must be a valid URL';\n }\n};\n","import React, {useEffect, 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';\nimport { validateKubernetesName, validateEmail, validateURL } from '../../utils/validation';\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 const [httpRoutesRetry, setHttpRoutesRetry] = useState(0);\n const [nameError, setNameError] = useState<string | null>(null);\n const [contactEmailError, setContactEmailError] = useState<string | null>(null);\n const [docsURLError, setDocsURLError] = useState<string | null>(null);\n const [openAPISpecError, setOpenAPISpecError] = useState<string | null>(null);\n const {\n value: httpRoutes,\n loading: httpRoutesLoading,\n error: httpRoutesError\n } = 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, httpRoutesRetry]);\n\n // load planpolicies with full details to show associated plans\n const {\n value: planPolicies,\n error: planPoliciesError\n } = 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 useEffect(() => {\n if (open) {\n setNameError(null);\n setContactEmailError(null);\n setDocsURLError(null);\n setOpenAPISpecError(null);\n }\n }, [open]);\n\n // validate handlers\n const handleNameChange = (value: string) => {\n setName(value);\n setNameError(validateKubernetesName(value));\n };\n\n const handleContactEmailChange = (value: string) => {\n setContactEmail(value);\n setContactEmailError(validateEmail(value));\n };\n\n const handleDocsURLChange = (value: string) => {\n setDocsURL(value);\n setDocsURLError(validateURL(value));\n };\n\n const handleOpenAPISpecChange = (value: string) => {\n setOpenAPISpec(value);\n setOpenAPISpecError(validateURL(value));\n };\n\n const handleAddTag = () => {\n if (tagInput.trim() && !tags.includes(tagInput.trim())) {\n setTags([...tags, tagInput.trim()]);\n setTagInput('');\n }\n };\n\n const handleDeleteTag = (tagToDelete: string) => {\n setTags(tags.filter(tag => tag !== tagToDelete));\n };\n\n const 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 setNameError(null);\n setContactEmailError(null);\n setDocsURLError(null);\n setOpenAPISpecError(null);\n onClose();\n };\n\n const hasValidationErrors = !!nameError || !!contactEmailError || !!docsURLError || !!openAPISpecError;\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 {httpRoutesError && (\n <Alert severity=\"error\" style={{ marginBottom: 16 }}>\n <strong>Failed to load HTTPRoutes:</strong> {httpRoutesError.message}\n <Box mt={1}>\n <Button\n size=\"small\"\n variant=\"outlined\"\n onClick={() => setHttpRoutesRetry(prev => prev + 1)}\n >\n Retry\n </Button>\n </Box>\n </Alert>\n )}\n\n {planPoliciesError && (\n <Alert severity=\"warning\" style={{ marginBottom: 16 }}>\n <strong>Failed to load PlanPolicies:</strong> {planPoliciesError.message}\n <Typography variant=\"body2\" style={{ marginTop: 8 }}>\n You can still create the API Product, but plan information may be incomplete.\n </Typography>\n </Alert>\n )}\n <Grid container spacing={2}>\n <Grid item xs={6}>\n <TextField\n fullWidth\n label=\"Name\"\n value={name}\n onChange={e => handleNameChange(e.target.value)}\n placeholder=\"my-api\"\n helperText={nameError || \"Kubernetes resource name (lowercase, hyphens)\"}\n error={!!nameError}\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={\n httpRoutesError\n ? \"Unable to load HTTPRoutes. Please retry.\"\n : \"Select an HTTPRoute (backstage.io/expose: true). APIProduct will be created in the same namespace.\"\n }\n error={!!httpRoutesError}\n disabled={httpRoutesLoading || creating || !!httpRoutesError}\n InputLabelProps={{\n classes: {\n asterisk: classes.asterisk,\n },\n }}\n SelectProps={{\n 'data-testid': 'httproute-select',\n } as any}\n >\n {httpRoutesLoading && (\n <MenuItem value=\"\">Loading...</MenuItem>\n )}\n {httpRoutesError && (\n <MenuItem value=\"\">Error loading routes</MenuItem>\n )}\n {!httpRoutesLoading && !httpRoutesError && httpRoutes && httpRoutes.length === 0 && (\n <MenuItem value=\"\">No HTTPRoutes available</MenuItem>\n )}\n {!httpRoutesLoading && !httpRoutesError && 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 => handleContactEmailChange(e.target.value)}\n placeholder=\"api-team@example.com\"\n helperText={contactEmailError || \"Contact email for API support\"}\n error={!!contactEmailError}\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 => handleDocsURLChange(e.target.value)}\n placeholder=\"https://api.example.com/docs\"\n helperText={docsURLError || \"Link to API documentation\"}\n error={!!docsURLError}\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 => handleOpenAPISpecChange(e.target.value)}\n placeholder=\"https://api.example.com/openapi.json\"\n helperText={openAPISpecError || \"Link to OpenAPI specification\"}\n error={!!openAPISpecError}\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 || hasValidationErrors}\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';\nimport { validateEmail, validateURL } from '../../utils/validation';\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 // valid error states\n const [contactEmailError, setContactEmailError] = useState<string | null>(null);\n const [docsURLError, setDocsURLError] = useState<string | null>(null);\n const [openAPISpecError, setOpenAPISpecError] = useState<string | null>(null);\n\n // Load APIProduct data when dialog opens\n useEffect(() => {\n if (open && namespace && name) {\n setLoading(true);\n setError('');\n\n fetchApi.fetch(`${backendUrl}/api/kuadrant/apiproducts/${namespace}/${name}`)\n .then(async res => {\n if (!res.ok) {\n const errorData = await res.json();\n throw new Error(errorData.error || `Failed to fetch API product: ${res.status}`);\n }\n return res.json();\n })\n .then(data => {\n setDisplayName(data.spec.displayName || '');\n setDescription(data.spec.description || '');\n setVersion(data.spec.version || 'v1');\n setPublishStatus(data.spec.publishStatus || 'Draft');\n setApprovalMode(data.spec.approvalMode || 'manual');\n setTags(data.spec.tags || []);\n setTargetRef(data.spec.targetRef || null);\n setContactEmail(data.spec.contact?.email || '');\n setContactTeam(data.spec.contact?.team || '');\n setDocsURL(data.spec.documentation?.docsURL || '');\n setOpenAPISpec(data.spec.documentation?.openAPISpec || '');\n setContactEmailError(null);\n setDocsURLError(null);\n setOpenAPISpecError(null);\n setLoading(false);\n })\n .catch(err => {\n setError(err.message || 'Failed to load API product');\n setLoading(false);\n });\n }\n }, [open, namespace, name, backendUrl, fetchApi]);\n\n // load planpolicies with full details to show associated plans\n const {\n value: planPolicies,\n error: planPoliciesError\n } = useAsync(async () => {\n if (!open) return null;\n const response = await fetchApi.fetch(`${backendUrl}/api/kuadrant/planpolicies`);\n return await response.json();\n }, [backendUrl, fetchApi, open]);\n\n // find planpolicy associated with targetRef\n const selectedPolicy = React.useMemo(() => {\n if (!planPolicies?.items || !targetRef) return null;\n\n return planPolicies.items.find((pp: any) => {\n const ref = pp.targetRef;\n return (\n ref?.kind === 'HTTPRoute' &&\n ref?.name === targetRef.name &&\n (!ref?.namespace || ref?.namespace === (targetRef.namespace || namespace))\n );\n });\n }, [planPolicies, targetRef, namespace]);\n\n // val handlers\n\n useEffect(() => {\n if (open) {\n setContactEmailError(null);\n setDocsURLError(null);\n setOpenAPISpecError(null);\n }\n }, [open]);\n\n const handleContactEmailChange = (value: string) => {\n setContactEmail(value);\n setContactEmailError(validateEmail(value));\n };\n\n const handleDocsURLChange = (value: string) => {\n setDocsURL(value);\n setDocsURLError(validateURL(value));\n };\n\n const handleOpenAPISpecChange = (value: string) => {\n setOpenAPISpec(value);\n setOpenAPISpecError(validateURL(value));\n };\n\n const handleAddTag = () => {\n if (tagInput.trim() && !tags.includes(tagInput.trim())) {\n setTags([...tags, tagInput.trim()]);\n setTagInput('');\n }\n };\n\n const handleDeleteTag = (tagToDelete: string) => {\n setTags(tags.filter(tag => tag !== tagToDelete));\n };\n\n const handleSave = async () => {\n setError('');\n setSaving(true);\n\n 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 {planPoliciesError && (\n <Alert severity=\"warning\" style={{ marginBottom: 16 }}>\n <strong>Failed to load PlanPolicies:</strong> {planPoliciesError.message}\n <Typography variant=\"body2\" style={{ marginTop: 8 }}>\n Plan information may be incomplete.\n </Typography>\n </Alert>\n )}\n {loading ? (\n <Progress />\n ) : (\n <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 => handleContactEmailChange(e.target.value)}\n placeholder=\"api-team@example.com\"\n helperText={contactEmailError || \"Contact email for API support\"}\n error={!!contactEmailError}\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 => handleDocsURLChange(e.target.value)}\n placeholder=\"https://api.example.com/docs\"\n helperText={docsURLError || \"Link to API documentation\"}\n error={!!docsURLError}\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 => handleOpenAPISpecChange(e.target.value)}\n placeholder=\"https://api.example.com/openapi.json\"\n helperText={openAPISpecError || \"Link to OpenAPI specification\"}\n error={!!openAPISpecError}\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 || !!contactEmailError || !!docsURLError || !!openAPISpecError}\n startIcon={saving ? <CircularProgress size={16} color=\"inherit\" /> : undefined}\n >\n {saving ? 'Saving...' : 'Save'}\n </Button>\n </DialogActions>\n </Dialog>\n );\n};\n","import React, { useState } from \"react\";\nimport {\n Typography,\n Grid,\n Box,\n Chip,\n Button,\n IconButton,\n} 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 VpnKeyIcon from \"@material-ui/icons/VpnKey\";\nimport LockIcon from \"@material-ui/icons/Lock\";\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 {\n useApi,\n configApiRef,\n fetchApiRef,\n alertApiRef,\n identityApiRef,\n} from \"@backstage/core-plugin-api\";\nimport { PermissionGate } from \"../PermissionGate\";\nimport { CreateAPIProductDialog } from \"../CreateAPIProductDialog\";\nimport {\n kuadrantApiProductCreatePermission,\n kuadrantApiProductDeleteOwnPermission,\n kuadrantApiProductDeleteAllPermission,\n kuadrantApiProductUpdateOwnPermission,\n kuadrantApiProductUpdateAllPermission,\n kuadrantApiProductListPermission,\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<{\n namespace: string;\n name: string;\n } | null>(null);\n const [apiProductToEdit, setApiProductToEdit] = useState<{\n namespace: string;\n name: string;\n } | null>(null);\n const [deleting, setDeleting] = useState(false);\n const [deleteStats, setDeleteStats] = useState<{\n requests: number;\n secrets: number;\n } | null>(null);\n\n const {\n allowed: canCreateApiProduct,\n loading: createPermissionLoading,\n error: createPermissionError,\n } = useKuadrantPermission(kuadrantApiProductCreatePermission);\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 { allowed: canUpdateOwnApiProduct } = useKuadrantPermission(\n kuadrantApiProductUpdateOwnPermission,\n );\n\n const { allowed: canUpdateAllApiProducts } = useKuadrantPermission(\n kuadrantApiProductUpdateAllPermission,\n );\n\n const deletePermissionLoading =\n 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 {\n value: apiProducts,\n loading: apiProductsLoading,\n error: apiProductsError,\n } = useAsync(async (): Promise<KuadrantList> => {\n const response = await fetchApi.fetch(\n `${backendUrl}/api/kuadrant/apiproducts`,\n );\n return await response.json();\n }, [backendUrl, fetchApi, refreshTrigger]);\n\n const {\n value: planPolicies,\n loading: planPoliciesLoading,\n error: planPoliciesError,\n } = useAsync(async (): Promise<KuadrantList> => {\n const response = await fetchApi.fetch(\n `${backendUrl}/api/kuadrant/planpolicies`,\n );\n return await response.json();\n }, [backendUrl, fetchApi, refreshTrigger]);\n\n const loading =\n apiProductsLoading ||\n planPoliciesLoading ||\n createPermissionLoading ||\n deletePermissionLoading ||\n planPolicyPermissionLoading;\n const error = apiProductsError || planPoliciesError;\n const permissionError =\n createPermissionError || deletePermissionError || planPolicyPermissionError;\n\n const handleCreateSuccess = () => {\n setRefreshTrigger((prev) => prev + 1);\n alertApi.post({\n message: \"API Product created\",\n severity: \"success\",\n display: \"transient\",\n });\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({\n message: \"API Product updated\",\n severity: \"success\",\n display: \"transient\",\n });\n };\n\n const handleDeleteClick = async (namespace: string, name: string) => {\n setApiProductToDelete({ namespace, name });\n setDeleteStats(null);\n\n try {\n const response = await fetchApi.fetch(\n `${backendUrl}/api/kuadrant/requests?namespace=${namespace}`,\n );\n if (response.ok) {\n const data = await response.json();\n const related = (data.items || []).filter(\n (r: any) =>\n r.spec.apiName === name && r.spec.apiNamespace === namespace,\n );\n const approved = related.filter(\n (r: any) => r.status?.phase === \"Approved\",\n ).length;\n setDeleteStats({ requests: related.length, secrets: approved });\n }\n } catch (err) {\n console.warn(\"Failed to fetch delete stats:\", err);\n }\n\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({\n message: \"API Product deleted\",\n severity: \"success\",\n display: \"transient\",\n });\n } catch (err) {\n console.error(\"error deleting apiproduct:\", err);\n alertApi.post({\n message: \"Failed to delete API Product\",\n severity: \"error\",\n display: \"transient\",\n });\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: \"Authentication\",\n field: \"status.discoveredAuthScheme\",\n render: (row: any) => {\n const authSchemes =\n row.status?.discoveredAuthScheme?.authentication || {};\n const schemeObjects = Object.values(authSchemes);\n\n const hasApiKey = schemeObjects.some((scheme: any) =>\n scheme.hasOwnProperty(\"apiKey\"),\n );\n const hasJwt = schemeObjects.some((scheme: any) =>\n scheme.hasOwnProperty(\"jwt\"),\n );\n\n if (!hasApiKey && !hasJwt) {\n return (\n <Typography variant=\"body2\" style={{ fontStyle: \"italic\" }}>\n unknown\n </Typography>\n );\n }\n\n return (\n <Box display=\"flex\" style={{ gap: 4 }}>\n {hasApiKey && (\n <Chip\n icon={<VpnKeyIcon />}\n label=\"API Key\"\n size=\"small\"\n color=\"primary\"\n />\n )}\n {hasJwt && (\n <Chip\n icon={<LockIcon />}\n label=\"OIDC\"\n size=\"small\"\n color=\"secondary\"\n />\n )}\n </Box>\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 =\n canUpdateAllApiProducts || (canUpdateOwnApiProduct && isOwner);\n const canDelete =\n 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={() =>\n handleEditClick(row.metadata.namespace, row.metadata.name)\n }\n title=\"Edit API Product\"\n >\n <EditIcon fontSize=\"small\" />\n </IconButton>\n )}\n\n {canDelete && (\n <IconButton\n size=\"small\"\n onClick={() =>\n handleDeleteClick(row.metadata.namespace, row.metadata.name)\n }\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\n to={`/kuadrant/planpolicy/${row.metadata.namespace}/${row.metadata.name}`}\n >\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 (\n <Typography variant=\"body2\" color=\"textSecondary\">\n No API products found\n </Typography>\n );\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 (\n <Typography variant=\"body2\" color=\"textSecondary\">\n No plan policies found\n </Typography>\n );\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\n title=\"API Products\"\n subtitle=\"Manage API products for Kubernetes\"\n >\n <SupportButton>Manage API products and plan policies</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:{\" \"}\n {createPermissionError\n ? \"kuadrant.apiproduct.create\"\n : deletePermissionError\n ? \"kuadrant.apiproduct.delete\"\n : planPolicyPermissionError\n ? \"kuadrant.planpolicy.list\"\n : \"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 <InfoCard\n title=\"API Products\"\n action={\n canCreateApiProduct ? (\n <Box\n display=\"flex\"\n alignItems=\"center\"\n height=\"100%\"\n mt={1}\n >\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 </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={\n deleteStats\n ? `Deleting \"${apiProductToDelete?.name}\" will also remove:\n\n• ${deleteStats.requests} API Key(s)\n• ${deleteStats.secrets} API Key Secret(s)\n\nThis action cannot be undone.`\n : `Deleting \"${apiProductToDelete?.name}\" will also remove all associated API Keys and Secrets.\nThis action cannot be undone.`\n }\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","import { createPermission } from '@backstage/plugin-permission-common';\n\n/**\n * permission definitions for the kuadrant plugin\n *\n * these permissions control access to kuadrant resources and operations.\n * they must match the permissions defined in the backend plugin.\n *\n * permission types:\n * - BasicPermission: standard permission that applies globally\n * - ResourcePermission: permission scoped to specific resource types (e.g., apiproduct)\n *\n * permission patterns:\n * - `.create` - create new resources\n * - `.read` - read resource details\n * - `.read.own` - read only resources owned by the user\n * - `.read.all` - read all resources regardless of ownership\n * - `.update` - modify existing resources\n * - `.delete` - delete resources\n * - `.delete.own` - delete only resources owned by the user\n * - `.delete.all` - delete any resource regardless of ownership\n * - `.list` - list/view collections of resources\n */\n\n// planpolicy permissions\nexport const kuadrantPlanPolicyCreatePermission = createPermission({\n name: 'kuadrant.planpolicy.create',\n attributes: { action: 'create' },\n});\n\nexport const kuadrantPlanPolicyReadPermission = createPermission({\n name: 'kuadrant.planpolicy.read',\n attributes: { action: 'read' },\n});\n\nexport const kuadrantPlanPolicyUpdatePermission = createPermission({\n name: 'kuadrant.planpolicy.update',\n attributes: { action: 'update' },\n});\n\nexport const kuadrantPlanPolicyDeletePermission = createPermission({\n name: 'kuadrant.planpolicy.delete',\n attributes: { action: 'delete' },\n});\n\nexport const kuadrantPlanPolicyListPermission = createPermission({\n name: 'kuadrant.planpolicy.list',\n attributes: { action: 'read' },\n});\n\n// apiproduct permissions\n\n/**\n * permission to create new API products\n * granted to api owners and admins\n */\nexport const kuadrantApiProductCreatePermission = createPermission({\n name: 'kuadrant.apiproduct.create',\n attributes: { action: 'create' },\n});\n\n/**\n * permission to read API products owned by the current user\n * for api owners to view their own products\n */\nexport const kuadrantApiProductReadOwnPermission = createPermission({\n name: 'kuadrant.apiproduct.read.own',\n attributes: { action: 'read' },\n});\n\n/**\n * permission to read all API products regardless of ownership\n * for platform engineers/admins who need to view all products\n */\nexport const kuadrantApiProductReadAllPermission = createPermission({\n name: 'kuadrant.apiproduct.read.all',\n attributes: { action: 'read' },\n});\n\n/**\n * permission to update API products owned by the current user\n * for api owners to modify their own products\n */\nexport const kuadrantApiProductUpdateOwnPermission = createPermission({\n name: 'kuadrant.apiproduct.update.own',\n attributes: { action: 'update' },\n});\n\n/**\n * permission to update any API product regardless of ownership\n * for platform engineers/admins\n */\nexport const kuadrantApiProductUpdateAllPermission = createPermission({\n name: 'kuadrant.apiproduct.update.all',\n attributes: { action: 'update' },\n});\n\n/**\n * permission to delete API products owned by the current user\n * for api owners to remove their own products\n */\nexport const kuadrantApiProductDeleteOwnPermission = createPermission({\n name: 'kuadrant.apiproduct.delete.own',\n attributes: { action: 'delete' },\n});\n\n/**\n * permission to delete any API product regardless of ownership\n * for platform engineers/admins\n */\nexport const kuadrantApiProductDeleteAllPermission = createPermission({\n name: 'kuadrant.apiproduct.delete.all',\n attributes: { action: 'delete' },\n});\n\n/**\n * permission to list API products\n * backend filters results based on .own vs .all read permissions\n */\nexport const kuadrantApiProductListPermission = createPermission({\n name: 'kuadrant.apiproduct.list',\n attributes: { action: 'read' },\n});\n\n// apikey permissions\n\n/**\n * permission to create API keys (request API access)\n *\n * this is a ResourcePermission scoped to 'apiproduct', allowing\n * fine-grained control over which API products users can request access to.\n *\n * use in frontend: useKuadrantPermission(kuadrantApiKeyCreatePermission)\n * use in backend with resource: { permission, resourceRef: 'apiproduct:namespace/name' }\n */\nexport const kuadrantApiKeyCreatePermission = createPermission({\n name: 'kuadrant.apikey.create',\n attributes: { action: 'create' },\n resourceType: 'apiproduct',\n});\n\n/**\n * permission to read API keys owned by the current user\n * allows users to view their own API keys and request history\n */\nexport const kuadrantApiKeyReadOwnPermission = createPermission({\n name: 'kuadrant.apikey.read.own',\n attributes: { action: 'read' },\n});\n\n/**\n * permission to read all API keys regardless of ownership\n * for platform engineers/admins who need to view the approval queue and audit keys\n */\nexport const kuadrantApiKeyReadAllPermission = createPermission({\n name: 'kuadrant.apikey.read.all',\n attributes: { action: 'read' },\n});\n\n/**\n * permission to update API keys owned by the current user\n * allows users to edit their own pending requests (change plan tier, use case)\n */\nexport const kuadrantApiKeyUpdateOwnPermission = createPermission({\n name: 'kuadrant.apikey.update.own',\n attributes: { action: 'update' },\n});\n\n/**\n * permission to update any API key regardless of ownership\n * typically granted to API owners and platform engineers for approving/rejecting requests\n */\nexport const kuadrantApiKeyUpdateAllPermission = createPermission({\n name: 'kuadrant.apikey.update.all',\n attributes: { action: 'update' },\n});\n\n/**\n * permission to delete API keys owned by the current user\n * allows users to cancel their own requests or revoke their own access\n */\nexport const kuadrantApiKeyDeleteOwnPermission = createPermission({\n name: 'kuadrant.apikey.delete.own',\n attributes: { action: 'delete' },\n});\n\n/**\n * permission to delete any API key regardless of ownership\n * for platform engineers/admins who need to revoke access\n */\nexport const kuadrantApiKeyDeleteAllPermission = createPermission({\n name: 'kuadrant.apikey.delete.all',\n attributes: { action: 'delete' },\n});\n\n/**\n * permission to approve/reject API key requests\n * grants access to the approval queue - for API owners and admins only\n * separate from update.own which consumers use to edit their pending requests\n */\nexport const kuadrantApiKeyApprovePermission = createPermission({\n name: 'kuadrant.apikey.approve',\n attributes: { action: 'update' },\n});\n\nexport const kuadrantPermissions = [\n kuadrantPlanPolicyCreatePermission,\n kuadrantPlanPolicyReadPermission,\n kuadrantPlanPolicyUpdatePermission,\n kuadrantPlanPolicyDeletePermission,\n kuadrantPlanPolicyListPermission,\n kuadrantApiProductCreatePermission,\n kuadrantApiProductReadOwnPermission,\n kuadrantApiProductReadAllPermission,\n kuadrantApiProductUpdateOwnPermission,\n kuadrantApiProductUpdateAllPermission,\n kuadrantApiProductDeleteOwnPermission,\n kuadrantApiProductDeleteAllPermission,\n kuadrantApiProductListPermission,\n kuadrantApiKeyCreatePermission,\n kuadrantApiKeyReadOwnPermission,\n kuadrantApiKeyReadAllPermission,\n kuadrantApiKeyUpdateOwnPermission,\n kuadrantApiKeyUpdateAllPermission,\n kuadrantApiKeyDeleteOwnPermission,\n kuadrantApiKeyDeleteAllPermission,\n kuadrantApiKeyApprovePermission,\n];\n","import 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 { usePermission } from '@backstage/plugin-permission-react';\nimport { Permission, ResourcePermission } from '@backstage/plugin-permission-common';\n\n/**\n * result of a permission check including error state\n */\nexport interface PermissionCheckResult {\n allowed: boolean;\n loading: boolean;\n error?: Error;\n}\n\n/**\n * custom hook for checking kuadrant permissions that handles both\n * BasicPermission and ResourcePermission types without type bypasses\n *\n * @param permission - the permission to check\n * @param resourceRef - optional resource reference for ResourcePermissions\n * @returns permission check result with error handling\n *\n * @example\n * // basic permission\n * const { allowed, loading, error } = useKuadrantPermission(\n * kuadrantApiProductListPermission\n * );\n *\n * @example\n * // resource permission\n * const { allowed, loading, error } = useKuadrantPermission(\n * kuadrantApiKeyCreatePermission,\n * 'apiproduct:namespace/name'\n * );\n */\nexport function useKuadrantPermission(\n permission: Permission,\n resourceRef?: string,\n): PermissionCheckResult {\n // construct the permission request based on whether it's a ResourcePermission\n const permissionRequest = 'resourceType' in permission\n ? { permission: permission as ResourcePermission, resourceRef }\n : { permission };\n\n const result = usePermission(permissionRequest as any);\n\n return {\n allowed: result.allowed,\n loading: result.loading,\n error: result.error,\n };\n}\n\n/**\n * helper to determine if a user can delete a specific API key or request\n *\n * @param ownerId - the user id who owns the key/request\n * @param currentUserId - the current user's id\n * @param canDeleteOwn - whether user has permission to delete their own keys\n * @param canDeleteAll - whether user has permission to delete all keys\n * @returns true if user can delete this specific key/request\n */\nexport function canDeleteResource(\n ownerId: string,\n currentUserId: string,\n canDeleteOwn: boolean,\n canDeleteAll: boolean,\n): boolean {\n if (canDeleteAll) return true;\n if (canDeleteOwn && ownerId === currentUserId) return true;\n return false;\n}\n","import React, { useState, useEffect } from 'react';\nimport {\n Dialog,\n DialogTitle,\n DialogContent,\n DialogContentText,\n DialogActions,\n Button,\n TextField,\n Typography,\n Box,\n CircularProgress,\n} from '@material-ui/core';\nimport WarningIcon from '@material-ui/icons/Warning';\n\nexport interface ConfirmDeleteDialogProps {\n open: boolean;\n title: string;\n description: string;\n // for dangerous deletes, require typing this text to confirm\n confirmText?: string;\n // severity affects styling - 'high' shows warning icon and requires text confirmation\n severity?: 'normal' | 'high';\n deleting?: boolean;\n onConfirm: () => void;\n onCancel: () => void;\n}\n\nexport const ConfirmDeleteDialog = ({\n open,\n title,\n description,\n confirmText,\n severity = 'normal',\n deleting = false,\n onConfirm,\n onCancel,\n}: ConfirmDeleteDialogProps) => {\n const [inputValue, setInputValue] = useState('');\n\n // reset input when dialog opens/closes\n useEffect(() => {\n if (!open) {\n setInputValue('');\n }\n }, [open]);\n\n const requiresTextConfirmation = severity === 'high' && confirmText;\n const canConfirm = requiresTextConfirmation ? inputValue === confirmText : true;\n\n const handleConfirm = () => {\n if (canConfirm) {\n onConfirm();\n }\n };\n\n return (\n <Dialog\n open={open}\n onClose={deleting ? undefined : onCancel}\n maxWidth=\"sm\"\n fullWidth\n >\n <DialogTitle>\n {severity === 'high' && (\n <Box display=\"flex\" alignItems=\"center\" style={{ gap: 8 }}>\n <WarningIcon color=\"error\" />\n <span>{title}</span>\n </Box>\n )}\n {severity !== 'high' && title}\n </DialogTitle>\n <DialogContent>\n <DialogContentText style={{ whiteSpace: 'pre-line' }}>\n {description}\n </DialogContentText>\n {requiresTextConfirmation && (\n <Box mt={2}>\n <Typography variant=\"body2\" color=\"textSecondary\" gutterBottom>\n Type <strong>{confirmText}</strong> to confirm:\n </Typography>\n <TextField\n fullWidth\n variant=\"outlined\"\n size=\"small\"\n value={inputValue}\n onChange={e => setInputValue(e.target.value)}\n disabled={deleting}\n autoFocus\n placeholder={confirmText}\n />\n </Box>\n )}\n </DialogContent>\n <DialogActions>\n <Button onClick={onCancel} disabled={deleting}>\n Cancel\n </Button>\n <Button\n onClick={handleConfirm}\n color=\"secondary\"\n variant=\"contained\"\n disabled={deleting || !canConfirm}\n startIcon={deleting ? <CircularProgress size={16} color=\"inherit\" /> : undefined}\n >\n {deleting ? 'Deleting...' : 'Delete'}\n </Button>\n </DialogActions>\n </Dialog>\n );\n};\n"],"names":["PlanPolicyDetails","selectedPolicy","alertSeverity","alertMessage","includeTopMargin","theme","useTheme","Box","mt","p","bgcolor","palette","background","default","borderRadius","border","divider","Typography","variant","gutterBottom","style","fontWeight","strong","metadata","name","plans","length","display","color","marginTop","flexWrap","gap","map","plan","idx","limitText","limits","daily","monthly","yearly","Chip","label","tier","size","Alert","severity","validateEmail","value","test","validateURL","url","URL","includes","protocol","useStyles","makeStyles","asterisk","CreateAPIProductDialog","open","onClose","onSuccess","classes","config","useApi","configApiRef","fetchApi","fetchApiRef","backendUrl","getString","setName","useState","displayName","setDisplayName","description","setDescription","version","setVersion","approvalMode","setApprovalMode","publishStatus","setPublishStatus","tags","setTags","tagInput","setTagInput","selectedHTTPRoute","setSelectedHTTPRoute","contactEmail","setContactEmail","contactTeam","setContactTeam","docsURL","setDocsURL","openAPISpec","setOpenAPISpec","error","setError","creating","setCreating","httpRoutesRetry","setHttpRoutesRetry","nameError","setNameError","contactEmailError","setContactEmailError","docsURLError","setDocsURLError","openAPISpecError","setOpenAPISpecError","httpRoutes","loading","httpRoutesLoading","httpRoutesError","useAsync","async","response","fetch","json","items","filter","route","annotations","planPolicies","planPoliciesError","selectedRouteInfo","split","routeNamespace","routeName","find","pp","ref","targetRef","kind","namespace","useEffect","handleAddTag","trim","handleClose","hasValidationErrors","Dialog","maxWidth","fullWidth","DialogTitle","DialogContent","marginBottom","message","Button","onClick","prev","Grid","container","spacing","item","xs","TextField","onChange","e","handleNameChange","target","validateKubernetesName","placeholder","helperText","margin","required","disabled","InputLabelProps","select","MenuItem","multiline","rows","tag","onDelete","undefined","handleDeleteTag","tagToDelete","onKeyPress","key","SelectProps","handleContactEmailChange","handleDocsURLChange","handleOpenAPISpecChange","DialogActions","Error","selectedRouteNamespace","selectedRouteName","apiProduct","apiVersion","spec","group","contact","email","team","documentation","method","headers","body","JSON","stringify","ok","errorData","err","String","startIcon","CircularProgress","EditAPIProductDialog","setLoading","setTargetRef","saving","setSaving","then","res","status","data","catch","React","Progress","patch","ResourceList","alertApi","alertApiRef","identityApi","identityApiRef","userEntityRef","setUserEntityRef","createDialogOpen","setCreateDialogOpen","editDialogOpen","setEditDialogOpen","refreshTrigger","setRefreshTrigger","deleteDialogOpen","setDeleteDialogOpen","apiProductToDelete","setApiProductToDelete","apiProductToEdit","setApiProductToEdit","deleting","setDeleting","deleteStats","setDeleteStats","allowed","canCreateApiProduct","createPermissionLoading","createPermissionError","useKuadrantPermission","kuadrantApiProductCreatePermission","canDeleteOwnApiProduct","deleteOwnPermissionLoading","kuadrantApiProductDeleteOwnPermission","canDeleteAllApiProducts","deleteAllPermissionLoading","deletePermissionError","kuadrantApiProductDeleteAllPermission","canUpdateOwnApiProduct","kuadrantApiProductUpdateOwnPermission","canUpdateAllApiProducts","kuadrantApiProductUpdateAllPermission","deletePermissionLoading","canListPlanPolicies","planPolicyPermissionLoading","planPolicyPermissionError","kuadrantPlanPolicyListPermission","identity","getBackstageIdentity","apiProducts","apiProductsLoading","apiProductsError","planPoliciesLoading","permissionError","columns","title","field","render","row","isPublished","Link","to","span","className","customFilterAndSearch","term","toLowerCase","mode","authSchemes","discoveredAuthScheme","authentication","schemeObjects","Object","values","hasApiKey","some","scheme","hasOwnProperty","hasJwt","icon","VpnKeyIcon","LockIcon","fontStyle","formatDate","dateString","creationTimestamp","Date","toLocaleDateString","year","month","day","filtering","isOwner","canEdit","canDelete","IconButton","handleEditClick","EditIcon","fontSize","related","r","apiName","apiNamespace","approved","phase","requests","secrets","console","warn","handleDeleteClick","DeleteIcon","planPolicyColumns","Page","themeId","Header","subtitle","SupportButton","Content","ResponseErrorPanel","direction","InfoCard","action","alignItems","height","AddIcon","resources","Table","options","paging","pageSize","search","debounceInterval","toolbar","emptyRowsWhenPaging","renderPlanPolicies","post","ConfirmDeleteDialog","confirmText","onConfirm","onCancel","KuadrantPage","PermissionGate","permission","kuadrantApiProductListPermission","errorMessage","createPermission","attributes","kuadrantApiProductReadAllPermission","kuadrantApiKeyCreatePermission","resourceType","kuadrantApiKeyReadOwnPermission","kuadrantApiKeyUpdateOwnPermission","kuadrantApiKeyUpdateAllPermission","kuadrantApiKeyDeleteOwnPermission","kuadrantApiKeyDeleteAllPermission","kuadrantApiKeyApprovePermission","children","fallback","resourceRef","permissionRequest","result","usePermission","canDeleteResource","ownerId","currentUserId","canDeleteOwn","canDeleteAll","inputValue","setInputValue","requiresTextConfirmation","canConfirm","WarningIcon","DialogContentText","whiteSpace","autoFocus"],"sourceRoot":""}
|
|
@@ -1,2 +0,0 @@
|
|
|
1
|
-
"use strict";(self.webpackChunkinternal_plugin_kuadrant=self.webpackChunkinternal_plugin_kuadrant||[]).push([[8416],{5030:(e,t,n)=>{Object.defineProperty(t,"__esModule",{value:!0});var r=n(85608),a=n(95478),o=r.__importDefault(n(10009));t.default=function(e,t,n){void 0===t&&(t=[]),void 0===n&&(n={loading:!1});var i=a.useRef(0),l=o.default(),s=a.useState(n),c=s[0],d=s[1],p=a.useCallback(function(){for(var t=[],n=0;n<arguments.length;n++)t[n]=arguments[n];var a=++i.current;return c.loading||d(function(e){return r.__assign(r.__assign({},e),{loading:!0})}),e.apply(void 0,t).then(function(e){return l()&&a===i.current&&d({value:e,loading:!1}),e},function(e){return l()&&a===i.current&&d({error:e,loading:!1}),e})},t);return[c,p]}},10009:(e,t,n)=>{Object.defineProperty(t,"__esModule",{value:!0});var r=n(95478);t.default=function(){var e=r.useRef(!1),t=r.useCallback(function(){return e.current},[]);return r.useEffect(function(){return e.current=!0,function(){e.current=!1}},[]),t}},18466:(e,t,n)=>{var r=n(4293),a=n(78920);t.A=void 0;var o=a(n(95478)),i=(0,r(n(74044)).default)(o.createElement("path",{d:"M19 13h-6v6h-2v-6H5v-2h6V5h2v6h6v2z"}),"Add");t.A=i},21702:(e,t,n)=>{var r=n(4293),a=n(78920);t.A=void 0;var o=a(n(95478)),i=(0,r(n(74044)).default)(o.createElement("path",{d:"M12.65 10C11.83 7.67 9.61 6 7 6c-3.31 0-6 2.69-6 6s2.69 6 6 6c2.61 0 4.83-1.67 5.65-4H17v4h4v-4h2v-4H12.65zM7 14c-1.1 0-2-.9-2-2s.9-2 2-2 2 .9 2 2-.9 2-2 2z"}),"VpnKey");t.A=i},27799:(e,t,n)=>{var r=n(4293),a=n(78920);t.A=void 0;var o=a(n(95478)),i=(0,r(n(74044)).default)(o.createElement("path",{d:"M21 5v6.59l-3-3.01-4 4.01-4-4-4 4-3-3.01V5c0-1.1.9-2 2-2h14c1.1 0 2 .9 2 2zm-3 6.42l3 3.01V19c0 1.1-.9 2-2 2H5c-1.1 0-2-.9-2-2v-6.58l3 2.99 4-4 4 4 4-3.99z"}),"BrokenImage");t.A=i},39590:(e,t,n)=>{var r=n(4293),a=n(78920);t.A=void 0;var o=a(n(95478)),i=(0,r(n(74044)).default)(o.createElement("path",{d:"M6 19c0 1.1.9 2 2 2h8c1.1 0 2-.9 2-2V7H6v12zM19 4h-3.5l-1-1h-5l-1 1H5v2h14V4z"}),"Delete");t.A=i},45210:(e,t,n)=>{n.d(t,{Y:()=>I});var r=n(31085),a=n(22097),o=n(10394),i=n(64947),l=n(93453),s=n(29365),c=n(46423),d=n(5951),p=n(26343),u=n(28233),h=n(55197),m=n(37976),f=n(72501),g=n(5893),v=n(95478),A=(n(45250),n(42469),n(16098)),x=n(67550);const y=()=>{const{t:e}=(0,x.i)(A.O);return{url:"https://github.com/backstage/backstage/issues",items:[{title:e("supportConfig.default.title"),icon:"warning",links:[{title:e("supportConfig.default.linkTitle"),url:"https://github.com/backstage/backstage/blob/master/app-config.yaml"}]}]}};var b=n(27799);function C(e){const{id:t,Fallback:n=b.A,...o}=e,i=(0,a.useApp)().getSystemIcon(t)??n;return(0,r.jsx)(i,{...o})}function k(e){return(0,r.jsx)(C,{id:"help",...e})}var j=n(37725);const S=(0,m.makeStyles)({popoverList:{minWidth:260,maxWidth:400},menuItem:{whiteSpace:"normal"}},{name:"BackstageSupportButton"}),z=({icon:e})=>{const t=(0,a.useApp)(),n=e?t.getSystemIcon(e)??k:k;return(0,r.jsx)(n,{})},E=({link:e})=>(0,r.jsx)(j.N_,{to:e.url,children:e.title??e.url}),M=({item:e})=>(0,r.jsxs)(p.A,{button:!1,children:[(0,r.jsx)(c.A,{children:(0,r.jsx)(z,{icon:e.icon})}),(0,r.jsx)(d.A,{primary:e.title,secondary:e.links?.reduce((e,t,n)=>[...e,n>0&&(0,r.jsx)("br",{},n),(0,r.jsx)(E,{link:t},t.url)],[])})]});function I(e){const{t}=(0,x.i)(A.O),{title:n,items:c,children:d}=e,{items:m}=function(){const e=(0,a.useApiHolder)().get(a.configApiRef),t=e?.getOptionalConfig("app.support"),n=y();return t?{url:t.getString("url"),items:t.getConfigArray("items").flatMap(e=>({title:e.getString("title"),icon:e.getOptionalString("icon"),links:(e.getOptionalConfigArray("links")??[]).flatMap(e=>({url:e.getString("url"),title:e.getOptionalString("title")??""}))}))}:n}(),[b,C]=(0,v.useState)(!1),[j,z]=(0,v.useState)(null),E=S(),I=(0,a.useApi)(a.configApiRef).getOptionalConfig("app.support"),N=(0,g.A)(e=>e.breakpoints.down("sm")),w=e=>{z(e.currentTarget),C(!0)},L=()=>{C(!1)};return I?(0,r.jsxs)(r.Fragment,{children:[(0,r.jsx)(o.A,{display:"flex",ml:1,children:N?(0,r.jsx)(s.A,{color:"primary",size:"small",onClick:w,"data-testid":"support-button","aria-label":"Support",children:(0,r.jsx)(k,{})}):(0,r.jsx)(i.A,{"data-testid":"support-button","aria-label":"Support",color:"primary",onClick:w,startIcon:(0,r.jsx)(k,{}),children:t("supportButton.title")})}),(0,r.jsxs)(h.Ay,{"data-testid":"support-button-popover",open:b,anchorEl:j,anchorOrigin:{vertical:"bottom",horizontal:"right"},transformOrigin:{vertical:"top",horizontal:"right"},onClose:L,children:[(0,r.jsxs)(u.A,{className:E.popoverList,autoFocusItem:Boolean(j),children:[n&&(0,r.jsx)(p.A,{button:!1,alignItems:"flex-start",className:E.menuItem,children:(0,r.jsx)(f.A,{variant:"subtitle1",children:n})}),v.Children.map(d,(e,t)=>(0,r.jsx)(p.A,{button:!1,alignItems:"flex-start",className:E.menuItem,children:e},`child-${t}`)),(c??m).map((e,t)=>(0,r.jsx)(M,{item:e},`item-${t}`))]}),(0,r.jsx)(l.A,{children:(0,r.jsx)(i.A,{color:"primary",onClick:L,"aria-label":"Close",children:t("supportButton.close")})})]})]}):null}},52235:(e,t,n)=>{var r=n(4293),a=n(78920);t.A=void 0;var o=a(n(95478)),i=(0,r(n(74044)).default)(o.createElement("path",{d:"M12 4l-1.41 1.41L16.17 11H4v2h12.17l-5.58 5.59L12 20l8-8z"}),"ArrowForward");t.A=i},77225:(e,t,n)=>{var r=n(4293),a=n(78920);t.A=void 0;var o=a(n(95478)),i=(0,r(n(74044)).default)(o.createElement("path",{d:"M1 21h22L12 2 1 21zm12-3h-2v-2h2v2zm0-4h-2v-4h2v4z"}),"Warning");t.A=i},84441:(e,t,n)=>{n.d(t,{A:()=>b});var r=n(39850),a=n(89575),o=n(95478);function i(e){var t,n,r="";if("string"==typeof e||"number"==typeof e)r+=e;else if("object"==typeof e)if(Array.isArray(e))for(t=0;t<e.length;t++)e[t]&&(n=i(e[t]))&&(r&&(r+=" "),r+=n);else for(t in e)e[t]&&(r&&(r+=" "),r+=t);return r}const l=function(){for(var e,t,n=0,r="";n<arguments.length;)(e=arguments[n++])&&(t=i(e))&&(r&&(r+=" "),r+=t);return r};var s=n(37976),c=n(4321),d=n(38483);const p=(0,d.A)(o.createElement("path",{d:"M20,12A8,8 0 0,1 12,20A8,8 0 0,1 4,12A8,8 0 0,1 12,4C12.76,4 13.5,4.11 14.2, 4.31L15.77,2.74C14.61,2.26 13.34,2 12,2A10,10 0 0,0 2,12A10,10 0 0,0 12,22A10,10 0 0, 0 22,12M7.91,10.08L6.5,11.5L11,16L21,6L19.59,4.58L11,13.17L7.91,10.08Z"}),"SuccessOutlined"),u=(0,d.A)(o.createElement("path",{d:"M12 5.99L19.53 19H4.47L12 5.99M12 2L1 21h22L12 2zm1 14h-2v2h2v-2zm0-6h-2v4h2v-4z"}),"ReportProblemOutlined"),h=(0,d.A)(o.createElement("path",{d:"M11 15h2v2h-2zm0-8h2v6h-2zm.99-5C6.47 2 2 6.48 2 12s4.47 10 9.99 10C17.52 22 22 17.52 22 12S17.52 2 11.99 2zM12 20c-4.42 0-8-3.58-8-8s3.58-8 8-8 8 3.58 8 8-3.58 8-8 8z"}),"ErrorOutline"),m=(0,d.A)(o.createElement("path",{d:"M11,9H13V7H11M12,20C7.59,20 4,16.41 4,12C4,7.59 7.59,4 12,4C16.41,4 20,7.59 20, 12C20,16.41 16.41,20 12,20M12,2A10,10 0 0,0 2,12A10,10 0 0,0 12,22A10,10 0 0,0 22,12A10, 10 0 0,0 12,2M11,17H13V11H11V17Z"}),"InfoOutlined"),f=(0,d.A)(o.createElement("path",{d:"M19 6.41L17.59 5 12 10.59 6.41 5 5 6.41 10.59 12 5 17.59 6.41 19 12 13.41 17.59 19 19 17.59 13.41 12z"}),"Close");var g=n(29365),v=n(11192),A={success:o.createElement(p,{fontSize:"inherit"}),warning:o.createElement(u,{fontSize:"inherit"}),error:o.createElement(h,{fontSize:"inherit"}),info:o.createElement(m,{fontSize:"inherit"})},x=o.createElement(f,{fontSize:"small"}),y=o.forwardRef(function(e,t){var n=e.action,i=e.children,s=e.classes,d=e.className,p=e.closeText,u=void 0===p?"Close":p,h=e.color,m=e.icon,f=e.iconMapping,y=void 0===f?A:f,b=e.onClose,C=e.role,k=void 0===C?"alert":C,j=e.severity,S=void 0===j?"success":j,z=e.variant,E=void 0===z?"standard":z,M=(0,r.A)(e,["action","children","classes","className","closeText","color","icon","iconMapping","onClose","role","severity","variant"]);return o.createElement(c.A,(0,a.A)({role:k,square:!0,elevation:0,className:l(s.root,s["".concat(E).concat((0,v.A)(h||S))],d),ref:t},M),!1!==m?o.createElement("div",{className:s.icon},m||y[S]||A[S]):null,o.createElement("div",{className:s.message},i),null!=n?o.createElement("div",{className:s.action},n):null,null==n&&b?o.createElement("div",{className:s.action},o.createElement(g.A,{size:"small","aria-label":u,title:u,color:"inherit",onClick:b},x)):null)});const b=(0,s.withStyles)(function(e){var t="light"===e.palette.type?s.darken:s.lighten,n="light"===e.palette.type?s.lighten:s.darken;return{root:(0,a.A)({},e.typography.body2,{borderRadius:e.shape.borderRadius,backgroundColor:"transparent",display:"flex",padding:"6px 16px"}),standardSuccess:{color:t(e.palette.success.main,.6),backgroundColor:n(e.palette.success.main,.9),"& $icon":{color:e.palette.success.main}},standardInfo:{color:t(e.palette.info.main,.6),backgroundColor:n(e.palette.info.main,.9),"& $icon":{color:e.palette.info.main}},standardWarning:{color:t(e.palette.warning.main,.6),backgroundColor:n(e.palette.warning.main,.9),"& $icon":{color:e.palette.warning.main}},standardError:{color:t(e.palette.error.main,.6),backgroundColor:n(e.palette.error.main,.9),"& $icon":{color:e.palette.error.main}},outlinedSuccess:{color:t(e.palette.success.main,.6),border:"1px solid ".concat(e.palette.success.main),"& $icon":{color:e.palette.success.main}},outlinedInfo:{color:t(e.palette.info.main,.6),border:"1px solid ".concat(e.palette.info.main),"& $icon":{color:e.palette.info.main}},outlinedWarning:{color:t(e.palette.warning.main,.6),border:"1px solid ".concat(e.palette.warning.main),"& $icon":{color:e.palette.warning.main}},outlinedError:{color:t(e.palette.error.main,.6),border:"1px solid ".concat(e.palette.error.main),"& $icon":{color:e.palette.error.main}},filledSuccess:{color:"#fff",fontWeight:e.typography.fontWeightMedium,backgroundColor:e.palette.success.main},filledInfo:{color:"#fff",fontWeight:e.typography.fontWeightMedium,backgroundColor:e.palette.info.main},filledWarning:{color:"#fff",fontWeight:e.typography.fontWeightMedium,backgroundColor:e.palette.warning.main},filledError:{color:"#fff",fontWeight:e.typography.fontWeightMedium,backgroundColor:e.palette.error.main},icon:{marginRight:12,padding:"7px 0",display:"flex",fontSize:22,opacity:.9},message:{padding:"8px 0"},action:{display:"flex",alignItems:"center",marginLeft:"auto",paddingLeft:16,marginRight:-8}}},{name:"MuiAlert"})(y)},85142:(e,t,n)=>{var r=n(4293),a=n(78920);t.A=void 0;var o=a(n(95478)),i=(0,r(n(74044)).default)(o.createElement("path",{d:"M18 8h-1V6c0-2.76-2.24-5-5-5S7 3.24 7 6v2H6c-1.1 0-2 .9-2 2v10c0 1.1.9 2 2 2h12c1.1 0 2-.9 2-2V10c0-1.1-.9-2-2-2zm-6 9c-1.1 0-2-.9-2-2s.9-2 2-2 2 .9 2 2-.9 2-2 2zm3.1-9H8.9V6c0-1.71 1.39-3.1 3.1-3.1 1.71 0 3.1 1.39 3.1 3.1v2z"}),"Lock");t.A=i},91638:(e,t,n)=>{var r=n(85608),a=n(95478),o=r.__importDefault(n(5030));t.A=function(e,t){void 0===t&&(t=[]);var n=o.default(e,t,{loading:!0}),r=n[0],i=n[1];return a.useEffect(function(){i()},[i]),r}},96040:(e,t,n)=>{n.d(t,{n:()=>N});var r=n(31085),a=n(40703),o=n(59469),i=n(48653),l=n(45685),s=n(37197),c=n(37976),d=n(53373),p=n.n(d),u=n(10394),h=n(72501),m=n(52235),f=n(37725);const g=(0,c.makeStyles)(e=>({root:{maxWidth:"fit-content",padding:e.spacing(2,2,2,2.5)},boxTitle:{margin:0,color:e.palette.textSubtle},arrow:{color:e.palette.textSubtle}}),{name:"BackstageBottomLink"});function v(e){const{link:t,title:n,onClick:a}=e,o=g();return(0,r.jsxs)(u.A,{children:[(0,r.jsx)(s.A,{}),(0,r.jsx)(f.N_,{to:t,onClick:a,underline:"none",children:(0,r.jsxs)(u.A,{display:"flex",alignItems:"center",className:o.root,children:[(0,r.jsx)(u.A,{className:o.boxTitle,fontWeight:"fontWeightBold",m:1,children:(0,r.jsx)(h.A,{children:(0,r.jsx)("strong",{children:n})})}),(0,r.jsx)(m.A,{className:o.arrow})]})})]})}var A=n(95478),x=n(64947);const y=(0,A.forwardRef)((e,t)=>(0,r.jsx)(f.N_,{ref:t,...e,color:"initial"})),b=(0,A.forwardRef)((e,t)=>(0,r.jsx)(x.A,{ref:t,component:y,...e}));var C=n(34169),k=n(16098),j=n(67550);const S=e=>{const{slackChannel:t}=e,{t:n}=(0,j.i)(k.O);return t?"string"==typeof t?(0,r.jsx)(h.A,{children:n("errorBoundary.title",{slackChannel:t})}):t.href?(0,r.jsx)(b,{to:t.href,variant:"contained",children:t.name}):(0,r.jsx)(h.A,{children:n("errorBoundary.title",{slackChannel:t.name})}):null},z=class extends A.Component{constructor(e){super(e),this.state={error:void 0,errorInfo:void 0}}componentDidCatch(e,t){console.error(`ErrorBoundary, error: ${e}`,{error:e,errorInfo:t}),this.setState({error:e,errorInfo:t})}render(){const{slackChannel:e,children:t}=this.props,{error:n}=this.state;return n?(0,r.jsx)(C.b,{title:"Something Went Wrong",error:n,children:(0,r.jsx)(S,{slackChannel:e})}):t}},E=(0,c.makeStyles)(e=>({noPadding:{padding:0,"&:last-child":{paddingBottom:0}},contentAlignBottom:{display:"flex",alignItems:"self-end"},header:{padding:e.spacing(2,2,2,2.5)},headerTitle:{fontWeight:e.typography.fontWeightBold},headerSubheader:{paddingTop:e.spacing(1)},headerAvatar:{},headerAction:{},headerContent:{},subheader:{display:"flex"}}),{name:"BackstageInfoCard"}),M=(0,c.withStyles)(e=>({root:{display:"inline-block",padding:e.spacing(8,8,0,0),float:"right"}}),{name:"BackstageInfoCardCardActionsTopRight"})(o.A),I={card:{flex:{display:"flex",flexDirection:"column"},fullHeight:{display:"flex",flexDirection:"column",height:"100%"},gridItem:{display:"flex",flexDirection:"column",height:"calc(100% - 10px)",marginBottom:"10px",breakInside:"avoid-page","@media print":{height:"auto"}}},cardContent:{fullHeight:{flex:1},gridItem:{flex:1}}};function N(e){const{title:t,subheader:n,divider:c=!0,deepLink:d,slackChannel:u,errorBoundaryProps:h,variant:m,alignContent:f="normal",children:g,headerStyle:A,headerProps:x,icon:y,action:b,actionsClassName:C,actions:k,cardClassName:j,actionsTopRight:S,className:N,noPadding:w,titleTypographyProps:L,subheaderTypographyProps:W}=e,B=E();let _={},H={};m&&m.split(/[\s]+/g).forEach(e=>{_={..._,...I.card[e]},H={...H,...I.cardContent[e]}});const O=h||(u?{slackChannel:u}:{});return(0,r.jsx)(a.A,{style:_,className:N,children:(0,r.jsxs)(z,{...O,children:[t&&(0,r.jsx)(l.A,{classes:{root:p()(B.header),title:B.headerTitle,subheader:B.headerSubheader,avatar:B.headerAvatar,action:B.headerAction,content:B.headerContent},title:t,subheader:n||y?(0,r.jsxs)("div",{"data-testid":"info-card-subheader",children:[n&&(0,r.jsx)("div",{className:B.subheader,children:n}),y]}):null,action:b,style:{...A},titleTypographyProps:L,subheaderTypographyProps:W,...x}),S&&(0,r.jsx)(M,{children:S}),c&&(0,r.jsx)(s.A,{}),(0,r.jsx)(i.A,{className:p()(j,{[B.noPadding]:w,[B.contentAlignBottom]:"bottom"===f}),style:H,children:g}),k&&(0,r.jsx)(o.A,{className:C,children:k}),d&&(0,r.jsx)(v,{...d})]})})}}}]);
|
|
2
|
-
//# sourceMappingURL=8416.3604a311.chunk.js.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"static/8416.3604a311.chunk.js","mappings":"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","sources":["webpack://internal.plugin-kuadrant/../../node_modules/react-use/lib/useAsyncFn.js","webpack://internal.plugin-kuadrant/../../node_modules/react-use/lib/useMountedState.js","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/icons/Add.js","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/icons/VpnKey.js","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/icons/BrokenImage.js","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/icons/Delete.js","webpack://internal.plugin-kuadrant/./node_modules/@backstage/core-components/dist/hooks/useSupportConfig.esm.js","webpack://internal.plugin-kuadrant/./node_modules/@backstage/core-components/dist/icons/icons.esm.js","webpack://internal.plugin-kuadrant/./node_modules/@backstage/core-components/dist/components/SupportButton/SupportButton.esm.js","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/icons/ArrowForward.js","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/icons/Warning.js","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/lab/node_modules/clsx/dist/clsx.m.js","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/lab/esm/internal/svg-icons/SuccessOutlined.js","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/lab/esm/internal/svg-icons/ReportProblemOutlined.js","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/lab/esm/internal/svg-icons/ErrorOutline.js","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/lab/esm/internal/svg-icons/InfoOutlined.js","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/lab/esm/internal/svg-icons/Close.js","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/lab/esm/Alert/Alert.js","webpack://internal.plugin-kuadrant/../../node_modules/@material-ui/icons/Lock.js","webpack://internal.plugin-kuadrant/../../node_modules/react-use/lib/useAsync.js","webpack://internal.plugin-kuadrant/./node_modules/@backstage/core-components/dist/layout/BottomLink/BottomLink.esm.js","webpack://internal.plugin-kuadrant/./node_modules/@backstage/core-components/dist/components/LinkButton/LinkButton.esm.js","webpack://internal.plugin-kuadrant/./node_modules/@backstage/core-components/dist/layout/ErrorBoundary/ErrorBoundary.esm.js","webpack://internal.plugin-kuadrant/./node_modules/@backstage/core-components/dist/layout/InfoCard/InfoCard.esm.js"],"sourcesContent":["\"use strict\";\nObject.defineProperty(exports, \"__esModule\", { value: true });\nvar tslib_1 = require(\"tslib\");\nvar react_1 = require(\"react\");\nvar useMountedState_1 = tslib_1.__importDefault(require(\"./useMountedState\"));\nfunction useAsyncFn(fn, deps, initialState) {\n if (deps === void 0) { deps = []; }\n if (initialState === void 0) { initialState = { loading: false }; }\n var lastCallId = react_1.useRef(0);\n var isMounted = useMountedState_1.default();\n var _a = react_1.useState(initialState), state = _a[0], set = _a[1];\n var callback = react_1.useCallback(function () {\n var args = [];\n for (var _i = 0; _i < arguments.length; _i++) {\n args[_i] = arguments[_i];\n }\n var callId = ++lastCallId.current;\n if (!state.loading) {\n set(function (prevState) { return (tslib_1.__assign(tslib_1.__assign({}, prevState), { loading: true })); });\n }\n return fn.apply(void 0, args).then(function (value) {\n isMounted() && callId === lastCallId.current && set({ value: value, loading: false });\n return value;\n }, function (error) {\n isMounted() && callId === lastCallId.current && set({ error: error, loading: false });\n return error;\n });\n }, deps);\n return [state, callback];\n}\nexports.default = useAsyncFn;\n","\"use strict\";\nObject.defineProperty(exports, \"__esModule\", { value: true });\nvar react_1 = require(\"react\");\nfunction useMountedState() {\n var mountedRef = react_1.useRef(false);\n var get = react_1.useCallback(function () { return mountedRef.current; }, []);\n react_1.useEffect(function () {\n mountedRef.current = true;\n return function () {\n mountedRef.current = false;\n };\n }, []);\n return get;\n}\nexports.default = useMountedState;\n","\"use strict\";\n\nvar _interopRequireDefault = require(\"@babel/runtime/helpers/interopRequireDefault\");\n\nvar _interopRequireWildcard = require(\"@babel/runtime/helpers/interopRequireWildcard\");\n\nObject.defineProperty(exports, \"__esModule\", {\n value: true\n});\nexports.default = void 0;\n\nvar React = _interopRequireWildcard(require(\"react\"));\n\nvar _createSvgIcon = _interopRequireDefault(require(\"./utils/createSvgIcon\"));\n\nvar _default = (0, _createSvgIcon.default)( /*#__PURE__*/React.createElement(\"path\", {\n d: \"M19 13h-6v6h-2v-6H5v-2h6V5h2v6h6v2z\"\n}), 'Add');\n\nexports.default = _default;","\"use strict\";\n\nvar _interopRequireDefault = require(\"@babel/runtime/helpers/interopRequireDefault\");\n\nvar _interopRequireWildcard = require(\"@babel/runtime/helpers/interopRequireWildcard\");\n\nObject.defineProperty(exports, \"__esModule\", {\n value: true\n});\nexports.default = void 0;\n\nvar React = _interopRequireWildcard(require(\"react\"));\n\nvar _createSvgIcon = _interopRequireDefault(require(\"./utils/createSvgIcon\"));\n\nvar _default = (0, _createSvgIcon.default)( /*#__PURE__*/React.createElement(\"path\", {\n d: \"M12.65 10C11.83 7.67 9.61 6 7 6c-3.31 0-6 2.69-6 6s2.69 6 6 6c2.61 0 4.83-1.67 5.65-4H17v4h4v-4h2v-4H12.65zM7 14c-1.1 0-2-.9-2-2s.9-2 2-2 2 .9 2 2-.9 2-2 2z\"\n}), 'VpnKey');\n\nexports.default = _default;","\"use strict\";\n\nvar _interopRequireDefault = require(\"@babel/runtime/helpers/interopRequireDefault\");\n\nvar _interopRequireWildcard = require(\"@babel/runtime/helpers/interopRequireWildcard\");\n\nObject.defineProperty(exports, \"__esModule\", {\n value: true\n});\nexports.default = void 0;\n\nvar React = _interopRequireWildcard(require(\"react\"));\n\nvar _createSvgIcon = _interopRequireDefault(require(\"./utils/createSvgIcon\"));\n\nvar _default = (0, _createSvgIcon.default)( /*#__PURE__*/React.createElement(\"path\", {\n d: \"M21 5v6.59l-3-3.01-4 4.01-4-4-4 4-3-3.01V5c0-1.1.9-2 2-2h14c1.1 0 2 .9 2 2zm-3 6.42l3 3.01V19c0 1.1-.9 2-2 2H5c-1.1 0-2-.9-2-2v-6.58l3 2.99 4-4 4 4 4-3.99z\"\n}), 'BrokenImage');\n\nexports.default = _default;","\"use strict\";\n\nvar _interopRequireDefault = require(\"@babel/runtime/helpers/interopRequireDefault\");\n\nvar _interopRequireWildcard = require(\"@babel/runtime/helpers/interopRequireWildcard\");\n\nObject.defineProperty(exports, \"__esModule\", {\n value: true\n});\nexports.default = void 0;\n\nvar React = _interopRequireWildcard(require(\"react\"));\n\nvar _createSvgIcon = _interopRequireDefault(require(\"./utils/createSvgIcon\"));\n\nvar _default = (0, _createSvgIcon.default)( /*#__PURE__*/React.createElement(\"path\", {\n d: \"M6 19c0 1.1.9 2 2 2h8c1.1 0 2-.9 2-2V7H6v12zM19 4h-3.5l-1-1h-5l-1 1H5v2h14V4z\"\n}), 'Delete');\n\nexports.default = _default;","import { useApiHolder, configApiRef } from '@backstage/core-plugin-api';\nimport { coreComponentsTranslationRef } from '../translation.esm.js';\nimport { useTranslationRef } from '@backstage/core-plugin-api/alpha';\n\nconst useDefaultSupportConfig = () => {\n const { t } = useTranslationRef(coreComponentsTranslationRef);\n return {\n url: \"https://github.com/backstage/backstage/issues\",\n items: [\n {\n title: t(\"supportConfig.default.title\"),\n icon: \"warning\",\n links: [\n {\n // TODO: Update to dedicated support page on backstage.io/docs\n title: t(\"supportConfig.default.linkTitle\"),\n url: \"https://github.com/backstage/backstage/blob/master/app-config.yaml\"\n }\n ]\n }\n ]\n };\n};\nfunction useSupportConfig() {\n const apiHolder = useApiHolder();\n const config = apiHolder.get(configApiRef);\n const supportConfig = config?.getOptionalConfig(\"app.support\");\n const defaultSupportConfig = useDefaultSupportConfig();\n if (!supportConfig) {\n return defaultSupportConfig;\n }\n return {\n url: supportConfig.getString(\"url\"),\n items: supportConfig.getConfigArray(\"items\").flatMap((itemConf) => ({\n title: itemConf.getString(\"title\"),\n icon: itemConf.getOptionalString(\"icon\"),\n links: (itemConf.getOptionalConfigArray(\"links\") ?? []).flatMap(\n (linkConf) => ({\n url: linkConf.getString(\"url\"),\n title: linkConf.getOptionalString(\"title\") ?? \"\"\n })\n )\n }))\n };\n}\n\nexport { useSupportConfig };\n//# sourceMappingURL=useSupportConfig.esm.js.map\n","import { jsx } from 'react/jsx-runtime';\nimport { useApp } from '@backstage/core-plugin-api';\nimport MuiBrokenImageIcon from '@material-ui/icons/BrokenImage';\n\nfunction AppIcon(props) {\n const { id: key, Fallback = MuiBrokenImageIcon, ...rest } = props;\n const app = useApp();\n const Icon = app.getSystemIcon(key) ?? Fallback;\n return /* @__PURE__ */ jsx(Icon, { ...rest });\n}\nfunction BrokenImageIcon(props) {\n return /* @__PURE__ */ jsx(AppIcon, { id: \"brokenImage\", ...props });\n}\nfunction CatalogIcon(props) {\n return /* @__PURE__ */ jsx(AppIcon, { id: \"catalog\", ...props });\n}\nfunction ChatIcon(props) {\n return /* @__PURE__ */ jsx(AppIcon, { id: \"chat\", ...props });\n}\nfunction DashboardIcon(props) {\n return /* @__PURE__ */ jsx(AppIcon, { id: \"dashboard\", ...props });\n}\nfunction DocsIcon(props) {\n return /* @__PURE__ */ jsx(AppIcon, { id: \"docs\", ...props });\n}\nfunction EmailIcon(props) {\n return /* @__PURE__ */ jsx(AppIcon, { id: \"email\", ...props });\n}\nfunction GitHubIcon(props) {\n return /* @__PURE__ */ jsx(AppIcon, { id: \"github\", ...props });\n}\nfunction GroupIcon(props) {\n return /* @__PURE__ */ jsx(AppIcon, { id: \"group\", ...props });\n}\nfunction HelpIcon(props) {\n return /* @__PURE__ */ jsx(AppIcon, { id: \"help\", ...props });\n}\nfunction UserIcon(props) {\n return /* @__PURE__ */ jsx(AppIcon, { id: \"user\", ...props });\n}\nfunction WarningIcon(props) {\n return /* @__PURE__ */ jsx(AppIcon, { id: \"warning\", ...props });\n}\nfunction StarIcon(props) {\n return /* @__PURE__ */ jsx(AppIcon, { id: \"star\", ...props });\n}\nfunction UnstarredIcon(props) {\n return /* @__PURE__ */ jsx(AppIcon, { id: \"unstarred\", ...props });\n}\n\nexport { AppIcon, BrokenImageIcon, CatalogIcon, ChatIcon, DashboardIcon, DocsIcon, EmailIcon, GitHubIcon, GroupIcon, HelpIcon, StarIcon, UnstarredIcon, UserIcon, WarningIcon };\n//# sourceMappingURL=icons.esm.js.map\n","import { jsxs, Fragment, jsx } from 'react/jsx-runtime';\nimport { useApi, configApiRef, useApp } from '@backstage/core-plugin-api';\nimport Box from '@material-ui/core/Box';\nimport Button from '@material-ui/core/Button';\nimport DialogActions from '@material-ui/core/DialogActions';\nimport IconButton from '@material-ui/core/IconButton';\nimport ListItemIcon from '@material-ui/core/ListItemIcon';\nimport ListItemText from '@material-ui/core/ListItemText';\nimport MenuItem from '@material-ui/core/MenuItem';\nimport MenuList from '@material-ui/core/MenuList';\nimport Popover from '@material-ui/core/Popover';\nimport { makeStyles } from '@material-ui/core/styles';\nimport Typography from '@material-ui/core/Typography';\nimport useMediaQuery from '@material-ui/core/useMediaQuery';\nimport { useState, Children } from 'react';\nimport 'lodash';\nimport 'qs';\nimport 'react-router-dom';\nimport '@react-hookz/web';\nimport { useSupportConfig } from '../../hooks/useSupportConfig.esm.js';\nimport { HelpIcon } from '../../icons/icons.esm.js';\nimport { Link } from '../Link/Link.esm.js';\nimport { coreComponentsTranslationRef } from '../../translation.esm.js';\nimport { useTranslationRef } from '@backstage/core-plugin-api/alpha';\n\nconst useStyles = makeStyles(\n {\n popoverList: {\n minWidth: 260,\n maxWidth: 400\n },\n menuItem: {\n whiteSpace: \"normal\"\n }\n },\n { name: \"BackstageSupportButton\" }\n);\nconst SupportIcon = ({ icon }) => {\n const app = useApp();\n const Icon = icon ? app.getSystemIcon(icon) ?? HelpIcon : HelpIcon;\n return /* @__PURE__ */ jsx(Icon, {});\n};\nconst SupportLink = ({ link }) => /* @__PURE__ */ jsx(Link, { to: link.url, children: link.title ?? link.url });\nconst SupportListItem = ({ item }) => {\n return /* @__PURE__ */ jsxs(MenuItem, { button: false, children: [\n /* @__PURE__ */ jsx(ListItemIcon, { children: /* @__PURE__ */ jsx(SupportIcon, { icon: item.icon }) }),\n /* @__PURE__ */ jsx(\n ListItemText,\n {\n primary: item.title,\n secondary: item.links?.reduce(\n (prev, link, idx) => [\n ...prev,\n idx > 0 && /* @__PURE__ */ jsx(\"br\", {}, idx),\n /* @__PURE__ */ jsx(SupportLink, { link }, link.url)\n ],\n []\n )\n }\n )\n ] });\n};\nfunction SupportButton(props) {\n const { t } = useTranslationRef(coreComponentsTranslationRef);\n const { title, items, children } = props;\n const { items: configItems } = useSupportConfig();\n const [popoverOpen, setPopoverOpen] = useState(false);\n const [anchorEl, setAnchorEl] = useState(null);\n const classes = useStyles();\n const supportConfig = useApi(configApiRef).getOptionalConfig(\"app.support\");\n const isSmallScreen = useMediaQuery(\n (theme) => theme.breakpoints.down(\"sm\")\n );\n const onClickHandler = (event) => {\n setAnchorEl(event.currentTarget);\n setPopoverOpen(true);\n };\n const popoverCloseHandler = () => {\n setPopoverOpen(false);\n };\n if (!supportConfig) {\n return null;\n }\n return /* @__PURE__ */ jsxs(Fragment, { children: [\n /* @__PURE__ */ jsx(Box, { display: \"flex\", ml: 1, children: isSmallScreen ? /* @__PURE__ */ jsx(\n IconButton,\n {\n color: \"primary\",\n size: \"small\",\n onClick: onClickHandler,\n \"data-testid\": \"support-button\",\n \"aria-label\": \"Support\",\n children: /* @__PURE__ */ jsx(HelpIcon, {})\n }\n ) : /* @__PURE__ */ jsx(\n Button,\n {\n \"data-testid\": \"support-button\",\n \"aria-label\": \"Support\",\n color: \"primary\",\n onClick: onClickHandler,\n startIcon: /* @__PURE__ */ jsx(HelpIcon, {}),\n children: t(\"supportButton.title\")\n }\n ) }),\n /* @__PURE__ */ jsxs(\n Popover,\n {\n \"data-testid\": \"support-button-popover\",\n open: popoverOpen,\n anchorEl,\n anchorOrigin: {\n vertical: \"bottom\",\n horizontal: \"right\"\n },\n transformOrigin: {\n vertical: \"top\",\n horizontal: \"right\"\n },\n onClose: popoverCloseHandler,\n children: [\n /* @__PURE__ */ jsxs(\n MenuList,\n {\n className: classes.popoverList,\n autoFocusItem: Boolean(anchorEl),\n children: [\n title && /* @__PURE__ */ jsx(\n MenuItem,\n {\n button: false,\n alignItems: \"flex-start\",\n className: classes.menuItem,\n children: /* @__PURE__ */ jsx(Typography, { variant: \"subtitle1\", children: title })\n }\n ),\n Children.map(children, (child, i) => /* @__PURE__ */ jsx(\n MenuItem,\n {\n button: false,\n alignItems: \"flex-start\",\n className: classes.menuItem,\n children: child\n },\n `child-${i}`\n )),\n (items ?? configItems).map((item, i) => /* @__PURE__ */ jsx(SupportListItem, { item }, `item-${i}`))\n ]\n }\n ),\n /* @__PURE__ */ jsx(DialogActions, { children: /* @__PURE__ */ jsx(\n Button,\n {\n color: \"primary\",\n onClick: popoverCloseHandler,\n \"aria-label\": \"Close\",\n children: t(\"supportButton.close\")\n }\n ) })\n ]\n }\n )\n ] });\n}\n\nexport { SupportButton };\n//# sourceMappingURL=SupportButton.esm.js.map\n","\"use strict\";\n\nvar _interopRequireDefault = require(\"@babel/runtime/helpers/interopRequireDefault\");\n\nvar _interopRequireWildcard = require(\"@babel/runtime/helpers/interopRequireWildcard\");\n\nObject.defineProperty(exports, \"__esModule\", {\n value: true\n});\nexports.default = void 0;\n\nvar React = _interopRequireWildcard(require(\"react\"));\n\nvar _createSvgIcon = _interopRequireDefault(require(\"./utils/createSvgIcon\"));\n\nvar _default = (0, _createSvgIcon.default)( /*#__PURE__*/React.createElement(\"path\", {\n d: \"M12 4l-1.41 1.41L16.17 11H4v2h12.17l-5.58 5.59L12 20l8-8z\"\n}), 'ArrowForward');\n\nexports.default = _default;","\"use strict\";\n\nvar _interopRequireDefault = require(\"@babel/runtime/helpers/interopRequireDefault\");\n\nvar _interopRequireWildcard = require(\"@babel/runtime/helpers/interopRequireWildcard\");\n\nObject.defineProperty(exports, \"__esModule\", {\n value: true\n});\nexports.default = void 0;\n\nvar React = _interopRequireWildcard(require(\"react\"));\n\nvar _createSvgIcon = _interopRequireDefault(require(\"./utils/createSvgIcon\"));\n\nvar _default = (0, _createSvgIcon.default)( /*#__PURE__*/React.createElement(\"path\", {\n d: \"M1 21h22L12 2 1 21zm12-3h-2v-2h2v2zm0-4h-2v-4h2v4z\"\n}), 'Warning');\n\nexports.default = _default;","function r(e){var t,f,n=\"\";if(\"string\"==typeof e||\"number\"==typeof e)n+=e;else if(\"object\"==typeof e)if(Array.isArray(e))for(t=0;t<e.length;t++)e[t]&&(f=r(e[t]))&&(n&&(n+=\" \"),n+=f);else for(t in e)e[t]&&(n&&(n+=\" \"),n+=t);return n}export function clsx(){for(var e,t,f=0,n=\"\";f<arguments.length;)(e=arguments[f++])&&(t=r(e))&&(n&&(n+=\" \"),n+=t);return n}export default clsx;","import * as React from 'react';\nimport { createSvgIcon } from '@material-ui/core/utils';\n/**\n * @ignore - internal component.\n */\n\nexport default createSvgIcon( /*#__PURE__*/React.createElement(\"path\", {\n d: \"M20,12A8,8 0 0,1 12,20A8,8 0 0,1 4,12A8,8 0 0,1 12,4C12.76,4 13.5,4.11 14.2, 4.31L15.77,2.74C14.61,2.26 13.34,2 12,2A10,10 0 0,0 2,12A10,10 0 0,0 12,22A10,10 0 0, 0 22,12M7.91,10.08L6.5,11.5L11,16L21,6L19.59,4.58L11,13.17L7.91,10.08Z\"\n}), 'SuccessOutlined');","import * as React from 'react';\nimport { createSvgIcon } from '@material-ui/core/utils';\n/**\n * @ignore - internal component.\n */\n\nexport default createSvgIcon( /*#__PURE__*/React.createElement(\"path\", {\n d: \"M12 5.99L19.53 19H4.47L12 5.99M12 2L1 21h22L12 2zm1 14h-2v2h2v-2zm0-6h-2v4h2v-4z\"\n}), 'ReportProblemOutlined');","import * as React from 'react';\nimport { createSvgIcon } from '@material-ui/core/utils';\n/**\n * @ignore - internal component.\n */\n\nexport default createSvgIcon( /*#__PURE__*/React.createElement(\"path\", {\n d: \"M11 15h2v2h-2zm0-8h2v6h-2zm.99-5C6.47 2 2 6.48 2 12s4.47 10 9.99 10C17.52 22 22 17.52 22 12S17.52 2 11.99 2zM12 20c-4.42 0-8-3.58-8-8s3.58-8 8-8 8 3.58 8 8-3.58 8-8 8z\"\n}), 'ErrorOutline');","import * as React from 'react';\nimport { createSvgIcon } from '@material-ui/core/utils';\n/**\n * @ignore - internal component.\n */\n\nexport default createSvgIcon( /*#__PURE__*/React.createElement(\"path\", {\n d: \"M11,9H13V7H11M12,20C7.59,20 4,16.41 4,12C4,7.59 7.59,4 12,4C16.41,4 20,7.59 20, 12C20,16.41 16.41,20 12,20M12,2A10,10 0 0,0 2,12A10,10 0 0,0 12,22A10,10 0 0,0 22,12A10, 10 0 0,0 12,2M11,17H13V11H11V17Z\"\n}), 'InfoOutlined');","import * as React from 'react';\nimport { createSvgIcon } from '@material-ui/core/utils';\n/**\n * @ignore - internal component.\n */\n\nexport default createSvgIcon( /*#__PURE__*/React.createElement(\"path\", {\n d: \"M19 6.41L17.59 5 12 10.59 6.41 5 5 6.41 10.59 12 5 17.59 6.41 19 12 13.41 17.59 19 19 17.59 13.41 12z\"\n}), 'Close');","import _objectWithoutProperties from \"@babel/runtime/helpers/esm/objectWithoutProperties\";\nimport _extends from \"@babel/runtime/helpers/esm/extends\";\nimport * as React from 'react';\nimport PropTypes from 'prop-types';\nimport clsx from 'clsx';\nimport { withStyles, lighten, darken } from '@material-ui/core/styles';\nimport Paper from '@material-ui/core/Paper';\nimport SuccessOutlinedIcon from '../internal/svg-icons/SuccessOutlined';\nimport ReportProblemOutlinedIcon from '../internal/svg-icons/ReportProblemOutlined';\nimport ErrorOutlineIcon from '../internal/svg-icons/ErrorOutline';\nimport InfoOutlinedIcon from '../internal/svg-icons/InfoOutlined';\nimport CloseIcon from '../internal/svg-icons/Close';\nimport IconButton from '@material-ui/core/IconButton';\nimport { capitalize } from '@material-ui/core/utils';\nexport var styles = function styles(theme) {\n var getColor = theme.palette.type === 'light' ? darken : lighten;\n var getBackgroundColor = theme.palette.type === 'light' ? lighten : darken;\n return {\n /* Styles applied to the root element. */\n root: _extends({}, theme.typography.body2, {\n borderRadius: theme.shape.borderRadius,\n backgroundColor: 'transparent',\n display: 'flex',\n padding: '6px 16px'\n }),\n\n /* Styles applied to the root element if `variant=\"standard\"` and `color=\"success\"`. */\n standardSuccess: {\n color: getColor(theme.palette.success.main, 0.6),\n backgroundColor: getBackgroundColor(theme.palette.success.main, 0.9),\n '& $icon': {\n color: theme.palette.success.main\n }\n },\n\n /* Styles applied to the root element if `variant=\"standard\"` and `color=\"info\"`. */\n standardInfo: {\n color: getColor(theme.palette.info.main, 0.6),\n backgroundColor: getBackgroundColor(theme.palette.info.main, 0.9),\n '& $icon': {\n color: theme.palette.info.main\n }\n },\n\n /* Styles applied to the root element if `variant=\"standard\"` and `color=\"warning\"`. */\n standardWarning: {\n color: getColor(theme.palette.warning.main, 0.6),\n backgroundColor: getBackgroundColor(theme.palette.warning.main, 0.9),\n '& $icon': {\n color: theme.palette.warning.main\n }\n },\n\n /* Styles applied to the root element if `variant=\"standard\"` and `color=\"error\"`. */\n standardError: {\n color: getColor(theme.palette.error.main, 0.6),\n backgroundColor: getBackgroundColor(theme.palette.error.main, 0.9),\n '& $icon': {\n color: theme.palette.error.main\n }\n },\n\n /* Styles applied to the root element if `variant=\"outlined\"` and `color=\"success\"`. */\n outlinedSuccess: {\n color: getColor(theme.palette.success.main, 0.6),\n border: \"1px solid \".concat(theme.palette.success.main),\n '& $icon': {\n color: theme.palette.success.main\n }\n },\n\n /* Styles applied to the root element if `variant=\"outlined\"` and `color=\"info\"`. */\n outlinedInfo: {\n color: getColor(theme.palette.info.main, 0.6),\n border: \"1px solid \".concat(theme.palette.info.main),\n '& $icon': {\n color: theme.palette.info.main\n }\n },\n\n /* Styles applied to the root element if `variant=\"outlined\"` and `color=\"warning\"`. */\n outlinedWarning: {\n color: getColor(theme.palette.warning.main, 0.6),\n border: \"1px solid \".concat(theme.palette.warning.main),\n '& $icon': {\n color: theme.palette.warning.main\n }\n },\n\n /* Styles applied to the root element if `variant=\"outlined\"` and `color=\"error\"`. */\n outlinedError: {\n color: getColor(theme.palette.error.main, 0.6),\n border: \"1px solid \".concat(theme.palette.error.main),\n '& $icon': {\n color: theme.palette.error.main\n }\n },\n\n /* Styles applied to the root element if `variant=\"filled\"` and `color=\"success\"`. */\n filledSuccess: {\n color: '#fff',\n fontWeight: theme.typography.fontWeightMedium,\n backgroundColor: theme.palette.success.main\n },\n\n /* Styles applied to the root element if `variant=\"filled\"` and `color=\"info\"`. */\n filledInfo: {\n color: '#fff',\n fontWeight: theme.typography.fontWeightMedium,\n backgroundColor: theme.palette.info.main\n },\n\n /* Styles applied to the root element if `variant=\"filled\"` and `color=\"warning\"`. */\n filledWarning: {\n color: '#fff',\n fontWeight: theme.typography.fontWeightMedium,\n backgroundColor: theme.palette.warning.main\n },\n\n /* Styles applied to the root element if `variant=\"filled\"` and `color=\"error\"`. */\n filledError: {\n color: '#fff',\n fontWeight: theme.typography.fontWeightMedium,\n backgroundColor: theme.palette.error.main\n },\n\n /* Styles applied to the icon wrapper element. */\n icon: {\n marginRight: 12,\n padding: '7px 0',\n display: 'flex',\n fontSize: 22,\n opacity: 0.9\n },\n\n /* Styles applied to the message wrapper element. */\n message: {\n padding: '8px 0'\n },\n\n /* Styles applied to the action wrapper element if `action` is provided. */\n action: {\n display: 'flex',\n alignItems: 'center',\n marginLeft: 'auto',\n paddingLeft: 16,\n marginRight: -8\n }\n };\n};\nvar defaultIconMapping = {\n success: /*#__PURE__*/React.createElement(SuccessOutlinedIcon, {\n fontSize: \"inherit\"\n }),\n warning: /*#__PURE__*/React.createElement(ReportProblemOutlinedIcon, {\n fontSize: \"inherit\"\n }),\n error: /*#__PURE__*/React.createElement(ErrorOutlineIcon, {\n fontSize: \"inherit\"\n }),\n info: /*#__PURE__*/React.createElement(InfoOutlinedIcon, {\n fontSize: \"inherit\"\n })\n};\n\nvar _ref = /*#__PURE__*/React.createElement(CloseIcon, {\n fontSize: \"small\"\n});\n\nvar Alert = /*#__PURE__*/React.forwardRef(function Alert(props, ref) {\n var action = props.action,\n children = props.children,\n classes = props.classes,\n className = props.className,\n _props$closeText = props.closeText,\n closeText = _props$closeText === void 0 ? 'Close' : _props$closeText,\n color = props.color,\n icon = props.icon,\n _props$iconMapping = props.iconMapping,\n iconMapping = _props$iconMapping === void 0 ? defaultIconMapping : _props$iconMapping,\n onClose = props.onClose,\n _props$role = props.role,\n role = _props$role === void 0 ? 'alert' : _props$role,\n _props$severity = props.severity,\n severity = _props$severity === void 0 ? 'success' : _props$severity,\n _props$variant = props.variant,\n variant = _props$variant === void 0 ? 'standard' : _props$variant,\n other = _objectWithoutProperties(props, [\"action\", \"children\", \"classes\", \"className\", \"closeText\", \"color\", \"icon\", \"iconMapping\", \"onClose\", \"role\", \"severity\", \"variant\"]);\n\n return /*#__PURE__*/React.createElement(Paper, _extends({\n role: role,\n square: true,\n elevation: 0,\n className: clsx(classes.root, classes[\"\".concat(variant).concat(capitalize(color || severity))], className),\n ref: ref\n }, other), icon !== false ? /*#__PURE__*/React.createElement(\"div\", {\n className: classes.icon\n }, icon || iconMapping[severity] || defaultIconMapping[severity]) : null, /*#__PURE__*/React.createElement(\"div\", {\n className: classes.message\n }, children), action != null ? /*#__PURE__*/React.createElement(\"div\", {\n className: classes.action\n }, action) : null, action == null && onClose ? /*#__PURE__*/React.createElement(\"div\", {\n className: classes.action\n }, /*#__PURE__*/React.createElement(IconButton, {\n size: \"small\",\n \"aria-label\": closeText,\n title: closeText,\n color: \"inherit\",\n onClick: onClose\n }, _ref)) : null);\n});\nprocess.env.NODE_ENV !== \"production\" ? Alert.propTypes = {\n // ----------------------------- Warning --------------------------------\n // | These PropTypes are generated from the TypeScript type definitions |\n // | To update them edit the d.ts file and run \"yarn proptypes\" |\n // ----------------------------------------------------------------------\n\n /**\n * The action to display. It renders after the message, at the end of the alert.\n */\n action: PropTypes.node,\n\n /**\n * The content of the component.\n */\n children: PropTypes.node,\n\n /**\n * Override or extend the styles applied to the component.\n * See [CSS API](#css) below for more details.\n */\n classes: PropTypes.object,\n\n /**\n * @ignore\n */\n className: PropTypes.string,\n\n /**\n * Override the default label for the *close popup* icon button.\n *\n * For localization purposes, you can use the provided [translations](/guides/localization/).\n */\n closeText: PropTypes.string,\n\n /**\n * The main color for the alert. Unless provided, the value is taken from the `severity` prop.\n */\n color: PropTypes.oneOf(['error', 'info', 'success', 'warning']),\n\n /**\n * Override the icon displayed before the children.\n * Unless provided, the icon is mapped to the value of the `severity` prop.\n */\n icon: PropTypes.node,\n\n /**\n * The component maps the `severity` prop to a range of different icons,\n * for instance success to `<SuccessOutlined>`.\n * If you wish to change this mapping, you can provide your own.\n * Alternatively, you can use the `icon` prop to override the icon displayed.\n */\n iconMapping: PropTypes.shape({\n error: PropTypes.node,\n info: PropTypes.node,\n success: PropTypes.node,\n warning: PropTypes.node\n }),\n\n /**\n * Callback fired when the component requests to be closed.\n * When provided and no `action` prop is set, a close icon button is displayed that triggers the callback when clicked.\n *\n * @param {object} event The event source of the callback.\n */\n onClose: PropTypes.func,\n\n /**\n * The ARIA role attribute of the element.\n */\n role: PropTypes.string,\n\n /**\n * The severity of the alert. This defines the color and icon used.\n */\n severity: PropTypes.oneOf(['error', 'info', 'success', 'warning']),\n\n /**\n * The variant to use.\n */\n variant: PropTypes.oneOf(['filled', 'outlined', 'standard'])\n} : void 0;\nexport default withStyles(styles, {\n name: 'MuiAlert'\n})(Alert);","\"use strict\";\n\nvar _interopRequireDefault = require(\"@babel/runtime/helpers/interopRequireDefault\");\n\nvar _interopRequireWildcard = require(\"@babel/runtime/helpers/interopRequireWildcard\");\n\nObject.defineProperty(exports, \"__esModule\", {\n value: true\n});\nexports.default = void 0;\n\nvar React = _interopRequireWildcard(require(\"react\"));\n\nvar _createSvgIcon = _interopRequireDefault(require(\"./utils/createSvgIcon\"));\n\nvar _default = (0, _createSvgIcon.default)( /*#__PURE__*/React.createElement(\"path\", {\n d: \"M18 8h-1V6c0-2.76-2.24-5-5-5S7 3.24 7 6v2H6c-1.1 0-2 .9-2 2v10c0 1.1.9 2 2 2h12c1.1 0 2-.9 2-2V10c0-1.1-.9-2-2-2zm-6 9c-1.1 0-2-.9-2-2s.9-2 2-2 2 .9 2 2-.9 2-2 2zm3.1-9H8.9V6c0-1.71 1.39-3.1 3.1-3.1 1.71 0 3.1 1.39 3.1 3.1v2z\"\n}), 'Lock');\n\nexports.default = _default;","\"use strict\";\nObject.defineProperty(exports, \"__esModule\", { value: true });\nvar tslib_1 = require(\"tslib\");\nvar react_1 = require(\"react\");\nvar useAsyncFn_1 = tslib_1.__importDefault(require(\"./useAsyncFn\"));\nfunction useAsync(fn, deps) {\n if (deps === void 0) { deps = []; }\n var _a = useAsyncFn_1.default(fn, deps, {\n loading: true,\n }), state = _a[0], callback = _a[1];\n react_1.useEffect(function () {\n callback();\n }, [callback]);\n return state;\n}\nexports.default = useAsync;\n","import { jsxs, jsx } from 'react/jsx-runtime';\nimport Box from '@material-ui/core/Box';\nimport Divider from '@material-ui/core/Divider';\nimport { makeStyles } from '@material-ui/core/styles';\nimport Typography from '@material-ui/core/Typography';\nimport ArrowIcon from '@material-ui/icons/ArrowForward';\nimport { Link } from '../../components/Link/Link.esm.js';\n\nconst useStyles = makeStyles(\n (theme) => ({\n root: {\n maxWidth: \"fit-content\",\n padding: theme.spacing(2, 2, 2, 2.5)\n },\n boxTitle: {\n margin: 0,\n color: theme.palette.textSubtle\n },\n arrow: {\n color: theme.palette.textSubtle\n }\n }),\n { name: \"BackstageBottomLink\" }\n);\nfunction BottomLink(props) {\n const { link, title, onClick } = props;\n const classes = useStyles();\n return /* @__PURE__ */ jsxs(Box, { children: [\n /* @__PURE__ */ jsx(Divider, {}),\n /* @__PURE__ */ jsx(Link, { to: link, onClick, underline: \"none\", children: /* @__PURE__ */ jsxs(Box, { display: \"flex\", alignItems: \"center\", className: classes.root, children: [\n /* @__PURE__ */ jsx(Box, { className: classes.boxTitle, fontWeight: \"fontWeightBold\", m: 1, children: /* @__PURE__ */ jsx(Typography, { children: /* @__PURE__ */ jsx(\"strong\", { children: title }) }) }),\n /* @__PURE__ */ jsx(ArrowIcon, { className: classes.arrow })\n ] }) })\n ] });\n}\n\nexport { BottomLink };\n//# sourceMappingURL=BottomLink.esm.js.map\n","import { jsx } from 'react/jsx-runtime';\nimport Button$1 from '@material-ui/core/Button';\nimport { forwardRef } from 'react';\nimport { Link } from '../Link/Link.esm.js';\n\nconst LinkWrapper = forwardRef((props, ref) => /* @__PURE__ */ jsx(Link, { ref, ...props, color: \"initial\" }));\nconst LinkButton = forwardRef((props, ref) => /* @__PURE__ */ jsx(Button$1, { ref, component: LinkWrapper, ...props }));\nconst Button = LinkButton;\n\nexport { Button, LinkButton };\n//# sourceMappingURL=LinkButton.esm.js.map\n","import { jsx } from 'react/jsx-runtime';\nimport Typography from '@material-ui/core/Typography';\nimport { Component } from 'react';\nimport { LinkButton } from '../../components/LinkButton/LinkButton.esm.js';\nimport { ErrorPanel } from '../../components/ErrorPanel/ErrorPanel.esm.js';\nimport { coreComponentsTranslationRef } from '../../translation.esm.js';\nimport { useTranslationRef } from '@backstage/core-plugin-api/alpha';\n\nconst SlackLink = (props) => {\n const { slackChannel } = props;\n const { t } = useTranslationRef(coreComponentsTranslationRef);\n if (!slackChannel) {\n return null;\n } else if (typeof slackChannel === \"string\") {\n return /* @__PURE__ */ jsx(Typography, { children: t(\"errorBoundary.title\", { slackChannel }) });\n } else if (!slackChannel.href) {\n return /* @__PURE__ */ jsx(Typography, { children: t(\"errorBoundary.title\", {\n slackChannel: slackChannel.name\n }) });\n }\n return /* @__PURE__ */ jsx(LinkButton, { to: slackChannel.href, variant: \"contained\", children: slackChannel.name });\n};\nconst ErrorBoundary = class ErrorBoundary2 extends Component {\n constructor(props) {\n super(props);\n this.state = {\n error: void 0,\n errorInfo: void 0\n };\n }\n componentDidCatch(error, errorInfo) {\n console.error(`ErrorBoundary, error: ${error}`, { error, errorInfo });\n this.setState({ error, errorInfo });\n }\n render() {\n const { slackChannel, children } = this.props;\n const { error } = this.state;\n if (!error) {\n return children;\n }\n return /* @__PURE__ */ jsx(ErrorPanel, { title: \"Something Went Wrong\", error, children: /* @__PURE__ */ jsx(SlackLink, { slackChannel }) });\n }\n};\n\nexport { ErrorBoundary };\n//# sourceMappingURL=ErrorBoundary.esm.js.map\n","import { jsx, jsxs } from 'react/jsx-runtime';\nimport Card from '@material-ui/core/Card';\nimport CardActions from '@material-ui/core/CardActions';\nimport CardContent from '@material-ui/core/CardContent';\nimport CardHeader from '@material-ui/core/CardHeader';\nimport Divider from '@material-ui/core/Divider';\nimport { makeStyles, withStyles } from '@material-ui/core/styles';\nimport classNames from 'classnames';\nimport { BottomLink } from '../BottomLink/BottomLink.esm.js';\nimport { ErrorBoundary } from '../ErrorBoundary/ErrorBoundary.esm.js';\n\nconst useStyles = makeStyles(\n (theme) => ({\n noPadding: {\n padding: 0,\n \"&:last-child\": {\n paddingBottom: 0\n }\n },\n contentAlignBottom: {\n display: \"flex\",\n alignItems: \"self-end\"\n },\n header: {\n padding: theme.spacing(2, 2, 2, 2.5)\n },\n headerTitle: {\n fontWeight: theme.typography.fontWeightBold\n },\n headerSubheader: {\n paddingTop: theme.spacing(1)\n },\n headerAvatar: {},\n headerAction: {},\n headerContent: {},\n subheader: {\n display: \"flex\"\n }\n }),\n { name: \"BackstageInfoCard\" }\n);\nconst CardActionsTopRight = withStyles(\n (theme) => ({\n root: {\n display: \"inline-block\",\n padding: theme.spacing(8, 8, 0, 0),\n float: \"right\"\n }\n }),\n { name: \"BackstageInfoCardCardActionsTopRight\" }\n)(CardActions);\nconst VARIANT_STYLES = {\n card: {\n flex: {\n display: \"flex\",\n flexDirection: \"column\"\n },\n fullHeight: {\n display: \"flex\",\n flexDirection: \"column\",\n height: \"100%\"\n },\n gridItem: {\n display: \"flex\",\n flexDirection: \"column\",\n height: \"calc(100% - 10px)\",\n // for pages without content header\n marginBottom: \"10px\",\n breakInside: \"avoid-page\",\n \"@media print\": {\n height: \"auto\"\n }\n }\n },\n cardContent: {\n fullHeight: {\n flex: 1\n },\n gridItem: {\n flex: 1\n }\n }\n};\nfunction InfoCard(props) {\n const {\n title,\n subheader,\n divider = true,\n deepLink,\n slackChannel,\n errorBoundaryProps,\n variant,\n alignContent = \"normal\",\n children,\n headerStyle,\n headerProps,\n icon,\n action,\n actionsClassName,\n actions,\n cardClassName,\n actionsTopRight,\n className,\n noPadding,\n titleTypographyProps,\n subheaderTypographyProps\n } = props;\n const classes = useStyles();\n let calculatedStyle = {};\n let calculatedCardStyle = {};\n if (variant) {\n const variants = variant.split(/[\\s]+/g);\n variants.forEach((name) => {\n calculatedStyle = {\n ...calculatedStyle,\n ...VARIANT_STYLES.card[name]\n };\n calculatedCardStyle = {\n ...calculatedCardStyle,\n ...VARIANT_STYLES.cardContent[name]\n };\n });\n }\n const cardSubTitle = () => {\n if (!subheader && !icon) {\n return null;\n }\n return /* @__PURE__ */ jsxs(\"div\", { \"data-testid\": \"info-card-subheader\", children: [\n subheader && /* @__PURE__ */ jsx(\"div\", { className: classes.subheader, children: subheader }),\n icon\n ] });\n };\n const errProps = errorBoundaryProps || (slackChannel ? { slackChannel } : {});\n return /* @__PURE__ */ jsx(Card, { style: calculatedStyle, className, children: /* @__PURE__ */ jsxs(ErrorBoundary, { ...errProps, children: [\n title && /* @__PURE__ */ jsx(\n CardHeader,\n {\n classes: {\n root: classNames(classes.header),\n title: classes.headerTitle,\n subheader: classes.headerSubheader,\n avatar: classes.headerAvatar,\n action: classes.headerAction,\n content: classes.headerContent\n },\n title,\n subheader: cardSubTitle(),\n action,\n style: { ...headerStyle },\n titleTypographyProps,\n subheaderTypographyProps,\n ...headerProps\n }\n ),\n actionsTopRight && /* @__PURE__ */ jsx(CardActionsTopRight, { children: actionsTopRight }),\n divider && /* @__PURE__ */ jsx(Divider, {}),\n /* @__PURE__ */ jsx(\n CardContent,\n {\n className: classNames(cardClassName, {\n [classes.noPadding]: noPadding,\n [classes.contentAlignBottom]: alignContent === \"bottom\"\n }),\n style: calculatedCardStyle,\n children\n }\n ),\n actions && /* @__PURE__ */ jsx(CardActions, { className: actionsClassName, children: actions }),\n deepLink && /* @__PURE__ */ jsx(BottomLink, { ...deepLink })\n ] }) });\n}\n\nexport { InfoCard };\n//# sourceMappingURL=InfoCard.esm.js.map\n"],"names":["Object","defineProperty","exports","value","tslib_1","react_1","useMountedState_1","__importDefault","fn","deps","initialState","loading","lastCallId","useRef","isMounted","default","_a","useState","state","set","callback","useCallback","args","_i","arguments","length","callId","current","prevState","__assign","apply","then","error","mountedRef","get","useEffect","_interopRequireDefault","_interopRequireWildcard","React","_default","createElement","d","useDefaultSupportConfig","t","url","items","title","icon","links","AppIcon","props","id","key","Fallback","rest","Icon","useApp","getSystemIcon","jsx","HelpIcon","useStyles","makeStyles","popoverList","minWidth","maxWidth","menuItem","whiteSpace","name","SupportIcon","app","SupportLink","link","to","children","SupportListItem","item","jsxs","MenuItem","button","ListItemIcon","ListItemText","primary","secondary","reduce","prev","idx","SupportButton","configItems","config","useApiHolder","configApiRef","supportConfig","getOptionalConfig","defaultSupportConfig","getString","getConfigArray","flatMap","itemConf","getOptionalString","getOptionalConfigArray","linkConf","useSupportConfig","popoverOpen","setPopoverOpen","anchorEl","setAnchorEl","classes","useApi","isSmallScreen","useMediaQuery","theme","breakpoints","down","onClickHandler","event","currentTarget","popoverCloseHandler","Fragment","Box","display","ml","IconButton","color","size","onClick","Button","startIcon","Popover","open","anchorOrigin","vertical","horizontal","transformOrigin","onClose","MenuList","className","autoFocusItem","Boolean","alignItems","Typography","variant","Children","map","child","i","DialogActions","r","e","f","n","Array","isArray","createSvgIcon","defaultIconMapping","success","SuccessOutlined","fontSize","warning","ReportProblemOutlined","ErrorOutline","info","InfoOutlined","_ref","Close","Alert","ref","action","_props$closeText","closeText","_props$iconMapping","iconMapping","_props$role","role","_props$severity","severity","_props$variant","other","Paper","square","elevation","root","concat","capitalize","message","withStyles","getColor","palette","type","darken","lighten","getBackgroundColor","typography","body2","borderRadius","shape","backgroundColor","padding","standardSuccess","main","standardInfo","standardWarning","standardError","outlinedSuccess","border","outlinedInfo","outlinedWarning","outlinedError","filledSuccess","fontWeight","fontWeightMedium","filledInfo","filledWarning","filledError","marginRight","opacity","marginLeft","paddingLeft","useAsyncFn_1","spacing","boxTitle","margin","textSubtle","arrow","BottomLink","Divider","underline","m","A","LinkWrapper","forwardRef","LinkButton","component","SlackLink","slackChannel","href","ErrorBoundary","Component","constructor","super","this","errorInfo","componentDidCatch","console","setState","render","noPadding","paddingBottom","contentAlignBottom","header","headerTitle","fontWeightBold","headerSubheader","paddingTop","headerAvatar","headerAction","headerContent","subheader","CardActionsTopRight","float","CardActions","VARIANT_STYLES","card","flex","flexDirection","fullHeight","height","gridItem","marginBottom","breakInside","cardContent","InfoCard","divider","deepLink","errorBoundaryProps","alignContent","headerStyle","headerProps","actionsClassName","actions","cardClassName","actionsTopRight","titleTypographyProps","subheaderTypographyProps","calculatedStyle","calculatedCardStyle","split","forEach","errProps","Card","style","CardHeader","avatar","content","CardContent"],"sourceRoot":""}
|
|
@@ -1,2 +0,0 @@
|
|
|
1
|
-
"use strict";(self.webpackChunkinternal_plugin_kuadrant=self.webpackChunkinternal_plugin_kuadrant||[]).push([[879],{72627:(e,n,t)=>{t.r(n),t.d(n,{ApiKeyDetailPage:()=>y,ApiKeysPage:()=>l,ApiProductsPage:()=>d,EntityKuadrantApiAccessCard:()=>u,EntityKuadrantApiApprovalTab:()=>K,EntityKuadrantApiKeyManagementTab:()=>m,EntityKuadrantApiKeysContent:()=>c,EntityKuadrantApiProductInfoContent:()=>P,EntityKuadrantApiProductOpenApiAlert:()=>h,KuadrantApprovalQueueCard:()=>s,KuadrantPage:()=>p,PlanPolicyDetailPage:()=>A,kuadrantPlugin:()=>r});var a=t(22097);const o=(0,a.createRouteRef)({id:"kuadrant"}),i=(0,a.createSubRouteRef)({id:"kuadrant/resource",parent:o,path:"/:kind/:namespace/:name"}),r=(0,a.createPlugin)({id:"kuadrant",routes:{root:o,resource:i}}),p=r.provide((0,a.createRoutableExtension)({name:"KuadrantPage",component:()=>Promise.all([t.e(7556),t.e(9400),t.e(2946),t.e(2628),t.e(2198),t.e(2120),t.e(6979),t.e(5010),t.e(8563),t.e(8416),t.e(5478),t.e(2469),t.e(4218),t.e(484),t.e(1942),t.e(7976),t.e(7094),t.e(7684)]).then(t.bind(t,27684)).then(e=>e.KuadrantPage),mountPoint:o})),d=r.provide((0,a.createRoutableExtension)({name:"ApiProductsPage",component:()=>Promise.all([t.e(7556),t.e(9400),t.e(2946),t.e(2628),t.e(2198),t.e(2120),t.e(6979),t.e(5010),t.e(8563),t.e(8416),t.e(5478),t.e(2469),t.e(4218),t.e(484),t.e(1942),t.e(7976),t.e(7094),t.e(7684)]).then(t.bind(t,27684)).then(e=>e.ApiProductsPage),mountPoint:o})),l=r.provide((0,a.createRoutableExtension)({name:"ApiKeysPage",component:()=>Promise.all([t.e(7556),t.e(9400),t.e(2946),t.e(2628),t.e(2198),t.e(2120),t.e(6979),t.e(5010),t.e(8563),t.e(5478),t.e(2469),t.e(4218),t.e(484),t.e(1942),t.e(7976),t.e(7094),t.e(6840),t.e(6813)]).then(t.bind(t,16813)).then(e=>e.ApiKeysPage),mountPoint:o})),u=r.provide((0,a.createComponentExtension)({name:"EntityKuadrantApiAccessCard",component:{lazy:()=>Promise.all([t.e(7556),t.e(9400),t.e(2946),t.e(2628),t.e(5478),t.e(2469),t.e(4218),t.e(484),t.e(1942),t.e(7976),t.e(6659),t.e(441)]).then(t.bind(t,60441)).then(e=>e.ApiAccessCard)}})),m=r.provide((0,a.createComponentExtension)({name:"EntityKuadrantApiKeyManagementTab",component:{lazy:()=>Promise.all([t.e(7556),t.e(9400),t.e(2946),t.e(2628),t.e(2198),t.e(2120),t.e(6979),t.e(5010),t.e(5478),t.e(4218),t.e(484),t.e(1942),t.e(7976),t.e(7094),t.e(6659),t.e(6840),t.e(3650)]).then(t.bind(t,23650)).then(e=>e.ApiKeyManagementTab)}})),c=r.provide((0,a.createComponentExtension)({name:"EntityKuadrantApiKeysContent",component:{lazy:()=>Promise.all([t.e(7556),t.e(9400),t.e(2946),t.e(2628),t.e(2198),t.e(2120),t.e(6979),t.e(5010),t.e(5478),t.e(4218),t.e(484),t.e(1942),t.e(7976),t.e(7094),t.e(6659),t.e(6840),t.e(3650)]).then(t.bind(t,23650)).then(e=>e.ApiKeyManagementTab)}})),s=r.provide((0,a.createComponentExtension)({name:"KuadrantApprovalQueueCard",component:{lazy:()=>Promise.all([t.e(7556),t.e(9400),t.e(2946),t.e(2628),t.e(2198),t.e(2120),t.e(6979),t.e(5010),t.e(5478),t.e(2469),t.e(4218),t.e(484),t.e(1942),t.e(7976),t.e(7094),t.e(3097)]).then(t.bind(t,43097)).then(e=>e.ApprovalQueueCard)}})),P=r.provide((0,a.createComponentExtension)({name:"EntityKuadrantApiProductInfoContent",component:{lazy:()=>Promise.all([t.e(7556),t.e(9400),t.e(2946),t.e(2628),t.e(2120),t.e(5478),t.e(2469),t.e(4218),t.e(484),t.e(1942),t.e(7976),t.e(7094),t.e(6659),t.e(5453)]).then(t.bind(t,75453)).then(e=>e.ApiProductInfoCard)}})),A=r.provide((0,a.createComponentExtension)({name:"PlanPolicyDetailPage",component:{lazy:()=>Promise.all([t.e(7556),t.e(9400),t.e(2946),t.e(2628),t.e(2198),t.e(8563),t.e(5478),t.e(2469),t.e(4218),t.e(484),t.e(1942),t.e(7976),t.e(8365)]).then(t.bind(t,58365)).then(e=>e.PlanPolicyDetailPage)}})),y=r.provide((0,a.createRoutableExtension)({name:"ApiKeyDetailPage",component:()=>Promise.all([t.e(7556),t.e(9400),t.e(2946),t.e(2628),t.e(2198),t.e(6979),t.e(8563),t.e(5478),t.e(2469),t.e(4218),t.e(484),t.e(1942),t.e(7976),t.e(2967)]).then(t.bind(t,72967)).then(e=>e.ApiKeyDetailPage),mountPoint:o})),K=r.provide((0,a.createComponentExtension)({name:"EntityKuadrantApiApprovalTab",component:{lazy:()=>Promise.all([t.e(7556),t.e(9400),t.e(2946),t.e(2628),t.e(2198),t.e(2120),t.e(6979),t.e(5010),t.e(5478),t.e(4218),t.e(484),t.e(1942),t.e(7976),t.e(7094),t.e(6659),t.e(6800)]).then(t.bind(t,46800)).then(e=>e.EntityApiApprovalTab)}})),h=r.provide((0,a.createComponentExtension)({name:"EntityKuadrantApiProductOpenApiAlert",component:{lazy:()=>Promise.all([t.e(9400),t.e(2946),t.e(5478),t.e(2469),t.e(4218),t.e(484),t.e(1942),t.e(7976),t.e(6659),t.e(6371)]).then(t.bind(t,36371)).then(e=>e.ApiProductOpenApiAlert)}}))}}]);
|
|
2
|
-
//# sourceMappingURL=exposed-PluginRoot.16bf7897.chunk.js.map
|