@cobaltcore-dev/aurora 0.3.0 → 0.4.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 (134) hide show
  1. package/dist/client/{ContentHeader-B_PWrxbw.mjs → ContentHeader-BXZoN3B9.mjs} +16 -16
  2. package/dist/client/{ContentHeader-B_PWrxbw.mjs.map → ContentHeader-BXZoN3B9.mjs.map} +1 -1
  3. package/dist/client/{DeleteFlavorModal-BkPrQKyE.mjs → DeleteFlavorModal-BusYn32r.mjs} +152 -152
  4. package/dist/client/{DeleteFlavorModal-BkPrQKyE.mjs.map → DeleteFlavorModal-BusYn32r.mjs.map} +1 -1
  5. package/dist/client/{EditSecurityGroupModal-ad4uzlt0.mjs → EditSecurityGroupModal-DK3WYikA.mjs} +15 -15
  6. package/dist/client/{EditSecurityGroupModal-ad4uzlt0.mjs.map → EditSecurityGroupModal-DK3WYikA.mjs.map} +1 -1
  7. package/dist/client/FiltersInput-OVeIJzIo.mjs +81 -0
  8. package/dist/client/FiltersInput-OVeIJzIo.mjs.map +1 -0
  9. package/dist/client/{FloatingIpActionModals-lKMwLuL8.mjs → FloatingIpActionModals-Dok7fJss.mjs} +52 -52
  10. package/dist/client/{FloatingIpActionModals-lKMwLuL8.mjs.map → FloatingIpActionModals-Dok7fJss.mjs.map} +1 -1
  11. package/dist/client/{ImageToastNotifications--U13YiQ_.mjs → ImageToastNotifications-BG9LPnXf.mjs} +324 -324
  12. package/dist/client/{ImageToastNotifications--U13YiQ_.mjs.map → ImageToastNotifications-BG9LPnXf.mjs.map} +1 -1
  13. package/dist/client/ListToolbar-BojRTNbo.mjs +129 -0
  14. package/dist/client/ListToolbar-BojRTNbo.mjs.map +1 -0
  15. package/dist/client/{RouteError-iP1Vd6bY.mjs → RouteError-pDEWC_k7.mjs} +2 -2
  16. package/dist/client/{RouteError-iP1Vd6bY.mjs.map → RouteError-pDEWC_k7.mjs.map} +1 -1
  17. package/dist/client/SortInput-DXWSqSny.mjs +34 -0
  18. package/dist/client/SortInput-DXWSqSny.mjs.map +1 -0
  19. package/dist/client/{_flavorId-BaNXUJhA.mjs → _flavorId-BRonXvCo.mjs} +40 -40
  20. package/dist/client/{_flavorId-BaNXUJhA.mjs.map → _flavorId-BRonXvCo.mjs.map} +1 -1
  21. package/dist/client/{_flavorId-CnO76tuy.mjs → _flavorId-BoNcxYmF.mjs} +9 -9
  22. package/dist/client/{_flavorId-CnO76tuy.mjs.map → _flavorId-BoNcxYmF.mjs.map} +1 -1
  23. package/dist/client/{_floatingIpId-C8G20nNt.mjs → _floatingIpId-D33bOEmH.mjs} +2 -2
  24. package/dist/client/{_floatingIpId-C8G20nNt.mjs.map → _floatingIpId-D33bOEmH.mjs.map} +1 -1
  25. package/dist/client/_floatingIpId-DF_BSJN6.mjs +228 -0
  26. package/dist/client/{_floatingIpId-DdKnjdIV.mjs.map → _floatingIpId-DF_BSJN6.mjs.map} +1 -1
  27. package/dist/client/{_imageId-DdSbxFqG.mjs → _imageId-BL0I5_pv.mjs} +127 -127
  28. package/dist/client/{_imageId-DdSbxFqG.mjs.map → _imageId-BL0I5_pv.mjs.map} +1 -1
  29. package/dist/client/{_pcaId-DX_S-3hE.mjs → _pcaId-BYCoeK6_.mjs} +2 -2
  30. package/dist/client/{_pcaId-DX_S-3hE.mjs.map → _pcaId-BYCoeK6_.mjs.map} +1 -1
  31. package/dist/client/_pcaId-CbBhBrX1.mjs +466 -0
  32. package/dist/client/_pcaId-CbBhBrX1.mjs.map +1 -0
  33. package/dist/client/{_projectId-Dha4XqX4.mjs → _projectId-5NiasyXm.mjs} +3 -3
  34. package/dist/client/{_projectId-Dha4XqX4.mjs.map → _projectId-5NiasyXm.mjs.map} +1 -1
  35. package/dist/client/{_projectId-DYrcZ3E3.mjs → _projectId-BwLMEMGC.mjs} +3 -3
  36. package/dist/client/{_projectId-DYrcZ3E3.mjs.map → _projectId-BwLMEMGC.mjs.map} +1 -1
  37. package/dist/client/{_projectId-DsSVV2EA.mjs → _projectId-D35MN1kY.mjs} +70 -70
  38. package/dist/client/{_projectId-DsSVV2EA.mjs.map → _projectId-D35MN1kY.mjs.map} +1 -1
  39. package/dist/client/{_projectId-KH5si25Q.mjs → _projectId-OW2xkK43.mjs} +2 -2
  40. package/dist/client/{_projectId-KH5si25Q.mjs.map → _projectId-OW2xkK43.mjs.map} +1 -1
  41. package/dist/client/{_securityGroupId-DgaSqYex.mjs → _securityGroupId-B-Z-CzLp.mjs} +2 -2
  42. package/dist/client/{_securityGroupId-DgaSqYex.mjs.map → _securityGroupId-B-Z-CzLp.mjs.map} +1 -1
  43. package/dist/client/{_securityGroupId-O7FXfx0M.mjs → _securityGroupId-B1bOYRbX.mjs} +307 -307
  44. package/dist/client/{_securityGroupId-O7FXfx0M.mjs.map → _securityGroupId-B1bOYRbX.mjs.map} +1 -1
  45. package/dist/client/{about-DN8n8sN8.mjs → about-DLn1ShhF.mjs} +2 -2
  46. package/dist/client/{about-DN8n8sN8.mjs.map → about-DLn1ShhF.mjs.map} +1 -1
  47. package/dist/client/{build-CZRvXrAL.mjs → build-BJDfnAyi.mjs} +541 -537
  48. package/dist/client/{build-CZRvXrAL.mjs.map → build-BJDfnAyi.mjs.map} +1 -1
  49. package/dist/client/{buildFilterParams-ngVK3ybs.mjs → buildFilterParams-TeyosGyK.mjs} +1 -1
  50. package/dist/client/{buildFilterParams-ngVK3ybs.mjs.map → buildFilterParams-TeyosGyK.mjs.map} +1 -1
  51. package/dist/client/{constants-akdIBeTX.mjs → constants-B-P2r5F1.mjs} +11 -11
  52. package/dist/client/{constants-akdIBeTX.mjs.map → constants-B-P2r5F1.mjs.map} +1 -1
  53. package/dist/client/{containers-DmwhE9Uz.mjs → containers-BjWqjNOx.mjs} +2 -2
  54. package/dist/client/containers-BjWqjNOx.mjs.map +1 -0
  55. package/dist/client/{containers-Cs5vOeR2.mjs → containers-DsRWc1L5.mjs} +1 -1
  56. package/dist/client/containers-DsRWc1L5.mjs.map +1 -0
  57. package/dist/client/containers-J7WFA18U.mjs +3120 -0
  58. package/dist/client/containers-J7WFA18U.mjs.map +1 -0
  59. package/dist/client/{flavors-CY7A6--v.mjs → flavors-D8oElC2K.mjs} +2 -2
  60. package/dist/client/{flavors-CY7A6--v.mjs.map → flavors-D8oElC2K.mjs.map} +1 -1
  61. package/dist/client/flavors-qvgPSI7J.mjs +613 -0
  62. package/dist/client/flavors-qvgPSI7J.mjs.map +1 -0
  63. package/dist/client/{floatingips-BtL4d4m-.mjs → floatingips-Fa6ocNUu.mjs} +86 -86
  64. package/dist/client/{floatingips-BtL4d4m-.mjs.map → floatingips-Fa6ocNUu.mjs.map} +1 -1
  65. package/dist/client/{formatBytes-D6oa0wU9.mjs → formatBytes-tQBEnPoL.mjs} +1 -1
  66. package/dist/client/{formatBytes-D6oa0wU9.mjs.map → formatBytes-tQBEnPoL.mjs.map} +1 -1
  67. package/dist/client/{hooks-D0krAKvo.mjs → hooks-DEjb9d1F.mjs} +1 -1
  68. package/dist/client/images-CSFfefAu.mjs +1901 -0
  69. package/dist/client/images-CSFfefAu.mjs.map +1 -0
  70. package/dist/client/{images-I9gQfRa7.mjs → images-CTLCY-yY.mjs} +2 -2
  71. package/dist/client/{images-I9gQfRa7.mjs.map → images-CTLCY-yY.mjs.map} +1 -1
  72. package/dist/client/images-DM9I8G0p.mjs.map +1 -1
  73. package/dist/client/images-tYfyOkX8.mjs +8 -0
  74. package/dist/client/images-tYfyOkX8.mjs.map +1 -0
  75. package/dist/client/index.js +200 -198
  76. package/dist/client/index.js.map +1 -1
  77. package/dist/client/{network-SCVadZsv.mjs → network-rYLHyf15.mjs} +1 -1
  78. package/dist/client/{network-SCVadZsv.mjs.map → network-rYLHyf15.mjs.map} +1 -1
  79. package/dist/client/{objects-BrYe_RaJ.mjs → objects-BciXwZ00.mjs} +2 -2
  80. package/dist/client/objects-BciXwZ00.mjs.map +1 -0
  81. package/dist/client/{objects-B4yrYf_a.mjs → objects-Cdew99tK.mjs} +1 -1
  82. package/dist/client/objects-Cdew99tK.mjs.map +1 -0
  83. package/dist/client/objects-DaElrban.mjs +5340 -0
  84. package/dist/client/objects-DaElrban.mjs.map +1 -0
  85. package/dist/client/{overview-CmQkJ4Hh.mjs → overview-BMhjFMIV.mjs} +2 -2
  86. package/dist/client/{overview-CmQkJ4Hh.mjs.map → overview-BMhjFMIV.mjs.map} +1 -1
  87. package/dist/client/{overview-BjRSFSBh.mjs → overview-BYIRj7_X.mjs} +2 -2
  88. package/dist/client/{overview-BjRSFSBh.mjs.map → overview-BYIRj7_X.mjs.map} +1 -1
  89. package/dist/client/{overview-DTLIAKkJ.mjs → overview-DRCKNBH2.mjs} +2 -2
  90. package/dist/client/{overview-DTLIAKkJ.mjs.map → overview-DRCKNBH2.mjs.map} +1 -1
  91. package/dist/client/{overview-C4gjtc2q.mjs → overview-urYLOVQE.mjs} +3 -3
  92. package/dist/client/{overview-C4gjtc2q.mjs.map → overview-urYLOVQE.mjs.map} +1 -1
  93. package/dist/client/{pca-DzixU9Dl.mjs → pca-COmKvp3J.mjs} +2 -2
  94. package/dist/client/{pca-DzixU9Dl.mjs.map → pca-COmKvp3J.mjs.map} +1 -1
  95. package/dist/client/{pca-CYFJxSZ2.mjs → pca-oc7J0_Xd.mjs} +62 -62
  96. package/dist/client/pca-oc7J0_Xd.mjs.map +1 -0
  97. package/dist/client/{projects-D3hOC1cy.mjs → projects-BUabCzvw.mjs} +33 -33
  98. package/dist/client/projects-BUabCzvw.mjs.map +1 -0
  99. package/dist/client/{projects-BtyjXGq2.mjs → projects-DI_L4oDw.mjs} +2 -2
  100. package/dist/client/{projects-BtyjXGq2.mjs.map → projects-DI_L4oDw.mjs.map} +1 -1
  101. package/dist/client/{projects-DwVawmll.mjs → projects-Dl5XkXUP.mjs} +2 -2
  102. package/dist/client/{projects-DwVawmll.mjs.map → projects-Dl5XkXUP.mjs.map} +1 -1
  103. package/dist/client/{projects-BsN4bvU2.mjs → projects-HoQ0gE5Y.mjs} +1 -1
  104. package/dist/client/{projects-BsN4bvU2.mjs.map → projects-HoQ0gE5Y.mjs.map} +1 -1
  105. package/dist/client/{securitygroups-DMCIDHQS.mjs → securitygroups-BjkmHk2J.mjs} +102 -102
  106. package/dist/client/{securitygroups-DMCIDHQS.mjs.map → securitygroups-BjkmHk2J.mjs.map} +1 -1
  107. package/dist/client/{useListWithFiltering-CEDh1LO-.mjs → useListWithFiltering-CbhHJO4V.mjs} +1 -1
  108. package/dist/client/{useListWithFiltering-CEDh1LO-.mjs.map → useListWithFiltering-CbhHJO4V.mjs.map} +1 -1
  109. package/dist/client/{useProjectId-CgOTejka.mjs → useProjectId-OQv2KBbG.mjs} +1 -1
  110. package/dist/client/{useProjectId-CgOTejka.mjs.map → useProjectId-OQv2KBbG.mjs.map} +1 -1
  111. package/dist/server/index.d.ts +7 -1
  112. package/dist/server/index.js +116 -55
  113. package/package.json +3 -4
  114. package/dist/client/ListToolbar-C5lzTrit.mjs +0 -223
  115. package/dist/client/ListToolbar-C5lzTrit.mjs.map +0 -1
  116. package/dist/client/_floatingIpId-DdKnjdIV.mjs +0 -228
  117. package/dist/client/_pcaId-DFkYJEb5.mjs +0 -369
  118. package/dist/client/_pcaId-DFkYJEb5.mjs.map +0 -1
  119. package/dist/client/containers-BE2QiLBs.mjs +0 -3031
  120. package/dist/client/containers-BE2QiLBs.mjs.map +0 -1
  121. package/dist/client/containers-Cs5vOeR2.mjs.map +0 -1
  122. package/dist/client/containers-DmwhE9Uz.mjs.map +0 -1
  123. package/dist/client/flavors-pEcGkCut.mjs +0 -565
  124. package/dist/client/flavors-pEcGkCut.mjs.map +0 -1
  125. package/dist/client/images-C19gCFSy.mjs +0 -1797
  126. package/dist/client/images-C19gCFSy.mjs.map +0 -1
  127. package/dist/client/objects-B4yrYf_a.mjs.map +0 -1
  128. package/dist/client/objects-BrYe_RaJ.mjs.map +0 -1
  129. package/dist/client/objects-DOYFFn3Y.mjs +0 -4760
  130. package/dist/client/objects-DOYFFn3Y.mjs.map +0 -1
  131. package/dist/client/pca-CYFJxSZ2.mjs.map +0 -1
  132. package/dist/client/projects-D3hOC1cy.mjs.map +0 -1
  133. package/permission_policies/compute.yaml +0 -975
  134. package/permission_policies/image.yaml +0 -71
