@cobaltcore-dev/aurora 0.5.0 → 0.6.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (113) hide show
  1. package/dist/client/{ContentHeader-xQVhO2yT.mjs → ContentHeader-kx1Th5Sq.mjs} +18 -18
  2. package/dist/client/{ContentHeader-xQVhO2yT.mjs.map → ContentHeader-kx1Th5Sq.mjs.map} +1 -1
  3. package/dist/client/{DeleteFlavorModal-CHTUZ3YV.mjs → DeleteFlavorModal-C3cb7YiJ.mjs} +155 -155
  4. package/dist/client/{DeleteFlavorModal-CHTUZ3YV.mjs.map → DeleteFlavorModal-C3cb7YiJ.mjs.map} +1 -1
  5. package/dist/client/{EditSecurityGroupModal-CWHHx2Xk.mjs → EditSecurityGroupModal-CpP54WIK.mjs} +18 -18
  6. package/dist/client/{EditSecurityGroupModal-CWHHx2Xk.mjs.map → EditSecurityGroupModal-CpP54WIK.mjs.map} +1 -1
  7. package/dist/client/{FiltersInput-UKJpNFdr.mjs → FiltersInput-DxcyR6Bp.mjs} +18 -18
  8. package/dist/client/{FiltersInput-UKJpNFdr.mjs.map → FiltersInput-DxcyR6Bp.mjs.map} +1 -1
  9. package/dist/client/{FloatingIpActionModals-caXn6bYo.mjs → FloatingIpActionModals-BP8RWHbu.mjs} +55 -55
  10. package/dist/client/{FloatingIpActionModals-caXn6bYo.mjs.map → FloatingIpActionModals-BP8RWHbu.mjs.map} +1 -1
  11. package/dist/client/{ImageToastNotifications-BWimIzu_.mjs → ImageToastNotifications-TZ3EfQg-.mjs} +323 -371
  12. package/dist/client/ImageToastNotifications-TZ3EfQg-.mjs.map +1 -0
  13. package/dist/client/{RouteError-CUj_m3gu.mjs → RouteError-QSV7qOoJ.mjs} +2 -2
  14. package/dist/client/{RouteError-CUj_m3gu.mjs.map → RouteError-QSV7qOoJ.mjs.map} +1 -1
  15. package/dist/client/{SortInput-GQlQBlAj.mjs → SortInput-CYv2_Pur.mjs} +5 -5
  16. package/dist/client/{SortInput-GQlQBlAj.mjs.map → SortInput-CYv2_Pur.mjs.map} +1 -1
  17. package/dist/client/{_flavorId-sUWG0xuy.mjs → _flavorId-C2x43-6S.mjs} +9 -9
  18. package/dist/client/{_flavorId-sUWG0xuy.mjs.map → _flavorId-C2x43-6S.mjs.map} +1 -1
  19. package/dist/client/{_flavorId-C5dc1N_j.mjs → _flavorId-CR8ZUI-P.mjs} +46 -46
  20. package/dist/client/{_flavorId-C5dc1N_j.mjs.map → _flavorId-CR8ZUI-P.mjs.map} +1 -1
  21. package/dist/client/{_floatingIpId-CT4y-60o.mjs → _floatingIpId-BCk41_Lb.mjs} +2 -2
  22. package/dist/client/{_floatingIpId-CT4y-60o.mjs.map → _floatingIpId-BCk41_Lb.mjs.map} +1 -1
  23. package/dist/client/{_floatingIpId-COK_TBrz2.mjs → _floatingIpId-BGrOAmPT.mjs} +33 -33
  24. package/dist/client/_floatingIpId-BGrOAmPT.mjs.map +1 -0
  25. package/dist/client/_imageId-CvfD832b.mjs +534 -0
  26. package/dist/client/{_imageId-n3RcnhAR.mjs.map → _imageId-CvfD832b.mjs.map} +1 -1
  27. package/dist/client/_pcaId-BxBt5DXi.mjs +459 -0
  28. package/dist/client/_pcaId-BxBt5DXi.mjs.map +1 -0
  29. package/dist/client/{_pcaId-ChnM_t-9.mjs → _pcaId-DOHycvCf.mjs} +2 -2
  30. package/dist/client/{_pcaId-ChnM_t-9.mjs.map → _pcaId-DOHycvCf.mjs.map} +1 -1
  31. package/dist/client/{_projectId-26bw-_rm.mjs → _projectId-BDSWnMGj.mjs} +3 -3
  32. package/dist/client/{_projectId-26bw-_rm.mjs.map → _projectId-BDSWnMGj.mjs.map} +1 -1
  33. package/dist/client/{_projectId-Dhb4AyBD.mjs → _projectId-DOgwFiqD.mjs} +2 -2
  34. package/dist/client/{_projectId-Dhb4AyBD.mjs.map → _projectId-DOgwFiqD.mjs.map} +1 -1
  35. package/dist/client/_projectId-DS4nR59B.mjs +299 -0
  36. package/dist/client/_projectId-DS4nR59B.mjs.map +1 -0
  37. package/dist/client/{_projectId-BK9UqeYw.mjs → _projectId-MxcHrXW4.mjs} +12 -12
  38. package/dist/client/_projectId-MxcHrXW4.mjs.map +1 -0
  39. package/dist/client/{_securityGroupId-CR1mKICQ.mjs → _securityGroupId-CJJanWiY.mjs} +2 -2
  40. package/dist/client/{_securityGroupId-CR1mKICQ.mjs.map → _securityGroupId-CJJanWiY.mjs.map} +1 -1
  41. package/dist/client/{_securityGroupId-DroYG6cA.mjs → _securityGroupId-KKw4RPdH.mjs} +328 -328
  42. package/dist/client/{_securityGroupId-DroYG6cA.mjs.map → _securityGroupId-KKw4RPdH.mjs.map} +1 -1
  43. package/dist/client/{containers-DGY2hoWw.mjs → _storageType-4wSxI__0.mjs} +18 -15
  44. package/dist/client/_storageType-4wSxI__0.mjs.map +1 -0
  45. package/dist/client/{containers-ZMFTRaQL.mjs → _storageType-DYjo-6ej.mjs} +1054 -1088
  46. package/dist/client/_storageType-DYjo-6ej.mjs.map +1 -0
  47. package/dist/client/{containers-xfsYgRyf.mjs → _storageType-zeSZe--V.mjs} +2 -2
  48. package/dist/client/_storageType-zeSZe--V.mjs.map +1 -0
  49. package/dist/client/{about-969pIiZ9.mjs → about-Bo9vxGHy.mjs} +8 -8
  50. package/dist/client/{about-969pIiZ9.mjs.map → about-Bo9vxGHy.mjs.map} +1 -1
  51. package/dist/client/{build-DracvfrJ.mjs → build-DeJcDjPi.mjs} +3798 -3372
  52. package/dist/client/build-DeJcDjPi.mjs.map +1 -0
  53. package/dist/client/constants-BmcGYeR-.mjs +153 -0
  54. package/dist/client/constants-BmcGYeR-.mjs.map +1 -0
  55. package/dist/client/{flavors-C4GtoybP.mjs → flavors-BxFVqgnb.mjs} +2 -2
  56. package/dist/client/{flavors-C4GtoybP.mjs.map → flavors-BxFVqgnb.mjs.map} +1 -1
  57. package/dist/client/{flavors-CpirO_nr.mjs → flavors-CfdgjsZY.mjs} +192 -192
  58. package/dist/client/{flavors-CpirO_nr.mjs.map → flavors-CfdgjsZY.mjs.map} +1 -1
  59. package/dist/client/{floatingips-BUf_oLRl.mjs → floatingips-ByRb82wS.mjs} +123 -123
  60. package/dist/client/{floatingips-BUf_oLRl.mjs.map → floatingips-ByRb82wS.mjs.map} +1 -1
  61. package/dist/client/{images-Da1t5KPh.mjs → images-CKqIXUq52.mjs} +613 -613
  62. package/dist/client/images-CKqIXUq52.mjs.map +1 -0
  63. package/dist/client/{images-CsonlmFx.mjs → images-C_dX7nY6.mjs} +2 -2
  64. package/dist/client/{images-CsonlmFx.mjs.map → images-C_dX7nY6.mjs.map} +1 -1
  65. package/dist/client/{images-QnWf63uj.mjs → images-CenluYV8.mjs} +2 -2
  66. package/dist/client/{images-QnWf63uj.mjs.map → images-CenluYV8.mjs.map} +1 -1
  67. package/dist/client/index.js +382 -425
  68. package/dist/client/index.js.map +1 -1
  69. package/dist/client/{md-sBiSNVSQ.mjs → md-CYTrL5dq.mjs} +52 -16
  70. package/dist/client/{md-sBiSNVSQ.mjs.map → md-CYTrL5dq.mjs.map} +1 -1
  71. package/dist/client/{objects-H0NN_Sja.mjs → objects-BJM6YeuF.mjs} +2131 -1883
  72. package/dist/client/objects-BJM6YeuF.mjs.map +1 -0
  73. package/dist/client/{objects-B89dYCBq.mjs → objects-gxSjvbvF.mjs} +36 -21
  74. package/dist/client/objects-gxSjvbvF.mjs.map +1 -0
  75. package/dist/client/{objects-CuFLUOe1.mjs → objects-o2Cj_ndZ.mjs} +2 -2
  76. package/dist/client/objects-o2Cj_ndZ.mjs.map +1 -0
  77. package/dist/client/{pca-CtU6REww.mjs → pca-Bl8NmoVZ.mjs} +2 -2
  78. package/dist/client/{pca-CtU6REww.mjs.map → pca-Bl8NmoVZ.mjs.map} +1 -1
  79. package/dist/client/pca-RSiWpJs9.mjs +182 -0
  80. package/dist/client/pca-RSiWpJs9.mjs.map +1 -0
  81. package/dist/client/{projects-DNXsDnJM.mjs → projects-CgclWI16.mjs} +33 -27
  82. package/dist/client/{projects-DNXsDnJM.mjs.map → projects-CgclWI16.mjs.map} +1 -1
  83. package/dist/client/{projects-dhnQkuvV.mjs → projects-D2iewAzu.mjs} +2 -2
  84. package/dist/client/{projects-dhnQkuvV.mjs.map → projects-D2iewAzu.mjs.map} +1 -1
  85. package/dist/client/{projects-C-sjd9T5.mjs → projects-pe2_dCnV.mjs} +2 -2
  86. package/dist/client/{projects-C-sjd9T5.mjs.map → projects-pe2_dCnV.mjs.map} +1 -1
  87. package/dist/client/projects-yiK0HGSA.mjs.map +1 -1
  88. package/dist/client/{securitygroups-wHdrxZXd.mjs → securitygroups-DahZkVYQ.mjs} +111 -111
  89. package/dist/client/{securitygroups-wHdrxZXd.mjs.map → securitygroups-DahZkVYQ.mjs.map} +1 -1
  90. package/dist/client/{useListWithFiltering-DrgUwXef.mjs → useListWithFiltering-DaYcu5AB.mjs} +12 -12
  91. package/dist/client/{useListWithFiltering-DrgUwXef.mjs.map → useListWithFiltering-DaYcu5AB.mjs.map} +1 -1
  92. package/dist/server/index.js +7 -6
  93. package/package.json +4 -4
  94. package/dist/client/ImageToastNotifications-BWimIzu_.mjs.map +0 -1
  95. package/dist/client/_floatingIpId-COK_TBrz2.mjs.map +0 -1
  96. package/dist/client/_imageId-n3RcnhAR.mjs +0 -534
  97. package/dist/client/_pcaId-B-f_6kqs.mjs +0 -466
  98. package/dist/client/_pcaId-B-f_6kqs.mjs.map +0 -1
  99. package/dist/client/_projectId-BK9UqeYw.mjs.map +0 -1
  100. package/dist/client/_projectId-CCtpAQ8m.mjs +0 -299
  101. package/dist/client/_projectId-CCtpAQ8m.mjs.map +0 -1
  102. package/dist/client/build-DracvfrJ.mjs.map +0 -1
  103. package/dist/client/constants-4lVQXort.mjs +0 -128
  104. package/dist/client/constants-4lVQXort.mjs.map +0 -1
  105. package/dist/client/containers-DGY2hoWw.mjs.map +0 -1
  106. package/dist/client/containers-ZMFTRaQL.mjs.map +0 -1
  107. package/dist/client/containers-xfsYgRyf.mjs.map +0 -1
  108. package/dist/client/images-Da1t5KPh.mjs.map +0 -1
  109. package/dist/client/objects-B89dYCBq.mjs.map +0 -1
  110. package/dist/client/objects-CuFLUOe1.mjs.map +0 -1
  111. package/dist/client/objects-H0NN_Sja.mjs.map +0 -1
  112. package/dist/client/pca-C-UFjicP.mjs +0 -167
  113. package/dist/client/pca-C-UFjicP.mjs.map +0 -1
