@cobaltcore-dev/aurora 0.1.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 (142) hide show
  1. package/LICENSE +201 -0
  2. package/README.md +141 -0
  3. package/dist/client/AuroraApp.d.ts +7 -0
  4. package/dist/client/AuthProvider-D-5Jpa6F.mjs +100 -0
  5. package/dist/client/AuthProvider-D-5Jpa6F.mjs.map +1 -0
  6. package/dist/client/ContentHeader-H8KGY3Wd.mjs +81 -0
  7. package/dist/client/ContentHeader-H8KGY3Wd.mjs.map +1 -0
  8. package/dist/client/DeleteFlavorModal-B98oiHWx.mjs +629 -0
  9. package/dist/client/DeleteFlavorModal-B98oiHWx.mjs.map +1 -0
  10. package/dist/client/EditSecurityGroupModal-wQVNIVg1.mjs +137 -0
  11. package/dist/client/EditSecurityGroupModal-wQVNIVg1.mjs.map +1 -0
  12. package/dist/client/FloatingIpActionModals-qu1NMI5a.mjs +417 -0
  13. package/dist/client/FloatingIpActionModals-qu1NMI5a.mjs.map +1 -0
  14. package/dist/client/ImageToastNotifications-wsQDNEh7.mjs +1267 -0
  15. package/dist/client/ImageToastNotifications-wsQDNEh7.mjs.map +1 -0
  16. package/dist/client/ListToolbar-CHlkZrpl.mjs +223 -0
  17. package/dist/client/ListToolbar-CHlkZrpl.mjs.map +1 -0
  18. package/dist/client/RouteError-BwgDIwJE.mjs +25 -0
  19. package/dist/client/RouteError-BwgDIwJE.mjs.map +1 -0
  20. package/dist/client/_auth-CsliQdkJ.mjs +11 -0
  21. package/dist/client/_auth-CsliQdkJ.mjs.map +1 -0
  22. package/dist/client/_flavorId-D_A53VYa.mjs +56 -0
  23. package/dist/client/_flavorId-D_A53VYa.mjs.map +1 -0
  24. package/dist/client/_flavorId-DbhYLFxY.mjs +190 -0
  25. package/dist/client/_flavorId-DbhYLFxY.mjs.map +1 -0
  26. package/dist/client/_floatingIpId-BGgftRBQ.mjs +21 -0
  27. package/dist/client/_floatingIpId-BGgftRBQ.mjs.map +1 -0
  28. package/dist/client/_floatingIpId-D5myuLFz.mjs +228 -0
  29. package/dist/client/_floatingIpId-D5myuLFz.mjs.map +1 -0
  30. package/dist/client/_imageId-BoHX155h.mjs +27 -0
  31. package/dist/client/_imageId-BoHX155h.mjs.map +1 -0
  32. package/dist/client/_imageId-CTa0c3Av.mjs +530 -0
  33. package/dist/client/_imageId-CTa0c3Av.mjs.map +1 -0
  34. package/dist/client/_pcaId-C7Lrv1H_.mjs +242 -0
  35. package/dist/client/_pcaId-C7Lrv1H_.mjs.map +1 -0
  36. package/dist/client/_pcaId-DBgz5V_9.mjs +21 -0
  37. package/dist/client/_pcaId-DBgz5V_9.mjs.map +1 -0
  38. package/dist/client/_projectId-B9fln31N.mjs +8 -0
  39. package/dist/client/_projectId-B9fln31N.mjs.map +1 -0
  40. package/dist/client/_projectId-Be1Erj68.mjs +300 -0
  41. package/dist/client/_projectId-Be1Erj68.mjs.map +1 -0
  42. package/dist/client/_projectId-C-E4NNgo.mjs +84 -0
  43. package/dist/client/_projectId-C-E4NNgo.mjs.map +1 -0
  44. package/dist/client/_projectId-INhedXor.mjs +26 -0
  45. package/dist/client/_projectId-INhedXor.mjs.map +1 -0
  46. package/dist/client/_securityGroupId-DQoRQ-yA.mjs +1647 -0
  47. package/dist/client/_securityGroupId-DQoRQ-yA.mjs.map +1 -0
  48. package/dist/client/_securityGroupId-ihjy8Lcd.mjs +21 -0
  49. package/dist/client/_securityGroupId-ihjy8Lcd.mjs.map +1 -0
  50. package/dist/client/about-oT6ccz8T.mjs +92 -0
  51. package/dist/client/about-oT6ccz8T.mjs.map +1 -0
  52. package/dist/client/aurora-D_NPTbo-.mjs +19 -0
  53. package/dist/client/aurora-D_NPTbo-.mjs.map +1 -0
  54. package/dist/client/build-eu9eg0zF.mjs +14619 -0
  55. package/dist/client/build-eu9eg0zF.mjs.map +1 -0
  56. package/dist/client/buildFilterParams-BDOIRDeD.mjs +15 -0
  57. package/dist/client/buildFilterParams-BDOIRDeD.mjs.map +1 -0
  58. package/dist/client/cn-C3laVXMm.mjs +10 -0
  59. package/dist/client/cn-C3laVXMm.mjs.map +1 -0
  60. package/dist/client/constants-ByHCdNsI.mjs +128 -0
  61. package/dist/client/constants-ByHCdNsI.mjs.map +1 -0
  62. package/dist/client/containers-B_ozmVlx.mjs +74 -0
  63. package/dist/client/containers-B_ozmVlx.mjs.map +1 -0
  64. package/dist/client/containers-Dx7TYruP.mjs +7 -0
  65. package/dist/client/containers-Dx7TYruP.mjs.map +1 -0
  66. package/dist/client/containers-rn_ntCJu.mjs +3029 -0
  67. package/dist/client/containers-rn_ntCJu.mjs.map +1 -0
  68. package/dist/client/flavors-CT4auvLO.mjs +565 -0
  69. package/dist/client/flavors-CT4auvLO.mjs.map +1 -0
  70. package/dist/client/flavors-DRZb9LJP.mjs +8 -0
  71. package/dist/client/flavors-DRZb9LJP.mjs.map +1 -0
  72. package/dist/client/flavors-DtgMd0Ii.mjs +12 -0
  73. package/dist/client/flavors-DtgMd0Ii.mjs.map +1 -0
  74. package/dist/client/floatingips-DG5cFJSZ.mjs +12 -0
  75. package/dist/client/floatingips-DG5cFJSZ.mjs.map +1 -0
  76. package/dist/client/floatingips-iCMR0ZiL.mjs +436 -0
  77. package/dist/client/floatingips-iCMR0ZiL.mjs.map +1 -0
  78. package/dist/client/formatBytes-GYujK0dP.mjs +38 -0
  79. package/dist/client/formatBytes-GYujK0dP.mjs.map +1 -0
  80. package/dist/client/helpers--JWXi40U.mjs +6 -0
  81. package/dist/client/helpers--JWXi40U.mjs.map +1 -0
  82. package/dist/client/hooks-s-I8vWww.mjs +2 -0
  83. package/dist/client/images-BTqRflJv2.mjs +1794 -0
  84. package/dist/client/images-BTqRflJv2.mjs.map +1 -0
  85. package/dist/client/images-DRTfx8k2.mjs +8 -0
  86. package/dist/client/images-DRTfx8k2.mjs.map +1 -0
  87. package/dist/client/images-xBfsjxkX.mjs +12 -0
  88. package/dist/client/images-xBfsjxkX.mjs.map +1 -0
  89. package/dist/client/index.d.ts +1 -0
  90. package/dist/client/index.js +1033 -0
  91. package/dist/client/index.js.map +1 -0
  92. package/dist/client/md-CI9FmfYv.mjs +390 -0
  93. package/dist/client/md-CI9FmfYv.mjs.map +1 -0
  94. package/dist/client/network-DFVVVNS5.mjs +8 -0
  95. package/dist/client/network-DFVVVNS5.mjs.map +1 -0
  96. package/dist/client/objects-CKk6cST_.mjs +4762 -0
  97. package/dist/client/objects-CKk6cST_.mjs.map +1 -0
  98. package/dist/client/objects-DkDKVSmQ.mjs +8 -0
  99. package/dist/client/objects-DkDKVSmQ.mjs.map +1 -0
  100. package/dist/client/objects-r_Vl31oj.mjs +80 -0
  101. package/dist/client/objects-r_Vl31oj.mjs.map +1 -0
  102. package/dist/client/overview-B7pXx6bt.mjs +173 -0
  103. package/dist/client/overview-B7pXx6bt.mjs.map +1 -0
  104. package/dist/client/overview-CKGLIu6W.mjs +12 -0
  105. package/dist/client/overview-CKGLIu6W.mjs.map +1 -0
  106. package/dist/client/overview-Ca8r3SAz.mjs +16 -0
  107. package/dist/client/overview-Ca8r3SAz.mjs.map +1 -0
  108. package/dist/client/overview-DkPM0Od5.mjs +12 -0
  109. package/dist/client/overview-DkPM0Od5.mjs.map +1 -0
  110. package/dist/client/overview-Dxm7Ef3X.mjs +12 -0
  111. package/dist/client/overview-Dxm7Ef3X.mjs.map +1 -0
  112. package/dist/client/overview-ag4Envez.mjs +16 -0
  113. package/dist/client/overview-ag4Envez.mjs.map +1 -0
  114. package/dist/client/pca-BGv7Mprl.mjs +12 -0
  115. package/dist/client/pca-BGv7Mprl.mjs.map +1 -0
  116. package/dist/client/pca-DpULpMu5.mjs +167 -0
  117. package/dist/client/pca-DpULpMu5.mjs.map +1 -0
  118. package/dist/client/projects-BuN69cxO.mjs +144 -0
  119. package/dist/client/projects-BuN69cxO.mjs.map +1 -0
  120. package/dist/client/projects-D1pP0XdA.mjs +12 -0
  121. package/dist/client/projects-D1pP0XdA.mjs.map +1 -0
  122. package/dist/client/projects-MbS1USl2.mjs +7 -0
  123. package/dist/client/projects-MbS1USl2.mjs.map +1 -0
  124. package/dist/client/projects-_Dfn6eQT.mjs +22 -0
  125. package/dist/client/projects-_Dfn6eQT.mjs.map +1 -0
  126. package/dist/client/securitygroups-DURjFfYK.mjs +12 -0
  127. package/dist/client/securitygroups-DURjFfYK.mjs.map +1 -0
  128. package/dist/client/securitygroups-KC2qvmH8.mjs +442 -0
  129. package/dist/client/securitygroups-KC2qvmH8.mjs.map +1 -0
  130. package/dist/client/trpcClient-BxguzNYF.mjs +57 -0
  131. package/dist/client/trpcClient-BxguzNYF.mjs.map +1 -0
  132. package/dist/client/useErrorTranslation-TZVwIAzq.mjs +83 -0
  133. package/dist/client/useErrorTranslation-TZVwIAzq.mjs.map +1 -0
  134. package/dist/client/useListWithFiltering-mMX_EfyI.mjs +32 -0
  135. package/dist/client/useListWithFiltering-mMX_EfyI.mjs.map +1 -0
  136. package/dist/client/useModal-Dg4CBeqL.mjs +12 -0
  137. package/dist/client/useModal-Dg4CBeqL.mjs.map +1 -0
  138. package/dist/client/useProjectId-BWaeJZOy.mjs +11 -0
  139. package/dist/client/useProjectId-BWaeJZOy.mjs.map +1 -0
  140. package/dist/server/index.d.ts +35 -0
  141. package/dist/server/index.js +36514 -0
  142. package/package.json +129 -0