@@ -1 +1 @@
1
- {"version":3,"file":"_projectId-DYrcZ3E3.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","name","length","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 - always show Ceph, optionally add Swift\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 storageLinks.push({ label: \"Ceph\", to: `${base}/storage/ceph/containers` })\n cards.push({ title: \"Storage\", links: storageLinks })\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;AACrD,QACE,kBAAC,GAAA;EAAI,WAAU;aACb,kBAAC,MAAA;GAAG,WAAU;aAAgDD;MAC9D,kBAAC,MAAA;GAAG,WAAU;aACXC,EAAMI,KAAK,EAAEH,UAAOC,YACnB,kBAAC,MAAA,EAAA,UACC,kBAAC,GAAA;IAASA;IAAI,WAAU;cACrBD;SAFIA,EAAAA,CAAAA;;;;AAWnB,SAASI,IAAAA;CACP,IAAM,EAAEC,iBAAcC,sBAAmBC,iBAAcjB,EAAc,EACnEkB,MAAM,8BACR,CAAA,EACM,EAAA,MAAA,GAAA,GAAA,MAAQb,GAAAA,EAERe,IAAejB,EAAgBa,KAAqB,EAAE,CAAA,EACtDK,IAAO,aAAaJ,KAEpBK,IAA4B,EAAE;AAEpC,KAAIF,EAAa,OAAW,UAAaA,EAAa,SAAa,MAAS;EAC1E,IAAMX,IAAyC,EAAE;AAGjDa,EAFIF,EAAa,OAAW,UAAWX,EAAMc,KAAK;GAAEb,OAAO;GAAUC,IAAI,GAAGU,EAAI;GAAkB,CAAA,EAC9FD,EAAa,SAAa,QAASX,EAAMc,KAAK;GAAEb,OAAO;GAAWC,IAAI,GAAGU,EAAI;GAAmB,CAAA,EACpGC,EAAMC,KAAK;GAAEf,OAAO;GAAWC;GAAM,CAAA;;AAGvC,CAAIW,EAAa,WACfE,EAAMC,KAAK;EACTf,OAAO;EACPC,OAAO,CACL;GAAEC,OAAO;GAAmBC,IAAI,GAAGU,EAAI;GAA0B,EACjE;GAAEX,OAAO;GAAgBC,IAAI,GAAGU,EAAI;GAAuB,CAAC;EAEhE,CAAA;CAIF,IAAMG,IAAgD,EAAE;AAexD,QAdIJ,EAAa,iBAAkB,SACjCI,EAAaD,KAAK;EAAEb,OAAO;EAASC,IAAI,GAAGU,EAAI;EAA4B,CAAA,EAE7EG,EAAaD,KAAK;EAAEb,OAAO;EAAQC,IAAI,GAAGU,EAAI;EAA2B,CAAA,EACzEC,EAAMC,KAAK;EAAEf,OAAO;EAAWC,OAAOe;EAAa,CAAA,GAG/CJ,EAAa,MAAS,iBAAiBA,EAAa,MAAS,mBAC/DE,EAAMC,KAAK;EACTf,OAAO;EACPC,OAAO,CAAC;GAAEC,OAAO;GAAgBC,IAAI,GAAGU,EAAI;GAAgB,CAAC;EAC/D,CAAA,EAIA,kBAAC,GAAA;EAAM,WAAU;EAAW,KAAI;EAAI,WAAU;aAC5C,kBAAC,GAAA;GAAc,OAAON,GAAcU,QAAQN,EAAAA,EAAC,EAAA,IAAA,UAAA,CAAA;GAAsBF;MAClEK,EAAMI,SAAS,IACd,kBAAC,OAAA;GAAI,WAAU;aACZJ,EAAMT,KAAKc,MACV,kBAAC,GAAA,EAA6B,GAAIA,GAAAA,EAAhBA,EAAKnB,MAAK,CAAA;OAIhC,kBAAC,KAAA;GAAE,WAAU;aACX,kBAAA,GAAA,EAAA,IAAA,UAAA,CAAA"}
1
+ {"version":3,"file":"_projectId-BwLMEMGC.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","name","length","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 - always show Ceph, optionally add Swift\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 storageLinks.push({ label: \"Ceph\", to: `${base}/storage/ceph/containers` })\n cards.push({ title: \"Storage\", links: storageLinks })\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;AACrD,QACE,kBAAC,GAAA;EAAI,WAAU;aACb,kBAAC,MAAA;GAAG,WAAU;aAAgDD;MAC9D,kBAAC,MAAA;GAAG,WAAU;aACXC,EAAMI,KAAK,EAAEH,UAAOC,YACnB,kBAAC,MAAA,EAAA,UACC,kBAAC,GAAA;IAASA;IAAI,WAAU;cACrBD;SAFIA,EAAAA,CAAAA;;;;AAWnB,SAASI,IAAAA;CACP,IAAM,EAAEC,iBAAcC,sBAAmBC,iBAAcjB,EAAc,EACnEkB,MAAM,8BACR,CAAA,EACM,EAAA,MAAA,GAAA,GAAA,MAAQb,GAAAA,EAERe,IAAejB,EAAgBa,KAAqB,EAAE,CAAA,EACtDK,IAAO,aAAaJ,KAEpBK,IAA4B,EAAE;AAEpC,KAAIF,EAAa,OAAW,UAAaA,EAAa,SAAa,MAAS;EAC1E,IAAMX,IAAyC,EAAE;AAGjDa,EAFIF,EAAa,OAAW,UAAWX,EAAMc,KAAK;GAAEb,OAAO;GAAUC,IAAI,GAAGU,EAAI;GAAkB,CAAA,EAC9FD,EAAa,SAAa,QAASX,EAAMc,KAAK;GAAEb,OAAO;GAAWC,IAAI,GAAGU,EAAI;GAAmB,CAAA,EACpGC,EAAMC,KAAK;GAAEf,OAAO;GAAWC;GAAM,CAAA;;AAGvC,CAAIW,EAAa,WACfE,EAAMC,KAAK;EACTf,OAAO;EACPC,OAAO,CACL;GAAEC,OAAO;GAAmBC,IAAI,GAAGU,EAAI;GAA0B,EACjE;GAAEX,OAAO;GAAgBC,IAAI,GAAGU,EAAI;GAAuB,CAAC;EAEhE,CAAA;CAIF,IAAMG,IAAgD,EAAE;AAexD,QAdIJ,EAAa,iBAAkB,SACjCI,EAAaD,KAAK;EAAEb,OAAO;EAASC,IAAI,GAAGU,EAAI;EAA4B,CAAA,EAE7EG,EAAaD,KAAK;EAAEb,OAAO;EAAQC,IAAI,GAAGU,EAAI;EAA2B,CAAA,EACzEC,EAAMC,KAAK;EAAEf,OAAO;EAAWC,OAAOe;EAAa,CAAA,GAG/CJ,EAAa,MAAS,iBAAiBA,EAAa,MAAS,mBAC/DE,EAAMC,KAAK;EACTf,OAAO;EACPC,OAAO,CAAC;GAAEC,OAAO;GAAgBC,IAAI,GAAGU,EAAI;GAAgB,CAAC;EAC/D,CAAA,EAIA,kBAAC,GAAA;EAAM,WAAU;EAAW,KAAI;EAAI,WAAU;aAC5C,kBAAC,GAAA;GAAc,OAAON,GAAcU,QAAQN,EAAAA,EAAC,EAAA,IAAA,UAAA,CAAA;GAAsBF;MAClEK,EAAMI,SAAS,IACd,kBAAC,OAAA;GAAI,WAAU;aACZJ,EAAMT,KAAKc,MACV,kBAAC,GAAA,EAA6B,GAAIA,GAAAA,EAAhBA,EAAKnB,MAAK,CAAA;OAIhC,kBAAC,KAAA;GAAE,WAAU;aACX,kBAAA,GAAA,EAAA,IAAA,UAAA,CAAA"}
@@ -1,5 +1,5 @@
1
- import { B as e, E as t, G as n, O as r, R as i, Y as a, g as o, tt as s } from "./build-CZRvXrAL.mjs";
2
- import { t as c } from "./_projectId-Dha4XqX4.mjs";
1
+ import { D as e, K as t, V as n, X as r, g as i, k as a, nt as o, z as s } from "./build-BJDfnAyi.mjs";
2
+ import { t as c } from "./_projectId-5NiasyXm.mjs";
3
3
  import { t as l } from "./helpers--JWXi40U.mjs";