@@ -1 +1 @@
1
- {"version":3,"file":"_projectId-26bw-_rm.mjs","names":["useRef","useState","useEffect","createPortal","useRouteContext","SlotShadowRoot","children","ref","root","setRoot","current","shadowRoot","attachShadow","mode","div","style","display","Slot","component","Component","useShadowDOM","trpcClient","strict","content","auroraContext","client","createFileRoute","Route","component","lazyRouteComponent","$$splitComponentImporter","errorComponent","$$splitErrorComponentImporter","loader","options","context","params","data","trpcClient","auth","setCurrentScope","mutate","type","projectId","availableServices","getAvailableServices","query","accountId","domain","id","crumbDomain","path","name","crumbProject","project"],"sources":["../../src/client/components/Slot.tsx","../../src/client/routes/_auth/projects/$projectId.tsx"],"sourcesContent":["import { useRef, useState, useEffect, type ReactNode, type FC } from \"react\"\nimport { createPortal } from \"react-dom\"\nimport { useRouteContext } from \"@tanstack/react-router\"\nimport type { SlotProps } from \"../AuroraApp\"\n\nfunction SlotShadowRoot({ children }: { children: ReactNode }) {\n const ref = useRef<HTMLDivElement>(null)\n const [root, setRoot] = useState<ShadowRoot | null>(null)\n\n useEffect(() => {\n if (!ref.current) return\n setRoot(ref.current.shadowRoot ?? ref.current.attachShadow({ mode: \"open\" }))\n }, [])\n\n return (\n <div ref={ref} style={{ display: \"contents\" }}>\n {root && createPortal(children, root)}\n </div>\n )\n}\n\nexport function Slot({\n component: Component,\n useShadowDOM = true,\n}: {\n component: FC<SlotProps>\n useShadowDOM?: boolean\n}) {\n const { trpcClient } = useRouteContext({ strict: false })\n\n if (!trpcClient) return null\n\n const content = <Component auroraContext={{ client: trpcClient }} />\n\n if (!useShadowDOM) {\n return content\n }\n\n return <SlotShadowRoot>{content}</SlotShadowRoot>\n}\n","import { createFileRoute, Outlet, useLoaderData } from \"@tanstack/react-router\"\nimport { AppShell, Container, Stack } from \"@cloudoperators/juno-ui-components\"\nimport { SideNavBar } from \"@/client/routes/_auth/projects/-components/SideNavBar\"\nimport { ProjectInfoBox } from \"@/client/components/ProjectView/ProjectInfoBox\"\nimport { RouteError } from \"@/client/components/Error/RouteError\"\n\nexport const Route = createFileRoute(\"/_auth/projects/$projectId\")({\n component: RouteComponent,\n errorComponent: ({ error }) => {\n return <RouteError error={error} />\n },\n loader: async (options) => {\n const { context, params } = options\n const data = await context.trpcClient?.auth.setCurrentScope.mutate({\n type: \"project\",\n projectId: params.projectId || \"\",\n })\n const availableServices = await context.trpcClient?.auth.getAvailableServices.query()\n\n // Extract accountId (domain id) from the rescoped token\n // This is needed for SideNavBar navigation until we refactor it\n const accountId = data?.domain?.id || \"\"\n\n return {\n trpcClient: context.trpcClient,\n crumbDomain: { path: `/projects`, name: data?.domain?.name },\n crumbProject: data?.project,\n availableServices,\n accountId, // Keep for SideNavBar compatibility\n projectId: params.projectId,\n }\n },\n})\n\nfunction RouteComponent() {\n const { availableServices, projectId, crumbProject } = useLoaderData({ from: Route.id })\n\n return (\n <AppShell\n embedded\n sideNavigation={\n <SideNavBar\n availableServices={availableServices!}\n projectId={projectId}\n projectName={crumbProject?.name || projectId}\n />\n }\n className=\"h-min-screen\"\n >\n <Container>\n <Stack direction=\"vertical\" distribution=\"start\" alignment=\"stretch\" className=\"xl:flex-row\" gap=\"6\">\n {/* Main content area */}\n <div className=\"min-w-0 flex-1\">\n <ProjectInfoBox\n projectInfo={{\n id: projectId,\n name: crumbProject?.name || projectId,\n domain: crumbProject?.domain,\n }}\n />\n <Outlet />\n </div>\n </Stack>\n </Container>\n </AppShell>\n )\n}\n"],"mappings":";;;;;AAKA,SAASK,EAAe,EAAEC,eAAmC;CAC3D,IAAMC,IAAMP,EAAuB,IAAA,GAC7B,CAACQ,GAAMC,KAAWR,EAA4B,IAAA;CAOpD,OALAC,QAAU;EACHK,EAAIG,WACTD,EAAQF,EAAIG,QAAQC,cAAcJ,EAAIG,QAAQE,aAAa,EAAEC,MAAM,OAAO,CAAA,CAAA;CAC5E,GAAG,CAAA,CAAE,GAGH,gBAACC,OAAAA;EAASP;EAAKQ,OAAO,EAAEC,SAAS,WAAW;YACzCR,KAAQL,gBAAaG,GAAUE,CAAAA;;AAGtC;AAEA,SAAgBS,EAAK,EACnBC,WAAWC,GACXC,kBAAe,MAIhB;CACC,IAAM,EAAEC,kBAAejB,EAAgB,EAAEkB,QAAQ,GAAM,CAAA;CAEvD,IAAI,CAACD,GAAY,OAAO;CAExB,IAAME,IAAU,gBAACJ,GAAAA,EAAUK,eAAe,EAAEC,QAAQJ,EAAW,EAAA,CAAA;CAM/D,OAJKD,IAIE,gBAACf,GAAAA,EAAAA,UAAgBkB,EAAAA,CAAAA,IAHfA;AAIX;ACjCA,IAAaI,IAAQD,EAAgB,4BAAA,EAA8B;CACjEE,WAASC,6CAAA,WAAA;CACTE,gBAAcF,6CAAA,gBAAA;CAGdI,QAAQ,OAAOC,MAAAA;EACb,IAAM,EAAEC,YAASC,cAAWF,GACtBG,IAAO,MAAMF,EAAQG,YAAYC,KAAKC,gBAAgBC,OAAO;GACjEC,MAAM;GACNC,WAAWP,EAAOO,aAAa;EACjC,CAAA,GACMC,IAAoB,MAAMT,EAAQG,YAAYC,KAAKM,qBAAqBC,MAAAA,GAIxEC,IAAYV,GAAMW,QAAQC,MAAM;EAEtC,OAAO;GACLX,YAAYH,EAAQG;GACpBY,aAAa;IAAEC,MAAM;IAAaC,MAAMf,GAAMW,QAAQI;GAAK;GAC3DC,cAAchB,GAAMiB;GACpBV;GACAG;GACAJ,WAAWP,EAAOO;EACpB;CACF;AACF,CAAA"}
1
+ {"version":3,"file":"_projectId-BDSWnMGj.mjs","names":["useRef","useState","useEffect","createPortal","useRouteContext","SlotShadowRoot","children","ref","root","setRoot","current","shadowRoot","attachShadow","mode","div","style","display","Slot","component","Component","useShadowDOM","trpcClient","strict","content","auroraContext","client","createFileRoute","Route","component","lazyRouteComponent","$$splitComponentImporter","errorComponent","$$splitErrorComponentImporter","loader","options","context","params","data","trpcClient","auth","setCurrentScope","mutate","type","projectId","availableServices","getAvailableServices","query","accountId","domain","id","crumbDomain","path","name","crumbProject","project"],"sources":["../../src/client/components/Slot.tsx","../../src/client/routes/_auth/projects/$projectId.tsx"],"sourcesContent":["import { useRef, useState, useEffect, type ReactNode, type FC } from \"react\"\nimport { createPortal } from \"react-dom\"\nimport { useRouteContext } from \"@tanstack/react-router\"\nimport type { SlotProps } from \"../AuroraApp\"\n\nfunction SlotShadowRoot({ children }: { children: ReactNode }) {\n const ref = useRef<HTMLDivElement>(null)\n const [root, setRoot] = useState<ShadowRoot | null>(null)\n\n useEffect(() => {\n if (!ref.current) return\n setRoot(ref.current.shadowRoot ?? ref.current.attachShadow({ mode: \"open\" }))\n }, [])\n\n return (\n <div ref={ref} style={{ display: \"contents\" }}>\n {root && createPortal(children, root)}\n </div>\n )\n}\n\nexport function Slot({\n component: Component,\n useShadowDOM = true,\n}: {\n component: FC<SlotProps>\n useShadowDOM?: boolean\n}) {\n const { trpcClient } = useRouteContext({ strict: false })\n\n if (!trpcClient) return null\n\n const content = <Component auroraContext={{ client: trpcClient }} />\n\n if (!useShadowDOM) {\n return content\n }\n\n return <SlotShadowRoot>{content}</SlotShadowRoot>\n}\n","import { createFileRoute, Outlet, useLoaderData } from \"@tanstack/react-router\"\nimport { AppShell, Container, Stack } from \"@cloudoperators/juno-ui-components\"\nimport { SideNavBar } from \"@/client/routes/_auth/projects/-components/SideNavBar\"\nimport { ProjectInfoBox } from \"@/client/components/ProjectView/ProjectInfoBox\"\nimport { RouteError } from \"@/client/components/Error/RouteError\"\n\nexport const Route = createFileRoute(\"/_auth/projects/$projectId\")({\n component: RouteComponent,\n errorComponent: ({ error }) => {\n return <RouteError error={error} />\n },\n loader: async (options) => {\n const { context, params } = options\n const data = await context.trpcClient?.auth.setCurrentScope.mutate({\n type: \"project\",\n projectId: params.projectId || \"\",\n })\n const availableServices = await context.trpcClient?.auth.getAvailableServices.query()\n\n // Extract accountId (domain id) from the rescoped token\n // This is needed for SideNavBar navigation until we refactor it\n const accountId = data?.domain?.id || \"\"\n\n return {\n trpcClient: context.trpcClient,\n crumbDomain: { path: `/projects`, name: data?.domain?.name },\n crumbProject: data?.project,\n availableServices,\n accountId, // Keep for SideNavBar compatibility\n projectId: params.projectId,\n }\n },\n})\n\nfunction RouteComponent() {\n const { availableServices, projectId, crumbProject, crumbDomain } = useLoaderData({ from: Route.id })\n\n return (\n <AppShell\n embedded\n sideNavigation={\n <SideNavBar\n availableServices={availableServices!}\n projectId={projectId}\n projectName={crumbProject?.name || projectId}\n domainName={crumbDomain?.name}\n />\n }\n className=\"h-min-screen\"\n >\n <Container>\n <Stack direction=\"vertical\" distribution=\"start\" alignment=\"stretch\" className=\"xl:flex-row\" gap=\"6\">\n {/* Main content area */}\n <div className=\"min-w-0 flex-1\">\n <ProjectInfoBox\n projectInfo={{\n id: projectId,\n name: crumbProject?.name || projectId,\n domain: crumbProject?.domain,\n }}\n />\n <Outlet />\n </div>\n </Stack>\n </Container>\n </AppShell>\n )\n}\n"],"mappings":";;;;;AAKA,SAASK,EAAe,EAAEC,eAAmC;CAC3D,IAAMC,IAAMP,EAAuB,IAAA,GAC7B,CAACQ,GAAMC,KAAWR,EAA4B,IAAA;CAOpD,OALAC,QAAU;EACHK,EAAIG,WACTD,EAAQF,EAAIG,QAAQC,cAAcJ,EAAIG,QAAQE,aAAa,EAAEC,MAAM,OAAO,CAAA,CAAA;CAC5E,GAAG,CAAA,CAAE,GAGH,gBAACC,OAAAA;EAASP;EAAKQ,OAAO,EAAEC,SAAS,WAAW;YACzCR,KAAQL,gBAAaG,GAAUE,CAAAA;;AAGtC;AAEA,SAAgBS,EAAK,EACnBC,WAAWC,GACXC,kBAAe,MAIhB;CACC,IAAM,EAAEC,kBAAejB,EAAgB,EAAEkB,QAAQ,GAAM,CAAA;CAEvD,IAAI,CAACD,GAAY,OAAO;CAExB,IAAME,IAAU,gBAACJ,GAAAA,EAAUK,eAAe,EAAEC,QAAQJ,EAAW,EAAA,CAAA;CAM/D,OAJKD,IAIE,gBAACf,GAAAA,EAAAA,UAAgBkB,EAAAA,CAAAA,IAHfA;AAIX;ACjCA,IAAaI,IAAQD,EAAgB,4BAAA,EAA8B;CACjEE,WAASC,6CAAA,WAAA;CACTE,gBAAcF,6CAAA,gBAAA;CAGdI,QAAQ,OAAOC,MAAAA;EACb,IAAM,EAAEC,YAASC,cAAWF,GACtBG,IAAO,MAAMF,EAAQG,YAAYC,KAAKC,gBAAgBC,OAAO;GACjEC,MAAM;GACNC,WAAWP,EAAOO,aAAa;EACjC,CAAA,GACMC,IAAoB,MAAMT,EAAQG,YAAYC,KAAKM,qBAAqBC,MAAAA,GAIxEC,IAAYV,GAAMW,QAAQC,MAAM;EAEtC,OAAO;GACLX,YAAYH,EAAQG;GACpBY,aAAa;IAAEC,MAAM;IAAaC,MAAMf,GAAMW,QAAQI;GAAK;GAC3DC,cAAchB,GAAMiB;GACpBV;GACAG;GACAJ,WAAWP,EAAOO;EACpB;CACF;AACF,CAAA"}
@@ -1,8 +1,8 @@
1
- import { t as e } from "./RouteError-CUj_m3gu.mjs";
1
+ import { t as e } from "./RouteError-QSV7qOoJ.mjs";
2
2
  import { jsx as t } from "react/jsx-runtime";