@@ -0,0 +1,300 @@
1
+ import { B as e, J as t, M as n, Y as r, nt as i, q as a, y as o, z as s } from "./build-eu9eg0zF.mjs";
2
+ import { t as c } from "./_projectId-INhedXor.mjs";
3
+ import { t as l } from "./helpers--JWXi40U.mjs";
4
+ import { Fragment as u, jsx as d, jsxs as f } from "react/jsx-runtime";
5
+ import { useEffect as p, useState as m } from "react";
6
+ import { Outlet as h, useLoaderData as g, useMatches as _, useNavigate as v, useParams as y, useRouteContext as b } from "@tanstack/react-router";
7
+ import { useLingui as x } from "@lingui/react";
8
+ //#region src/client/routes/routeInfo.ts
9
+ function S(e) {
10
+ return typeof e == "object" && !!e && typeof e.section == "string" && (typeof e.service == "string" || e.service === void 0) && (typeof e.isDetail == "boolean" || e.isDetail === void 0);
11
+ }
12
+ //#endregion
13
+ //#region src/client/routes/_auth/projects/-components/SideNavBar.tsx
14
+ var C = ({ projectId: e, projectName: r, availableServices: i }) => {
15
+ let { i18n: a, _: s } = x(), c = v(), p = _(), { provider: h } = y({ strict: !1 }), [g, b] = m({
16
+ compute: !0,
17
+ network: !0,
18
+ storage: !0,
19
+ services: !0
20
+ }), C = l(i), w = [...p].reverse().find((e) => S(e.staticData)), T = w && S(w.staticData) ? w.staticData : void 0, E = T?.section ?? null, D = T?.service ?? null, O = [...C.image?.glance ? [{
21
+ service: "images",
22
+ label: a._({ id: "an5hVd" }),
23
+ to: "/projects/$projectId/compute/images",
24
+ params: { projectId: e }
25
+ }] : [], ...C?.compute?.nova ? [{
26
+ service: "flavors",
27
+ label: a._({ id: "neiJm0" }),
28
+ to: "/projects/$projectId/compute/flavors",
29
+ params: { projectId: e }
30
+ }] : []], k = [...C.network ? [{
31
+ service: "securitygroups",
32
+ label: a._({ id: "4opp4r" }),
33
+ to: "/projects/$projectId/network/securitygroups",
34
+ params: { projectId: e }
35
+ }, {
36
+ service: "floatingips",
37
+ label: a._({ id: "u77/s4" }),
38
+ to: "/projects/$projectId/network/floatingips",
39
+ params: { projectId: e }
40
+ }] : []], A = [...C?.["object-store"]?.swift ? [{
41
+ service: "containers",
42
+ label: a._({ id: "+OEi73" }),
43
+ to: "/projects/$projectId/storage/$provider/containers",
44
+ params: {
45
+ projectId: e,
46
+ provider: "swift"
47
+ }
48
+ }] : [], {
49
+ service: "ceph-containers",
50
+ label: a._({ id: "B4Jzm7" }),
51
+ to: "/projects/$projectId/storage/$provider/containers",
52
+ params: {
53
+ projectId: e,
54
+ provider: "ceph"
55
+ }
56
+ }], j = [...C.pca?.["clavis-beta"] || C.pca?.["clavis-dev"] ? [{
57
+ service: "pca",
58
+ label: a._({ id: "miy5mb" }),
59
+ to: "/projects/$projectId/services/pca",
60
+ params: { projectId: e }
61
+ }] : []], M = E === null, N = (e) => g[e] || E === e;
62
+ return /* @__PURE__ */ d(o, {
63
+ ariaLabel: "Project Side Navigation",
64
+ onActiveItemChange: () => {},
65
+ children: /* @__PURE__ */ d(n, { children: /* @__PURE__ */ f(u, { children: [
66
+ /* @__PURE__ */ d(t, {
67
+ icon: "home",
68
+ label: r,
69
+ onClick: () => c({
70
+ to: "/projects/$projectId",
71
+ params: { projectId: e }
72
+ }),
73
+ selected: M
74
+ }),
75
+ /* @__PURE__ */ d(t, {
76
+ label: a._({ id: "rp0Bd0" }),
77
+ open: N("compute"),
78
+ onClick: () => b((e) => ({
79
+ ...e,
80
+ compute: !e.compute
81
+ })),
82
+ children: O.map(({ service: e, label: n, to: r, params: i }) => /* @__PURE__ */ d(t, {
83
+ onClick: () => c({
84
+ to: r,
85
+ params: i
86
+ }),
87
+ label: n,
88
+ selected: E === "compute" && D === e
89
+ }, n))
90
+ }),
91
+ k.length > 0 && /* @__PURE__ */ d(t, {
92
+ label: a._({ id: "OR475H" }),
93
+ open: N("network"),
94
+ onClick: () => b((e) => ({
95
+ ...e,
96
+ network: !e.network
97
+ })),
98
+ children: k.map(({ service: e, label: n, to: r, params: i }) => /* @__PURE__ */ d(t, {
99
+ onClick: () => c({
100
+ to: r,
101
+ params: i
102
+ }),
103
+ label: n,
104
+ selected: E === "network" && D === e
105
+ }, n))
106
+ }),
107
+ A.length > 0 && /* @__PURE__ */ d(t, {
108
+ label: a._({ id: "BrrIs8" }),
109
+ open: N("storage"),
110
+ onClick: () => b((e) => ({
111
+ ...e,
112
+ storage: !e.storage
113
+ })),
114
+ children: A.map(({ service: e, label: n, to: r, params: i }) => /* @__PURE__ */ d(t, {
115
+ onClick: () => c({
116
+ to: r,
117
+ params: i
118
+ }),
119
+ label: n,
120
+ selected: E === "storage" && D === "containers" ? i.provider === h : D === e
121
+ }, n))
122
+ }),
123
+ j.length > 0 && /* @__PURE__ */ d(t, {
124
+ label: a._({ id: "MILoeL" }),
125
+ open: N("services"),
126
+ onClick: () => b((e) => ({
127
+ ...e,
128
+ services: !e.services
129
+ })),
130
+ children: j.map(({ service: e, label: n, to: r, params: i }) => /* @__PURE__ */ d(t, {
131
+ onClick: () => c({
132
+ to: r,
133
+ params: i
134
+ }),
135
+ label: n,
136
+ selected: E === "services" && D === e
137
+ }, n))
138
+ })
139
+ ] }) })
140
+ });
141
+ }, w = {
142
+ compute: "Compute",
143
+ network: "Network",
144
+ storage: "Storage",
145
+ services: "Services"
146
+ }, T = {
147
+ images: "Images",
148
+ flavors: "Flavors",
149
+ securitygroups: "Security Groups",
150
+ floatingips: "Floating IPs",
151
+ containers: "Object Storage (Swift)",
152
+ pca: "PCA (Clavis)"
153
+ };
154
+ function E({ projectInfo: e }) {
155
+ let { pageTitleRef: t } = b({ from: "__root__" }), [n, a] = m(t.current), o = v(), s = _(), { projectId: c, provider: l } = y({ strict: !1 });
156
+ return p(() => {
157
+ let e = (e) => {
158
+ a(e.detail.title);
159
+ };
160
+ return window.addEventListener("pageTitleChange", e), a(t.current), () => {
161
+ window.removeEventListener("pageTitleChange", e);
162
+ };
163
+ }, [t]), /* @__PURE__ */ d(i, {
164
+ className: "mt-8 mb-4",
165
+ children: (() => {
166
+ let t = "/_auth/projects/$projectId", r = s.filter((e) => e.routeId !== t && e.routeId.startsWith(t)), i = r[r.length - 1]?.staticData, a = S(i) ? i : void 0, u = a?.section, d = a?.service, f = a?.isDetail ?? !1, p = u ? w[u] : void 0, m = d ? T[d] : void 0;
167
+ d === "containers" && l && (m = `Object Storage (${l === "swift" ? "Swift" : "Ceph"})`);
168
+ let h = !f && d === "overview", g = !f && !!d && d !== "overview", _ = !f && !d && !!u, v = [];
169
+ return v.push({
170
+ icon: "home",
171
+ label: "Home",
172
+ onClick: () => o({ to: "/projects" })
173
+ }), e.domain?.name && v.push({ label: e.domain.name }), v.push({
174
+ label: e.name,
175
+ onClick: () => o({
176
+ to: "/projects/$projectId",
177
+ params: { projectId: c }
178
+ })
179
+ }), p && u && (_ || h ? v.push({
180
+ label: p,
181
+ active: !0
182
+ }) : u === "compute" ? v.push({
183
+ label: p,
184
+ onClick: () => o({
185
+ to: "/projects/$projectId/compute/overview",
186
+ params: { projectId: c }
187
+ })
188
+ }) : u === "network" ? v.push({
189
+ label: p,
190
+ onClick: () => o({
191
+ to: "/projects/$projectId/network/overview",
192
+ params: { projectId: c }
193
+ })
194
+ }) : u === "storage" ? v.push({
195
+ label: p,
196
+ onClick: () => o({
197
+ to: "/projects/$projectId/storage/$provider/containers",
198
+ params: {
199
+ projectId: c,
200
+ provider: l
201
+ }
202
+ })
203
+ }) : u === "services" && v.push({
204
+ label: p,
205
+ onClick: () => o({
206
+ to: "/projects/$projectId/services/overview",
207
+ params: { projectId: c }
208
+ })
209
+ })), m && u && d && (g ? v.push({
210
+ label: m,
211
+ active: !0
212
+ }) : f && (u === "compute" && d === "images" ? v.push({
213
+ label: m,
214
+ onClick: () => o({
215
+ to: "/projects/$projectId/compute/images",
216
+ params: { projectId: c }
217
+ })
218
+ }) : u === "compute" && d === "flavors" ? v.push({
219
+ label: m,
220
+ onClick: () => o({
221
+ to: "/projects/$projectId/compute/flavors",
222
+ params: { projectId: c }
223
+ })
224
+ }) : u === "network" && d === "securitygroups" ? v.push({
225
+ label: m,
226
+ onClick: () => o({
227
+ to: "/projects/$projectId/network/securitygroups",
228
+ params: { projectId: c }
229
+ })
230
+ }) : u === "network" && d === "floatingips" ? v.push({
231
+ label: m,
232
+ onClick: () => o({
233
+ to: "/projects/$projectId/network/floatingips",
234
+ params: { projectId: c }
235
+ })
236
+ }) : u === "storage" && d === "containers" ? v.push({
237
+ label: m,
238
+ onClick: () => o({
239
+ to: "/projects/$projectId/storage/$provider/containers",
240
+ params: {
241
+ projectId: c,
242
+ provider: l
243
+ }
244
+ })
245
+ }) : u === "services" && d === "pca" ? v.push({
246
+ label: m,
247
+ onClick: () => o({
248
+ to: "/projects/$projectId/services/pca",
249
+ params: { projectId: c }
250
+ })
251
+ }) : v.push({
252
+ label: m,
253
+ onClick: () => {
254
+ o({ to: `/projects/${c}/${u}/${d}` });
255
+ }
256
+ }))), f && v.push({
257
+ label: n,
258
+ active: !0
259
+ }), v;
260
+ })().map((e, t) => /* @__PURE__ */ d(r, {
261
+ label: e.label,
262
+ icon: e.icon,
263
+ onClick: e.onClick,
264
+ active: e.active
265
+ }, t))
266
+ });
267
+ }
268
+ //#endregion
269
+ //#region src/client/routes/_auth/projects/$projectId.tsx?tsr-split=component
270
+ function D() {
271
+ let { availableServices: t, projectId: n, crumbProject: r } = g({ from: c.id });
272
+ return /* @__PURE__ */ d(e, {
273
+ embedded: !0,
274
+ sideNavigation: /* @__PURE__ */ d(C, {
275
+ availableServices: t,
276
+ projectId: n,
277
+ projectName: r?.name || n
278
+ }),
279
+ className: "h-min-screen",
280
+ children: /* @__PURE__ */ d(s, { children: /* @__PURE__ */ d(a, {
281
+ direction: "vertical",
282
+ distribution: "start",
283
+ alignment: "stretch",
284
+ className: "xl:flex-row",
285
+ gap: "6",
286
+ children: /* @__PURE__ */ f("div", {
287
+ className: "min-w-0 flex-1",
288
+ children: [/* @__PURE__ */ d(E, { projectInfo: {
289
+ id: n,
290
+ name: r?.name || n,
291
+ domain: r?.domain
292
+ } }), /* @__PURE__ */ d(h, {})]
293
+ })
294
+ }) })
295
+ });
296
+ }
297
+ //#endregion
298
+ export { D as component };
299
+
300
+ //# sourceMappingURL=_projectId-Be1Erj68.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"_projectId-Be1Erj68.mjs","names":["isRouteInfo","data","section","service","undefined","isDetail","useNavigate","useMatches","useParams","useState","getServiceIndex","SideNavigation","SideNavigationList","SideNavigationItem","isRouteInfo","SideNavBar","projectId","projectName","availableServices","useLingui","navigate","matches","provider","strict","openSections","setOpenSections","compute","network","storage","services","serviceIndex","activeMatch","reverse","find","m","staticData","activeRouteInfo","undefined","activeSection","section","activeService","service","computeServices","label","t","to","params","networkServices","storageServices","pcaServices","clavisServices","isOverviewActive","isOpen","ariaLabel","onActiveItemChange","icon","onClick","selected","open","prev","map","length","isStorageContainers","isSelected","Breadcrumb","BreadcrumbItem","useRouteContext","useMatches","useNavigate","useParams","useState","useEffect","isRouteInfo","SECTION_LABELS","compute","network","storage","services","SERVICE_LABELS","images","flavors","securitygroups","floatingips","containers","pca","ProjectInfoBox","projectInfo","pageTitleRef","from","pageTitle","setPageTitle","current","navigate","matches","projectId","provider","strict","handleTitleChange","e","detail","title","window","addEventListener","removeEventListener","buildBreadcrumbs","projectIdRouteId","childMatches","filter","m","routeId","startsWith","deepestMatch","length","rawData","staticData","routeInfo","undefined","section","service","isDetail","sectionLabel","serviceLabel","isOverviewPage","isServicePage","isSectionPage","items","push","icon","label","onClick","to","domain","name","params","active","path","breadcrumbs","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/routes/_auth/projects/-components/SideNavBar.tsx","../../src/client/components/ProjectView/ProjectInfoBox.tsx","../../src/client/routes/_auth/projects/$projectId.tsx?tsr-split=component"],"sourcesContent":["export interface RouteInfo {\n section: string\n service: string | undefined\n isDetail?: boolean\n}\n\nexport function isRouteInfo(data: unknown): data is RouteInfo {\n return (\n typeof data === \"object\" &&\n data !== null &&\n typeof (data as RouteInfo).section === \"string\" &&\n (typeof (data as RouteInfo).service === \"string\" || (data as RouteInfo).service === undefined) &&\n (typeof (data as RouteInfo).isDetail === \"boolean\" || (data as RouteInfo).isDetail === undefined)\n )\n}\n","import { useNavigate, useMatches, useParams } from \"@tanstack/react-router\"\nimport { useState } 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\"\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\n const [openSections, setOpenSections] = useState({ compute: true, network: true, storage: true, services: true })\n\n const serviceIndex = getServiceIndex(availableServices)\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 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`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 const isOverviewActive = activeSection === null\n const isOpen = (section: string) => openSections[section as keyof typeof openSections] || activeSection === section\n\n return (\n <SideNavigation ariaLabel=\"Project Side Navigation\" onActiveItemChange={() => {}}>\n <SideNavigationList>\n <>\n <SideNavigationItem\n icon=\"home\"\n label={projectName}\n onClick={() => navigate({ to: \"/projects/$projectId\", params: { projectId } })}\n selected={isOverviewActive}\n />\n <SideNavigationItem\n label={t`Compute`}\n open={isOpen(\"compute\")}\n onClick={() => setOpenSections((prev) => ({ ...prev, compute: !prev.compute }))}\n >\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\n {networkServices.length > 0 && (\n <SideNavigationItem\n label={t`Network`}\n open={isOpen(\"network\")}\n onClick={() => setOpenSections((prev) => ({ ...prev, network: !prev.network }))}\n >\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 )}\n\n {storageServices.length > 0 && (\n <SideNavigationItem\n label={t`Storage`}\n open={isOpen(\"storage\")}\n onClick={() => setOpenSections((prev) => ({ ...prev, storage: !prev.storage }))}\n >\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 )}\n\n {clavisServices.length > 0 && (\n <SideNavigationItem\n label={t`Services`}\n open={isOpen(\"services\")}\n onClick={() => setOpenSections((prev) => ({ ...prev, services: !prev.services }))}\n >\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 )}\n </>\n </SideNavigationList>\n </SideNavigation>\n )\n}\n","import { Breadcrumb, BreadcrumbItem, KnownIcons } from \"@cloudoperators/juno-ui-components\"\nimport { useRouteContext, useMatches, useNavigate, useParams } from \"@tanstack/react-router\"\nimport { useState, useEffect } from \"react\"\nimport { isRouteInfo } 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\nconst SECTION_LABELS: Record<string, string> = {\n compute: \"Compute\",\n network: \"Network\",\n storage: \"Storage\",\n services: \"Services\",\n}\n\nconst SERVICE_LABELS: Record<string, string> = {\n images: \"Images\",\n flavors: \"Flavors\",\n securitygroups: \"Security Groups\",\n floatingips: \"Floating IPs\",\n containers: \"Object Storage (Swift)\",\n pca: \"PCA (Clavis)\",\n}\n\nexport function ProjectInfoBox({ projectInfo }: ProjectInfoBoxProps) {\n const { pageTitleRef } = useRouteContext({ from: \"__root__\" })\n const [pageTitle, setPageTitle] = useState(pageTitleRef.current)\n const navigate = useNavigate()\n const matches = useMatches()\n\n const { projectId, provider } = useParams({ strict: false }) as { projectId: string; provider: string }\n\n useEffect(() => {\n const handleTitleChange = (e: CustomEvent<{ title: string }>) => {\n setPageTitle(e.detail.title)\n }\n window.addEventListener(\"pageTitleChange\", handleTitleChange as EventListener)\n setPageTitle(pageTitleRef.current)\n return () => {\n window.removeEventListener(\"pageTitleChange\", handleTitleChange as EventListener)\n }\n }, [pageTitleRef])\n\n const buildBreadcrumbs = () => {\n const projectIdRouteId = \"/_auth/projects/$projectId\"\n\n const childMatches = matches.filter((m) => m.routeId !== projectIdRouteId && m.routeId.startsWith(projectIdRouteId))\n const deepestMatch = childMatches[childMatches.length - 1]\n\n const rawData = deepestMatch?.staticData\n const routeInfo = isRouteInfo(rawData) ? rawData : undefined\n const section = routeInfo?.section\n const service = routeInfo?.service\n const isDetail = routeInfo?.isDetail ?? false\n\n const sectionLabel = section ? SECTION_LABELS[section] : undefined\n // Dynamic service label for containers - depends on provider\n let serviceLabel = service ? SERVICE_LABELS[service] : undefined\n if (service === \"containers\" && provider) {\n serviceLabel = `Object Storage (${provider === \"swift\" ? \"Swift\" : \"Ceph\"})`\n }\n\n const isOverviewPage = !isDetail && service === \"overview\"\n const isServicePage = !isDetail && !!service && service !== \"overview\"\n const isSectionPage = !isDetail && !service && !!section\n\n const items: Array<{ label?: string; icon?: KnownIcons; onClick?: () => void; active?: boolean }> = []\n\n items.push({\n icon: \"home\",\n label: \"Home\",\n onClick: () => navigate({ to: \"/projects\" }),\n })\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 if (sectionLabel && section) {\n if (isSectionPage || isOverviewPage) {\n items.push({ label: sectionLabel, active: true })\n } else if (section === \"compute\") {\n items.push({\n label: sectionLabel,\n onClick: () =>\n navigate({\n to: \"/projects/$projectId/compute/overview\",\n params: { projectId },\n }),\n })\n } else if (section === \"network\") {\n items.push({\n label: sectionLabel,\n onClick: () =>\n navigate({\n to: \"/projects/$projectId/network/overview\",\n params: { projectId },\n }),\n })\n } else if (section === \"storage\") {\n items.push({\n label: sectionLabel,\n onClick: () =>\n navigate({\n to: \"/projects/$projectId/storage/$provider/containers\",\n params: { projectId, provider },\n }),\n })\n } else if (section === \"services\") {\n items.push({\n label: sectionLabel,\n onClick: () =>\n navigate({\n to: \"/projects/$projectId/services/overview\",\n params: { projectId },\n }),\n })\n }\n }\n\n if (serviceLabel && section && service) {\n if (isServicePage) {\n items.push({ label: serviceLabel, active: true })\n } else if (isDetail) {\n if (section === \"compute\" && service === \"images\") {\n items.push({\n label: serviceLabel,\n onClick: () =>\n navigate({\n to: \"/projects/$projectId/compute/images\",\n params: { projectId },\n }),\n })\n } else if (section === \"compute\" && service === \"flavors\") {\n items.push({\n label: serviceLabel,\n onClick: () =>\n navigate({\n to: \"/projects/$projectId/compute/flavors\",\n params: { projectId },\n }),\n })\n } else if (section === \"network\" && service === \"securitygroups\") {\n items.push({\n label: serviceLabel,\n onClick: () =>\n navigate({\n to: \"/projects/$projectId/network/securitygroups\",\n params: { projectId },\n }),\n })\n } else if (section === \"network\" && service === \"floatingips\") {\n items.push({\n label: serviceLabel,\n onClick: () =>\n navigate({\n to: \"/projects/$projectId/network/floatingips\",\n params: { projectId },\n }),\n })\n } else if (section === \"storage\" && service === \"containers\") {\n items.push({\n label: serviceLabel,\n onClick: () =>\n navigate({\n to: \"/projects/$projectId/storage/$provider/containers\",\n params: { projectId, provider },\n }),\n })\n } else if (section === \"services\" && service === \"pca\") {\n items.push({\n label: serviceLabel,\n onClick: () =>\n navigate({\n to: \"/projects/$projectId/services/pca\",\n params: { projectId },\n }),\n })\n } else {\n items.push({\n label: serviceLabel,\n onClick: () => {\n const path = `/projects/${projectId}/${section}/${service}` as Parameters<typeof navigate>[0][\"to\"]\n navigate({ to: path })\n },\n })\n }\n }\n }\n\n if (isDetail) {\n items.push({ label: pageTitle, active: true })\n }\n\n return items\n }\n\n const breadcrumbs = buildBreadcrumbs()\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":"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"}
@@ -0,0 +1,84 @@
1
+ import { E as e, q as t } from "./build-eu9eg0zF.mjs";
2
+ import { t as n } from "./helpers--JWXi40U.mjs";
3
+ import { t as r } from "./ContentHeader-H8KGY3Wd.mjs";
4
+ import { jsx as i, jsxs as a } from "react/jsx-runtime";
5
+ import { Link as o, useLoaderData as s } from "@tanstack/react-router";
6
+ import { Trans as c, useLingui as l } from "@lingui/react";
7
+ //#region src/client/routes/_auth/projects/$projectId/index.tsx?tsr-split=component
8
+ function u({ title: t, links: n }) {
9
+ return /* @__PURE__ */ a(e, {
10
+ className: "p-5",
11
+ children: [/* @__PURE__ */ i("h3", {
12
+ className: "text-theme-high mb-3 text-base font-semibold",
13
+ children: t
14
+ }), /* @__PURE__ */ i("ul", {
15
+ className: "space-y-1.5",
16
+ children: n.map(({ label: e, to: t }) => /* @__PURE__ */ i("li", { children: /* @__PURE__ */ i(o, {
17
+ to: t,
18
+ className: "text-theme-accent hover:text-theme-accent/80 text-sm",
19
+ children: e
20
+ }) }, e))
21
+ })]
22
+ });
23
+ }
24
+ function d() {
25
+ let { crumbProject: e, availableServices: o, projectId: d } = s({ from: "/_auth/projects/$projectId" }), { i18n: f, _: p } = l(), m = n(o ?? []), h = `/projects/${d}`, g = [];
26
+ if (m.image?.glance || m.compute?.nova) {
27
+ let e = [];
28
+ m.image?.glance && e.push({
29
+ label: "Images",
30
+ to: `${h}/compute/images`
31
+ }), m.compute?.nova && e.push({
32
+ label: "Flavors",
33
+ to: `${h}/compute/flavors`
34
+ }), g.push({
35
+ title: "Compute",
36
+ links: e
37
+ });
38
+ }
39
+ m.network && g.push({
40
+ title: "Network",
41
+ links: [{
42
+ label: "Security Groups",
43
+ to: `${h}/network/securitygroups`
44
+ }, {
45
+ label: "Floating IPs",
46
+ to: `${h}/network/floatingips`
47
+ }]
48
+ });
49
+ let _ = [];
50
+ return m["object-store"]?.swift && _.push({
51
+ label: "Swift",
52
+ to: `${h}/storage/swift/containers`
53
+ }), _.push({
54
+ label: "Ceph",
55
+ to: `${h}/storage/ceph/containers`
56
+ }), g.push({
57
+ title: "Storage",
58
+ links: _
59
+ }), (m.pca?.["clavis-dev"] || m.pca?.["clavis-beta"]) && g.push({
60
+ title: "Services",
61
+ links: [{
62
+ label: "PCA (Clavis)",
63
+ to: `${h}/services/pca`
64
+ }]
65
+ }), /* @__PURE__ */ a(t, {
66
+ direction: "vertical",
67
+ gap: "6",
68
+ className: "pb-4",
69
+ children: [/* @__PURE__ */ i(r, {
70
+ title: e?.name ?? f._({ id: "e0NrBM" }),
71
+ projectId: d
72
+ }), g.length > 0 ? /* @__PURE__ */ i("div", {
73
+ className: "grid grid-cols-1 gap-4 sm:grid-cols-2 lg:grid-cols-3",
74
+ children: g.map((e) => /* @__PURE__ */ i(u, { ...e }, e.title))
75
+ }) : /* @__PURE__ */ i("p", {
76
+ className: "text-theme-light text-sm",
77
+ children: /* @__PURE__ */ i(c, { id: "Q1W//7" })
78
+ })]
79
+ });
80
+ }
81
+ //#endregion
82
+ export { d as component };
83
+
84
+ //# sourceMappingURL=_projectId-C-E4NNgo.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"_projectId-C-E4NNgo.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"}
@@ -0,0 +1,26 @@
1
+ import { createFileRoute as e, lazyRouteComponent as t } from "@tanstack/react-router";
2
+ var n = e("/_auth/projects/$projectId")({
3
+ component: t(() => import("./_projectId-Be1Erj68.mjs"), "component"),
4
+ errorComponent: t(() => import("./_projectId-B9fln31N.mjs"), "errorComponent"),
5
+ loader: async (e) => {
6
+ let { context: t, params: n } = e, r = await t.trpcClient?.auth.setCurrentScope.mutate({
7
+ type: "project",
8
+ projectId: n.projectId || ""
9
+ }), i = await t.trpcClient?.auth.getAvailableServices.query(), a = r?.domain?.id || "";
10
+ return {
11
+ trpcClient: t.trpcClient,
12
+ crumbDomain: {
13
+ path: "/projects",
14
+ name: r?.domain?.name
15
+ },
16
+ crumbProject: r?.project,
17
+ availableServices: i,
18
+ accountId: a,
19
+ projectId: n.projectId
20
+ };
21
+ }
22
+ });
23
+ //#endregion
24
+ export { n as t };
25
+
26
+ //# sourceMappingURL=_projectId-INhedXor.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"_projectId-INhedXor.mjs","names":["createFileRoute","Route","component","lazyRouteComponent","$$splitComponentImporter","errorComponent","$$splitErrorComponentImporter","loader","options","context","params","data","trpcClient","auth","setCurrentScope","mutate","type","projectId","availableServices","getAvailableServices","query","accountId","domain","id","crumbDomain","path","name","crumbProject","project"],"sources":["../../src/client/routes/_auth/projects/$projectId.tsx"],"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":";AAMA,IAAaC,IAAQD,EAAgB,6BAAA,CAA8B;CACjEE,WAASC,6CAAA,YAAA;CACTE,gBAAcF,6CAAA,iBAAA;CAGdI,QAAQ,OAAOC,MAAAA;EACb,IAAM,EAAEC,YAASC,cAAWF,GACtBG,IAAO,MAAMF,EAAQG,YAAYC,KAAKC,gBAAgBC,OAAO;GACjEC,MAAM;GACNC,WAAWP,EAAOO,aAAa;GACjC,CAAA,EACMC,IAAoB,MAAMT,EAAQG,YAAYC,KAAKM,qBAAqBC,OAAAA,EAIxEC,IAAYV,GAAMW,QAAQC,MAAM;AAEtC,SAAO;GACLX,YAAYH,EAAQG;GACpBY,aAAa;IAAEC,MAAM;IAAaC,MAAMf,GAAMW,QAAQI;IAAK;GAC3DC,cAAchB,GAAMiB;GACpBV;GACAG;GACAJ,WAAWP,EAAOO;GACpB;;CAEJ,CAAA"}