4
4
  import { Fragment as u, jsx as d, jsxs as f } from "react/jsx-runtime";
5
5
  import { useEffect as p, useMemo as m, useRef as h, useState as g } from "react";
@@ -49,8 +49,8 @@ function A({ component: e }) {
49
49
  }
50
50
  //#endregion
51
51
  //#region src/client/routes/_auth/projects/-components/SideNavBar.tsx
52
- var j = ({ projectId: t, projectName: i, availableServices: a }) => {
53
- let { i18n: o, _: s } = w(), c = x(), m = b(), { provider: h } = S({ strict: !1 }), { slots: _ } = C({ strict: !1 }), v = [...m].reverse().find((e) => O(e.staticData)), y = v && O(v.staticData) ? v.staticData : void 0, T = y?.section ?? null, E = y?.service ?? null, D = l(a), [k, j] = g({
52
+ var j = ({ projectId: e, projectName: r, availableServices: i }) => {
53
+ let { i18n: o, _: s } = w(), c = x(), m = b(), { provider: h } = S({ strict: !1 }), { slots: _ } = C({ strict: !1 }), v = [...m].reverse().find((e) => O(e.staticData)), y = v && O(v.staticData) ? v.staticData : void 0, T = y?.section ?? null, E = y?.service ?? null, D = l(i), [k, j] = g({
54
54
  compute: !0,
55
55
  network: !0,
56
56
  storage: !0,
@@ -80,28 +80,28 @@ var j = ({ projectId: t, projectName: i, availableServices: a }) => {
80
80
  service: "images",
81
81
  label: o._({ id: "an5hVd" }),
82
82
  to: "/projects/$projectId/compute/images",
83
- params: { projectId: t }
83
+ params: { projectId: e }
84
84
  }] : [], ...D?.compute?.nova ? [{
85
85
  service: "flavors",
86
86
  label: o._({ id: "neiJm0" }),
87
87
  to: "/projects/$projectId/compute/flavors",
88
- params: { projectId: t }
88
+ params: { projectId: e }
89
89
  }] : []], R = [...D.network ? [{
90
90
  service: "securitygroups",
91
91
  label: o._({ id: "4opp4r" }),
92
92
  to: "/projects/$projectId/network/securitygroups",
93
- params: { projectId: t }
93
+ params: { projectId: e }
94
94
  }, {
95
95
  service: "floatingips",
96
96
  label: o._({ id: "u77/s4" }),
97
97
  to: "/projects/$projectId/network/floatingips",
98
- params: { projectId: t }
98
+ params: { projectId: e }
99
99
  }] : []], z = [...D?.["object-store"]?.swift ? [{
100
100
  service: "containers",
101
101
  label: o._({ id: "+OEi73" }),
102
102
  to: "/projects/$projectId/storage/$provider/containers",
103
103
  params: {
104
- projectId: t,
104
+ projectId: e,
105
105
  provider: "swift"
106
106
  }
107
107
  }] : [], {
@@ -109,85 +109,85 @@ var j = ({ projectId: t, projectName: i, availableServices: a }) => {
109
109
  label: o._({ id: "KhNDX4" }),
110
110
  to: "/projects/$projectId/storage/$provider/containers",
111
111
  params: {
112
- projectId: t,
112
+ projectId: e,
113
113
  provider: "ceph"
114
114
  }
115
115
  }], B = [...D.pca?.["clavis-beta"] || D.pca?.["clavis-dev"] ? [{
116
116
  service: "pca",
117
117
  label: o._({ id: "miy5mb" }),
118
118
  to: "/projects/$projectId/services/pca",
119
- params: { projectId: t }
119
+ params: { projectId: e }
120
120
  }] : []];
121
- return /* @__PURE__ */ d(n, {
121
+ return /* @__PURE__ */ d(t, {
122
122
  ariaLabel: "Project Side Navigation",
123
- children: /* @__PURE__ */ f(u, { children: [/* @__PURE__ */ d(r, { children: /* @__PURE__ */ f(u, { children: [
124
- /* @__PURE__ */ d(e, {
123
+ children: /* @__PURE__ */ f(u, { children: [/* @__PURE__ */ d(a, { children: /* @__PURE__ */ f(u, { children: [
124
+ /* @__PURE__ */ d(n, {
125
125
  icon: "home",
126
- label: i,
126
+ label: r,
127
127
  onClick: () => c({
128
128
  to: "/projects/$projectId",
129
- params: { projectId: t }
129
+ params: { projectId: e }
130
130
  }),
131
131
  selected: F
132
132
  }),
133
133
  /* @__PURE__ */ d("div", {
134
134
  onClickCapture: I("compute"),
135
- children: /* @__PURE__ */ d(e, {
135
+ children: /* @__PURE__ */ d(n, {
136
136
  label: o._({ id: "rp0Bd0" }),
137
137
  open: k.compute,
138
- children: L.map(({ service: t, label: n, to: r, params: i }) => /* @__PURE__ */ d(e, {
138
+ children: L.map(({ service: e, label: t, to: r, params: i }) => /* @__PURE__ */ d(n, {
139
139
  onClick: () => c({
140
140
  to: r,
141
141
  params: i
142
142
  }),
143
- label: n,
144
- selected: T === "compute" && E === t
145
- }, n))
143
+ label: t,
144
+ selected: T === "compute" && E === e
145
+ }, t))
146
146
  }, N.compute)
147
147
  }),
148
148
  R.length > 0 && /* @__PURE__ */ d("div", {
149
149
  onClickCapture: I("network"),
150
- children: /* @__PURE__ */ d(e, {
150
+ children: /* @__PURE__ */ d(n, {
151
151
  label: o._({ id: "OR475H" }),
152
152
  open: k.network,
153
- children: R.map(({ service: t, label: n, to: r, params: i }) => /* @__PURE__ */ d(e, {
153
+ children: R.map(({ service: e, label: t, to: r, params: i }) => /* @__PURE__ */ d(n, {
154
154
  onClick: () => c({
155
155
  to: r,
156
156
  params: i
157
157
  }),
158
- label: n,
159
- selected: T === "network" && E === t
160
- }, n))
158
+ label: t,
159
+ selected: T === "network" && E === e
160
+ }, t))
161
161
  }, N.network)
162
162
  }),
163
163
  z.length > 0 && /* @__PURE__ */ d("div", {
164
164
  onClickCapture: I("storage"),
165
- children: /* @__PURE__ */ d(e, {
165
+ children: /* @__PURE__ */ d(n, {
166
166
  label: o._({ id: "BrrIs8" }),
167
167
  open: k.storage,
168
- children: z.map(({ service: t, label: n, to: r, params: i }) => /* @__PURE__ */ d(e, {
168
+ children: z.map(({ service: e, label: t, to: r, params: i }) => /* @__PURE__ */ d(n, {
169
169
  onClick: () => c({
170
170
  to: r,
171
171
  params: i
172
172
  }),
173
- label: n,
174
- selected: T === "storage" && E === "containers" ? i.provider === h : E === t
175
- }, n))
173
+ label: t,
174
+ selected: T === "storage" && E === "containers" ? i.provider === h : E === e
175
+ }, t))
176
176
  }, N.storage)
177
177
  }),
178
178
  B.length > 0 && /* @__PURE__ */ d("div", {
179
179
  onClickCapture: I("services"),
180
- children: /* @__PURE__ */ d(e, {
180
+ children: /* @__PURE__ */ d(n, {
181
181
  label: o._({ id: "MILoeL" }),
182
182
  open: k.services,
183
- children: B.map(({ service: t, label: n, to: r, params: i }) => /* @__PURE__ */ d(e, {
183
+ children: B.map(({ service: e, label: t, to: r, params: i }) => /* @__PURE__ */ d(n, {
184
184
  onClick: () => c({
185
185
  to: r,
186
186
  params: i
187
187
  }),
188
- label: n,
189
- selected: T === "services" && E === t
190
- }, n))
188
+ label: t,
189
+ selected: T === "services" && E === e
190
+ }, t))
191
191
  }, N.services)
192
192
  })
193
193
  ] }) }), _?.sideNavBanner && /* @__PURE__ */ d(A, { component: _.sideNavBanner })] })
@@ -195,12 +195,12 @@ var j = ({ projectId: t, projectName: i, availableServices: a }) => {
195
195
  };
196
196
  //#endregion
197
197
  //#region src/client/components/ProjectView/ProjectInfoBox.tsx
198
- function M({ projectInfo: e }) {
199
- let { i18n: n, _: r } = w(), i = x(), o = b(), { projectId: s } = S({ strict: !1 });
200
- return /* @__PURE__ */ d(a, {
201
- className: "mt-8 mb-4",
198
+ function M({ projectInfo: t }) {
199
+ let { i18n: n, _: i } = w(), a = x(), o = b(), { projectId: s } = S({ strict: !1 });
200
+ return /* @__PURE__ */ d(r, {
201
+ className: "relative z-1 mt-8 mb-4",
202
202
  children: m(() => {
203
- let t = {
203
+ let e = {
204
204
  Compute: n._({ id: "rp0Bd0" }),
205
205
  Network: n._({ id: "OR475H" }),
206
206
  Storage: n._({ id: "BrrIs8" }),
@@ -210,28 +210,28 @@ function M({ projectInfo: e }) {
210
210
  "Security Groups": n._({ id: "4opp4r" }),
211
211
  "Floating IPs": n._({ id: "u77/s4" }),
212
212
  "PCA (Clavis)": n._({ id: "miy5mb" })
213
- }, r = (e) => e === "swift" ? n._({ id: "+OEi73" }) : e === "ceph" ? n._({ id: "KhNDX4" }) : n._({ id: "BrrIs8" }), a = [];
214
- a.push({
213
+ }, r = (e) => e === "swift" ? n._({ id: "+OEi73" }) : e === "ceph" ? n._({ id: "KhNDX4" }) : n._({ id: "BrrIs8" }), i = [];
214
+ i.push({
215
215
  icon: "home",
216
216
  label: n._({ id: "i0qMbr" }),
217
- onClick: () => i({ to: "/projects" })
218
- }), e.domain?.name && a.push({ label: e.domain.name }), a.push({
219
- label: e.name,
220
- onClick: () => i({
217
+ onClick: () => a({ to: "/projects" })
218
+ }), t.domain?.name && i.push({ label: t.domain.name }), i.push({
219
+ label: t.name,
220
+ onClick: () => a({
221
221
  to: "/projects/$projectId",
222
222
  params: { projectId: s }
223
223
  })
224
224
  });
225
225
  let c = o.filter((e) => e.routeId !== "/_auth/projects/$projectId" && e.routeId.startsWith("/_auth/projects/$projectId")), l = c[c.length - 1];
226
- if (!l) return a;
226
+ if (!l) return i;
227
227
  let u = O(l.staticData) ? l.staticData : void 0;
228
- if (!u) return a;
228
+ if (!u) return i;
229
229
  let d = l.params;
230
230
  if (u.sectionCrumb) {
231
- let { labelKey: e, to: n } = u.sectionCrumb, r = e ? t[e] : void 0, o = !u.crumb;
232
- a.push(n ? {
231
+ let { labelKey: t, to: n } = u.sectionCrumb, r = t ? e[t] : void 0, o = !u.crumb;
232
+ i.push(n ? {
233
233
  label: r,
234
- onClick: () => i({
234
+ onClick: () => a({
235
235
  to: n,
236
236
  params: d
237
237
  })
@@ -241,23 +241,23 @@ function M({ projectInfo: e }) {
241
241
  });
242
242
  }
243
243
  if (u.crumb) {
244
- let { labelKey: e, to: n, useParamAsLabel: o } = u.crumb, s = o ? r(d[o]) : e ? t[e] : void 0;
244
+ let { labelKey: t, to: n, useParamAsLabel: o } = u.crumb, s = o ? r(d[o]) : t ? e[t] : void 0;
245
245
  if (u.isDetail) {
246
- a.push({
246
+ i.push({
247
247
  label: s,
248
- onClick: () => i({
248
+ onClick: () => a({
249
249
  to: n,
250
250
  params: d
251
251
  })
252
252
  });
253
253
  let e = l.meta?.find((e) => e != null && "title" in e)?.title;
254
- e && a.push({
254
+ e && i.push({
255
255
  label: e,
256
256
  active: !0
257
257
  });
258
- } else a.push(n ? {
258
+ } else i.push(n ? {
259
259
  label: s,
260
- onClick: () => i({
260
+ onClick: () => a({
261
261
  to: n,
262
262
  params: d
263
263
  })
@@ -266,18 +266,18 @@ function M({ projectInfo: e }) {
266
266
  active: !0
267
267
  });
268
268
  }
269
- return a;
269
+ return i;
270
270
  }, [
271
271
  o,
272
- e,
272
+ t,
273
273
  s,
274
- i,
275
- r
276
- ]).map((e, n) => /* @__PURE__ */ d(t, {
277
- label: e.label,
278
- icon: e.icon,
279
- onClick: e.onClick,
280
- active: e.active
274
+ a,
275
+ i
276
+ ]).map((t, n) => /* @__PURE__ */ d(e, {
277
+ label: t.label,
278
+ icon: t.icon,
279
+ onClick: t.onClick,
280
+ active: t.active
281
281
  }, n))
282
282
  });
283
283
  }
@@ -285,7 +285,7 @@ function M({ projectInfo: e }) {
285
285
  //#region src/client/routes/_auth/projects/$projectId.tsx?tsr-split=component
286
286
  function N() {
287
287
  let { availableServices: e, projectId: t, crumbProject: n } = y({ from: c.id });
288
- return /* @__PURE__ */ d(o, {
288
+ return /* @__PURE__ */ d(i, {
289
289
  embedded: !0,
290
290
  sideNavigation: /* @__PURE__ */ d(j, {
291
291
  availableServices: e,
@@ -293,7 +293,7 @@ function N() {
293
293
  projectName: n?.name || t
294
294
  }),
295
295
  className: "h-min-screen",
296
- children: /* @__PURE__ */ d(i, { children: /* @__PURE__ */ d(s, {
296
+ children: /* @__PURE__ */ d(s, { children: /* @__PURE__ */ d(o, {
297
297
  direction: "vertical",
298
298
  distribution: "start",
299
299
  alignment: "stretch",
@@ -313,4 +313,4 @@ function N() {
313
313
  //#endregion
314
314
  export { N as component };
315
315
 
316
- //# sourceMappingURL=_projectId-DsSVV2EA.mjs.map
316
+ //# sourceMappingURL=_projectId-D35MN1kY.mjs.map
@@ -1 +1 @@
1
- {"version":3,"file":"_projectId-DsSVV2EA.mjs","names":["z","CRUMB_LABEL_KEYS","CrumbSchema","object","labelKey","enum","optional","to","string","useParamAsLabel","RouteInfoSchema","section","service","isDetail","boolean","crumb","sectionCrumb","isRouteInfo","data","safeParse","success","useRef","useState","useEffect","createPortal","useRouteContext","SlotShadowRoot","children","ref","root","setRoot","current","shadowRoot","attachShadow","mode","div","style","display","Slot","component","Component","trpcClient","strict","auroraContext","client","useNavigate","useMatches","useParams","useRouteContext","useState","useEffect","getServiceIndex","SideNavigation","SideNavigationList","SideNavigationItem","isRouteInfo","Slot","SideNavBar","projectId","projectName","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","toggle","prev","sectionKeys","setSectionKeys","isOverviewActive","handleSectionClick","e","clickedItem","target","closest","parentElement","querySelector","computeServices","label","t","to","params","networkServices","storageServices","pcaServices","clavisServices","ariaLabel","icon","onClick","selected","div","onClickCapture","open","map","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","isLeaf","crumb","active","useParamAsLabel","resolvedLabel","isDetail","title","meta","find","className","map","item","index","Outlet","useLoaderData","AppShell","Container","Stack","SideNavBar","ProjectInfoBox","Route","RouteComponent","availableServices","projectId","crumbProject","from","id","name","domain","component"],"sources":["../../src/client/routes/routeInfo.ts","../../src/client/components/Slot.tsx","../../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})\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})\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 { 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({ component: Component }: { component: FC<SlotProps> }) {\n const { trpcClient } = useRouteContext({ strict: false })\n\n if (!trpcClient) return null\n\n return (\n <SlotShadowRoot>\n <Component auroraContext={{ client: trpcClient }} />\n </SlotShadowRoot>\n )\n}\n","import { useNavigate, useMatches, useParams, useRouteContext } from \"@tanstack/react-router\"\nimport { type MouseEvent, useState, useEffect } from \"react\"\nimport { getServiceIndex } from \"@/server/Authentication/helpers\"\nimport { SideNavigation, SideNavigationList, SideNavigationItem } 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 availableServices: {\n type: string\n name: string\n }[]\n}\n\nexport const SideNavBar = ({ projectId, projectName, 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 toggle = (section: keyof typeof openSections) =>\n setOpenSections((prev) => ({ ...prev, [section]: !prev[section] }))\n\n // When navigating into a section, force Juno to re-open it by resetting the key\n const [sectionKeys, setSectionKeys] = useState({ compute: 0, network: 0, storage: 0, services: 0 })\n\n useEffect(() => {\n if (activeSection && activeSection in sectionKeys) {\n setSectionKeys((prev) => ({ ...prev, [activeSection]: prev[activeSection as keyof typeof sectionKeys] + 1 }))\n setOpenSections((prev) => ({ ...prev, [activeSection]: true }))\n }\n }, [activeSection])\n\n const isOverviewActive = activeSection === null\n\n const handleSectionClick = (section: keyof typeof openSections) => (e: MouseEvent<HTMLDivElement>) => {\n // Only toggle if the click is on the header row (has expand-icon sibling), not on child items\n const clickedItem = (e.target as HTMLElement).closest(\".juno-sidenavigation-item\")\n if (clickedItem && clickedItem.parentElement?.querySelector(\".expand-icon\")) {\n toggle(section)\n }\n }\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/containers\" as const,\n params: { projectId, provider: \"swift\" },\n },\n ]\n : []),\n {\n service: \"ceph-containers\",\n label: t`Object Storage (Ceph)`,\n to: \"/projects/$projectId/storage/$provider/containers\" as const,\n params: { projectId, provider: \"ceph\" },\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 icon=\"home\"\n label={projectName}\n onClick={() => navigate({ to: \"/projects/$projectId\", params: { projectId } })}\n selected={isOverviewActive}\n />\n {/* onClickCapture fires before Juno's chevron stopPropagation, keeping our state in sync */}\n <div onClickCapture={handleSectionClick(\"compute\")}>\n <SideNavigationItem key={sectionKeys.compute} 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 </SideNavigationItem>\n </div>\n\n {networkServices.length > 0 && (\n <div onClickCapture={handleSectionClick(\"network\")}>\n <SideNavigationItem key={sectionKeys.network} 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 </SideNavigationItem>\n </div>\n )}\n\n {storageServices.length > 0 && (\n <div onClickCapture={handleSectionClick(\"storage\")}>\n <SideNavigationItem key={sectionKeys.storage} 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 </SideNavigationItem>\n </div>\n )}\n\n {clavisServices.length > 0 && (\n <div onClickCapture={handleSectionClick(\"services\")}>\n <SideNavigationItem key={sectionKeys.services} 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 </SideNavigationItem>\n </div>\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) {\n const { labelKey, to } = info.sectionCrumb\n const label = labelKey ? crumbLabels[labelKey] : undefined\n const isLeaf = !info.crumb\n items.push(\n to\n ? { label, onClick: () => navigate({ to: to as never, params: params as never }) }\n : { label, active: isLeaf }\n )\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 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=\"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 } = 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":";;;;;;;;;AAgBA,IAAME,IAAcF,EAAEG,OAAO;CAC3BC,UAAUJ,EAAEK,KAfkB;EAC9B;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACD,CAKkBJ,CAAkBK,UAAQ;CAC3CC,IAAIP,EAAEQ,QAAM,CAAGF,UAAQ;CACvBG,iBAAiBT,EAAEQ,QAAM,CAAGF,UAAQ;CACtC,CAAA,EAEMI,IAAkBV,EAAEG,OAAO;CAC/BQ,SAASX,EAAEQ,QAAM;CACjBI,SAASZ,EAAEQ,QAAM,CAAGF,UAAQ;CAC5BO,UAAUb,EAAEc,SAAO,CAAGR,UAAQ;CAC9BS,OAAOb,EAAYI,UAAQ;CAC3BU,cAAcd,EAAYI,UAAQ;CACpC,CAAA;AAKA,SAAgBW,EAAYC,GAAa;AACvC,QAAOR,EAAgBS,UAAUD,EAAAA,CAAME;;;;AC7BzC,SAASM,EAAe,EAAEC,eAAmC;CAC3D,IAAMC,IAAMP,EAAuB,KAAA,EAC7B,CAACQ,GAAMC,KAAWR,EAA4B,KAAA;AAOpD,QALAC,QAAU;AACHK,IAAIG,WACTD,EAAQF,EAAIG,QAAQC,cAAcJ,EAAIG,QAAQE,aAAa,EAAEC,MAAM,QAAO,CAAA,CAAA;IACzE,EAAE,CAAA,EAGH,kBAACC,OAAAA;EAASP;EAAKQ,OAAO,EAAEC,SAAS,YAAW;YACzCR,KAAQL,kBAAaG,GAAUE,EAAAA;;;AAKtC,SAAgBS,EAAK,EAAEC,WAAWC,KAAyC;CACzE,IAAM,EAAEC,kBAAehB,EAAgB,EAAEiB,QAAQ,IAAM,CAAA;AAIvD,QAFKD,IAGH,kBAACf,GAAAA,EAAAA,UACC,kBAACc,GAAAA,EAAUG,eAAe,EAAEC,QAAQH,GAAW,EAAA,CAAA,EAAA,CAAA,GAJ3B;;;;ACP1B,IAAagB,KAAc,EAAEC,cAAWC,gBAAaC,2BAAoC;CACvF,IAAM,EAAA,MAAA,GAAA,GAAA,MAAQC,GAAAA,EACRC,IAAWjB,GAAAA,EACXkB,IAAUjB,GAAAA,EACV,EAAEkB,gBAAajB,EAAU,EAAEkB,QAAQ,IAAM,CAAA,EACzC,EAAEC,aAAUlB,EAAgB,EAAEiB,QAAQ,IAAM,CAAA,EAG5CE,IAAc,CAAA,GAAIJ,EAAQ,CAACK,SAAO,CAAGC,MAAMC,MAAMf,EAAYe,EAAEC,WAAU,CAAA,EACzEC,IAAkBL,KAAeZ,EAAYY,EAAYI,WAAU,GAAIJ,EAAYI,aAAaE,KAAAA,GAChGC,IAAgBF,GAAiBG,WAAW,MAC5CC,IAAgBJ,GAAiBK,WAAW,MAE5CC,IAAe3B,EAAgBS,EAAAA,EAE/B,CAACmB,GAAcC,KAAmB/B,EAAS;EAAEgC,SAAS;EAAMC,SAAS;EAAMC,SAAS;EAAMC,UAAU;EAAK,CAAA,EACzGC,KAAUV,MACdK,GAAiBM,OAAU;EAAE,GAAGA;GAAOX,IAAU,CAACW,EAAKX;EAAS,EAAA,EAG5D,CAACY,GAAaC,KAAkBvC,EAAS;EAAEgC,SAAS;EAAGC,SAAS;EAAGC,SAAS;EAAGC,UAAU;EAAE,CAAA;AAEjGlC,SAAU;AACR,EAAIwB,KAAiBA,KAAiBa,MACpCC,GAAgBF,OAAU;GAAE,GAAGA;IAAOZ,IAAgBY,EAAKZ,KAA6C;GAAE,EAAA,EAC1GM,GAAiBM,OAAU;GAAE,GAAGA;IAAOZ,IAAgB;GAAK,EAAA;IAE7D,CAACA,EAAc,CAAA;CAElB,IAAMe,IAAmBf,MAAkB,MAErCgB,KAAsBf,OAAwCgB,MAAAA;EAElE,IAAMC,IAAc,EAAGC,OAAuBC,QAAQ,4BAAA;AACtD,EAAIF,KAAeA,EAAYG,eAAeC,cAAc,eAAA,IAC1DX,EAAOV,EAAAA;IAILsB,IAAkB,CAAA,GAClBnB,EAAa,OAAW,SACxB,CACE;EACED,SAAS;EACTqB,OAAOC,EAAAA,EAAC,EAAA,IAAA,UAAO,CAAA;EACfC,IAAI;EACJC,QAAQ,EAAE3C,cAAU;EACtB,CACD,GACD,EAAE,EAAA,GACFoB,GAAe,SAAa,OAC5B,CACE;EACED,SAAS;EACTqB,OAAOC,EAAAA,EAAC,EAAA,IAAA,UAAQ,CAAA;EAChBC,IAAI;EACJC,QAAQ,EAAE3C,cAAU;EACtB,CACD,GACD,EAAE,CACP,EAEK4C,IAAkB,CAAA,GAClBxB,EAAa,UACb,CACE;EACED,SAAS;EACTqB,OAAOC,EAAAA,EAAC,EAAA,IAAA,UAAgB,CAAA;EACxBC,IAAI;EACJC,QAAQ,EAAE3C,cAAU;EACtB,EACA;EACEmB,SAAS;EACTqB,OAAOC,EAAAA,EAAC,EAAA,IAAA,UAAa,CAAA;EACrBC,IAAI;EACJC,QAAQ,EAAE3C,cAAU;EACtB,CACD,GACD,EAAE,CACP,EAEK6C,IAAkB,CAAA,GAClBzB,IAAe,iBAAkB,QACjC,CACE;EACED,SAAS;EACTqB,OAAOC,EAAAA,EAAC,EAAA,IAAA,UAAuB,CAAA;EAC/BC,IAAI;EACJC,QAAQ;GAAE3C;GAAWM,UAAU;GAAQ;EACzC,CACD,GACD,EAAE,EACN;EACEa,SAAS;EACTqB,OAAOC,EAAAA,EAAC,EAAA,IAAA,UAAsB,CAAA;EAC9BC,IAAI;EACJC,QAAQ;GAAE3C;GAAWM,UAAU;GAAO;EACxC,CACD,EAIKyC,IAAiB,CAAA,GADH3B,EAAa,MAAS,kBAAkBA,EAAa,MAAS,gBAG5E,CACE;EACED,SAAS;EACTqB,OAAOC,EAAAA,EAAC,EAAA,IAAA,UAAa,CAAA;EACrBC,IAAI;EACJC,QAAQ,EAAE3C,cAAU;EACtB,CACD,GACD,EAAE,CACP;AAED,QACE,kBAACN,GAAAA;EAAesD,WAAU;YACxB,kBAAA,GAAA,EAAA,UAAA,CACE,kBAACrD,GAAAA,EAAAA,UACC,kBAAA,GAAA,EAAA,UAAA;GACE,kBAACC,GAAAA;IACCqD,MAAK;IACLT,OAAOvC;IACPiD,eAAe9C,EAAS;KAAEsC,IAAI;KAAwBC,QAAQ,EAAE3C,cAAU;KAAE,CAAA;IAC5EmD,UAAUpB;;GAGZ,kBAACqB,OAAAA;IAAIC,gBAAgBrB,EAAmB,UAAA;cACtC,kBAACpC,GAAAA;KAA6C4C,OAAOC,EAAAA,EAAC,EAAA,IAAA,UAAQ,CAAA;KAAGa,MAAMjC,EAAaE;eACjFgB,EAAgBgB,KAAK,EAAEpC,YAASqB,UAAOE,OAAIC,gBAC1C,kBAAC/C,GAAAA;MAECsD,eAAe9C,EAAS;OAAEsC;OAAIC;OAAO,CAAA;MAC9BH;MACPW,UAAUnC,MAAkB,aAAaE,MAAkBC;QAHtDqB,EAAAA,CAAAA;OAHcX,EAAYN,QAAO;;GAY7CqB,EAAgBY,SAAS,KACxB,kBAACJ,OAAAA;IAAIC,gBAAgBrB,EAAmB,UAAA;cACtC,kBAACpC,GAAAA;KAA6C4C,OAAOC,EAAAA,EAAC,EAAA,IAAA,UAAQ,CAAA;KAAGa,MAAMjC,EAAaG;eACjFoB,EAAgBW,KAAK,EAAEpC,YAASqB,UAAOE,OAAIC,gBAC1C,kBAAC/C,GAAAA;MAECsD,eAAe9C,EAAS;OAAEsC;OAAIC;OAAO,CAAA;MAC9BH;MACPW,UAAUnC,MAAkB,aAAaE,MAAkBC;QAHtDqB,EAAAA,CAAAA;OAHcX,EAAYL,QAAO;;GAa/CqB,EAAgBW,SAAS,KACxB,kBAACJ,OAAAA;IAAIC,gBAAgBrB,EAAmB,UAAA;cACtC,kBAACpC,GAAAA;KAA6C4C,OAAOC,EAAAA,EAAC,EAAA,IAAA,UAAQ,CAAA;KAAGa,MAAMjC,EAAaI;eACjFoB,EAAgBU,KAAK,EAAEpC,YAASqB,UAAOE,OAAIC,gBAMxC,kBAAC/C,GAAAA;MAECsD,eAAe9C,EAAS;OAAEsC;OAAIC;OAAO,CAAA;MAC9BH;MACPW,UARwBnC,MAAkB,aAAaE,MAAkB,eACpCyB,EAAOrC,aAAaA,IAAWY,MAAkBC;QAIjFqB,EAAAA,CAMX;OAduBX,EAAYJ,QAAO;;GAmB/CsB,EAAeS,SAAS,KACvB,kBAACJ,OAAAA;IAAIC,gBAAgBrB,EAAmB,WAAA;cACtC,kBAACpC,GAAAA;KAA8C4C,OAAOC,EAAAA,EAAC,EAAA,IAAA,UAAS,CAAA;KAAGa,MAAMjC,EAAaK;eACnFqB,EAAeQ,KAAK,EAAEpC,YAASqB,UAAOE,OAAIC,gBACzC,kBAAC/C,GAAAA;MAECsD,eAAe9C,EAAS;OAAEsC;OAAIC;OAAO,CAAA;MAC9BH;MACPW,UAAUnC,MAAkB,cAAcE,MAAkBC;QAHvDqB,EAAAA,CAAAA;OAHcX,EAAYH,SAAQ;;WAcpDlB,GAAOmD,iBAAiB,kBAAC7D,GAAAA,EAAK8D,WAAWpD,EAAMmD,eAAAA,CAAAA,CAAAA,EAAAA,CAAAA;;;;;AChMxD,SAAgBS,EAAe,EAAEC,kBAAkC;CACjE,IAAM,EAAA,MAAA,GAAA,GAAA,MAAQC,GAAAA,EACRC,IAAWP,GAAAA,EACXQ,IAAUT,GAAAA,EACV,EAAEU,iBAAcR,EAAU,EAAES,QAAQ,IAAM,CAAA;AAgFhD,QACE,kBAACb,GAAAA;EAAWuD,WAAU;YA/EJlD,QAAQ;GAC1B,IAAMU,IAA6C;IACjDC,SAASC,EAAAA,EAAC,EAAA,IAAA,UAAQ,CAAA;IAClBC,SAASD,EAAAA,EAAC,EAAA,IAAA,UAAQ,CAAA;IAClBE,SAASF,EAAAA,EAAC,EAAA,IAAA,UAAQ,CAAA;IAClBG,UAAUH,EAAAA,EAAC,EAAA,IAAA,UAAS,CAAA;IACpBI,QAAQJ,EAAAA,EAAC,EAAA,IAAA,UAAO,CAAA;IAChBK,SAASL,EAAAA,EAAC,EAAA,IAAA,UAAQ,CAAA;IAClB,mBAAmBA,EAAAA,EAAC,EAAA,IAAA,UAAgB,CAAA;IACpC,gBAAgBA,EAAAA,EAAC,EAAA,IAAA,UAAa,CAAA;IAC9B,gBAAgBA,EAAAA,EAAC,EAAA,IAAA,UAAa,CAAA;IAChC,EAEMM,KAAwBC,MACxBA,MAAa,UAAgBP,EAAAA,EAAC,EAAA,IAAA,UAAuB,CAAA,GACrDO,MAAa,SAAeP,EAAAA,EAAC,EAAA,IAAA,UAAsB,CAAA,GAChDA,EAAAA,EAAC,EAAA,IAAA,UAAQ,CAAA,EAGZQ,IAA8F,EAAE;AAQtGA,GANAA,EAAMC,KAAK;IAAEC,MAAM;IAAQC,OAAOX,EAAAA,EAAC,EAAA,IAAA,UAAK,CAAA;IAAGY,eAAenB,EAAS,EAAEoB,IAAI,aAAY,CAAA;IAAG,CAAA,EAEpFtB,EAAYuB,QAAQC,QACtBP,EAAMC,KAAK,EAAEE,OAAOpB,EAAYuB,OAAOC,MAAK,CAAA,EAG9CP,EAAMC,KAAK;IACTE,OAAOpB,EAAYwB;IACnBH,eAAenB,EAAS;KAAEoB,IAAI;KAAwBG,QAAQ,EAAErB,cAAU;KAAE,CAAA;IAC9E,CAAA;GAEA,IAAMsB,IAAiBvB,EAAQwB,QAC5BC,MAAMA,EAAEC,YAAY,gCAAgCD,EAAEC,QAAQC,WAAW,6BAAA,CAAA,EAEtEC,IAAUL,EAAeA,EAAeM,SAAS;AACvD,OAAI,CAACD,EAAS,QAAOd;GAErB,IAAMgB,IAAOnC,EAAYiC,EAAQG,WAAU,GAAIH,EAAQG,aAAaC,KAAAA;AACpE,OAAI,CAACF,EAAM,QAAOhB;GAElB,IAAMQ,IAASM,EAAQN;AAEvB,OAAIQ,EAAKG,cAAc;IACrB,IAAM,EAAEC,aAAUf,UAAOW,EAAKG,cACxBhB,IAAQiB,IAAW9B,EAAY8B,KAAYF,KAAAA,GAC3CG,IAAS,CAACL,EAAKM;AACrBtB,MAAMC,KACJI,IACI;KAAEF;KAAOC,eAAenB,EAAS;MAAMoB;MAAqBG;MAAgB,CAAA;KAAG,GAC/E;KAAEL;KAAOoB,QAAQF;KAAO,CAAA;;AAIhC,OAAIL,EAAKM,OAAO;IACd,IAAM,EAAEF,aAAUf,OAAImB,uBAAoBR,EAAKM,OACzCG,IAAgBD,IAClB1B,EAAqBU,EAAOgB,GAAgB,GAC5CJ,IACE9B,EAAY8B,KACZF,KAAAA;AAEN,QAAIF,EAAKU,UAAU;AACjB1B,OAAMC,KAAK;MAAEE,OAAOsB;MAAerB,eAAenB,EAAS;OAAMoB;OAAqBG;OAAgB,CAAA;MAAG,CAAA;KACzG,IAAMmB,IAAQb,EAAQc,MAAMC,MAAMlB,MAAMA,KAAK,QAAQ,WAAWA,EAAAA,EAAIgB;AACpE,KAAIA,KAAO3B,EAAMC,KAAK;MAAEE,OAAOwB;MAAOJ,QAAQ;MAAK,CAAA;UAEnDvB,GAAMC,KACJI,IACI;KAAEF,OAAOsB;KAAerB,eAAenB,EAAS;MAAMoB;MAAqBG;MAAgB,CAAA;KAAG,GAC9F;KAAEL,OAAOsB;KAAeF,QAAQ;KAAK,CAAA;;AAK/C,UAAOvB;KACN;GAACd;GAASH;GAAaI;GAAWF;;GAAY,CAAA,CAIhC8C,KAAKC,GAAMC,MACtB,kBAACzD,GAAAA;GAA2B2B,OAAO6B,EAAK7B;GAAOD,MAAM8B,EAAK9B;GAAME,SAAS4B,EAAK5B;GAASmB,QAAQS,EAAKT;KAA/EU,EAAAA,CAAAA;;;;;ACtE7B,SAASS,IAAAA;CACP,IAAM,EAAEC,sBAAmBC,cAAWC,oBAAiBV,EAAc,EAAEW,MAAML,EAAMM,IAAG,CAAA;AAEtF,QACE,kBAAC,GAAA;EACC,UAAQ;EACR,gBACE,kBAAC,GAAA;GACoBJ;GACRC;GACX,aAAaC,GAAcG,QAAQJ;;EAGvC,WAAU;YAEV,kBAAC,GAAA,EAAA,UACC,kBAAC,GAAA;GAAM,WAAU;GAAW,cAAa;GAAQ,WAAU;GAAU,WAAU;GAAc,KAAI;aAE/F,kBAAC,OAAA;IAAI,WAAU;eACb,kBAAC,GAAA,EACC,aAAa;KACXG,IAAIH;KACJI,MAAMH,GAAcG,QAAQJ;KAC5BK,QAAQJ,GAAcI;KACxB,EAAA,CAAA,EAEF,kBAAC,GAAA,EAAA,CAAA,CAAA"}
1
+ {"version":3,"file":"_projectId-D35MN1kY.mjs","names":["z","CRUMB_LABEL_KEYS","CrumbSchema","object","labelKey","enum","optional","to","string","useParamAsLabel","RouteInfoSchema","section","service","isDetail","boolean","crumb","sectionCrumb","isRouteInfo","data","safeParse","success","useRef","useState","useEffect","createPortal","useRouteContext","SlotShadowRoot","children","ref","root","setRoot","current","shadowRoot","attachShadow","mode","div","style","display","Slot","component","Component","trpcClient","strict","auroraContext","client","useNavigate","useMatches","useParams","useRouteContext","useState","useEffect","getServiceIndex","SideNavigation","SideNavigationList","SideNavigationItem","isRouteInfo","Slot","SideNavBar","projectId","projectName","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","toggle","prev","sectionKeys","setSectionKeys","isOverviewActive","handleSectionClick","e","clickedItem","target","closest","parentElement","querySelector","computeServices","label","t","to","params","networkServices","storageServices","pcaServices","clavisServices","ariaLabel","icon","onClick","selected","div","onClickCapture","open","map","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","isLeaf","crumb","active","useParamAsLabel","resolvedLabel","isDetail","title","meta","find","className","map","item","index","Outlet","useLoaderData","AppShell","Container","Stack","SideNavBar","ProjectInfoBox","Route","RouteComponent","availableServices","projectId","crumbProject","from","id","name","domain","component"],"sources":["../../src/client/routes/routeInfo.ts","../../src/client/components/Slot.tsx","../../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})\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})\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 { 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({ component: Component }: { component: FC<SlotProps> }) {\n const { trpcClient } = useRouteContext({ strict: false })\n\n if (!trpcClient) return null\n\n return (\n <SlotShadowRoot>\n <Component auroraContext={{ client: trpcClient }} />\n </SlotShadowRoot>\n )\n}\n","import { useNavigate, useMatches, useParams, useRouteContext } from \"@tanstack/react-router\"\nimport { type MouseEvent, useState, useEffect } from \"react\"\nimport { getServiceIndex } from \"@/server/Authentication/helpers\"\nimport { SideNavigation, SideNavigationList, SideNavigationItem } 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 availableServices: {\n type: string\n name: string\n }[]\n}\n\nexport const SideNavBar = ({ projectId, projectName, 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 toggle = (section: keyof typeof openSections) =>\n setOpenSections((prev) => ({ ...prev, [section]: !prev[section] }))\n\n // When navigating into a section, force Juno to re-open it by resetting the key\n const [sectionKeys, setSectionKeys] = useState({ compute: 0, network: 0, storage: 0, services: 0 })\n\n useEffect(() => {\n if (activeSection && activeSection in sectionKeys) {\n setSectionKeys((prev) => ({ ...prev, [activeSection]: prev[activeSection as keyof typeof sectionKeys] + 1 }))\n setOpenSections((prev) => ({ ...prev, [activeSection]: true }))\n }\n }, [activeSection])\n\n const isOverviewActive = activeSection === null\n\n const handleSectionClick = (section: keyof typeof openSections) => (e: MouseEvent<HTMLDivElement>) => {\n // Only toggle if the click is on the header row (has expand-icon sibling), not on child items\n const clickedItem = (e.target as HTMLElement).closest(\".juno-sidenavigation-item\")\n if (clickedItem && clickedItem.parentElement?.querySelector(\".expand-icon\")) {\n toggle(section)\n }\n }\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/containers\" as const,\n params: { projectId, provider: \"swift\" },\n },\n ]\n : []),\n {\n service: \"ceph-containers\",\n label: t`Object Storage (Ceph)`,\n to: \"/projects/$projectId/storage/$provider/containers\" as const,\n params: { projectId, provider: \"ceph\" },\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 icon=\"home\"\n label={projectName}\n onClick={() => navigate({ to: \"/projects/$projectId\", params: { projectId } })}\n selected={isOverviewActive}\n />\n {/* onClickCapture fires before Juno's chevron stopPropagation, keeping our state in sync */}\n <div onClickCapture={handleSectionClick(\"compute\")}>\n <SideNavigationItem key={sectionKeys.compute} 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 </SideNavigationItem>\n </div>\n\n {networkServices.length > 0 && (\n <div onClickCapture={handleSectionClick(\"network\")}>\n <SideNavigationItem key={sectionKeys.network} 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 </SideNavigationItem>\n </div>\n )}\n\n {storageServices.length > 0 && (\n <div onClickCapture={handleSectionClick(\"storage\")}>\n <SideNavigationItem key={sectionKeys.storage} 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 </SideNavigationItem>\n </div>\n )}\n\n {clavisServices.length > 0 && (\n <div onClickCapture={handleSectionClick(\"services\")}>\n <SideNavigationItem key={sectionKeys.services} 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 </SideNavigationItem>\n </div>\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) {\n const { labelKey, to } = info.sectionCrumb\n const label = labelKey ? crumbLabels[labelKey] : undefined\n const isLeaf = !info.crumb\n items.push(\n to\n ? { label, onClick: () => navigate({ to: to as never, params: params as never }) }\n : { label, active: isLeaf }\n )\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 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 } = 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":"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"}
@@ -1,8 +1,8 @@
1
- import { t as e } from "./RouteError-iP1Vd6bY.mjs";
1
+ import { t as e } from "./RouteError-pDEWC_k7.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-KH5si25Q.mjs.map
8
+ //# sourceMappingURL=_projectId-OW2xkK43.mjs.map
@@ -1 +1 @@
1
- {"version":3,"file":"_projectId-KH5si25Q.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,kBAAC,GAAA,EAAkBA,UAAAA,CAAAA"}
1
+ {"version":3,"file":"_projectId-OW2xkK43.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,kBAAC,GAAA,EAAkBA,UAAAA,CAAAA"}
@@ -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-O7FXfx0M.mjs"), "component"),
22
+ component: n(() => import("./_securityGroupId-B1bOYRbX.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-DgaSqYex.mjs.map
34
+ //# sourceMappingURL=_securityGroupId-B-Z-CzLp.mjs.map
@@ -1 +1 @@
1
- {"version":3,"file":"_securityGroupId-DgaSqYex.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,sEAAA,CAAuE;CAC1GI,YAAY;EACVC,SAAS;EACTC,SAAS;EACTC,UAAU;EACVC,cAAc,EAAEC,UAAU,WAAU;EACpCC,OAAO;GAAED,UAAU;GAAmBE,IAAI;GAA8C;EAC1F;CACAE,QAAQ,OAAO,EAAEC,YAASC,gBAAQ;EAChC,IAAMC,IAAK,MAAMF,EAAQG,YAAYC,QAAQC,cAAcC,QAAQC,MAAM;GACvEC,YAAYP,EAAOQ;GACnBC,iBAAiBT,EAAOS;GAC1B,CAAA;AACA,SAAO,EAAEC,SAAST,GAAIU,QAAQV,GAAIW,MAAM,MAAK;;CAE/CC,OAAO,EAAEC,qBAAkB,EACzBC,MAAM,CAAC,EAAEC,OAAOF,GAAYJ,WAAW,kBAAiB,CAAC,EAC3D;CACAO,WAASC,mDAAA,YAAA;CACTE,YAAY,OAAO,EAAErB,YAASC,gBAAQ;EACpC,IAAM,EAAEE,kBAAeH,GAIjByB,IAAerC,EAFK,MAAOe,GAAYoB,KAAKC,qBAAqBjB,OAAAA,IAAY,EAAE,CAEhDe;AAUrC,MAPI,CAACG,EAAa,WAOd,CAACA,EAAa,QAAW,QAE3B,OAAMtC,EAAS;GACbU,IAAI;GACJI,QAAQ,EAAEQ,WAAWR,EAAOQ,WAAU;GACxC,CAAA;;CAGN,CAAA"}
1
+ {"version":3,"file":"_securityGroupId-B-Z-CzLp.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,sEAAA,CAAuE;CAC1GI,YAAY;EACVC,SAAS;EACTC,SAAS;EACTC,UAAU;EACVC,cAAc,EAAEC,UAAU,WAAU;EACpCC,OAAO;GAAED,UAAU;GAAmBE,IAAI;GAA8C;EAC1F;CACAE,QAAQ,OAAO,EAAEC,YAASC,gBAAQ;EAChC,IAAMC,IAAK,MAAMF,EAAQG,YAAYC,QAAQC,cAAcC,QAAQC,MAAM;GACvEC,YAAYP,EAAOQ;GACnBC,iBAAiBT,EAAOS;GAC1B,CAAA;AACA,SAAO,EAAEC,SAAST,GAAIU,QAAQV,GAAIW,MAAM,MAAK;;CAE/CC,OAAO,EAAEC,qBAAkB,EACzBC,MAAM,CAAC,EAAEC,OAAOF,GAAYJ,WAAW,kBAAiB,CAAC,EAC3D;CACAO,WAASC,mDAAA,YAAA;CACTE,YAAY,OAAO,EAAErB,YAASC,gBAAQ;EACpC,IAAM,EAAEE,kBAAeH,GAIjByB,IAAerC,EAFK,MAAOe,GAAYoB,KAAKC,qBAAqBjB,OAAAA,IAAY,EAAE,CAEhDe;AAUrC,MAPI,CAACG,EAAa,WAOd,CAACA,EAAa,QAAW,QAE3B,OAAMtC,EAAS;GACbU,IAAI;GACJI,QAAQ,EAAEQ,WAAWR,EAAOQ,WAAU;GACxC,CAAA;;CAGN,CAAA"}