3
3
  //#region src/client/routes/_auth/projects/$projectId.tsx?tsr-split=errorComponent
4
4
  var n = ({ error: n }) => /*#__PURE__*/ t(e, { error: n });
5
5
  //#endregion
6
6
  export { n as errorComponent };
7
7
 
8
- //# sourceMappingURL=_projectId-Dhb4AyBD.mjs.map
8
+ //# sourceMappingURL=_projectId-DOgwFiqD.mjs.map
@@ -1 +1 @@
1
- {"version":3,"file":"_projectId-Dhb4AyBD.mjs","names":["RouteError","SplitErrorComponent","error","errorComponent"],"sources":["../../src/client/routes/_auth/projects/$projectId.tsx?tsr-split=errorComponent"],"sourcesContent":["import { createFileRoute, Outlet, useLoaderData } from \"@tanstack/react-router\"\nimport { AppShell, Container, Stack } from \"@cloudoperators/juno-ui-components\"\nimport { SideNavBar } from \"@/client/routes/_auth/projects/-components/SideNavBar\"\nimport { ProjectInfoBox } from \"@/client/components/ProjectView/ProjectInfoBox\"\nimport { RouteError } from \"@/client/components/Error/RouteError\"\n\nexport const Route = createFileRoute(\"/_auth/projects/$projectId\")({\n component: RouteComponent,\n errorComponent: ({ error }) => {\n return <RouteError error={error} />\n },\n loader: async (options) => {\n const { context, params } = options\n const data = await context.trpcClient?.auth.setCurrentScope.mutate({\n type: \"project\",\n projectId: params.projectId || \"\",\n })\n const availableServices = await context.trpcClient?.auth.getAvailableServices.query()\n\n // Extract accountId (domain id) from the rescoped token\n // This is needed for SideNavBar navigation until we refactor it\n const accountId = data?.domain?.id || \"\"\n\n return {\n trpcClient: context.trpcClient,\n crumbDomain: { path: `/projects`, name: data?.domain?.name },\n crumbProject: data?.project,\n availableServices,\n accountId, // Keep for SideNavBar compatibility\n projectId: params.projectId,\n }\n },\n})\n\nfunction RouteComponent() {\n const { availableServices, projectId, crumbProject } = useLoaderData({ from: Route.id })\n\n return (\n <AppShell\n embedded\n sideNavigation={\n <SideNavBar\n availableServices={availableServices!}\n projectId={projectId}\n projectName={crumbProject?.name || projectId}\n />\n }\n className=\"h-min-screen\"\n >\n <Container>\n <Stack direction=\"vertical\" distribution=\"start\" alignment=\"stretch\" className=\"xl:flex-row\" gap=\"6\">\n {/* Main content area */}\n <div className=\"min-w-0 flex-1\">\n <ProjectInfoBox\n projectInfo={{\n id: projectId,\n name: crumbProject?.name || projectId,\n domain: crumbProject?.domain,\n }}\n />\n <Outlet />\n </div>\n </Stack>\n </Container>\n </AppShell>\n )\n}\n"],"mappings":";;;AAIiE,IAAAC,KAI9C,EAAEC,eACV,gBAAC,GAAA,EAAkBA,SAAAA,CAAAA"}
1
+ {"version":3,"file":"_projectId-DOgwFiqD.mjs","names":["RouteError","SplitErrorComponent","error","errorComponent"],"sources":["../../src/client/routes/_auth/projects/$projectId.tsx?tsr-split=errorComponent"],"sourcesContent":["import { createFileRoute, Outlet, useLoaderData } from \"@tanstack/react-router\"\nimport { AppShell, Container, Stack } from \"@cloudoperators/juno-ui-components\"\nimport { SideNavBar } from \"@/client/routes/_auth/projects/-components/SideNavBar\"\nimport { ProjectInfoBox } from \"@/client/components/ProjectView/ProjectInfoBox\"\nimport { RouteError } from \"@/client/components/Error/RouteError\"\n\nexport const Route = createFileRoute(\"/_auth/projects/$projectId\")({\n component: RouteComponent,\n errorComponent: ({ error }) => {\n return <RouteError error={error} />\n },\n loader: async (options) => {\n const { context, params } = options\n const data = await context.trpcClient?.auth.setCurrentScope.mutate({\n type: \"project\",\n projectId: params.projectId || \"\",\n })\n const availableServices = await context.trpcClient?.auth.getAvailableServices.query()\n\n // Extract accountId (domain id) from the rescoped token\n // This is needed for SideNavBar navigation until we refactor it\n const accountId = data?.domain?.id || \"\"\n\n return {\n trpcClient: context.trpcClient,\n crumbDomain: { path: `/projects`, name: data?.domain?.name },\n crumbProject: data?.project,\n availableServices,\n accountId, // Keep for SideNavBar compatibility\n projectId: params.projectId,\n }\n },\n})\n\nfunction RouteComponent() {\n const { availableServices, projectId, crumbProject, crumbDomain } = useLoaderData({ from: Route.id })\n\n return (\n <AppShell\n embedded\n sideNavigation={\n <SideNavBar\n availableServices={availableServices!}\n projectId={projectId}\n projectName={crumbProject?.name || projectId}\n domainName={crumbDomain?.name}\n />\n }\n className=\"h-min-screen\"\n >\n <Container>\n <Stack direction=\"vertical\" distribution=\"start\" alignment=\"stretch\" className=\"xl:flex-row\" gap=\"6\">\n {/* Main content area */}\n <div className=\"min-w-0 flex-1\">\n <ProjectInfoBox\n projectInfo={{\n id: projectId,\n name: crumbProject?.name || projectId,\n domain: crumbProject?.domain,\n }}\n />\n <Outlet />\n </div>\n </Stack>\n </Container>\n </AppShell>\n )\n}\n"],"mappings":";;;AAIiE,IAAAC,KAI9C,EAAEC,eACV,gBAAC,GAAA,EAAkBA,SAAAA,CAAAA"}
@@ -0,0 +1,299 @@
1
+ import { $ as e, D as t, I as n, L as r, O as i, Q as a, Y as o, h as s, o as c, p as l } from "./build-DeJcDjPi.mjs";
2
+ import { n as u, t as d } from "./_projectId-BDSWnMGj.mjs";
3
+ import { t as f } from "./helpers-1PpYf-fC.mjs";
4
+ import { Fragment as p, jsx as m, jsxs as h } from "react/jsx-runtime";
5
+ import { useEffect as g, useMemo as _, useRef as v, useState as y } from "react";
6
+ import { Outlet as b, useLoaderData as x, useMatches as S, useNavigate as C, useParams as w, useRouteContext as T } from "@tanstack/react-router";
7
+ import { useLingui as E } from "@lingui/react";
8
+ import { z as D } from "zod";
9
+ var O = D.object({
10
+ labelKey: D.enum([
11
+ "Compute",
12
+ "Network",
13
+ "Storage",
14
+ "Services",
15
+ "Images",
16
+ "Flavors",
17
+ "Security Groups",
18
+ "Floating IPs",
19
+ "PCA (Clavis)"
20
+ ]).optional(),
21
+ to: D.string().optional(),
22
+ useParamAsLabel: D.string().optional(),
23
+ useParentTitleAsLabel: D.boolean().optional()
24
+ }), k = D.object({
25
+ section: D.string(),
26
+ service: D.string().optional(),
27
+ isDetail: D.boolean().optional(),
28
+ crumb: O.optional(),
29
+ sectionCrumb: O.optional(),
30
+ intermediateCrumb: O.optional()
31
+ });
32
+ function A(e) {
33
+ return k.safeParse(e).success;
34
+ }
35
+ //#endregion
36
+ //#region src/client/routes/_auth/projects/-components/SideNavBar.tsx
37
+ var j = ({ projectId: e, projectName: n, domainName: i, availableServices: a }) => {
38
+ let { i18n: o, _: d } = E(), _ = C(), b = S(), { provider: x } = w({ strict: !1 }), { slots: D } = T({ strict: !1 }), O = [...b].reverse().find((e) => A(e.staticData)), k = O && A(O.staticData) ? O.staticData : void 0, j = k?.section ?? null, M = k?.service ?? null, N = f(a), [P, F] = y({
39
+ compute: !0,
40
+ network: !0,
41
+ storage: !0,
42
+ services: !0
43
+ }), I = v(null), L = v(!1);
44
+ g(() => {
45
+ let e = I.current;
46
+ I.current = j;
47
+ let t = L.current;
48
+ if (L.current = !0, t && j && j !== e && j in P) {
49
+ F((e) => ({
50
+ ...e,
51
+ [j]: !1
52
+ }));
53
+ let e = j;
54
+ setTimeout(() => F((t) => ({
55
+ ...t,
56
+ [e]: !0
57
+ })), 0);
58
+ }
59
+ }, [j]);
60
+ let R = [...N.image?.glance ? [{
61
+ service: "images",
62
+ label: o._({ id: "an5hVd" }),
63
+ to: "/projects/$projectId/compute/images",
64
+ params: { projectId: e }
65
+ }] : [], ...N?.compute?.nova ? [{
66
+ service: "flavors",
67
+ label: o._({ id: "neiJm0" }),
68
+ to: "/projects/$projectId/compute/flavors",
69
+ params: { projectId: e }
70
+ }] : []], z = [...N.network ? [{
71
+ service: "securitygroups",
72
+ label: o._({ id: "4opp4r" }),
73
+ to: "/projects/$projectId/network/securitygroups",
74
+ params: { projectId: e }
75
+ }, {
76
+ service: "floatingips",
77
+ label: o._({ id: "u77/s4" }),
78
+ to: "/projects/$projectId/network/floatingips",
79
+ params: { projectId: e }
80
+ }] : []], B = [...N?.["object-store"]?.swift ? [{
81
+ service: "containers",
82
+ label: o._({ id: "+OEi73" }),
83
+ to: "/projects/$projectId/storage/$provider/$storageType",
84
+ params: {
85
+ projectId: e,
86
+ provider: "swift",
87
+ storageType: "containers"
88
+ }
89
+ }] : [], {
90
+ service: "ceph-containers",
91
+ label: o._({ id: "KhNDX4" }),
92
+ to: "/projects/$projectId/storage/$provider/$storageType",
93
+ params: {
94
+ projectId: e,
95
+ provider: "ceph",
96
+ storageType: "buckets"
97
+ }
98
+ }], V = [...N.pca?.["clavis-beta"] || N.pca?.["clavis-dev"] ? [{
99
+ service: "pca",
100
+ label: o._({ id: "miy5mb" }),
101
+ to: "/projects/$projectId/services/pca",
102
+ params: { projectId: e }
103
+ }] : []];
104
+ return /*#__PURE__*/ m(r, {
105
+ ariaLabel: "Project Side Navigation",
106
+ children: /*#__PURE__*/ h(p, { children: [/*#__PURE__*/ m(l, { children: /*#__PURE__*/ h(p, { children: [
107
+ /*#__PURE__*/ m(c, {
108
+ onClick: () => _({
109
+ to: "/projects/$projectId",
110
+ params: { projectId: e }
111
+ }),
112
+ label: /*#__PURE__*/ h(p, { children: [i && /*#__PURE__*/ h("p", {
113
+ className: "text-theme-light text-xs leading-4 font-bold",
114
+ children: [i, " /"]
115
+ }), /*#__PURE__*/ m("p", {
116
+ className: "leading-5 font-normal",
117
+ children: n
118
+ })] })
119
+ }),
120
+ /*#__PURE__*/ m(t, { spacing: "1" }),
121
+ /*#__PURE__*/ m(s, {
122
+ label: o._({ id: "rp0Bd0" }),
123
+ open: P.compute,
124
+ children: R.map(({ service: e, label: t, to: n, params: r }) => /*#__PURE__*/ m(c, {
125
+ onClick: () => _({
126
+ to: n,
127
+ params: r
128
+ }),
129
+ label: t,
130
+ selected: j === "compute" && M === e
131
+ }, t))
132
+ }),
133
+ z.length > 0 && /*#__PURE__*/ m(s, {
134
+ label: o._({ id: "OR475H" }),
135
+ open: P.network,
136
+ children: z.map(({ service: e, label: t, to: n, params: r }) => /*#__PURE__*/ m(c, {
137
+ onClick: () => _({
138
+ to: n,
139
+ params: r
140
+ }),
141
+ label: t,
142
+ selected: j === "network" && M === e
143
+ }, t))
144
+ }),
145
+ B.length > 0 && /*#__PURE__*/ m(s, {
146
+ label: o._({ id: "BrrIs8" }),
147
+ open: P.storage,
148
+ children: B.map(({ service: e, label: t, to: n, params: r }) => /*#__PURE__*/ m(c, {
149
+ onClick: () => _({
150
+ to: n,
151
+ params: r
152
+ }),
153
+ label: t,
154
+ selected: j === "storage" && M === "containers" ? r.provider === x : M === e
155
+ }, t))
156
+ }),
157
+ V.length > 0 && /*#__PURE__*/ m(s, {
158
+ label: o._({ id: "MILoeL" }),
159
+ open: P.services,
160
+ children: V.map(({ service: e, label: t, to: n, params: r }) => /*#__PURE__*/ m(c, {
161
+ onClick: () => _({
162
+ to: n,
163
+ params: r
164
+ }),
165
+ label: t,
166
+ selected: j === "services" && M === e
167
+ }, t))
168
+ })
169
+ ] }) }), D?.sideNavBanner && /*#__PURE__*/ m(u, { component: D.sideNavBanner })] })
170
+ });
171
+ };
172
+ //#endregion
173
+ //#region src/client/components/ProjectView/ProjectInfoBox.tsx
174
+ function M({ projectInfo: t }) {
175
+ let { i18n: n, _: r } = E(), a = C(), o = S(), { projectId: s } = w({ strict: !1 });
176
+ return /*#__PURE__*/ m(i, {
177
+ className: "relative z-1 mt-8 mb-4",
178
+ children: _(() => {
179
+ let e = {
180
+ Compute: n._({ id: "rp0Bd0" }),
181
+ Network: n._({ id: "OR475H" }),
182
+ Storage: n._({ id: "BrrIs8" }),
183
+ Services: n._({ id: "MILoeL" }),
184
+ Images: n._({ id: "an5hVd" }),
185
+ Flavors: n._({ id: "neiJm0" }),
186
+ "Security Groups": n._({ id: "4opp4r" }),
187
+ "Floating IPs": n._({ id: "u77/s4" }),
188
+ "PCA (Clavis)": n._({ id: "miy5mb" })
189
+ }, r = (e) => e === "swift" ? n._({ id: "+OEi73" }) : e === "ceph" ? n._({ id: "KhNDX4" }) : n._({ id: "BrrIs8" }), i = [];
190
+ i.push({
191
+ icon: "home",
192
+ label: n._({ id: "i0qMbr" }),
193
+ onClick: () => a({ to: "/projects" })
194
+ }), t.domain?.name && i.push({ label: t.domain.name }), i.push({
195
+ label: t.name,
196
+ onClick: () => a({
197
+ to: "/projects/$projectId",
198
+ params: { projectId: s }
199
+ })
200
+ });
201
+ let c = o.filter((e) => e.routeId !== "/_auth/projects/$projectId" && e.routeId.startsWith("/_auth/projects/$projectId")), l = c[c.length - 1];
202
+ if (!l) return i;
203
+ let u = A(l.staticData) ? l.staticData : void 0;
204
+ if (!u) return i;
205
+ let d = l.params;
206
+ if (u.sectionCrumb?.to) {
207
+ let { labelKey: t, to: n } = u.sectionCrumb, r = t ? e[t] : void 0;
208
+ i.push({
209
+ label: r,
210
+ onClick: () => a({
211
+ to: n,
212
+ params: d
213
+ })
214
+ });
215
+ }
216
+ if (u.crumb) {
217
+ let { labelKey: t, to: n, useParamAsLabel: o } = u.crumb, s = o ? r(d[o]) : t ? e[t] : void 0;
218
+ if (u.isDetail) {
219
+ if (i.push({
220
+ label: s,
221
+ onClick: () => a({
222
+ to: n,
223
+ params: d
224
+ })
225
+ }), u.intermediateCrumb) {
226
+ let { to: e, useParamAsLabel: t, useParentTitleAsLabel: n } = u.intermediateCrumb, r = c[c.length - 2]?.meta?.find((e) => e != null && "title" in e)?.title, o = n ? r ?? (t ? d[t] : void 0) : t ? d[t] : void 0;
227
+ i.push(e ? {
228
+ label: o,
229
+ onClick: () => a({
230
+ to: e,
231
+ params: d
232
+ })
233
+ } : { label: o });
234
+ }
235
+ let e = l.meta?.find((e) => e != null && "title" in e)?.title;
236
+ e && i.push({
237
+ label: e,
238
+ active: !0
239
+ });
240
+ } else i.push(n ? {
241
+ label: s,
242
+ onClick: () => a({
243
+ to: n,
244
+ params: d
245
+ })
246
+ } : {
247
+ label: s,
248
+ active: !0
249
+ });
250
+ }
251
+ return i;
252
+ }, [
253
+ o,
254
+ t,
255
+ s,
256
+ a,
257
+ r
258
+ ]).map((t, n) => /*#__PURE__*/ m(e, {
259
+ label: t.label,
260
+ icon: t.icon,
261
+ onClick: t.onClick,
262
+ active: t.active
263
+ }, n))
264
+ });
265
+ }
266
+ //#endregion
267
+ //#region src/client/routes/_auth/projects/$projectId.tsx?tsr-split=component
268
+ function N() {
269
+ let { availableServices: e, projectId: t, crumbProject: r, crumbDomain: i } = x({ from: d.id });
270
+ return /*#__PURE__*/ m(n, {
271
+ embedded: !0,
272
+ sideNavigation: /*#__PURE__*/ m(j, {
273
+ availableServices: e,
274
+ projectId: t,
275
+ projectName: r?.name || t,
276
+ domainName: i?.name
277
+ }),
278
+ className: "h-min-screen",
279
+ children: /*#__PURE__*/ m(o, { children: /*#__PURE__*/ m(a, {
280
+ direction: "vertical",
281
+ distribution: "start",
282
+ alignment: "stretch",
283
+ className: "xl:flex-row",
284
+ gap: "6",
285
+ children: /*#__PURE__*/ h("div", {
286
+ className: "min-w-0 flex-1",
287
+ children: [/*#__PURE__*/ m(M, { projectInfo: {
288
+ id: t,
289
+ name: r?.name || t,
290
+ domain: r?.domain
291
+ } }), /*#__PURE__*/ m(b, {})]
292
+ })
293
+ }) })
294
+ });
295
+ }
296
+ //#endregion
297
+ export { N as component };
298
+
299
+ //# sourceMappingURL=_projectId-DS4nR59B.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"_projectId-DS4nR59B.mjs","names":["z","CRUMB_LABEL_KEYS","CrumbSchema","object","labelKey","enum","optional","to","string","useParamAsLabel","useParentTitleAsLabel","boolean","RouteInfoSchema","section","service","isDetail","crumb","sectionCrumb","intermediateCrumb","isRouteInfo","data","safeParse","success","useNavigate","useMatches","useParams","useRouteContext","useState","useEffect","useRef","getServiceIndex","SideNavigation","SideNavigationList","SideNavigationGroup","SideNavigationItem","Divider","isRouteInfo","Slot","SideNavBar","projectId","projectName","domainName","availableServices","useLingui","navigate","matches","provider","strict","slots","activeMatch","reverse","find","m","staticData","activeRouteInfo","undefined","activeSection","section","activeService","service","serviceIndex","openSections","setOpenSections","compute","network","storage","services","prevSectionRef","mountedRef","prev","current","wasMounted","s","setTimeout","computeServices","label","t","to","params","networkServices","storageServices","storageType","pcaServices","clavisServices","ariaLabel","onClick","p","className","spacing","open","map","selected","length","isStorageContainers","isSelected","sideNavBanner","component","Breadcrumb","BreadcrumbItem","useMatches","useNavigate","useParams","useMemo","isRouteInfo","ProjectInfoBox","projectInfo","useLingui","navigate","matches","projectId","strict","breadcrumbs","crumbLabels","Compute","t","Network","Storage","Services","Images","Flavors","resolveProviderLabel","provider","items","push","icon","label","onClick","to","domain","name","params","projectMatches","filter","m","routeId","startsWith","deepest","length","info","staticData","undefined","sectionCrumb","labelKey","crumb","useParamAsLabel","resolvedLabel","isDetail","intermediateCrumb","iTo","iParam","useParentTitleAsLabel","parentMatch","parentTitle","meta","find","title","iLabel","active","className","map","item","index","Outlet","useLoaderData","AppShell","Container","Stack","SideNavBar","ProjectInfoBox","Route","RouteComponent","availableServices","projectId","crumbProject","crumbDomain","from","id","name","domain","component"],"sources":["../../src/client/routes/routeInfo.ts","../../src/client/routes/_auth/projects/-components/SideNavBar.tsx","../../src/client/components/ProjectView/ProjectInfoBox.tsx","../../src/client/routes/_auth/projects/$projectId.tsx?tsr-split=component"],"sourcesContent":["import { z } from \"zod\"\n\nexport const CRUMB_LABEL_KEYS = [\n \"Compute\",\n \"Network\",\n \"Storage\",\n \"Services\",\n \"Images\",\n \"Flavors\",\n \"Security Groups\",\n \"Floating IPs\",\n \"PCA (Clavis)\",\n] as const\n\nexport type CrumbLabelKey = (typeof CRUMB_LABEL_KEYS)[number]\n\nconst CrumbSchema = z.object({\n labelKey: z.enum(CRUMB_LABEL_KEYS).optional(),\n to: z.string().optional(),\n useParamAsLabel: z.string().optional(),\n useParentTitleAsLabel: z.boolean().optional(),\n})\n\nconst RouteInfoSchema = z.object({\n section: z.string(),\n service: z.string().optional(),\n isDetail: z.boolean().optional(),\n crumb: CrumbSchema.optional(),\n sectionCrumb: CrumbSchema.optional(),\n intermediateCrumb: CrumbSchema.optional(),\n})\n\nexport type Crumb = z.infer<typeof CrumbSchema>\nexport type RouteInfo = z.infer<typeof RouteInfoSchema>\n\nexport function isRouteInfo(data: unknown): data is RouteInfo {\n return RouteInfoSchema.safeParse(data).success\n}\n","import { useNavigate, useMatches, useParams, useRouteContext } from \"@tanstack/react-router\"\nimport { useState, useEffect, useRef } from \"react\"\nimport { getServiceIndex } from \"@/server/Authentication/helpers\"\nimport {\n SideNavigation,\n SideNavigationList,\n SideNavigationGroup,\n SideNavigationItem,\n Divider,\n} from \"@cloudoperators/juno-ui-components/index\"\nimport { useLingui } from \"@lingui/react/macro\"\nimport { isRouteInfo } from \"@/client/routes/routeInfo\"\nimport { Slot } from \"@/client/components/Slot\"\n\ninterface SideNavBarProps {\n projectId: string\n projectName: string\n domainName?: string\n availableServices: {\n type: string\n name: string\n }[]\n}\n\nexport const SideNavBar = ({ projectId, projectName, domainName, availableServices }: SideNavBarProps) => {\n const { t } = useLingui()\n const navigate = useNavigate()\n const matches = useMatches()\n const { provider } = useParams({ strict: false }) as { provider?: string }\n const { slots } = useRouteContext({ strict: false })\n\n // Read active section/service from the deepest match that has valid RouteInfo staticData\n const activeMatch = [...matches].reverse().find((m) => isRouteInfo(m.staticData))\n const activeRouteInfo = activeMatch && isRouteInfo(activeMatch.staticData) ? activeMatch.staticData : undefined\n const activeSection = activeRouteInfo?.section ?? null\n const activeService = activeRouteInfo?.service ?? null\n\n const serviceIndex = getServiceIndex(availableServices)\n\n const [openSections, setOpenSections] = useState({ compute: true, network: true, storage: true, services: true })\n const prevSectionRef = useRef<string | null>(null)\n const mountedRef = useRef(false)\n\n useEffect(() => {\n const prev = prevSectionRef.current\n prevSectionRef.current = activeSection\n const wasMounted = mountedRef.current\n mountedRef.current = true\n // Skip on initial mount: all sections start open, Juno initializes correctly from the open prop.\n // Only re-open when navigating to a section that Juno may have internally collapsed.\n if (!wasMounted) return\n if (activeSection && activeSection !== prev && activeSection in openSections) {\n // Set false first, then true in the next tick so Juno's useEffect([open]) sees the change\n // even if the section was already true in our state (Juno may have internally collapsed it).\n setOpenSections((s) => ({ ...s, [activeSection]: false }))\n const section = activeSection\n setTimeout(() => setOpenSections((s) => ({ ...s, [section]: true })), 0)\n }\n }, [activeSection])\n\n const computeServices = [\n ...(serviceIndex[\"image\"]?.[\"glance\"]\n ? [\n {\n service: \"images\",\n label: t`Images`,\n to: \"/projects/$projectId/compute/images\" as const,\n params: { projectId },\n },\n ]\n : []),\n ...(serviceIndex?.[\"compute\"]?.[\"nova\"]\n ? [\n {\n service: \"flavors\",\n label: t`Flavors`,\n to: \"/projects/$projectId/compute/flavors\" as const,\n params: { projectId },\n },\n ]\n : []),\n ]\n\n const networkServices = [\n ...(serviceIndex[\"network\"]\n ? [\n {\n service: \"securitygroups\",\n label: t`Security Groups`,\n to: \"/projects/$projectId/network/securitygroups\" as const,\n params: { projectId },\n },\n {\n service: \"floatingips\",\n label: t`Floating IPs`,\n to: \"/projects/$projectId/network/floatingips\" as const,\n params: { projectId },\n },\n ]\n : []),\n ]\n\n const storageServices = [\n ...(serviceIndex?.[\"object-store\"]?.[\"swift\"]\n ? [\n {\n service: \"containers\",\n label: t`Object Storage (Swift)`,\n to: \"/projects/$projectId/storage/$provider/$storageType\" as const,\n params: { projectId, provider: \"swift\", storageType: \"containers\" },\n },\n ]\n : []),\n {\n service: \"ceph-containers\",\n label: t`Object Storage (Ceph)`,\n to: \"/projects/$projectId/storage/$provider/$storageType\" as const,\n params: { projectId, provider: \"ceph\", storageType: \"buckets\" },\n },\n ]\n\n // temporary as clavis is not fully GA, after GA replace with [\"pca\"]?.[\"clavis\"]\n const pcaServices = serviceIndex[\"pca\"]?.[\"clavis-beta\"] || serviceIndex[\"pca\"]?.[\"clavis-dev\"]\n const clavisServices = [\n ...(pcaServices\n ? [\n {\n service: \"pca\",\n label: t`PCA (Clavis)`,\n to: \"/projects/$projectId/services/pca\" as const,\n params: { projectId },\n },\n ]\n : []),\n ]\n\n return (\n <SideNavigation ariaLabel=\"Project Side Navigation\">\n <>\n <SideNavigationList>\n <>\n <SideNavigationItem\n onClick={() => navigate({ to: \"/projects/$projectId\", params: { projectId } })}\n label={\n <>\n {domainName && <p className=\"text-theme-light text-xs leading-4 font-bold\">{domainName} /</p>}\n <p className=\"leading-5 font-normal\">{projectName}</p>\n </>\n }\n />\n <Divider spacing=\"1\" />\n <SideNavigationGroup label={t`Compute`} open={openSections.compute}>\n {computeServices.map(({ service, label, to, params }) => (\n <SideNavigationItem\n key={label}\n onClick={() => navigate({ to, params })}\n label={label}\n selected={activeSection === \"compute\" && activeService === service}\n />\n ))}\n </SideNavigationGroup>\n\n {networkServices.length > 0 && (\n <SideNavigationGroup label={t`Network`} open={openSections.network}>\n {networkServices.map(({ service, label, to, params }) => (\n <SideNavigationItem\n key={label}\n onClick={() => navigate({ to, params })}\n label={label}\n selected={activeSection === \"network\" && activeService === service}\n />\n ))}\n </SideNavigationGroup>\n )}\n\n {storageServices.length > 0 && (\n <SideNavigationGroup label={t`Storage`} open={openSections.storage}>\n {storageServices.map(({ service, label, to, params }) => {\n // For storage services with provider param, match against current provider\n const isStorageContainers = activeSection === \"storage\" && activeService === \"containers\"\n const isSelected = isStorageContainers ? params.provider === provider : activeService === service\n\n return (\n <SideNavigationItem\n key={label}\n onClick={() => navigate({ to, params })}\n label={label}\n selected={isSelected}\n />\n )\n })}\n </SideNavigationGroup>\n )}\n\n {clavisServices.length > 0 && (\n <SideNavigationGroup label={t`Services`} open={openSections.services}>\n {clavisServices.map(({ service, label, to, params }) => (\n <SideNavigationItem\n key={label}\n onClick={() => navigate({ to, params })}\n label={label}\n selected={activeSection === \"services\" && activeService === service}\n />\n ))}\n </SideNavigationGroup>\n )}\n </>\n </SideNavigationList>\n {slots?.sideNavBanner && <Slot component={slots.sideNavBanner} />}\n </>\n </SideNavigation>\n )\n}\n","import { Breadcrumb, BreadcrumbItem, KnownIcons } from \"@cloudoperators/juno-ui-components\"\nimport { useMatches, useNavigate, useParams } from \"@tanstack/react-router\"\nimport { useMemo } from \"react\"\nimport { useLingui } from \"@lingui/react/macro\"\nimport { isRouteInfo, CrumbLabelKey } from \"@/client/routes/routeInfo\"\n\ninterface ProjectInfoBoxProps {\n projectInfo: {\n id: string\n name: string\n description?: string\n domain?: {\n name?: string\n }\n }\n}\n\nexport function ProjectInfoBox({ projectInfo }: ProjectInfoBoxProps) {\n const { t } = useLingui()\n const navigate = useNavigate()\n const matches = useMatches()\n const { projectId } = useParams({ strict: false }) as { projectId: string }\n\n const breadcrumbs = useMemo(() => {\n const crumbLabels: Record<CrumbLabelKey, string> = {\n Compute: t`Compute`,\n Network: t`Network`,\n Storage: t`Storage`,\n Services: t`Services`,\n Images: t`Images`,\n Flavors: t`Flavors`,\n \"Security Groups\": t`Security Groups`,\n \"Floating IPs\": t`Floating IPs`,\n \"PCA (Clavis)\": t`PCA (Clavis)`,\n }\n\n const resolveProviderLabel = (provider: string | undefined) => {\n if (provider === \"swift\") return t`Object Storage (Swift)`\n if (provider === \"ceph\") return t`Object Storage (Ceph)`\n return t`Storage`\n }\n\n const items: Array<{ label?: string; icon?: KnownIcons; onClick?: () => void; active?: boolean }> = []\n\n items.push({ icon: \"home\", label: t`Home`, onClick: () => navigate({ to: \"/projects\" }) })\n\n if (projectInfo.domain?.name) {\n items.push({ label: projectInfo.domain.name })\n }\n\n items.push({\n label: projectInfo.name,\n onClick: () => navigate({ to: \"/projects/$projectId\", params: { projectId } }),\n })\n\n const projectMatches = matches.filter(\n (m) => m.routeId !== \"/_auth/projects/$projectId\" && m.routeId.startsWith(\"/_auth/projects/$projectId\")\n )\n const deepest = projectMatches[projectMatches.length - 1]\n if (!deepest) return items\n\n const info = isRouteInfo(deepest.staticData) ? deepest.staticData : undefined\n if (!info) return items\n\n const params = deepest.params as Record<string, string>\n\n if (info.sectionCrumb?.to) {\n const { labelKey, to } = info.sectionCrumb\n const label = labelKey ? crumbLabels[labelKey] : undefined\n items.push({ label, onClick: () => navigate({ to: to as never, params: params as never }) })\n }\n\n if (info.crumb) {\n const { labelKey, to, useParamAsLabel } = info.crumb\n const resolvedLabel = useParamAsLabel\n ? resolveProviderLabel(params[useParamAsLabel])\n : labelKey\n ? crumbLabels[labelKey]\n : undefined\n\n if (info.isDetail) {\n items.push({ label: resolvedLabel, onClick: () => navigate({ to: to as never, params: params as never }) })\n\n if (info.intermediateCrumb) {\n const { to: iTo, useParamAsLabel: iParam, useParentTitleAsLabel } = info.intermediateCrumb\n const parentMatch = projectMatches[projectMatches.length - 2]\n const parentTitle = parentMatch?.meta?.find((m) => m != null && \"title\" in m)?.title as string | undefined\n const iLabel = useParentTitleAsLabel\n ? (parentTitle ?? (iParam ? params[iParam] : undefined))\n : iParam\n ? params[iParam]\n : undefined\n items.push(\n iTo\n ? { label: iLabel, onClick: () => navigate({ to: iTo as never, params: params as never }) }\n : { label: iLabel }\n )\n }\n\n const title = deepest.meta?.find((m) => m != null && \"title\" in m)?.title as string | undefined\n if (title) items.push({ label: title, active: true })\n } else {\n items.push(\n to\n ? { label: resolvedLabel, onClick: () => navigate({ to: to as never, params: params as never }) }\n : { label: resolvedLabel, active: true }\n )\n }\n }\n\n return items\n }, [matches, projectInfo, projectId, navigate, t])\n\n return (\n <Breadcrumb className=\"relative z-1 mt-8 mb-4\">\n {breadcrumbs.map((item, index) => (\n <BreadcrumbItem key={index} label={item.label} icon={item.icon} onClick={item.onClick} active={item.active} />\n ))}\n </Breadcrumb>\n )\n}\n","import { createFileRoute, Outlet, useLoaderData } from \"@tanstack/react-router\"\nimport { AppShell, Container, Stack } from \"@cloudoperators/juno-ui-components\"\nimport { SideNavBar } from \"@/client/routes/_auth/projects/-components/SideNavBar\"\nimport { ProjectInfoBox } from \"@/client/components/ProjectView/ProjectInfoBox\"\nimport { RouteError } from \"@/client/components/Error/RouteError\"\n\nexport const Route = createFileRoute(\"/_auth/projects/$projectId\")({\n component: RouteComponent,\n errorComponent: ({ error }) => {\n return <RouteError error={error} />\n },\n loader: async (options) => {\n const { context, params } = options\n const data = await context.trpcClient?.auth.setCurrentScope.mutate({\n type: \"project\",\n projectId: params.projectId || \"\",\n })\n const availableServices = await context.trpcClient?.auth.getAvailableServices.query()\n\n // Extract accountId (domain id) from the rescoped token\n // This is needed for SideNavBar navigation until we refactor it\n const accountId = data?.domain?.id || \"\"\n\n return {\n trpcClient: context.trpcClient,\n crumbDomain: { path: `/projects`, name: data?.domain?.name },\n crumbProject: data?.project,\n availableServices,\n accountId, // Keep for SideNavBar compatibility\n projectId: params.projectId,\n }\n },\n})\n\nfunction RouteComponent() {\n const { availableServices, projectId, crumbProject, crumbDomain } = useLoaderData({ from: Route.id })\n\n return (\n <AppShell\n embedded\n sideNavigation={\n <SideNavBar\n availableServices={availableServices!}\n projectId={projectId}\n projectName={crumbProject?.name || projectId}\n domainName={crumbDomain?.name}\n />\n }\n className=\"h-min-screen\"\n >\n <Container>\n <Stack direction=\"vertical\" distribution=\"start\" alignment=\"stretch\" className=\"xl:flex-row\" gap=\"6\">\n {/* Main content area */}\n <div className=\"min-w-0 flex-1\">\n <ProjectInfoBox\n projectInfo={{\n id: projectId,\n name: crumbProject?.name || projectId,\n domain: crumbProject?.domain,\n }}\n />\n <Outlet />\n </div>\n </Stack>\n </Container>\n </AppShell>\n )\n}\n"],"mappings":"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"}
@@ -1,16 +1,16 @@
1
- import { b as e, nt as t } from "./build-DracvfrJ.mjs";
1
+ import { Q as e, m as t } from "./build-DeJcDjPi.mjs";
2
2
  import { t as n } from "./helpers-1PpYf-fC.mjs";
3
- import { t as r } from "./ContentHeader-xQVhO2yT.mjs";
3
+ import { t as r } from "./ContentHeader-kx1Th5Sq.mjs";
4
4
  import { jsx as i, jsxs as a } from "react/jsx-runtime";
5
5
  import { Link as o, useLoaderData as s } from "@tanstack/react-router";
6
6
  import { Trans as c, useLingui as l } from "@lingui/react";
7
7
  //#region src/client/routes/_auth/projects/$projectId/index.tsx?tsr-split=component
8
- function u({ title: t, links: n }) {
9
- return /*#__PURE__*/ a(e, {
8
+ function u({ title: e, links: n }) {
9
+ return /*#__PURE__*/ a(t, {
10
10
  className: "p-5",
11
11
  children: [/*#__PURE__*/ i("h3", {
12
12
  className: "text-theme-high mb-3 text-base font-semibold",
13
- children: t
13
+ children: e
14
14
  }), /*#__PURE__*/ i("ul", {
15
15
  className: "space-y-1.5",
16
16
  children: n.map(({ label: e, to: t }) => /*#__PURE__*/ i("li", { children: /*#__PURE__*/ i(o, {
@@ -22,7 +22,7 @@ function u({ title: t, links: n }) {
22
22
  });
23
23
  }
24
24
  function d() {
25
- let { crumbProject: e, availableServices: o, projectId: d } = s({ from: "/_auth/projects/$projectId" }), { i18n: f, _: p } = l(), m = n(o ?? []), h = `/projects/${d}`, g = [];
25
+ let { crumbProject: t, availableServices: o, projectId: d } = s({ from: "/_auth/projects/$projectId" }), { i18n: f, _: p } = l(), m = n(o ?? []), h = `/projects/${d}`, g = [];
26
26
  if (m.image?.glance || m.compute?.nova) {
27
27
  let e = [];
28
28
  m.image?.glance && e.push({
@@ -50,10 +50,10 @@ function d() {
50
50
  return m["object-store"]?.swift && _.push({
51
51
  label: "Swift",
52
52
  to: `${h}/storage/swift/containers`
53
- }), _.push({
53
+ }), m["object-store-ceph"]?.ceph && _.push({
54
54
  label: "Ceph",
55
- to: `${h}/storage/ceph/containers`
56
- }), g.push({
55
+ to: `${h}/storage/ceph/buckets`
56
+ }), _.length > 0 && g.push({
57
57
  title: "Storage",
58
58
  links: _
59
59
  }), (m.pca?.["clavis-dev"] || m.pca?.["clavis-beta"]) && g.push({
@@ -62,12 +62,12 @@ function d() {
62
62
  label: "PCA (Clavis)",
63
63
  to: `${h}/services/pca`
64
64
  }]
65
- }), /*#__PURE__*/ a(t, {
65
+ }), /*#__PURE__*/ a(e, {
66
66
  direction: "vertical",
67
67
  gap: "6",
68
68
  className: "pb-4",
69
69
  children: [/*#__PURE__*/ i(r, {
70
- title: e?.name ?? f._({ id: "e0NrBM" }),
70
+ title: t?.name ?? f._({ id: "e0NrBM" }),
71
71
  projectId: d
72
72
  }), g.length > 0 ? /*#__PURE__*/ i("div", {
73
73
  className: "grid grid-cols-1 gap-4 sm:grid-cols-2 lg:grid-cols-3",
@@ -81,4 +81,4 @@ function d() {
81
81
  //#endregion
82
82
  export { d as component };
83
83
 
84
- //# sourceMappingURL=_projectId-BK9UqeYw.mjs.map
84
+ //# sourceMappingURL=_projectId-MxcHrXW4.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"_projectId-MxcHrXW4.mjs","names":["Link","useLoaderData","Box","Stack","getServiceIndex","Trans","useLingui","ContentHeader","ServiceCardProps","title","links","label","to","ServiceCard","map","RouteComponent","crumbProject","availableServices","projectId","from","t","serviceIndex","base","cards","push","storageLinks","length","name","card","component"],"sources":["../../src/client/routes/_auth/projects/$projectId/index.tsx?tsr-split=component"],"sourcesContent":["import { createFileRoute, Link, useLoaderData } from \"@tanstack/react-router\"\nimport { Box, Stack } from \"@cloudoperators/juno-ui-components\"\nimport { getServiceIndex } from \"@/server/Authentication/helpers\"\nimport { Trans } from \"@lingui/react/macro\"\nimport { useLingui } from \"@lingui/react/macro\"\nimport { ContentHeader } from \"@/client/components/ContentHeader/ContentHeader\"\n\nexport const Route = createFileRoute(\"/_auth/projects/$projectId/\")({\n component: RouteComponent,\n})\n\ninterface ServiceCardProps {\n title: string\n links: { label: string; to: string }[]\n}\n\nfunction ServiceCard({ title, links }: ServiceCardProps) {\n return (\n <Box className=\"p-5\">\n <h3 className=\"text-theme-high mb-3 text-base font-semibold\">{title}</h3>\n <ul className=\"space-y-1.5\">\n {links.map(({ label, to }) => (\n <li key={label}>\n <Link to={to} className=\"text-theme-accent hover:text-theme-accent/80 text-sm\">\n {label}\n </Link>\n </li>\n ))}\n </ul>\n </Box>\n )\n}\n\nfunction RouteComponent() {\n const { crumbProject, availableServices, projectId } = useLoaderData({\n from: \"/_auth/projects/$projectId\",\n })\n const { t } = useLingui()\n\n const serviceIndex = getServiceIndex(availableServices ?? [])\n const base = `/projects/${projectId}`\n\n const cards: ServiceCardProps[] = []\n\n if (serviceIndex[\"image\"]?.[\"glance\"] || serviceIndex[\"compute\"]?.[\"nova\"]) {\n const links: { label: string; to: string }[] = []\n if (serviceIndex[\"image\"]?.[\"glance\"]) links.push({ label: \"Images\", to: `${base}/compute/images` })\n if (serviceIndex[\"compute\"]?.[\"nova\"]) links.push({ label: \"Flavors\", to: `${base}/compute/flavors` })\n cards.push({ title: \"Compute\", links })\n }\n\n if (serviceIndex[\"network\"]) {\n cards.push({\n title: \"Network\",\n links: [\n { label: \"Security Groups\", to: `${base}/network/securitygroups` },\n { label: \"Floating IPs\", to: `${base}/network/floatingips` },\n ],\n })\n }\n\n // Storage section\n const storageLinks: { label: string; to: string }[] = []\n if (serviceIndex[\"object-store\"]?.[\"swift\"]) {\n storageLinks.push({ label: \"Swift\", to: `${base}/storage/swift/containers` })\n }\n\n if (serviceIndex[\"object-store-ceph\"]?.[\"ceph\"]) {\n storageLinks.push({ label: \"Ceph\", to: `${base}/storage/ceph/buckets` })\n }\n\n if (storageLinks.length > 0) {\n cards.push({ title: \"Storage\", links: storageLinks })\n }\n\n // temporary as clavis is not fully GA, after GA replace with [\"pca\"]?.[\"clavis\"]\n if (serviceIndex[\"pca\"]?.[\"clavis-dev\"] || serviceIndex[\"pca\"]?.[\"clavis-beta\"]) {\n cards.push({\n title: \"Services\",\n links: [{ label: \"PCA (Clavis)\", to: `${base}/services/pca` }],\n })\n }\n\n return (\n <Stack direction=\"vertical\" gap=\"6\" className=\"pb-4\">\n <ContentHeader title={crumbProject?.name ?? t`Project`} projectId={projectId} />\n {cards.length > 0 ? (\n <div className=\"grid grid-cols-1 gap-4 sm:grid-cols-2 lg:grid-cols-3\">\n {cards.map((card) => (\n <ServiceCard key={card.title} {...card} />\n ))}\n </div>\n ) : (\n <p className=\"text-theme-light text-sm\">\n <Trans>No services available for this project.</Trans>\n </p>\n )}\n </Stack>\n )\n}\n"],"mappings":";;;;;;;AAgBA,SAASa,EAAY,EAAEJ,UAAOC,YAAyB;CACrD,OACE,gBAAC,GAAA;EAAI,WAAU;aACb,gBAAC,MAAA;GAAG,WAAU;aAAgDD;MAC9D,gBAAC,MAAA;GAAG,WAAU;aACXC,EAAMI,KAAK,EAAEH,UAAOC,YACnB,gBAAC,MAAA,EAAA,UACC,gBAAC,GAAA;IAASA;IAAI,WAAU;cACrBD;SAFIA,CAAAA,CAAAA;;;AASnB;AAEA,SAASI,IAAAA;CACP,IAAM,EAAEC,iBAAcC,sBAAmBC,iBAAcjB,EAAc,EACnEkB,MAAM,6BACR,CAAA,GACM,EAAA,MAAA,GAAA,GAAA,MAAQb,EAAAA,GAERe,IAAejB,EAAgBa,KAAqB,CAAA,CAAE,GACtDK,IAAO,aAAaJ,KAEpBK,IAA4B,CAAA;CAElC,IAAIF,EAAa,OAAW,UAAaA,EAAa,SAAa,MAAS;EAC1E,IAAMX,IAAyC,CAAA;EAG/Ca,AAFIF,EAAa,OAAW,UAAWX,EAAMc,KAAK;GAAEb,OAAO;GAAUC,IAAI,GAAGU,EAAI;EAAkB,CAAA,GAC9FD,EAAa,SAAa,QAASX,EAAMc,KAAK;GAAEb,OAAO;GAAWC,IAAI,GAAGU,EAAI;EAAmB,CAAA,GACpGC,EAAMC,KAAK;GAAEf,OAAO;GAAWC;EAAM,CAAA;CACvC;CAEA,AAAIW,EAAa,WACfE,EAAMC,KAAK;EACTf,OAAO;EACPC,OAAO,CACL;GAAEC,OAAO;GAAmBC,IAAI,GAAGU,EAAI;EAA0B,GACjE;GAAEX,OAAO;GAAgBC,IAAI,GAAGU,EAAI;EAAuB,CAAA;CAE/D,CAAA;CAIF,IAAMG,IAAgD,CAAA;CAqBtD,OApBIJ,EAAa,iBAAkB,SACjCI,EAAaD,KAAK;EAAEb,OAAO;EAASC,IAAI,GAAGU,EAAI;CAA4B,CAAA,GAGzED,EAAa,sBAAuB,QACtCI,EAAaD,KAAK;EAAEb,OAAO;EAAQC,IAAI,GAAGU,EAAI;CAAwB,CAAA,GAGpEG,EAAaC,SAAS,KACxBH,EAAMC,KAAK;EAAEf,OAAO;EAAWC,OAAOe;CAAa,CAAA,IAIjDJ,EAAa,MAAS,iBAAiBA,EAAa,MAAS,mBAC/DE,EAAMC,KAAK;EACTf,OAAO;EACPC,OAAO,CAAC;GAAEC,OAAO;GAAgBC,IAAI,GAAGU,EAAI;EAAgB,CAAA;CAC9D,CAAA,GAIA,gBAAC,GAAA;EAAM,WAAU;EAAW,KAAI;EAAI,WAAU;aAC5C,gBAAC,GAAA;GAAc,OAAON,GAAcW,QAAQP,EAAAA,EAAC,EAAA,IAAA,SAAA,CAAA;GAAsBF;MAClEK,EAAMG,SAAS,IACd,gBAAC,OAAA;GAAI,WAAU;aACZH,EAAMT,KAAKc,MACV,gBAAC,GAAA,EAA6B,GAAIA,EAAAA,GAAhBA,EAAKnB,KAAK,CAAA;OAIhC,gBAAC,KAAA;GAAE,WAAU;aACX,gBAAA,GAAA,EAAA,IAAA,SAAA,CAAA;;;AAKV"}
@@ -19,7 +19,7 @@ var i = t("/_auth/projects/$projectId/network/securitygroups/$securityGroupId/")
19
19
  return { sgTitle: n?.name || n?.id || null };
20
20
  },
21
21
  head: ({ loaderData: e }) => ({ meta: [{ title: e?.sgTitle ?? "Security Group" }] }),
22
- component: n(() => import("./_securityGroupId-DroYG6cA.mjs"), "component"),
22
+ component: n(() => import("./_securityGroupId-KKw4RPdH.mjs"), "component"),
23
23
  beforeLoad: async ({ context: t, params: n }) => {
24
24
  let { trpcClient: i } = t, a = e(await i?.auth.getAvailableServices.query() || []);
25
25
  if (!a.network || !a.network.neutron) throw r({
@@ -31,4 +31,4 @@ var i = t("/_auth/projects/$projectId/network/securitygroups/$securityGroupId/")
31
31
  //#endregion
32
32
  export { i as t };
33
33
 
34
- //# sourceMappingURL=_securityGroupId-CR1mKICQ.mjs.map
34
+ //# sourceMappingURL=_securityGroupId-CJJanWiY.mjs.map
@@ -1 +1 @@
1
- {"version":3,"file":"_securityGroupId-CR1mKICQ.mjs","names":["createFileRoute","redirect","getServiceIndex","Route","staticData","section","service","isDetail","sectionCrumb","labelKey","crumb","to","RouteInfo","loader","context","params","sg","trpcClient","network","securityGroup","getById","query","project_id","projectId","securityGroupId","sgTitle","name","id","head","loaderData","meta","title","component","lazyRouteComponent","$$splitComponentImporter","beforeLoad","availableServices","auth","getAvailableServices","serviceIndex"],"sources":["../../src/client/routes/_auth/projects/$projectId/network/securitygroups/$securityGroupId/index.tsx"],"sourcesContent":["import {\n Breadcrumb,\n BreadcrumbItem,\n Button,\n ContentHeading,\n Stack,\n Spinner,\n} from \"@cloudoperators/juno-ui-components/index\"\nimport { createFileRoute, redirect, useNavigate } from \"@tanstack/react-router\"\nimport type { RouteInfo } from \"@/client/routes/routeInfo\"\nimport { Trans, useLingui } from \"@lingui/react/macro\"\nimport { useMemo } from \"react\"\nimport { getServiceIndex } from \"@/server/Authentication/helpers\"\nimport { useProjectId } from \"@/client/hooks\"\nimport { SecurityGroupDetailsView } from \"./-components/SecurityGroupDetailsView\"\nimport { EditSecurityGroupModal } from \"../-components/-modals/EditSecurityGroupModal\"\nimport { useSecurityGroupDetails } from \"./-hooks/useSecurityGroupDetails\"\nimport { useListWithFiltering } from \"@/client/utils/useListWithFiltering\"\nimport { trpcReact } from \"@/client/trpcClient\"\n\nexport const Route = createFileRoute(\"/_auth/projects/$projectId/network/securitygroups/$securityGroupId/\")({\n staticData: {\n section: \"network\",\n service: \"securitygroups\",\n isDetail: true,\n sectionCrumb: { labelKey: \"Network\" },\n crumb: { labelKey: \"Security Groups\", to: \"/projects/$projectId/network/securitygroups\" },\n } satisfies RouteInfo,\n loader: async ({ context, params }) => {\n const sg = await context.trpcClient?.network.securityGroup.getById.query({\n project_id: params.projectId,\n securityGroupId: params.securityGroupId,\n })\n return { sgTitle: sg?.name || sg?.id || null }\n },\n head: ({ loaderData }) => ({\n meta: [{ title: loaderData?.sgTitle ?? \"Security Group\" }],\n }),\n component: RouteComponent,\n beforeLoad: async ({ context, params }) => {\n const { trpcClient } = context\n\n const availableServices = (await trpcClient?.auth.getAvailableServices.query()) || []\n\n const serviceIndex = getServiceIndex(availableServices)\n\n // Redirect to the \"Projects Overview\" page if network service not available\n if (!serviceIndex[\"network\"]) {\n throw redirect({\n to: \"/projects/$projectId/network/securitygroups\",\n params: { projectId: params.projectId },\n })\n }\n\n if (!serviceIndex[\"network\"][\"neutron\"]) {\n // Redirect to the \"Network Services Overview\" page if the \"Neutron\" service is not available\n throw redirect({\n to: \"/projects/$projectId/network/securitygroups\",\n params: { projectId: params.projectId },\n })\n }\n },\n})\n\nfunction RouteComponent() {\n const { securityGroupId } = Route.useParams()\n const projectId = useProjectId()\n const navigate = useNavigate()\n const { t } = useLingui()\n\n // Rules filtering using the same pattern as List page\n const {\n searchTerm: rulesSearchTerm,\n sortSettings,\n filterSettings,\n handleSearchChange,\n handleSortChange,\n handleFilterChange,\n } = useListWithFiltering<\"direction\" | \"protocol\" | \"description\">({\n defaultSortKey: \"direction\",\n defaultSortDir: \"asc\",\n sortOptions: [\n { label: t`Direction`, value: \"direction\" },\n { label: t`Protocol`, value: \"protocol\" },\n { label: t`Description`, value: \"description\" },\n ],\n filterSettings: {\n filters: [\n {\n displayName: t`Direction`,\n filterName: \"direction\",\n values: [\"ingress\", \"egress\"],\n supportsMultiValue: false,\n },\n {\n displayName: t`Ethertype`,\n filterName: \"ethertype\",\n values: [\"IPv4\", \"IPv6\"],\n supportsMultiValue: false,\n },\n {\n displayName: t`Protocol`,\n filterName: \"protocol\",\n values: [\"tcp\", \"udp\", \"icmp\", \"ipv6-icmp\"],\n supportsMultiValue: false,\n },\n ],\n },\n })\n\n // Group filter controls for the hook\n const filterControls = {\n searchTerm: rulesSearchTerm,\n onSearchChange: handleSearchChange,\n sortSettings,\n onSortChange: handleSortChange,\n filterSettings,\n onFilterChange: handleFilterChange,\n }\n\n // Use custom hook for logic (now includes filtering/sorting)\n const {\n securityGroup,\n filteredAndSortedRules,\n isLoading,\n isError,\n error,\n isUpdating,\n updateError,\n isDeletingRule,\n deleteRuleError,\n isCreatingRule,\n createRuleError,\n editModalOpen,\n handleEdit,\n handleCloseEditModal,\n handleUpdate,\n handleDeleteRule,\n handleCreateRule,\n } = useSecurityGroupDetails({\n securityGroupId,\n filterControls,\n })\n\n // Fetch available security groups for the Add Rule dropdown\n const { data: securityGroups } = trpcReact.network.securityGroup.list.useQuery({ project_id: projectId })\n const availableSecurityGroups = useMemo(() => {\n return (securityGroups || [])\n .filter((sg) => sg.id !== securityGroupId) // Exclude current group\n .map((sg) => ({\n id: sg.id,\n name: sg.name || sg.id,\n }))\n }, [securityGroups, securityGroupId])\n\n const handleBack = () => {\n navigate({\n to: \"/projects/$projectId/network/securitygroups\",\n params: { projectId },\n })\n }\n\n // Handle loading state\n if (isLoading) {\n return (\n <Stack className=\"fixed inset-0\" distribution=\"center\" alignment=\"center\" direction=\"vertical\">\n <Spinner variant=\"primary\" size=\"large\" className=\"mb-2\" />\n <Trans>Loading Security Group Details...</Trans>\n </Stack>\n )\n }\n\n // Handle error state\n if (isError) {\n const errorMessage = error?.message || \"Unknown error\"\n\n return (\n <Stack className=\"fixed inset-0\" distribution=\"center\" alignment=\"center\" direction=\"vertical\" gap=\"5\">\n <p className=\"text-theme-error font-semibold\">\n <Trans>Error loading security group</Trans>\n </p>\n <p className=\"text-theme-highest\">{errorMessage}</p>\n <Button onClick={handleBack} variant=\"primary\">\n <Trans>Back to Security Groups</Trans>\n </Button>\n </Stack>\n )\n }\n\n // Handle no data state\n if (!securityGroup) {\n return (\n <Stack className=\"fixed inset-0\" distribution=\"center\" alignment=\"center\" direction=\"vertical\" gap=\"5\">\n <p className=\"text-theme-secondary\">\n <Trans>Security group not found</Trans>\n </p>\n <Button onClick={handleBack} variant=\"primary\">\n <Trans>Back to Security Groups</Trans>\n </Button>\n </Stack>\n )\n }\n\n // Render success state\n return (\n <Stack direction=\"vertical\">\n <ContentHeading>{securityGroup.name || securityGroup.id}</ContentHeading>\n <Breadcrumb className=\"my-6\">\n <BreadcrumbItem onClick={handleBack} label={t`Security Groups`} />\n <BreadcrumbItem active label={securityGroup.id} />\n </Breadcrumb>\n\n <SecurityGroupDetailsView\n securityGroup={securityGroup}\n filteredAndSortedRules={filteredAndSortedRules}\n onEdit={handleEdit}\n onDeleteRule={handleDeleteRule}\n isDeletingRule={isDeletingRule}\n deleteRuleError={deleteRuleError}\n filterControls={filterControls}\n onCreateRule={handleCreateRule}\n isCreatingRule={isCreatingRule}\n createRuleError={createRuleError}\n availableSecurityGroups={availableSecurityGroups}\n currentProjectId={projectId}\n />\n\n <EditSecurityGroupModal\n securityGroup={securityGroup}\n open={editModalOpen}\n onClose={handleCloseEditModal}\n onUpdate={handleUpdate}\n isLoading={isUpdating}\n error={updateError}\n />\n </Stack>\n )\n}\n"],"mappings":";;AAoBA,IAAaG,IAAQH,EAAgB,qEAAA,EAAuE;CAC1GI,YAAY;EACVC,SAAS;EACTC,SAAS;EACTC,UAAU;EACVC,cAAc,EAAEC,UAAU,UAAU;EACpCC,OAAO;GAAED,UAAU;GAAmBE,IAAI;EAA8C;CAC1F;CACAE,QAAQ,OAAO,EAAEC,YAASC,gBAAQ;EAChC,IAAMC,IAAK,MAAMF,EAAQG,YAAYC,QAAQC,cAAcC,QAAQC,MAAM;GACvEC,YAAYP,EAAOQ;GACnBC,iBAAiBT,EAAOS;EAC1B,CAAA;EACA,OAAO,EAAEC,SAAST,GAAIU,QAAQV,GAAIW,MAAM,KAAK;CAC/C;CACAC,OAAO,EAAEC,qBAAkB,EACzBC,MAAM,CAAC,EAAEC,OAAOF,GAAYJ,WAAW,iBAAiB,CAAA,EAC1D;CACAO,WAASC,mDAAA,WAAA;CACTE,YAAY,OAAO,EAAErB,YAASC,gBAAQ;EACpC,IAAM,EAAEE,kBAAeH,GAIjByB,IAAerC,EAFK,MAAOe,GAAYoB,KAAKC,qBAAqBjB,MAAAA,KAAY,CAAA,CAE9Ce;EAUrC,IAPI,CAACG,EAAa,WAOd,CAACA,EAAa,QAAW,SAE3B,MAAMtC,EAAS;GACbU,IAAI;GACJI,QAAQ,EAAEQ,WAAWR,EAAOQ,UAAU;EACxC,CAAA;CAEJ;AACF,CAAA"}
1
+ {"version":3,"file":"_securityGroupId-CJJanWiY.mjs","names":["createFileRoute","redirect","getServiceIndex","Route","staticData","section","service","isDetail","sectionCrumb","labelKey","crumb","to","RouteInfo","loader","context","params","sg","trpcClient","network","securityGroup","getById","query","project_id","projectId","securityGroupId","sgTitle","name","id","head","loaderData","meta","title","component","lazyRouteComponent","$$splitComponentImporter","beforeLoad","availableServices","auth","getAvailableServices","serviceIndex"],"sources":["../../src/client/routes/_auth/projects/$projectId/network/securitygroups/$securityGroupId/index.tsx"],"sourcesContent":["import {\n Breadcrumb,\n BreadcrumbItem,\n Button,\n ContentHeading,\n Stack,\n Spinner,\n} from \"@cloudoperators/juno-ui-components/index\"\nimport { createFileRoute, redirect, useNavigate } from \"@tanstack/react-router\"\nimport type { RouteInfo } from \"@/client/routes/routeInfo\"\nimport { Trans, useLingui } from \"@lingui/react/macro\"\nimport { useMemo } from \"react\"\nimport { getServiceIndex } from \"@/server/Authentication/helpers\"\nimport { useProjectId } from \"@/client/hooks\"\nimport { SecurityGroupDetailsView } from \"./-components/SecurityGroupDetailsView\"\nimport { EditSecurityGroupModal } from \"../-components/-modals/EditSecurityGroupModal\"\nimport { useSecurityGroupDetails } from \"./-hooks/useSecurityGroupDetails\"\nimport { useListWithFiltering } from \"@/client/utils/useListWithFiltering\"\nimport { trpcReact } from \"@/client/trpcClient\"\n\nexport const Route = createFileRoute(\"/_auth/projects/$projectId/network/securitygroups/$securityGroupId/\")({\n staticData: {\n section: \"network\",\n service: \"securitygroups\",\n isDetail: true,\n sectionCrumb: { labelKey: \"Network\" },\n crumb: { labelKey: \"Security Groups\", to: \"/projects/$projectId/network/securitygroups\" },\n } satisfies RouteInfo,\n loader: async ({ context, params }) => {\n const sg = await context.trpcClient?.network.securityGroup.getById.query({\n project_id: params.projectId,\n securityGroupId: params.securityGroupId,\n })\n return { sgTitle: sg?.name || sg?.id || null }\n },\n head: ({ loaderData }) => ({\n meta: [{ title: loaderData?.sgTitle ?? \"Security Group\" }],\n }),\n component: RouteComponent,\n beforeLoad: async ({ context, params }) => {\n const { trpcClient } = context\n\n const availableServices = (await trpcClient?.auth.getAvailableServices.query()) || []\n\n const serviceIndex = getServiceIndex(availableServices)\n\n // Redirect to the \"Projects Overview\" page if network service not available\n if (!serviceIndex[\"network\"]) {\n throw redirect({\n to: \"/projects/$projectId/network/securitygroups\",\n params: { projectId: params.projectId },\n })\n }\n\n if (!serviceIndex[\"network\"][\"neutron\"]) {\n // Redirect to the \"Network Services Overview\" page if the \"Neutron\" service is not available\n throw redirect({\n to: \"/projects/$projectId/network/securitygroups\",\n params: { projectId: params.projectId },\n })\n }\n },\n})\n\nfunction RouteComponent() {\n const { securityGroupId } = Route.useParams()\n const projectId = useProjectId()\n const navigate = useNavigate()\n const { t } = useLingui()\n\n // Rules filtering using the same pattern as List page\n const {\n searchTerm: rulesSearchTerm,\n sortSettings,\n filterSettings,\n handleSearchChange,\n handleSortChange,\n handleFilterChange,\n } = useListWithFiltering<\"direction\" | \"protocol\" | \"description\">({\n defaultSortKey: \"direction\",\n defaultSortDir: \"asc\",\n sortOptions: [\n { label: t`Direction`, value: \"direction\" },\n { label: t`Protocol`, value: \"protocol\" },\n { label: t`Description`, value: \"description\" },\n ],\n filterSettings: {\n filters: [\n {\n displayName: t`Direction`,\n filterName: \"direction\",\n values: [\"ingress\", \"egress\"],\n supportsMultiValue: false,\n },\n {\n displayName: t`Ethertype`,\n filterName: \"ethertype\",\n values: [\"IPv4\", \"IPv6\"],\n supportsMultiValue: false,\n },\n {\n displayName: t`Protocol`,\n filterName: \"protocol\",\n values: [\"tcp\", \"udp\", \"icmp\", \"ipv6-icmp\"],\n supportsMultiValue: false,\n },\n ],\n },\n })\n\n // Group filter controls for the hook\n const filterControls = {\n searchTerm: rulesSearchTerm,\n onSearchChange: handleSearchChange,\n sortSettings,\n onSortChange: handleSortChange,\n filterSettings,\n onFilterChange: handleFilterChange,\n }\n\n // Use custom hook for logic (now includes filtering/sorting)\n const {\n securityGroup,\n filteredAndSortedRules,\n isLoading,\n isError,\n error,\n isUpdating,\n updateError,\n isDeletingRule,\n deleteRuleError,\n isCreatingRule,\n createRuleError,\n editModalOpen,\n handleEdit,\n handleCloseEditModal,\n handleUpdate,\n handleDeleteRule,\n handleCreateRule,\n } = useSecurityGroupDetails({\n securityGroupId,\n filterControls,\n })\n\n // Fetch available security groups for the Add Rule dropdown\n const { data: securityGroups } = trpcReact.network.securityGroup.list.useQuery({ project_id: projectId })\n const availableSecurityGroups = useMemo(() => {\n return (securityGroups || [])\n .filter((sg) => sg.id !== securityGroupId) // Exclude current group\n .map((sg) => ({\n id: sg.id,\n name: sg.name || sg.id,\n }))\n }, [securityGroups, securityGroupId])\n\n const handleBack = () => {\n navigate({\n to: \"/projects/$projectId/network/securitygroups\",\n params: { projectId },\n })\n }\n\n // Handle loading state\n if (isLoading) {\n return (\n <Stack className=\"fixed inset-0\" distribution=\"center\" alignment=\"center\" direction=\"vertical\">\n <Spinner variant=\"primary\" size=\"large\" className=\"mb-2\" />\n <Trans>Loading Security Group Details...</Trans>\n </Stack>\n )\n }\n\n // Handle error state\n if (isError) {\n const errorMessage = error?.message || \"Unknown error\"\n\n return (\n <Stack className=\"fixed inset-0\" distribution=\"center\" alignment=\"center\" direction=\"vertical\" gap=\"5\">\n <p className=\"text-theme-error font-semibold\">\n <Trans>Error loading security group</Trans>\n </p>\n <p className=\"text-theme-highest\">{errorMessage}</p>\n <Button onClick={handleBack} variant=\"primary\">\n <Trans>Back to Security Groups</Trans>\n </Button>\n </Stack>\n )\n }\n\n // Handle no data state\n if (!securityGroup) {\n return (\n <Stack className=\"fixed inset-0\" distribution=\"center\" alignment=\"center\" direction=\"vertical\" gap=\"5\">\n <p className=\"text-theme-secondary\">\n <Trans>Security group not found</Trans>\n </p>\n <Button onClick={handleBack} variant=\"primary\">\n <Trans>Back to Security Groups</Trans>\n </Button>\n </Stack>\n )\n }\n\n // Render success state\n return (\n <Stack direction=\"vertical\">\n <ContentHeading>{securityGroup.name || securityGroup.id}</ContentHeading>\n <Breadcrumb className=\"my-6\">\n <BreadcrumbItem onClick={handleBack} label={t`Security Groups`} />\n <BreadcrumbItem active label={securityGroup.id} />\n </Breadcrumb>\n\n <SecurityGroupDetailsView\n securityGroup={securityGroup}\n filteredAndSortedRules={filteredAndSortedRules}\n onEdit={handleEdit}\n onDeleteRule={handleDeleteRule}\n isDeletingRule={isDeletingRule}\n deleteRuleError={deleteRuleError}\n filterControls={filterControls}\n onCreateRule={handleCreateRule}\n isCreatingRule={isCreatingRule}\n createRuleError={createRuleError}\n availableSecurityGroups={availableSecurityGroups}\n currentProjectId={projectId}\n />\n\n <EditSecurityGroupModal\n securityGroup={securityGroup}\n open={editModalOpen}\n onClose={handleCloseEditModal}\n onUpdate={handleUpdate}\n isLoading={isUpdating}\n error={updateError}\n />\n </Stack>\n )\n}\n"],"mappings":";;AAoBA,IAAaG,IAAQH,EAAgB,qEAAA,EAAuE;CAC1GI,YAAY;EACVC,SAAS;EACTC,SAAS;EACTC,UAAU;EACVC,cAAc,EAAEC,UAAU,UAAU;EACpCC,OAAO;GAAED,UAAU;GAAmBE,IAAI;EAA8C;CAC1F;CACAE,QAAQ,OAAO,EAAEC,YAASC,gBAAQ;EAChC,IAAMC,IAAK,MAAMF,EAAQG,YAAYC,QAAQC,cAAcC,QAAQC,MAAM;GACvEC,YAAYP,EAAOQ;GACnBC,iBAAiBT,EAAOS;EAC1B,CAAA;EACA,OAAO,EAAEC,SAAST,GAAIU,QAAQV,GAAIW,MAAM,KAAK;CAC/C;CACAC,OAAO,EAAEC,qBAAkB,EACzBC,MAAM,CAAC,EAAEC,OAAOF,GAAYJ,WAAW,iBAAiB,CAAA,EAC1D;CACAO,WAASC,mDAAA,WAAA;CACTE,YAAY,OAAO,EAAErB,YAASC,gBAAQ;EACpC,IAAM,EAAEE,kBAAeH,GAIjByB,IAAerC,EAFK,MAAOe,GAAYoB,KAAKC,qBAAqBjB,MAAAA,KAAY,CAAA,CAE9Ce;EAUrC,IAPI,CAACG,EAAa,WAOd,CAACA,EAAa,QAAW,SAE3B,MAAMtC,EAAS;GACbU,IAAI;GACJI,QAAQ,EAAEQ,WAAWR,EAAOQ,UAAU;EACxC,CAAA;CAEJ;AACF,CAAA"}