@cobaltcore-dev/aurora 0.3.1 → 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 (108) hide show
  1. package/dist/client/{EditSecurityGroupModal-Dl6m7wUe.mjs → EditSecurityGroupModal-DK3WYikA.mjs} +1 -1
  2. package/dist/client/{EditSecurityGroupModal-Dl6m7wUe.mjs.map → EditSecurityGroupModal-DK3WYikA.mjs.map} +1 -1
  3. package/dist/client/{FiltersInput-BgNaHFBt.mjs → FiltersInput-OVeIJzIo.mjs} +2 -2
  4. package/dist/client/FiltersInput-OVeIJzIo.mjs.map +1 -0
  5. package/dist/client/{FloatingIpActionModals-By2hXR9m.mjs → FloatingIpActionModals-Dok7fJss.mjs} +3 -3
  6. package/dist/client/{FloatingIpActionModals-By2hXR9m.mjs.map → FloatingIpActionModals-Dok7fJss.mjs.map} +1 -1
  7. package/dist/client/{ImageToastNotifications-fHI8jB2j.mjs → ImageToastNotifications-BG9LPnXf.mjs} +4 -4
  8. package/dist/client/{ImageToastNotifications-fHI8jB2j.mjs.map → ImageToastNotifications-BG9LPnXf.mjs.map} +1 -1
  9. package/dist/client/{ListToolbar-BvtCo8dk.mjs → ListToolbar-BojRTNbo.mjs} +2 -2
  10. package/dist/client/{ListToolbar-BvtCo8dk.mjs.map → ListToolbar-BojRTNbo.mjs.map} +1 -1
  11. package/dist/client/{_floatingIpId-CwHiXazi.mjs → _floatingIpId-D33bOEmH.mjs} +2 -2
  12. package/dist/client/{_floatingIpId-CwHiXazi.mjs.map → _floatingIpId-D33bOEmH.mjs.map} +1 -1
  13. package/dist/client/{_floatingIpId-BpUfL8Im2.mjs → _floatingIpId-DF_BSJN6.mjs} +5 -5
  14. package/dist/client/_floatingIpId-DF_BSJN6.mjs.map +1 -0
  15. package/dist/client/{_imageId-CdOOJjw0.mjs → _imageId-BL0I5_pv.mjs} +2 -2
  16. package/dist/client/{_imageId-CdOOJjw0.mjs.map → _imageId-BL0I5_pv.mjs.map} +1 -1
  17. package/dist/client/{_pcaId-D1ZEaCdp.mjs → _pcaId-BYCoeK6_.mjs} +2 -2
  18. package/dist/client/{_pcaId-D1ZEaCdp.mjs.map → _pcaId-BYCoeK6_.mjs.map} +1 -1
  19. package/dist/client/{_pcaId-CwlH1Kvl.mjs → _pcaId-CbBhBrX1.mjs} +153 -56
  20. package/dist/client/_pcaId-CbBhBrX1.mjs.map +1 -0
  21. package/dist/client/{_projectId-Dj_InfSc.mjs → _projectId-5NiasyXm.mjs} +2 -2
  22. package/dist/client/{_projectId-Dj_InfSc.mjs.map → _projectId-5NiasyXm.mjs.map} +1 -1
  23. package/dist/client/{_projectId-DhLpIalx.mjs → _projectId-BwLMEMGC.mjs} +1 -1
  24. package/dist/client/{_projectId-DhLpIalx.mjs.map → _projectId-BwLMEMGC.mjs.map} +1 -1
  25. package/dist/client/{_projectId-D1gGribM.mjs → _projectId-D35MN1kY.mjs} +3 -3
  26. package/dist/client/{_projectId-D1gGribM.mjs.map → _projectId-D35MN1kY.mjs.map} +1 -1
  27. package/dist/client/{_securityGroupId-B0llWH9A.mjs → _securityGroupId-B-Z-CzLp.mjs} +2 -2
  28. package/dist/client/{_securityGroupId-B0llWH9A.mjs.map → _securityGroupId-B-Z-CzLp.mjs.map} +1 -1
  29. package/dist/client/{_securityGroupId-gbUnd5Wv.mjs → _securityGroupId-B1bOYRbX.mjs} +7 -7
  30. package/dist/client/{_securityGroupId-gbUnd5Wv.mjs.map → _securityGroupId-B1bOYRbX.mjs.map} +1 -1
  31. package/dist/client/{about-DCe6LsKz.mjs → about-DLn1ShhF.mjs} +1 -1
  32. package/dist/client/{about-DCe6LsKz.mjs.map → about-DLn1ShhF.mjs.map} +1 -1
  33. package/dist/client/{buildFilterParams-By33pG59.mjs → buildFilterParams-TeyosGyK.mjs} +1 -1
  34. package/dist/client/{buildFilterParams-By33pG59.mjs.map → buildFilterParams-TeyosGyK.mjs.map} +1 -1
  35. package/dist/client/{constants-CAjjRTo_.mjs → constants-B-P2r5F1.mjs} +3 -3
  36. package/dist/client/{constants-CAjjRTo_.mjs.map → constants-B-P2r5F1.mjs.map} +1 -1
  37. package/dist/client/{containers-NW7RnHTI.mjs → containers-BjWqjNOx.mjs} +2 -2
  38. package/dist/client/containers-BjWqjNOx.mjs.map +1 -0
  39. package/dist/client/{containers-Ca5V1EBS.mjs → containers-DsRWc1L5.mjs} +1 -1
  40. package/dist/client/containers-DsRWc1L5.mjs.map +1 -0
  41. package/dist/client/{containers-BuXUVb1N.mjs → containers-J7WFA18U.mjs} +1140 -1051
  42. package/dist/client/containers-J7WFA18U.mjs.map +1 -0
  43. package/dist/client/{floatingips-Be3zLoaD.mjs → floatingips-Fa6ocNUu.mjs} +7 -7
  44. package/dist/client/{floatingips-Be3zLoaD.mjs.map → floatingips-Fa6ocNUu.mjs.map} +1 -1
  45. package/dist/client/{formatBytes-D6oa0wU9.mjs → formatBytes-tQBEnPoL.mjs} +1 -1
  46. package/dist/client/{formatBytes-D6oa0wU9.mjs.map → formatBytes-tQBEnPoL.mjs.map} +1 -1
  47. package/dist/client/{hooks-D0krAKvo.mjs → hooks-DEjb9d1F.mjs} +1 -1
  48. package/dist/client/{images-CCYBAphP2.mjs → images-CSFfefAu.mjs} +8 -7
  49. package/dist/client/images-CSFfefAu.mjs.map +1 -0
  50. package/dist/client/{images-BiEBENaj.mjs → images-CTLCY-yY.mjs} +2 -2
  51. package/dist/client/{images-BiEBENaj.mjs.map → images-CTLCY-yY.mjs.map} +1 -1
  52. package/dist/client/images-DM9I8G0p.mjs.map +1 -1
  53. package/dist/client/images-tYfyOkX8.mjs +8 -0
  54. package/dist/client/images-tYfyOkX8.mjs.map +1 -0
  55. package/dist/client/index.js +111 -109
  56. package/dist/client/index.js.map +1 -1
  57. package/dist/client/{network-nbSbl0X0.mjs → network-rYLHyf15.mjs} +1 -1
  58. package/dist/client/{network-nbSbl0X0.mjs.map → network-rYLHyf15.mjs.map} +1 -1
  59. package/dist/client/{objects-CU5ws07o.mjs → objects-BciXwZ00.mjs} +2 -2
  60. package/dist/client/objects-BciXwZ00.mjs.map +1 -0
  61. package/dist/client/{objects-GmuIOaHd.mjs → objects-Cdew99tK.mjs} +1 -1
  62. package/dist/client/objects-Cdew99tK.mjs.map +1 -0
  63. package/dist/client/objects-DaElrban.mjs +5340 -0
  64. package/dist/client/objects-DaElrban.mjs.map +1 -0
  65. package/dist/client/{overview-EhfPY8Je.mjs → overview-BMhjFMIV.mjs} +2 -2
  66. package/dist/client/{overview-EhfPY8Je.mjs.map → overview-BMhjFMIV.mjs.map} +1 -1
  67. package/dist/client/{overview-DzYBiNfD.mjs → overview-BYIRj7_X.mjs} +1 -1
  68. package/dist/client/{overview-DzYBiNfD.mjs.map → overview-BYIRj7_X.mjs.map} +1 -1
  69. package/dist/client/{overview-B3gdnWTG.mjs → overview-DRCKNBH2.mjs} +1 -1
  70. package/dist/client/{overview-B3gdnWTG.mjs.map → overview-DRCKNBH2.mjs.map} +1 -1
  71. package/dist/client/{overview-XueZI4LQ.mjs → overview-urYLOVQE.mjs} +2 -2
  72. package/dist/client/{overview-XueZI4LQ.mjs.map → overview-urYLOVQE.mjs.map} +1 -1
  73. package/dist/client/{pca-DSM71LhW.mjs → pca-COmKvp3J.mjs} +2 -2
  74. package/dist/client/{pca-DSM71LhW.mjs.map → pca-COmKvp3J.mjs.map} +1 -1
  75. package/dist/client/{pca-x9if8xU-.mjs → pca-oc7J0_Xd.mjs} +17 -17
  76. package/dist/client/pca-oc7J0_Xd.mjs.map +1 -0
  77. package/dist/client/{projects-CnmZIB2Q.mjs → projects-BUabCzvw.mjs} +16 -16
  78. package/dist/client/projects-BUabCzvw.mjs.map +1 -0
  79. package/dist/client/{projects-Bt0XptpG.mjs → projects-DI_L4oDw.mjs} +2 -2
  80. package/dist/client/{projects-Bt0XptpG.mjs.map → projects-DI_L4oDw.mjs.map} +1 -1
  81. package/dist/client/{projects-B6BPo2Ar.mjs → projects-Dl5XkXUP.mjs} +1 -1
  82. package/dist/client/{projects-B6BPo2Ar.mjs.map → projects-Dl5XkXUP.mjs.map} +1 -1
  83. package/dist/client/{projects-BilrmHLu.mjs → projects-HoQ0gE5Y.mjs} +1 -1
  84. package/dist/client/{projects-BilrmHLu.mjs.map → projects-HoQ0gE5Y.mjs.map} +1 -1
  85. package/dist/client/{securitygroups-BdzieS7Z.mjs → securitygroups-BjkmHk2J.mjs} +7 -7
  86. package/dist/client/{securitygroups-BdzieS7Z.mjs.map → securitygroups-BjkmHk2J.mjs.map} +1 -1
  87. package/dist/client/{useListWithFiltering-CqQbAjEe.mjs → useListWithFiltering-CbhHJO4V.mjs} +1 -1
  88. package/dist/client/{useListWithFiltering-CqQbAjEe.mjs.map → useListWithFiltering-CbhHJO4V.mjs.map} +1 -1
  89. package/dist/client/{useProjectId-CgOTejka.mjs → useProjectId-OQv2KBbG.mjs} +1 -1
  90. package/dist/client/{useProjectId-CgOTejka.mjs.map → useProjectId-OQv2KBbG.mjs.map} +1 -1
  91. package/dist/server/index.d.ts +7 -1
  92. package/dist/server/index.js +116 -55
  93. package/package.json +2 -3
  94. package/dist/client/FiltersInput-BgNaHFBt.mjs.map +0 -1
  95. package/dist/client/_floatingIpId-BpUfL8Im2.mjs.map +0 -1
  96. package/dist/client/_pcaId-CwlH1Kvl.mjs.map +0 -1
  97. package/dist/client/containers-BuXUVb1N.mjs.map +0 -1
  98. package/dist/client/containers-Ca5V1EBS.mjs.map +0 -1
  99. package/dist/client/containers-NW7RnHTI.mjs.map +0 -1
  100. package/dist/client/images-CCYBAphP2.mjs.map +0 -1
  101. package/dist/client/objects-CU5ws07o.mjs.map +0 -1
  102. package/dist/client/objects-FXN0VWLI.mjs +0 -4760
  103. package/dist/client/objects-FXN0VWLI.mjs.map +0 -1
  104. package/dist/client/objects-GmuIOaHd.mjs.map +0 -1
  105. package/dist/client/pca-x9if8xU-.mjs.map +0 -1
  106. package/dist/client/projects-CnmZIB2Q.mjs.map +0 -1
  107. package/permission_policies/compute.yaml +0 -975
  108. package/permission_policies/image.yaml +0 -71
@@ -1 +1 @@
1
- {"version":3,"file":"ListToolbar-BvtCo8dk.mjs","names":["useCallback","useRef","useEffect","SearchInput","Stack","TabNavigation","TabNavigationItem","SelectedFilters","FiltersInput","SortInput","ListToolbar","filterSettings","onFilter","sortSettings","onSort","searchTerm","onSearch","searchInputProps","actions","tabs","totalCount","filteredCount","itemName","lastUpdated","useLingui","debounceTimerRef","undefined","current","clearTimeout","formatLastUpdated","date","dateObj","Date","toLocaleString","showCountInfo","handleFilterDelete","filterToRemove","selectedFilters","filter","name","value","handleSelect","selectedFilter","filterExists","some","supportsMultiValue","filters","find","filterName","newSelected","handleSearch","searchValue","handleSearchInput","event","currentTarget","window","setTimeout","handleSearchClear","filtersProps","onChange","sortProps","options","sortBy","sortDirection","onSortByChange","param","onSortDirectionChange","direction","searchProps","placeholder","t","onInput","onClear","div","className","activeItem","onActiveItemChange","items","map","item","label","alignment","gap","length","onDelete","span","formattedDate"],"sources":["../../src/client/components/ListToolbar/index.tsx"],"sourcesContent":["import { ReactNode, useCallback, useRef, useEffect } from \"react\"\nimport { Trans, useLingui } from \"@lingui/react/macro\"\nimport {\n SearchInput,\n SearchInputProps,\n Stack,\n TabNavigation,\n TabNavigationItem,\n} from \"@cloudoperators/juno-ui-components\"\nimport { SelectedFilters } from \"./SelectedFilters\"\nimport { FiltersInput } from \"./FiltersInput\"\nimport { SortInput } from \"./SortInput\"\nimport { FilterSettings, SelectedFilter, SortSettings } from \"./types\"\n\nexport type ListToolbarProps = {\n filterSettings?: FilterSettings\n onFilter?: (filterSettings: FilterSettings) => void\n sortSettings?: SortSettings\n onSort?: (sortSettings: SortSettings) => void\n searchTerm?: string\n onSearch?: (searchTerm: string) => void\n searchInputProps?: Omit<SearchInputProps, \"value\" | \"onSearch\" | \"onClear\" | \"onInput\">\n actions?: ReactNode\n tabs?: {\n items: Array<{ label: string; value: string }>\n activeItem: string\n onActiveItemChange: (value: ReactNode) => void\n }\n // Count information\n totalCount?: number\n itemName?: string // e.g. \"items\", \"users\", \"projects\" - used in count display\n filteredCount?: number\n // Last updated timestamp\n lastUpdated?: Date | string\n}\n\nexport const ListToolbar = ({\n filterSettings,\n onFilter,\n sortSettings,\n onSort,\n searchTerm,\n onSearch,\n searchInputProps = {},\n actions,\n tabs,\n totalCount,\n filteredCount,\n itemName = \"items\",\n lastUpdated,\n}: ListToolbarProps) => {\n const { t } = useLingui()\n\n const debounceTimerRef = useRef<number | undefined>(undefined)\n\n useEffect(() => {\n return () => {\n if (debounceTimerRef.current) {\n clearTimeout(debounceTimerRef.current)\n }\n }\n }, [])\n\n // Format last updated time\n const formatLastUpdated = (date: Date | string | undefined): string => {\n if (!date) return \"\"\n const dateObj = typeof date === \"string\" ? new Date(date) : date\n return dateObj.toLocaleString()\n }\n\n const showCountInfo = totalCount !== undefined && filteredCount !== undefined\n\n const handleFilterDelete = useCallback(\n (filterToRemove: SelectedFilter) => {\n if (!onFilter || !filterSettings) return\n onFilter({\n ...filterSettings,\n selectedFilters: filterSettings.selectedFilters?.filter(\n (filter) => !(filter.name === filterToRemove.name && filter.value === filterToRemove.value)\n ),\n })\n },\n [filterSettings, onFilter]\n )\n\n const handleSelect = (selectedFilter: SelectedFilter) => {\n if (!onFilter || !filterSettings) return\n const filterExists = filterSettings.selectedFilters?.some(\n (filter) => filter.name === selectedFilter.name && filter.value === selectedFilter.value\n )\n if (filterExists) return\n\n const supportsMultiValue = filterSettings.filters.find(\n (filter) => selectedFilter.name === filter.filterName\n )?.supportsMultiValue\n\n const newSelected = supportsMultiValue\n ? [...(filterSettings.selectedFilters || []), selectedFilter]\n : [\n ...(filterSettings.selectedFilters || []).filter((filter) => filter.name !== selectedFilter.name),\n selectedFilter,\n ]\n\n onFilter({ ...filterSettings, selectedFilters: newSelected })\n }\n\n const handleSearch = useCallback(\n (value: string | number | string[] | undefined) => {\n const searchValue = typeof value === \"string\" ? value : \"\"\n\n if (debounceTimerRef.current) {\n clearTimeout(debounceTimerRef.current)\n }\n\n onSearch?.(searchValue)\n },\n [onSearch]\n )\n\n const handleSearchInput = useCallback(\n (event: React.FormEvent<HTMLInputElement>) => {\n const searchValue = event.currentTarget.value\n if (debounceTimerRef.current) clearTimeout(debounceTimerRef.current)\n debounceTimerRef.current = window.setTimeout(() => onSearch?.(searchValue), 500)\n },\n [onSearch]\n )\n\n const handleSearchClear = useCallback(() => {\n if (debounceTimerRef.current) clearTimeout(debounceTimerRef.current)\n onSearch?.(\"\")\n }, [onSearch])\n\n const filtersProps = filterSettings && onFilter ? { filters: filterSettings.filters, onChange: handleSelect } : null\n const sortProps =\n sortSettings && onSort\n ? {\n options: sortSettings.options,\n sortBy: sortSettings.sortBy,\n sortDirection: sortSettings.sortDirection || \"asc\",\n onSortByChange: (param: string | number | string[] | undefined) =>\n onSort({ ...sortSettings, sortBy: param, sortDirection: sortSettings.sortDirection || \"asc\" }),\n onSortDirectionChange: (direction: \"asc\" | \"desc\") => onSort({ ...sortSettings, sortDirection: direction }),\n }\n : null\n\n const searchProps: (SearchInputProps & { \"data-testid\"?: string }) | null = onSearch\n ? {\n placeholder: t`Search...`,\n \"data-testid\": \"searchbar\",\n value: searchTerm,\n onInput: handleSearchInput,\n onClear: handleSearchClear,\n onSearch: handleSearch,\n ...searchInputProps,\n }\n : null\n\n return (\n <>\n {tabs && (\n <div className=\"w-full\">\n <TabNavigation activeItem={tabs.activeItem} onActiveItemChange={tabs.onActiveItemChange}>\n {tabs.items.map((item) => (\n <TabNavigationItem key={item.value} label={item.label} value={item.value} />\n ))}\n </TabNavigation>\n </div>\n )}\n <Stack alignment=\"center\" gap=\"6\" className=\"bg-theme-background-lvl-1 flex w-full flex-col p-4\">\n {actions && (\n <Stack direction=\"horizontal\" className=\"w-full justify-end\">\n {actions}\n </Stack>\n )}\n\n <div className=\"flex w-full flex-col items-stretch gap-4 md:flex-row md:items-center\">\n {filtersProps && (\n <div className=\"w-full md:w-auto md:min-w-37.5\">\n <FiltersInput {...filtersProps} />\n </div>\n )}\n {sortProps && (\n <div className=\"w-full md:w-auto md:min-w-45\">\n <SortInput {...sortProps} />\n </div>\n )}\n {searchProps && (\n <div className=\"w-full md:ml-auto md:w-auto md:min-w-25\">\n <SearchInput {...searchProps} />\n </div>\n )}\n </div>\n\n {filterSettings?.selectedFilters && filterSettings.selectedFilters.length > 0 && onFilter && (\n <div className=\"w-full\">\n <SelectedFilters\n selectedFilters={filterSettings.selectedFilters}\n onDelete={handleFilterDelete}\n onClear={() => onFilter({ ...filterSettings, selectedFilters: [] })}\n />\n </div>\n )}\n {/* Count and Last Updated Info */}\n {(showCountInfo || lastUpdated) && (\n <div className=\"text-theme-secondary flex w-full items-center justify-between text-sm\">\n <div className=\"flex items-center gap-2\">\n {showCountInfo && (\n <span>\n <Trans>\n Showing {filteredCount} of {totalCount} {itemName}\n </Trans>\n </span>\n )}\n {lastUpdated &&\n (() => {\n const formattedDate = formatLastUpdated(lastUpdated)\n return (\n <span>\n <Trans>Last updated: {formattedDate}</Trans>\n </span>\n )\n })()}\n </div>\n </div>\n )}\n </Stack>\n </>\n )\n}\n"],"mappings":"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"}
1
+ {"version":3,"file":"ListToolbar-BojRTNbo.mjs","names":["useCallback","useRef","useEffect","SearchInput","Stack","TabNavigation","TabNavigationItem","SelectedFilters","FiltersInput","SortInput","ListToolbar","filterSettings","onFilter","sortSettings","onSort","searchTerm","onSearch","searchInputProps","actions","tabs","totalCount","filteredCount","itemName","lastUpdated","useLingui","debounceTimerRef","undefined","current","clearTimeout","formatLastUpdated","date","dateObj","Date","toLocaleString","showCountInfo","handleFilterDelete","filterToRemove","selectedFilters","filter","name","value","handleSelect","selectedFilter","filterExists","some","supportsMultiValue","filters","find","filterName","newSelected","handleSearch","searchValue","handleSearchInput","event","currentTarget","window","setTimeout","handleSearchClear","filtersProps","onChange","sortProps","options","sortBy","sortDirection","onSortByChange","param","onSortDirectionChange","direction","searchProps","placeholder","t","onInput","onClear","div","className","activeItem","onActiveItemChange","items","map","item","label","alignment","gap","length","onDelete","span","formattedDate"],"sources":["../../src/client/components/ListToolbar/index.tsx"],"sourcesContent":["import { ReactNode, useCallback, useRef, useEffect } from \"react\"\nimport { Trans, useLingui } from \"@lingui/react/macro\"\nimport {\n SearchInput,\n SearchInputProps,\n Stack,\n TabNavigation,\n TabNavigationItem,\n} from \"@cloudoperators/juno-ui-components\"\nimport { SelectedFilters } from \"./SelectedFilters\"\nimport { FiltersInput } from \"./FiltersInput\"\nimport { SortInput } from \"./SortInput\"\nimport { FilterSettings, SelectedFilter, SortSettings } from \"./types\"\n\nexport type ListToolbarProps = {\n filterSettings?: FilterSettings\n onFilter?: (filterSettings: FilterSettings) => void\n sortSettings?: SortSettings\n onSort?: (sortSettings: SortSettings) => void\n searchTerm?: string\n onSearch?: (searchTerm: string) => void\n searchInputProps?: Omit<SearchInputProps, \"value\" | \"onSearch\" | \"onClear\" | \"onInput\">\n actions?: ReactNode\n tabs?: {\n items: Array<{ label: string; value: string }>\n activeItem: string\n onActiveItemChange: (value: ReactNode) => void\n }\n // Count information\n totalCount?: number\n itemName?: string // e.g. \"items\", \"users\", \"projects\" - used in count display\n filteredCount?: number\n // Last updated timestamp\n lastUpdated?: Date | string\n}\n\nexport const ListToolbar = ({\n filterSettings,\n onFilter,\n sortSettings,\n onSort,\n searchTerm,\n onSearch,\n searchInputProps = {},\n actions,\n tabs,\n totalCount,\n filteredCount,\n itemName = \"items\",\n lastUpdated,\n}: ListToolbarProps) => {\n const { t } = useLingui()\n\n const debounceTimerRef = useRef<number | undefined>(undefined)\n\n useEffect(() => {\n return () => {\n if (debounceTimerRef.current) {\n clearTimeout(debounceTimerRef.current)\n }\n }\n }, [])\n\n // Format last updated time\n const formatLastUpdated = (date: Date | string | undefined): string => {\n if (!date) return \"\"\n const dateObj = typeof date === \"string\" ? new Date(date) : date\n return dateObj.toLocaleString()\n }\n\n const showCountInfo = totalCount !== undefined && filteredCount !== undefined\n\n const handleFilterDelete = useCallback(\n (filterToRemove: SelectedFilter) => {\n if (!onFilter || !filterSettings) return\n onFilter({\n ...filterSettings,\n selectedFilters: filterSettings.selectedFilters?.filter(\n (filter) => !(filter.name === filterToRemove.name && filter.value === filterToRemove.value)\n ),\n })\n },\n [filterSettings, onFilter]\n )\n\n const handleSelect = (selectedFilter: SelectedFilter) => {\n if (!onFilter || !filterSettings) return\n const filterExists = filterSettings.selectedFilters?.some(\n (filter) => filter.name === selectedFilter.name && filter.value === selectedFilter.value\n )\n if (filterExists) return\n\n const supportsMultiValue = filterSettings.filters.find(\n (filter) => selectedFilter.name === filter.filterName\n )?.supportsMultiValue\n\n const newSelected = supportsMultiValue\n ? [...(filterSettings.selectedFilters || []), selectedFilter]\n : [\n ...(filterSettings.selectedFilters || []).filter((filter) => filter.name !== selectedFilter.name),\n selectedFilter,\n ]\n\n onFilter({ ...filterSettings, selectedFilters: newSelected })\n }\n\n const handleSearch = useCallback(\n (value: string | number | string[] | undefined) => {\n const searchValue = typeof value === \"string\" ? value : \"\"\n\n if (debounceTimerRef.current) {\n clearTimeout(debounceTimerRef.current)\n }\n\n onSearch?.(searchValue)\n },\n [onSearch]\n )\n\n const handleSearchInput = useCallback(\n (event: React.FormEvent<HTMLInputElement>) => {\n const searchValue = event.currentTarget.value\n if (debounceTimerRef.current) clearTimeout(debounceTimerRef.current)\n debounceTimerRef.current = window.setTimeout(() => onSearch?.(searchValue), 500)\n },\n [onSearch]\n )\n\n const handleSearchClear = useCallback(() => {\n if (debounceTimerRef.current) clearTimeout(debounceTimerRef.current)\n onSearch?.(\"\")\n }, [onSearch])\n\n const filtersProps = filterSettings && onFilter ? { filters: filterSettings.filters, onChange: handleSelect } : null\n const sortProps =\n sortSettings && onSort\n ? {\n options: sortSettings.options,\n sortBy: sortSettings.sortBy,\n sortDirection: sortSettings.sortDirection || \"asc\",\n onSortByChange: (param: string | number | string[] | undefined) =>\n onSort({ ...sortSettings, sortBy: param, sortDirection: sortSettings.sortDirection || \"asc\" }),\n onSortDirectionChange: (direction: \"asc\" | \"desc\") => onSort({ ...sortSettings, sortDirection: direction }),\n }\n : null\n\n const searchProps: (SearchInputProps & { \"data-testid\"?: string }) | null = onSearch\n ? {\n placeholder: t`Search...`,\n \"data-testid\": \"searchbar\",\n value: searchTerm,\n onInput: handleSearchInput,\n onClear: handleSearchClear,\n onSearch: handleSearch,\n ...searchInputProps,\n }\n : null\n\n return (\n <>\n {tabs && (\n <div className=\"w-full\">\n <TabNavigation activeItem={tabs.activeItem} onActiveItemChange={tabs.onActiveItemChange}>\n {tabs.items.map((item) => (\n <TabNavigationItem key={item.value} label={item.label} value={item.value} />\n ))}\n </TabNavigation>\n </div>\n )}\n <Stack alignment=\"center\" gap=\"6\" className=\"bg-theme-background-lvl-1 flex w-full flex-col p-4\">\n {actions && (\n <Stack direction=\"horizontal\" className=\"w-full justify-end\">\n {actions}\n </Stack>\n )}\n\n <div className=\"flex w-full flex-col items-stretch gap-4 md:flex-row md:items-center\">\n {filtersProps && (\n <div className=\"w-full md:w-auto md:min-w-37.5\">\n <FiltersInput {...filtersProps} />\n </div>\n )}\n {sortProps && (\n <div className=\"w-full md:w-auto md:min-w-45\">\n <SortInput {...sortProps} />\n </div>\n )}\n {searchProps && (\n <div className=\"w-full md:ml-auto md:w-auto md:min-w-25\">\n <SearchInput {...searchProps} />\n </div>\n )}\n </div>\n\n {filterSettings?.selectedFilters && filterSettings.selectedFilters.length > 0 && onFilter && (\n <div className=\"w-full\">\n <SelectedFilters\n selectedFilters={filterSettings.selectedFilters}\n onDelete={handleFilterDelete}\n onClear={() => onFilter({ ...filterSettings, selectedFilters: [] })}\n />\n </div>\n )}\n {/* Count and Last Updated Info */}\n {(showCountInfo || lastUpdated) && (\n <div className=\"text-theme-secondary flex w-full items-center justify-between text-sm\">\n <div className=\"flex items-center gap-2\">\n {showCountInfo && (\n <span>\n <Trans>\n Showing {filteredCount} of {totalCount} {itemName}\n </Trans>\n </span>\n )}\n {lastUpdated &&\n (() => {\n const formattedDate = formatLastUpdated(lastUpdated)\n return (\n <span>\n <Trans>Last updated: {formattedDate}</Trans>\n </span>\n )\n })()}\n </div>\n </div>\n )}\n </Stack>\n </>\n )\n}\n"],"mappings":"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"}
@@ -16,7 +16,7 @@ var i = t("/_auth/projects/$projectId/network/floatingips/$floatingIpId/")({
16
16
  floatingip_id: t.floatingIpId
17
17
  }))?.floating_ip_address ?? null }),
18
18
  head: ({ loaderData: e }) => ({ meta: [{ title: e?.floatingIpAddress ?? "Floating IP" }] }),
19
- component: n(() => import("./_floatingIpId-BpUfL8Im2.mjs"), "component"),
19
+ component: n(() => import("./_floatingIpId-DF_BSJN6.mjs"), "component"),
20
20
  beforeLoad: async ({ context: t, params: n }) => {
21
21
  let { trpcClient: i } = t, a = e(await i?.auth.getAvailableServices.query() || []);
22
22
  if (!a.network || !a.network.neutron) throw r({
@@ -28,4 +28,4 @@ var i = t("/_auth/projects/$projectId/network/floatingips/$floatingIpId/")({
28
28
  //#endregion
29
29
  export { i as t };
30
30
 
31
- //# sourceMappingURL=_floatingIpId-CwHiXazi.mjs.map
31
+ //# sourceMappingURL=_floatingIpId-D33bOEmH.mjs.map
@@ -1 +1 @@
1
- {"version":3,"file":"_floatingIpId-CwHiXazi.mjs","names":["createFileRoute","redirect","getServiceIndex","Route","staticData","section","service","isDetail","sectionCrumb","labelKey","crumb","to","RouteInfo","loader","context","params","floatingIp","trpcClient","network","getById","query","project_id","projectId","floatingip_id","floatingIpId","floatingIpAddress","floating_ip_address","head","loaderData","meta","title","component","lazyRouteComponent","$$splitComponentImporter","beforeLoad","availableServices","auth","getAvailableServices","serviceIndex"],"sources":["../../src/client/routes/_auth/projects/$projectId/network/floatingips/$floatingIpId/index.tsx"],"sourcesContent":["import { createFileRoute, redirect, useNavigate } from \"@tanstack/react-router\"\nimport { Button, ContentHeading, Stack, Spinner } from \"@cloudoperators/juno-ui-components\"\nimport { Trans } from \"@lingui/react/macro\"\nimport type { RouteInfo } from \"@/client/routes/routeInfo\"\nimport { getServiceIndex } from \"@/server/Authentication/helpers\"\nimport { useProjectId } from \"@/client/hooks\"\nimport { trpcReact } from \"@/client/trpcClient\"\nimport { FloatingIpDetailsView } from \"./-components/-details/FloatingIpDetailsView\"\n\nexport const Route = createFileRoute(\"/_auth/projects/$projectId/network/floatingips/$floatingIpId/\")({\n staticData: {\n section: \"network\",\n service: \"floatingips\",\n isDetail: true,\n sectionCrumb: { labelKey: \"Network\" },\n crumb: { labelKey: \"Floating IPs\", to: \"/projects/$projectId/network/floatingips\" },\n } satisfies RouteInfo,\n loader: async ({ context, params }) => {\n const floatingIp = await context.trpcClient?.network.floatingIp.getById.query({\n project_id: params.projectId,\n floatingip_id: params.floatingIpId,\n })\n return { floatingIpAddress: floatingIp?.floating_ip_address ?? null }\n },\n head: ({ loaderData }) => ({\n meta: [{ title: loaderData?.floatingIpAddress ?? \"Floating IP\" }],\n }),\n component: RouteComponent,\n beforeLoad: async ({ context, params }) => {\n const { trpcClient } = context\n\n const availableServices = (await trpcClient?.auth.getAvailableServices.query()) || []\n const serviceIndex = getServiceIndex(availableServices)\n\n // Redirect if network service not available\n if (!serviceIndex[\"network\"]) {\n throw redirect({\n to: \"/projects/$projectId/network/floatingips\",\n params: { projectId: params.projectId },\n })\n }\n\n if (!serviceIndex[\"network\"][\"neutron\"]) {\n throw redirect({\n to: \"/projects/$projectId/network/floatingips\",\n params: { projectId: params.projectId },\n })\n }\n },\n})\n\nfunction RouteComponent() {\n const { floatingIpId } = Route.useParams()\n const projectId = useProjectId()\n const navigate = useNavigate()\n\n // Fetch floating IP details\n const {\n data: floatingIp,\n isLoading,\n isError,\n error,\n } = trpcReact.network.floatingIp.getById.useQuery({\n project_id: projectId,\n floatingip_id: floatingIpId,\n })\n\n const handleBack = () => {\n navigate({\n to: \"/projects/$projectId/network/floatingips\",\n params: { projectId },\n })\n }\n\n // 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 Floating IP Details...</Trans>\n </Stack>\n )\n }\n\n // Error state\n if (isError) {\n const errorMessage = error?.message || \"Unknown error\"\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 floating IP</Trans>\n </p>\n <p className=\"text-theme-highest\">{errorMessage}</p>\n <Button onClick={handleBack} variant=\"primary\">\n <Trans>Back to Floating IPs</Trans>\n </Button>\n </Stack>\n )\n }\n\n // No data state\n if (!floatingIp) {\n return (\n <Stack className=\"fixed inset-0\" distribution=\"center\" alignment=\"center\" direction=\"vertical\" gap=\"5\">\n <p className=\"text-theme-secondary\">\n <Trans>Floating IP not found</Trans>\n </p>\n <Button onClick={handleBack} variant=\"primary\">\n <Trans>Back to Floating IPs</Trans>\n </Button>\n </Stack>\n )\n }\n\n // Success state\n return (\n <>\n <ContentHeading>{floatingIp.floating_ip_address}</ContentHeading>\n <FloatingIpDetailsView floatingIp={floatingIp} />\n </>\n )\n}\n"],"mappings":";;AASA,IAAaG,IAAQH,EAAgB,gEAAA,CAAiE;CACpGI,YAAY;EACVC,SAAS;EACTC,SAAS;EACTC,UAAU;EACVC,cAAc,EAAEC,UAAU,WAAU;EACpCC,OAAO;GAAED,UAAU;GAAgBE,IAAI;GAA2C;EACpF;CACAE,QAAQ,OAAO,EAAEC,YAASC,iBAKjB,EAAEU,oBAJU,MAAMX,EAAQG,YAAYC,QAAQF,WAAWG,QAAQC,MAAM;EAC5EC,YAAYN,EAAOO;EACnBC,eAAeR,EAAOS;EACxB,CAAA,GACwCE,uBAAuB,MAAK;CAEtEC,OAAO,EAAEC,qBAAkB,EACzBC,MAAM,CAAC,EAAEC,OAAOF,GAAYH,qBAAqB,eAAc,CAAC,EAClE;CACAM,WAASC,iDAAA,YAAA;CACTE,YAAY,OAAO,EAAEpB,YAASC,gBAAQ;EACpC,IAAM,EAAEE,kBAAeH,GAGjBwB,IAAepC,EADK,MAAOe,GAAYmB,KAAKC,qBAAqBjB,OAAAA,IAAY,EAAE,CAChDe;AAUrC,MAPI,CAACG,EAAa,WAOd,CAACA,EAAa,QAAW,QAC3B,OAAMrC,EAAS;GACbU,IAAI;GACJI,QAAQ,EAAEO,WAAWP,EAAOO,WAAU;GACxC,CAAA;;CAGN,CAAA"}
1
+ {"version":3,"file":"_floatingIpId-D33bOEmH.mjs","names":["createFileRoute","redirect","getServiceIndex","Route","staticData","section","service","isDetail","sectionCrumb","labelKey","crumb","to","RouteInfo","loader","context","params","floatingIp","trpcClient","network","getById","query","project_id","projectId","floatingip_id","floatingIpId","floatingIpAddress","floating_ip_address","head","loaderData","meta","title","component","lazyRouteComponent","$$splitComponentImporter","beforeLoad","availableServices","auth","getAvailableServices","serviceIndex"],"sources":["../../src/client/routes/_auth/projects/$projectId/network/floatingips/$floatingIpId/index.tsx"],"sourcesContent":["import { createFileRoute, redirect, useNavigate } from \"@tanstack/react-router\"\nimport { Button, ContentHeading, Stack, Spinner } from \"@cloudoperators/juno-ui-components\"\nimport { Trans } from \"@lingui/react/macro\"\nimport type { RouteInfo } from \"@/client/routes/routeInfo\"\nimport { getServiceIndex } from \"@/server/Authentication/helpers\"\nimport { useProjectId } from \"@/client/hooks\"\nimport { trpcReact } from \"@/client/trpcClient\"\nimport { FloatingIpDetailsView } from \"./-components/-details/FloatingIpDetailsView\"\n\nexport const Route = createFileRoute(\"/_auth/projects/$projectId/network/floatingips/$floatingIpId/\")({\n staticData: {\n section: \"network\",\n service: \"floatingips\",\n isDetail: true,\n sectionCrumb: { labelKey: \"Network\" },\n crumb: { labelKey: \"Floating IPs\", to: \"/projects/$projectId/network/floatingips\" },\n } satisfies RouteInfo,\n loader: async ({ context, params }) => {\n const floatingIp = await context.trpcClient?.network.floatingIp.getById.query({\n project_id: params.projectId,\n floatingip_id: params.floatingIpId,\n })\n return { floatingIpAddress: floatingIp?.floating_ip_address ?? null }\n },\n head: ({ loaderData }) => ({\n meta: [{ title: loaderData?.floatingIpAddress ?? \"Floating IP\" }],\n }),\n component: RouteComponent,\n beforeLoad: async ({ context, params }) => {\n const { trpcClient } = context\n\n const availableServices = (await trpcClient?.auth.getAvailableServices.query()) || []\n const serviceIndex = getServiceIndex(availableServices)\n\n // Redirect if network service not available\n if (!serviceIndex[\"network\"]) {\n throw redirect({\n to: \"/projects/$projectId/network/floatingips\",\n params: { projectId: params.projectId },\n })\n }\n\n if (!serviceIndex[\"network\"][\"neutron\"]) {\n throw redirect({\n to: \"/projects/$projectId/network/floatingips\",\n params: { projectId: params.projectId },\n })\n }\n },\n})\n\nfunction RouteComponent() {\n const { floatingIpId } = Route.useParams()\n const projectId = useProjectId()\n const navigate = useNavigate()\n\n // Fetch floating IP details\n const {\n data: floatingIp,\n isLoading,\n isError,\n error,\n } = trpcReact.network.floatingIp.getById.useQuery({\n project_id: projectId,\n floatingip_id: floatingIpId,\n })\n\n const handleBack = () => {\n navigate({\n to: \"/projects/$projectId/network/floatingips\",\n params: { projectId },\n })\n }\n\n // 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 Floating IP Details...</Trans>\n </Stack>\n )\n }\n\n // Error state\n if (isError) {\n const errorMessage = error?.message || \"Unknown error\"\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 floating IP</Trans>\n </p>\n <p className=\"text-theme-highest\">{errorMessage}</p>\n <Button onClick={handleBack} variant=\"primary\">\n <Trans>Back to Floating IPs</Trans>\n </Button>\n </Stack>\n )\n }\n\n // No data state\n if (!floatingIp) {\n return (\n <Stack className=\"fixed inset-0\" distribution=\"center\" alignment=\"center\" direction=\"vertical\" gap=\"5\">\n <p className=\"text-theme-secondary\">\n <Trans>Floating IP not found</Trans>\n </p>\n <Button onClick={handleBack} variant=\"primary\">\n <Trans>Back to Floating IPs</Trans>\n </Button>\n </Stack>\n )\n }\n\n // Success state\n return (\n <>\n <ContentHeading>{floatingIp.floating_ip_address}</ContentHeading>\n <FloatingIpDetailsView floatingIp={floatingIp} />\n </>\n )\n}\n"],"mappings":";;AASA,IAAaG,IAAQH,EAAgB,gEAAA,CAAiE;CACpGI,YAAY;EACVC,SAAS;EACTC,SAAS;EACTC,UAAU;EACVC,cAAc,EAAEC,UAAU,WAAU;EACpCC,OAAO;GAAED,UAAU;GAAgBE,IAAI;GAA2C;EACpF;CACAE,QAAQ,OAAO,EAAEC,YAASC,iBAKjB,EAAEU,oBAJU,MAAMX,EAAQG,YAAYC,QAAQF,WAAWG,QAAQC,MAAM;EAC5EC,YAAYN,EAAOO;EACnBC,eAAeR,EAAOS;EACxB,CAAA,GACwCE,uBAAuB,MAAK;CAEtEC,OAAO,EAAEC,qBAAkB,EACzBC,MAAM,CAAC,EAAEC,OAAOF,GAAYH,qBAAqB,eAAc,CAAC,EAClE;CACAM,WAASC,gDAAA,YAAA;CACTE,YAAY,OAAO,EAAEpB,YAASC,gBAAQ;EACpC,IAAM,EAAEE,kBAAeH,GAGjBwB,IAAepC,EADK,MAAOe,GAAYmB,KAAKC,qBAAqBjB,OAAAA,IAAY,EAAE,CAChDe;AAUrC,MAPI,CAACG,EAAa,WAOd,CAACA,EAAa,QAAW,QAC3B,OAAMrC,EAAS;GACbU,IAAI;GACJI,QAAQ,EAAEO,WAAWP,EAAOO,WAAU;GACxC,CAAA;;CAGN,CAAA"}
@@ -1,9 +1,9 @@
1
1
  import { L as e, N as t, Q as n, W as r, c as i, h as a, j as o, nt as s } from "./build-BJDfnAyi.mjs";
2
2
  import { r as c } from "./trpcClient-BxguzNYF.mjs";
3
- import { t as l } from "./_floatingIpId-CwHiXazi.mjs";
4
- import { t as u } from "./useProjectId-CgOTejka.mjs";
5
- import "./hooks-D0krAKvo.mjs";
6
- import { t as d } from "./FloatingIpActionModals-By2hXR9m.mjs";
3
+ import { t as l } from "./_floatingIpId-D33bOEmH.mjs";
4
+ import { t as u } from "./useProjectId-OQv2KBbG.mjs";
5
+ import "./hooks-DEjb9d1F.mjs";
6
+ import { t as d } from "./FloatingIpActionModals-Dok7fJss.mjs";
7
7
  import { Fragment as f, jsx as p, jsxs as m } from "react/jsx-runtime";
8
8
  import { Fragment as h } from "react";
9
9
  import { useNavigate as g } from "@tanstack/react-router";
@@ -225,4 +225,4 @@ function S() {
225
225
  //#endregion
226
226
  export { S as component };
227
227
 
228
- //# sourceMappingURL=_floatingIpId-BpUfL8Im2.mjs.map
228
+ //# sourceMappingURL=_floatingIpId-DF_BSJN6.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"_floatingIpId-DF_BSJN6.mjs","names":["formatFloatingIpStatus","status","charAt","slice","toLowerCase","Fragment","DescriptionDefinition","DescriptionList","DescriptionTerm","Stack","TwoColumnDescriptionList","items","mid","Math","ceil","length","firstColumn","slice","secondColumn","gap","className","alignTerms","map","label","value","Stack","ButtonRow","Button","ContentHeading","formatFloatingIpStatus","TwoColumnDescriptionList","FloatingIpActionModals","FloatingIpDetailsView","floatingIp","useLingui","basicInfoItems","label","t","value","id","description","project_id","status","created_at","Date","toLocaleString","updated_at","tags","join","networkRoutingItems","floating_ip_address","floating_network_id","fixed_ip_address","port_details","name","mac_address","network_id","device_owner","device_id","router_id","port_id","qos_policy_id","port_forwardings","map","port","dnsItems","dns_domain","dns_name","p","className","toggleEditModal","toggleAttachModal","toggleDetachModal","toggleReleaseModal","onClick","direction","gap","items","useNavigate","Button","ContentHeading","Stack","Spinner","Trans","useProjectId","trpcReact","FloatingIpDetailsView","Route","RouteComponent","floatingIpId","useParams","projectId","navigate","data","floatingIp","isLoading","isError","error","network","getById","useQuery","project_id","floatingip_id","handleBack","to","params","errorMessage","message","floating_ip_address","component"],"sources":["../../src/client/utils/formatFloatingIpStatus.ts","../../src/client/routes/_auth/projects/$projectId/network/floatingips/$floatingIpId/-components/TwoColumnDescriptionList.tsx","../../src/client/routes/_auth/projects/$projectId/network/floatingips/$floatingIpId/-components/-details/FloatingIpDetailsView.tsx","../../src/client/routes/_auth/projects/$projectId/network/floatingips/$floatingIpId/index.tsx?tsr-split=component"],"sourcesContent":["import type { FloatingIpStatus } from \"@/server/Network/types/floatingIp\"\n\n/**\n * Formats a floating IP status value from uppercase enum to title case.\n * Example: \"ACTIVE\" → \"Active\", \"DOWN\" → \"Down\", \"ERROR\" → \"Error\"\n */\nexport const formatFloatingIpStatus = (status: FloatingIpStatus) => {\n return status.charAt(0) + status.slice(1).toLowerCase()\n}\n","import { Fragment } from \"react\"\nimport { DescriptionDefinition, DescriptionList, DescriptionTerm, Stack } from \"@cloudoperators/juno-ui-components\"\n\nexport type DetailListItem = {\n label: string\n value: string\n}\n\ninterface TwoColumnDescriptionListProps {\n items: DetailListItem[]\n}\n\nexport const TwoColumnDescriptionList = ({ items }: TwoColumnDescriptionListProps) => {\n const mid = Math.ceil(items.length / 2)\n const firstColumn = items.slice(0, mid)\n const secondColumn = items.slice(mid)\n\n return (\n <Stack gap=\"6\" className=\"grid grid-cols-2\">\n <DescriptionList alignTerms=\"right\">\n {firstColumn.map(({ label, value }) => (\n <Fragment key={label}>\n <DescriptionTerm>{label}</DescriptionTerm>\n <DescriptionDefinition>{value}</DescriptionDefinition>\n </Fragment>\n ))}\n </DescriptionList>\n\n <DescriptionList alignTerms=\"right\">\n {secondColumn.map(({ label, value }) => (\n <Fragment key={label}>\n <DescriptionTerm>{label}</DescriptionTerm>\n <DescriptionDefinition>{value}</DescriptionDefinition>\n </Fragment>\n ))}\n </DescriptionList>\n </Stack>\n )\n}\n","import { Trans, useLingui } from \"@lingui/react/macro\"\nimport { Stack, ButtonRow, Button, ContentHeading } from \"@cloudoperators/juno-ui-components\"\nimport type { FloatingIp } from \"@/server/Network/types/floatingIp\"\nimport { formatFloatingIpStatus } from \"@/client/utils/formatFloatingIpStatus\"\nimport { DetailListItem, TwoColumnDescriptionList } from \"../TwoColumnDescriptionList\"\nimport { FloatingIpActionModals } from \"../../../-components/-modals/FloatingIpActionModals\"\n\ninterface FloatingIpDetailsViewProps {\n floatingIp: FloatingIp\n}\n\nexport const FloatingIpDetailsView = ({ floatingIp }: FloatingIpDetailsViewProps) => {\n const { t } = useLingui()\n\n const basicInfoItems: DetailListItem[] = [\n { label: t`ID`, value: floatingIp.id },\n { label: t`Description`, value: floatingIp.description || `—` },\n { label: t`Project ID`, value: floatingIp.project_id || `—` },\n { label: t`Status`, value: formatFloatingIpStatus(floatingIp.status) },\n { label: t`Created At`, value: floatingIp.created_at ? new Date(floatingIp.created_at).toLocaleString() : `—` },\n { label: t`Updated At`, value: floatingIp.updated_at ? new Date(floatingIp.updated_at).toLocaleString() : `—` },\n { label: t`Tags`, value: floatingIp.tags?.join(\", \") || `—` },\n ]\n\n const networkRoutingItems: DetailListItem[] = [\n { label: t`Floating IP Address`, value: floatingIp.floating_ip_address || `—` },\n { label: t`Floating Network`, value: floatingIp.floating_network_id || `—` },\n { label: t`Fixed IP Address`, value: floatingIp.fixed_ip_address || `—` },\n { label: t`Port Name`, value: floatingIp.port_details?.name || `—` },\n { label: t`MAC Address`, value: floatingIp.port_details?.mac_address || `—` },\n { label: t`Network ID`, value: floatingIp.port_details?.network_id || `—` },\n { label: t`Device Owner`, value: floatingIp.port_details?.device_owner || `—` },\n { label: t`Device ID`, value: floatingIp.port_details?.device_id || `—` },\n { label: t`Router ID`, value: floatingIp.router_id || `—` },\n { label: t`Port ID`, value: floatingIp.port_id || `—` },\n { label: t`QoS Policy ID`, value: floatingIp.qos_policy_id || `—` },\n { label: t`Port Forwarding`, value: floatingIp.port_forwardings?.map((port) => port.id).join(\", \") || `—` },\n ]\n\n const dnsItems: DetailListItem[] = [\n { label: t`DNS Domain`, value: floatingIp.dns_domain || `—` },\n { label: t`DNS Name`, value: floatingIp.dns_name || `—` },\n ]\n\n return (\n <>\n <p className=\"text-theme-secondary mt-2 text-sm\">\n <Trans>\n Full lifecycle management of Floating IPs, including attachment, port association/disassociation, DNS\n settings, and deletion\n </Trans>\n </p>\n\n <FloatingIpActionModals floatingIp={floatingIp}>\n {({ toggleEditModal, toggleAttachModal, toggleDetachModal, toggleReleaseModal }) => (\n <ButtonRow>\n <Button onClick={toggleEditModal}>{t`Edit Description`}</Button>\n <Button onClick={toggleAttachModal}>{t`Attach`}</Button>\n <Button onClick={toggleDetachModal}>{t`Detach`}</Button>\n <Button onClick={toggleReleaseModal}>{t`Release`}</Button>\n </ButtonRow>\n )}\n </FloatingIpActionModals>\n\n <Stack direction=\"vertical\" gap=\"6\" className=\"my-6\">\n <Stack direction=\"vertical\" gap=\"2\">\n <ContentHeading>\n <Trans>Basic Info</Trans>\n </ContentHeading>\n <TwoColumnDescriptionList items={basicInfoItems} />\n </Stack>\n\n <Stack direction=\"vertical\" gap=\"2\">\n <ContentHeading>\n <Trans>Network & Routing</Trans>\n </ContentHeading>\n <TwoColumnDescriptionList items={networkRoutingItems} />\n </Stack>\n\n <Stack direction=\"vertical\" gap=\"2\">\n <ContentHeading>\n <Trans>DNS</Trans>\n </ContentHeading>\n <TwoColumnDescriptionList items={dnsItems} />\n </Stack>\n </Stack>\n </>\n )\n}\n","import { createFileRoute, redirect, useNavigate } from \"@tanstack/react-router\"\nimport { Button, ContentHeading, Stack, Spinner } from \"@cloudoperators/juno-ui-components\"\nimport { Trans } from \"@lingui/react/macro\"\nimport type { RouteInfo } from \"@/client/routes/routeInfo\"\nimport { getServiceIndex } from \"@/server/Authentication/helpers\"\nimport { useProjectId } from \"@/client/hooks\"\nimport { trpcReact } from \"@/client/trpcClient\"\nimport { FloatingIpDetailsView } from \"./-components/-details/FloatingIpDetailsView\"\n\nexport const Route = createFileRoute(\"/_auth/projects/$projectId/network/floatingips/$floatingIpId/\")({\n staticData: {\n section: \"network\",\n service: \"floatingips\",\n isDetail: true,\n sectionCrumb: { labelKey: \"Network\" },\n crumb: { labelKey: \"Floating IPs\", to: \"/projects/$projectId/network/floatingips\" },\n } satisfies RouteInfo,\n loader: async ({ context, params }) => {\n const floatingIp = await context.trpcClient?.network.floatingIp.getById.query({\n project_id: params.projectId,\n floatingip_id: params.floatingIpId,\n })\n return { floatingIpAddress: floatingIp?.floating_ip_address ?? null }\n },\n head: ({ loaderData }) => ({\n meta: [{ title: loaderData?.floatingIpAddress ?? \"Floating IP\" }],\n }),\n component: RouteComponent,\n beforeLoad: async ({ context, params }) => {\n const { trpcClient } = context\n\n const availableServices = (await trpcClient?.auth.getAvailableServices.query()) || []\n const serviceIndex = getServiceIndex(availableServices)\n\n // Redirect if network service not available\n if (!serviceIndex[\"network\"]) {\n throw redirect({\n to: \"/projects/$projectId/network/floatingips\",\n params: { projectId: params.projectId },\n })\n }\n\n if (!serviceIndex[\"network\"][\"neutron\"]) {\n throw redirect({\n to: \"/projects/$projectId/network/floatingips\",\n params: { projectId: params.projectId },\n })\n }\n },\n})\n\nfunction RouteComponent() {\n const { floatingIpId } = Route.useParams()\n const projectId = useProjectId()\n const navigate = useNavigate()\n\n // Fetch floating IP details\n const {\n data: floatingIp,\n isLoading,\n isError,\n error,\n } = trpcReact.network.floatingIp.getById.useQuery({\n project_id: projectId,\n floatingip_id: floatingIpId,\n })\n\n const handleBack = () => {\n navigate({\n to: \"/projects/$projectId/network/floatingips\",\n params: { projectId },\n })\n }\n\n // 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 Floating IP Details...</Trans>\n </Stack>\n )\n }\n\n // Error state\n if (isError) {\n const errorMessage = error?.message || \"Unknown error\"\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 floating IP</Trans>\n </p>\n <p className=\"text-theme-highest\">{errorMessage}</p>\n <Button onClick={handleBack} variant=\"primary\">\n <Trans>Back to Floating IPs</Trans>\n </Button>\n </Stack>\n )\n }\n\n // No data state\n if (!floatingIp) {\n return (\n <Stack className=\"fixed inset-0\" distribution=\"center\" alignment=\"center\" direction=\"vertical\" gap=\"5\">\n <p className=\"text-theme-secondary\">\n <Trans>Floating IP not found</Trans>\n </p>\n <Button onClick={handleBack} variant=\"primary\">\n <Trans>Back to Floating IPs</Trans>\n </Button>\n </Stack>\n )\n }\n\n // Success state\n return (\n <>\n <ContentHeading>{floatingIp.floating_ip_address}</ContentHeading>\n <FloatingIpDetailsView floatingIp={floatingIp} />\n </>\n )\n}\n"],"mappings":"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"}
@@ -1,7 +1,7 @@
1
1
  import { E as e, J as t, L as n, N as r, Q as i, R as a, W as o, Y as s, b as c, c as l, et as u, h as d, j as f, nt as p, q as m, z as h } from "./build-BJDfnAyi.mjs";
2
2
  import { r as g } from "./trpcClient-BxguzNYF.mjs";
3
3
  import { t as ee } from "./ContentHeader-BXZoN3B9.mjs";
4
- import { A as _, D as te, E as ne, F as v, I as y, N as re, O as ie, P as b, T as ae, d as oe, k as se, u as ce } from "./ImageToastNotifications-fHI8jB2j.mjs";
4
+ import { A as _, D as te, E as ne, F as v, I as y, N as re, O as ie, P as b, T as ae, d as oe, k as se, u as ce } from "./ImageToastNotifications-BG9LPnXf.mjs";
5
5
  import { Fragment as x, jsx as S, jsxs as C } from "react/jsx-runtime";
6
6
  import w, { useState as T } from "react";
7
7
  import { useNavigate as le, useParams as ue, useSearch as de } from "@tanstack/react-router";
@@ -524,4 +524,4 @@ function P() {
524
524
  //#endregion
525
525
  export { P as component };
526
526
 
527
- //# sourceMappingURL=_imageId-CdOOJjw0.mjs.map
527
+ //# sourceMappingURL=_imageId-BL0I5_pv.mjs.map
@@ -1 +1 @@
1
- {"version":3,"file":"_imageId-CdOOJjw0.mjs","names":["React","useState","DescriptionList","DescriptionTerm","DescriptionDefinition","Container","ContentHeading","Stack","Message","Box","Button","ButtonRow","SizeDisplay","trpcReact","MEMBER_STATUSES","ImageMembersTable","SharedImageBox","image","myMemberData","canUpdateMember","onStatusChange","isLoading","useLingui","isPending","status","PENDING","isRejected","REJECTED","isAccepted","ACCEPTED","sharedAt","created_at","Date","toLocaleString","t","updatedAt","updated_at","ownerProject","owner","p","className","span","ul","li","onClick","disabled","variant","GeneralImageData","px","py","alignTerms","id","name","size","min_disk","min_ram","disk_format","container_format","toLocaleDateString","SecuritySection","currentProjectId","isSharedWithMe","visibility","undefined","protected","checksum","CustomPropertiesSection","knownFields","Set","customProperties","Object","entries","filter","key","has","sort","a","b","localeCompare","hasProperties","length","map","value","Fragment","JSON","stringify","String","getTabClassName","active","base","SharingDetailsTab","permissions","isAddingMember","setIsAddingMember","message","setMessage","data","imageMembers","isMembersLoading","compute","listImageMembers","useQuery","project_id","imageId","enabled","text","type","onDismiss","canAdd","canCreateMember","canRemove","canDeleteMember","projectId","ImageDetailsView","activeTab","onTabChange","onMemberStatusChange","isMemberStatusChanging","actions","isImageOwner","showTabs","direction","gap","div","button","Button","ButtonRow","Stack","Spinner","PopupMenu","PopupMenuToggle","PopupMenuOptions","PopupMenuItem","Toast","ToastProps","useNavigate","useParams","useSearch","Trans","useLingui","trpcReact","ImageDetailsView","EditImageDetailsModal","EditImageMetadataModal","DeleteImageModal","ActivateImageModal","DeactivateImageModal","IMAGE_STATUSES","IMAGE_VISIBILITY","GlanceImage","MemberStatus","TRPCClientError","InferrableClientTypes","getImageAccessStatusUpdatedToast","getImageAccessStatusErrorToast","useState","ContentHeader","RouteComponent","projectId","imageId","from","tab","navigate","t","data","image","status","error","compute","getImageById","useQuery","project_id","permissionsData","canUser","permission","permissions","canDelete","canUpdate","canCreateMember","canDeleteMember","canUpdateMember","utils","useUtils","editDetailsModalOpen","setEditDetailsModalOpen","editMetadataModalOpen","setEditMetadataModalOpen","deleteModalOpen","setDeleteModalOpen","activateModalOpen","setActivateModalOpen","deactivateModalOpen","setDeactivateModalOpen","toastData","setToastData","updateImageMutation","updateImage","useMutation","onSuccess","updatedImage","setData","listImagesWithPagination","invalidate","deleteImageMutation","deleteImage","deactivateImageMutation","deactivateImage","reactivateImageMutation","reactivateImage","updateImageVisibilityMutation","updateImageVisibility","isSharedWithMe","visibility","SHARED","owner","undefined","myMemberData","getImageMember","memberId","enabled","updateMemberMutation","updateImageMember","listImageMembers","listSharedImagesByMemberStatus","handleMemberStatusChange","newStatus","mutateAsync","onDismiss","errorMessage","message","handleBack","to","params","isLoading","isPending","convertToJsonPatchOperations","updatedProperties","Partial","originalImage","Array","op","path","value","operations","Object","entries","forEach","key","push","propertyExists","handleSaveEdit","Promise","handleDelete","deletedImage","id","handleActivate","img","handleDeactivate","handleUpdateVisibility","newVisibility","isDeactivated","DEACTIVATED","isPrivate","PRIVATE","hasMoreActions","protected","headerActions","String","name","newTab","search","component"],"sources":["../../src/client/routes/_auth/projects/$projectId/compute/-components/Images/-components/ImageDetailsView.tsx","../../src/client/routes/_auth/projects/$projectId/compute/images/$imageId.tsx?tsr-split=component"],"sourcesContent":["import React, { useState } from \"react\"\nimport {\n DescriptionList,\n DescriptionTerm,\n DescriptionDefinition,\n Container,\n ContentHeading,\n Stack,\n Message,\n Box,\n Button,\n ButtonRow,\n} from \"@cloudoperators/juno-ui-components\"\nimport { Trans, useLingui } from \"@lingui/react/macro\"\nimport { GlanceImage, ImageMember, MemberStatus } from \"@/server/Compute/types/image\"\nimport { SizeDisplay } from \"./SizeDisplay\"\nimport { trpcReact } from \"@/client/trpcClient\"\nimport { MEMBER_STATUSES } from \"../../../-constants/filters\"\nimport { ImageMembersTable } from \"./ImageMembersTable\"\n\ninterface ImageDetailsViewProps {\n image: GlanceImage\n currentProjectId?: string\n activeTab?: \"details\" | \"sharing\"\n onTabChange?: (tab: \"details\" | \"sharing\") => void\n permissions?: {\n canCreateMember: boolean\n canDeleteMember: boolean\n canUpdateMember: boolean\n }\n myMemberData?: ImageMember\n onMemberStatusChange?: (status: MemberStatus) => void\n isMemberStatusChanging?: boolean\n actions?: React.ReactNode\n}\n\nconst SharedImageBox: React.FC<{\n image: GlanceImage\n myMemberData: ImageMember\n canUpdateMember: boolean\n onStatusChange: (status: MemberStatus) => void\n isLoading: boolean\n}> = ({ image, myMemberData, canUpdateMember, onStatusChange, isLoading }) => {\n const { t } = useLingui()\n const isPending = myMemberData.status === MEMBER_STATUSES.PENDING\n const isRejected = myMemberData.status === MEMBER_STATUSES.REJECTED\n const isAccepted = myMemberData.status === MEMBER_STATUSES.ACCEPTED\n\n const sharedAt = myMemberData.created_at ? new Date(myMemberData.created_at).toLocaleString() : t`N/A`\n const updatedAt = myMemberData.updated_at ? new Date(myMemberData.updated_at).toLocaleString() : t`N/A`\n const ownerProject = image.owner ?? \"\"\n\n return (\n <Box>\n {isPending && (\n <p className=\"text-theme-highest font-semibold\">\n <Trans>Your action is required</Trans>\n </p>\n )}\n <p>\n <Trans>\n This image was shared with you by <span className=\"font-semibold\">{ownerProject}</span> on {sharedAt}.\n </Trans>\n </p>\n <ul>\n <li>\n <span className=\"font-semibold\">\n <Trans>Access Status:</Trans>\n </span>{\" \"}\n {myMemberData.status}\n </li>\n <li>\n <span className=\"font-semibold\">\n <Trans>Shared:</Trans>\n </span>{\" \"}\n {sharedAt}\n </li>\n <li>\n <span className=\"font-semibold\">\n <Trans>Updated:</Trans>\n </span>{\" \"}\n {updatedAt}\n </li>\n </ul>\n\n {canUpdateMember && (\n <ButtonRow>\n {isPending && (\n <Button onClick={() => onStatusChange(MEMBER_STATUSES.REJECTED)} disabled={isLoading} variant=\"subdued\">\n <Trans>Reject</Trans>\n </Button>\n )}\n {(isPending || isRejected) && (\n <Button onClick={() => onStatusChange(MEMBER_STATUSES.ACCEPTED)} disabled={isLoading} variant=\"primary\">\n <Trans>Accept</Trans>\n </Button>\n )}\n {isAccepted && (\n <Button\n onClick={() => onStatusChange(MEMBER_STATUSES.REJECTED)}\n disabled={isLoading}\n variant=\"primary-danger\"\n >\n <Trans>Reject</Trans>\n </Button>\n )}\n </ButtonRow>\n )}\n </Box>\n )\n}\n\nexport const GeneralImageData: React.FC<{ image: GlanceImage }> = ({ image }) => {\n const { t } = useLingui()\n\n return (\n <Container px={false} py>\n <ContentHeading>{t`General Image Data`}</ContentHeading>\n <DescriptionList alignTerms=\"right\">\n <DescriptionTerm>{t`ID`}</DescriptionTerm>\n <DescriptionDefinition>{image.id}</DescriptionDefinition>\n\n <DescriptionTerm>{t`Name`}</DescriptionTerm>\n <DescriptionDefinition>{image.name}</DescriptionDefinition>\n\n <DescriptionTerm>{t`Status`}</DescriptionTerm>\n <DescriptionDefinition>{image.status}</DescriptionDefinition>\n\n <DescriptionTerm>{t`Size`}</DescriptionTerm>\n <DescriptionDefinition>\n <SizeDisplay size={image.size} />\n </DescriptionDefinition>\n\n <DescriptionTerm>{t`Min. Disk`}</DescriptionTerm>\n <DescriptionDefinition>{image.min_disk} GB</DescriptionDefinition>\n\n <DescriptionTerm>{t`Min. RAM`}</DescriptionTerm>\n <DescriptionDefinition>{image.min_ram} MB</DescriptionDefinition>\n\n <DescriptionTerm>{t`Disk Format`}</DescriptionTerm>\n <DescriptionDefinition>\n <span className=\"uppercase\">{image.disk_format}</span>\n </DescriptionDefinition>\n\n <DescriptionTerm>{t`Container Format`}</DescriptionTerm>\n <DescriptionDefinition>\n <span className=\"uppercase\">{image.container_format}</span>\n </DescriptionDefinition>\n\n <DescriptionTerm>{t`Created At`}</DescriptionTerm>\n <DescriptionDefinition>\n {image.created_at ? new Date(image.created_at).toLocaleDateString() : t`N/A`}\n </DescriptionDefinition>\n\n <DescriptionTerm>{t`Updated At`}</DescriptionTerm>\n <DescriptionDefinition>\n {image.updated_at ? new Date(image.updated_at).toLocaleDateString() : t`N/A`}\n </DescriptionDefinition>\n </DescriptionList>\n </Container>\n )\n}\n\nexport const SecuritySection: React.FC<{ image: GlanceImage; currentProjectId?: string }> = ({\n image,\n currentProjectId,\n}) => {\n const { t } = useLingui()\n\n const isSharedWithMe = image.visibility === \"shared\" && image.owner !== undefined && image.owner !== currentProjectId\n\n return (\n <Container px={false} py>\n <ContentHeading>{t`Security`}</ContentHeading>\n <DescriptionList alignTerms=\"right\">\n <DescriptionTerm>{isSharedWithMe ? t`Shared by Project` : t`Owner Project ID`}</DescriptionTerm>\n <DescriptionDefinition>{image.owner}</DescriptionDefinition>\n\n <DescriptionTerm>{t`Visibility`}</DescriptionTerm>\n <DescriptionDefinition>{image.visibility}</DescriptionDefinition>\n\n <DescriptionTerm>{t`Protected`}</DescriptionTerm>\n <DescriptionDefinition>{image.protected ? t`Yes` : t`No`}</DescriptionDefinition>\n\n <DescriptionTerm>{t`Checksum`}</DescriptionTerm>\n <DescriptionDefinition>{image?.checksum ? image.checksum : \"\"}</DescriptionDefinition>\n </DescriptionList>\n </Container>\n )\n}\n\nexport const CustomPropertiesSection: React.FC<{ image: GlanceImage }> = ({ image }) => {\n const { t } = useLingui()\n\n const knownFields = new Set([\n \"id\",\n \"name\",\n \"status\",\n \"visibility\",\n \"size\",\n \"disk_format\",\n \"container_format\",\n \"min_disk\",\n \"min_ram\",\n \"owner\",\n \"protected\",\n \"created_at\",\n \"updated_at\",\n \"checksum\",\n ])\n\n const customProperties = Object.entries(image)\n .filter(([key]) => !knownFields.has(key))\n .sort(([a], [b]) => a.localeCompare(b))\n\n const hasProperties = customProperties.length > 0\n\n return (\n <Container px={false} py>\n <ContentHeading>{t`Custom Properties / Metadata`}</ContentHeading>\n {hasProperties ? (\n <DescriptionList alignTerms=\"right\" className=\"grid-cols-4\">\n {customProperties.map(([key, value]) => (\n <React.Fragment key={key}>\n <DescriptionTerm className=\"col-span-1\">{key}</DescriptionTerm>\n <DescriptionDefinition className=\"col-span-1\">\n {value === null || value === undefined ? (\n <span>null</span>\n ) : typeof value === \"object\" ? (\n <span className=\"break-all\">{JSON.stringify(value)}</span>\n ) : typeof value === \"boolean\" ? (\n value ? (\n t`True`\n ) : (\n t`False`\n )\n ) : (\n <span className=\"break-all\">{String(value)}</span>\n )}\n </DescriptionDefinition>\n </React.Fragment>\n ))}\n </DescriptionList>\n ) : (\n <p className=\"text-theme-light\">{t`No custom properties defined`}</p>\n )}\n </Container>\n )\n}\n\nconst getTabClassName = (active: boolean) => {\n const base = \"px-6 py-3 font-semibold border-b-2 transition-colors\"\n return active\n ? `${base} border-theme-accent text-theme-highest`\n : `${base} border-transparent text-theme-secondary hover:text-theme-high`\n}\n\nconst SharingDetailsTab: React.FC<ImageDetailsViewProps> = ({ image, permissions, currentProjectId }) => {\n const [isAddingMember, setIsAddingMember] = useState(false)\n const [message, setMessage] = useState<{ text: string; type: \"error\" | \"info\" } | null>(null)\n\n const { data: imageMembers, isLoading: isMembersLoading } = trpcReact.compute.listImageMembers.useQuery(\n { project_id: currentProjectId!, imageId: image.id },\n { enabled: !!image.id && !!currentProjectId }\n )\n\n return (\n <Container px={false} py>\n {message && (\n <Message text={message.text} variant={message.type} onDismiss={() => setMessage(null)} className=\"mb-4\" />\n )}\n <ImageMembersTable\n image={image}\n imageMembers={imageMembers}\n isMembersLoading={isMembersLoading}\n canAdd={permissions?.canCreateMember ?? false}\n canRemove={permissions?.canDeleteMember ?? false}\n isAddingMember={isAddingMember}\n setIsAddingMember={setIsAddingMember}\n setMessage={setMessage}\n projectId={currentProjectId!}\n />\n </Container>\n )\n}\n\nexport const ImageDetailsView: React.FC<ImageDetailsViewProps> = ({\n image,\n currentProjectId,\n activeTab = \"details\",\n onTabChange,\n permissions,\n myMemberData,\n onMemberStatusChange,\n isMemberStatusChanging,\n actions,\n}) => {\n const { t } = useLingui()\n\n const isSharedWithMe = image.visibility === \"shared\" && image.owner !== undefined && image.owner !== currentProjectId\n const isImageOwner = image.owner === currentProjectId\n const showTabs = isImageOwner && image.visibility === \"shared\"\n\n return (\n <Stack direction=\"vertical\" gap=\"6\">\n {isSharedWithMe && myMemberData && onMemberStatusChange && (\n <SharedImageBox\n image={image}\n myMemberData={myMemberData}\n canUpdateMember={permissions?.canUpdateMember ?? false}\n onStatusChange={onMemberStatusChange}\n isLoading={isMemberStatusChanging ?? false}\n />\n )}\n\n {showTabs && (\n <div className=\"border-theme-background-lvl-3 border-b\">\n <Stack direction=\"horizontal\" gap=\"0\">\n <button className={getTabClassName(activeTab === \"details\")} onClick={() => onTabChange?.(\"details\")}>\n {t`Details`}\n </button>\n <button className={getTabClassName(activeTab === \"sharing\")} onClick={() => onTabChange?.(\"sharing\")}>\n {t`Sharing Details`}\n </button>\n </Stack>\n </div>\n )}\n\n {(activeTab === \"details\" || !showTabs) && (\n <>\n {actions}\n <GeneralImageData image={image} />\n <SecuritySection image={image} currentProjectId={currentProjectId} />\n <CustomPropertiesSection image={image} />\n </>\n )}\n\n {activeTab === \"sharing\" && showTabs && <SharingDetailsTab image={image} permissions={permissions} />}\n </Stack>\n )\n}\n","import {\n Button,\n ButtonRow,\n Stack,\n Spinner,\n PopupMenu,\n PopupMenuToggle,\n PopupMenuOptions,\n PopupMenuItem,\n Toast,\n ToastProps,\n} from \"@cloudoperators/juno-ui-components/index\"\nimport { createFileRoute, redirect, useNavigate, useParams, useSearch } from \"@tanstack/react-router\"\nimport { z } from \"zod\"\nimport type { RouteInfo } from \"@/client/routes/routeInfo\"\nimport { Trans, useLingui } from \"@lingui/react/macro\"\nimport { getServiceIndex } from \"@/server/Authentication/helpers\"\nimport { trpcReact } from \"@/client/trpcClient\"\nimport { ImageDetailsView } from \"../-components/Images/-components/ImageDetailsView\"\nimport { EditImageDetailsModal } from \"../-components/Images/-components/EditImageDetailsModal\"\nimport { EditImageMetadataModal } from \"../-components/Images/-components/EditImageMetadataModal\"\nimport { DeleteImageModal } from \"../-components/Images/-components/DeleteImageModal\"\nimport { ActivateImageModal } from \"../-components/Images/-components/ActivateImageModal\"\nimport { DeactivateImageModal } from \"../-components/Images/-components/DeactivateImageModal\"\nimport { IMAGE_STATUSES, IMAGE_VISIBILITY } from \"../-constants/filters\"\nimport { GlanceImage, MemberStatus } from \"@/server/Compute/types/image\"\nimport { TRPCClientError } from \"@trpc/client\"\nimport { InferrableClientTypes } from \"@trpc/server/unstable-core-do-not-import\"\nimport {\n getImageAccessStatusUpdatedToast,\n getImageAccessStatusErrorToast,\n} from \"../-components/Images/-components/ImageToastNotifications\"\nimport { useState } from \"react\"\nimport { ContentHeader } from \"@/client/components/ContentHeader/ContentHeader\"\n\nexport const Route = createFileRoute(\"/_auth/projects/$projectId/compute/images/$imageId\")({\n staticData: {\n section: \"compute\",\n service: \"images\",\n isDetail: true,\n sectionCrumb: { labelKey: \"Compute\" },\n crumb: { labelKey: \"Images\", to: \"/projects/$projectId/compute/images\" },\n } satisfies RouteInfo,\n validateSearch: z.object({\n tab: z.enum([\"details\", \"sharing\"]).optional(),\n }),\n loader: async ({ context, params }) => {\n try {\n const image = await context.trpcClient?.compute.getImageById.query({\n project_id: params.projectId,\n imageId: params.imageId,\n })\n return { imageTitle: (image?.name as string | undefined) ?? image?.id ?? null }\n } catch {\n return { imageTitle: null }\n }\n },\n head: ({ loaderData }) => ({\n meta: [{ title: loaderData?.imageTitle ?? \"Image Details\" }],\n }),\n component: RouteComponent,\n beforeLoad: async ({ context, params }) => {\n const { trpcClient } = context\n const { projectId } = params\n\n const availableServices = (await trpcClient?.auth.getAvailableServices.query()) || []\n\n const serviceIndex = getServiceIndex(availableServices)\n\n // Redirect to the \"Projects Overview\" page if none of compute services available\n if (!serviceIndex[\"image\"] && !serviceIndex[\"compute\"]) {\n throw redirect({\n to: \"/projects/$projectId\",\n params: { projectId },\n })\n }\n\n if (!serviceIndex[\"image\"][\"glance\"]) {\n // Redirect to the \"Compute Services Overview\" page if the \"Glance\" service is not available\n throw redirect({\n to: \"/projects/$projectId\",\n params: { projectId: params.projectId },\n })\n }\n },\n})\n\nfunction RouteComponent() {\n const { projectId, imageId } = useParams({\n from: \"/_auth/projects/$projectId/compute/images/$imageId\",\n })\n const { tab } = useSearch({\n from: \"/_auth/projects/$projectId/compute/images/$imageId\",\n })\n\n const navigate = useNavigate()\n const { t } = useLingui()\n\n const {\n data: image,\n status,\n error,\n } = trpcReact.compute.getImageById.useQuery({ project_id: projectId, imageId: imageId })\n\n const { data: permissionsData } = trpcReact.compute.canUser.useQuery({\n project_id: projectId,\n permission: [\n \"images:delete\",\n \"images:update\",\n \"images:create_member\",\n \"images:delete_member\",\n \"images:update_member\",\n ],\n })\n\n const permissions = {\n canDelete: permissionsData?.[0] ?? false,\n canUpdate: permissionsData?.[1] ?? false,\n canCreateMember: permissionsData?.[2] ?? false,\n canDeleteMember: permissionsData?.[3] ?? false,\n canUpdateMember: permissionsData?.[4] ?? false,\n }\n\n const utils = trpcReact.useUtils()\n\n const [editDetailsModalOpen, setEditDetailsModalOpen] = useState(false)\n const [editMetadataModalOpen, setEditMetadataModalOpen] = useState(false)\n const [deleteModalOpen, setDeleteModalOpen] = useState(false)\n const [activateModalOpen, setActivateModalOpen] = useState(false)\n const [deactivateModalOpen, setDeactivateModalOpen] = useState(false)\n const [toastData, setToastData] = useState<ToastProps | null>(null)\n\n const updateImageMutation = trpcReact.compute.updateImage.useMutation({\n onSuccess: (updatedImage) => {\n utils.compute.getImageById.setData({ project_id: projectId, imageId }, updatedImage)\n utils.compute.listImagesWithPagination.invalidate()\n },\n })\n\n const deleteImageMutation = trpcReact.compute.deleteImage.useMutation({\n onSuccess: () => {\n utils.compute.listImagesWithPagination.invalidate()\n },\n })\n\n const deactivateImageMutation = trpcReact.compute.deactivateImage.useMutation({\n onSuccess: () => {\n utils.compute.getImageById.invalidate({ project_id: projectId, imageId })\n },\n })\n\n const reactivateImageMutation = trpcReact.compute.reactivateImage.useMutation({\n onSuccess: () => {\n utils.compute.getImageById.invalidate({ project_id: projectId, imageId })\n },\n })\n\n const updateImageVisibilityMutation = trpcReact.compute.updateImageVisibility.useMutation({\n onSuccess: (updatedImage) => {\n utils.compute.getImageById.setData({ project_id: projectId, imageId }, updatedImage)\n },\n })\n\n const isSharedWithMe =\n image?.visibility === IMAGE_VISIBILITY.SHARED && image?.owner !== undefined && image?.owner !== projectId\n\n const { data: myMemberData } = trpcReact.compute.getImageMember.useQuery(\n { project_id: projectId, imageId: imageId, memberId: projectId },\n { enabled: isSharedWithMe && !!imageId && !!projectId }\n )\n\n const updateMemberMutation = trpcReact.compute.updateImageMember.useMutation({\n onSuccess: () => {\n utils.compute.getImageMember.invalidate({ project_id: projectId, imageId: imageId, memberId: projectId })\n utils.compute.listImageMembers.invalidate({ project_id: projectId, imageId: imageId })\n utils.compute.listImagesWithPagination.invalidate()\n utils.compute.listSharedImagesByMemberStatus.invalidate()\n },\n })\n\n const handleMemberStatusChange = async (newStatus: MemberStatus) => {\n try {\n await updateMemberMutation.mutateAsync({ project_id: projectId, imageId, memberId: projectId, status: newStatus })\n setToastData(getImageAccessStatusUpdatedToast(newStatus, { onDismiss: () => setToastData(null) }))\n } catch (error) {\n const errorMessage = (error as TRPCClientError<InferrableClientTypes>)?.message\n setToastData(getImageAccessStatusErrorToast(errorMessage, { onDismiss: () => setToastData(null) }))\n }\n }\n\n const handleBack = () => {\n navigate({\n to: \"/projects/$projectId/compute/images\",\n params: { projectId },\n })\n }\n\n const isLoading =\n updateImageMutation.isPending ||\n deleteImageMutation.isPending ||\n deactivateImageMutation.isPending ||\n reactivateImageMutation.isPending ||\n updateImageVisibilityMutation.isPending\n\n const convertToJsonPatchOperations = (\n updatedProperties: Partial<GlanceImage>,\n originalImage: GlanceImage\n ): Array<{ op: \"add\" | \"replace\" | \"remove\"; path: string; value?: unknown }> => {\n const operations: Array<{ op: \"add\" | \"replace\" | \"remove\"; path: string; value?: unknown }> = []\n Object.entries(updatedProperties).forEach(([key, value]) => {\n const path = `/${key}`\n if (value === null || value === undefined) {\n if (key in originalImage) operations.push({ op: \"remove\", path })\n } else {\n const propertyExists = key in originalImage\n operations.push({ op: propertyExists ? \"replace\" : \"add\", path, value })\n }\n })\n return operations\n }\n\n const handleSaveEdit = async (updatedProperties: Partial<GlanceImage>): Promise<boolean> => {\n if (!image) return false\n const operations = convertToJsonPatchOperations(updatedProperties, image)\n try {\n await updateImageMutation.mutateAsync({ project_id: projectId, imageId, operations })\n setEditDetailsModalOpen(false)\n return true\n } catch {\n return false\n }\n }\n\n const handleDelete = async (deletedImage: GlanceImage) => {\n try {\n await deleteImageMutation.mutateAsync({ project_id: projectId, imageId: deletedImage.id })\n setDeleteModalOpen(false)\n handleBack()\n } catch {\n setDeleteModalOpen(false)\n }\n }\n\n const handleActivate = async (img: GlanceImage) => {\n try {\n await reactivateImageMutation.mutateAsync({ project_id: projectId, imageId: img.id })\n setActivateModalOpen(false)\n } catch {\n setActivateModalOpen(false)\n }\n }\n\n const handleDeactivate = async (img: GlanceImage) => {\n try {\n await deactivateImageMutation.mutateAsync({ project_id: projectId, imageId: img.id })\n setDeactivateModalOpen(false)\n } catch {\n setDeactivateModalOpen(false)\n }\n }\n\n const handleUpdateVisibility = async (newVisibility: \"public\" | \"private\" | \"shared\" | \"community\") => {\n if (!image) return\n try {\n await updateImageVisibilityMutation.mutateAsync({\n project_id: projectId,\n imageId: image.id,\n visibility: newVisibility,\n })\n } catch {\n // error handled by mutation state\n }\n }\n\n // Handle loading state\n if (status === \"pending\") {\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 Image Details...</Trans>\n </Stack>\n )\n }\n\n // Handle error state\n if (status === \"error\") {\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 image</Trans>\n </p>\n <p className=\"text-theme-highest\">{errorMessage}</p>\n <Button onClick={handleBack} variant=\"primary\">\n <Trans>Back to Images</Trans>\n </Button>\n </Stack>\n )\n }\n\n // Handle no data state\n if (!image) {\n return (\n <Stack className=\"fixed inset-0\" distribution=\"center\" alignment=\"center\" direction=\"vertical\" gap=\"5\">\n <p className=\"text-theme-highest\">\n <Trans>Image not found</Trans>\n </p>\n <Button onClick={handleBack} variant=\"primary\">\n <Trans>Back to Images</Trans>\n </Button>\n </Stack>\n )\n }\n\n const isDeactivated = image.status === IMAGE_STATUSES.DEACTIVATED\n const isPrivate = image.visibility === IMAGE_VISIBILITY.PRIVATE\n const hasMoreActions = !isSharedWithMe && (permissions.canUpdate || (permissions.canDelete && !image.protected))\n\n const headerActions =\n !isSharedWithMe && (hasMoreActions || permissions.canUpdate) ? (\n <ButtonRow>\n {hasMoreActions && (\n <PopupMenu>\n <PopupMenuToggle as=\"div\">\n <Button icon=\"moreVert\" disabled={isLoading}>\n <Trans>More Actions</Trans>\n </Button>\n </PopupMenuToggle>\n <PopupMenuOptions>\n {permissions.canUpdate && (\n <PopupMenuItem\n label={isDeactivated ? t`Activate` : t`Deactivate`}\n onClick={() => (isDeactivated ? setActivateModalOpen(true) : setDeactivateModalOpen(true))}\n />\n )}\n {permissions.canUpdate && isPrivate && (\n <PopupMenuItem label={t`Set to \"Shared\"`} onClick={() => handleUpdateVisibility(\"shared\")} />\n )}\n {permissions.canDelete && !image.protected && (\n <PopupMenuItem label={t`Delete`} onClick={() => setDeleteModalOpen(true)} />\n )}\n </PopupMenuOptions>\n </PopupMenu>\n )}\n {permissions.canUpdate && (\n <Button onClick={() => setEditMetadataModalOpen(true)} disabled={isLoading}>\n <Trans>Edit Metadata</Trans>\n </Button>\n )}\n {permissions.canUpdate && (\n <Button onClick={() => setEditDetailsModalOpen(true)} variant=\"primary\" disabled={isLoading}>\n <Trans>Edit Details</Trans>\n </Button>\n )}\n </ButtonRow>\n ) : undefined\n\n // Render success state\n return (\n <>\n <ContentHeader title={String(image.name ?? image.id)} projectId={projectId} actions={headerActions} />\n <ImageDetailsView\n key={image.id}\n image={image}\n currentProjectId={projectId}\n activeTab={tab ?? \"details\"}\n onTabChange={(newTab) =>\n navigate({\n search: { tab: newTab === \"details\" ? undefined : newTab } as unknown as true,\n })\n }\n permissions={{\n canCreateMember: permissions.canCreateMember,\n canDeleteMember: permissions.canDeleteMember,\n canUpdateMember: permissions.canUpdateMember,\n }}\n myMemberData={myMemberData}\n onMemberStatusChange={handleMemberStatusChange}\n isMemberStatusChanging={updateMemberMutation.isPending}\n />\n\n {toastData && <Toast {...toastData} />}\n\n {editDetailsModalOpen && (\n <EditImageDetailsModal\n image={image}\n isOpen={editDetailsModalOpen}\n isLoading={updateImageMutation.isPending}\n onClose={() => setEditDetailsModalOpen(false)}\n onSave={handleSaveEdit}\n />\n )}\n\n {editMetadataModalOpen && (\n <EditImageMetadataModal\n image={image}\n isOpen={editMetadataModalOpen}\n isLoading={updateImageMutation.isPending}\n onClose={() => setEditMetadataModalOpen(false)}\n onSave={handleSaveEdit}\n />\n )}\n\n {deleteModalOpen && (\n <DeleteImageModal\n image={image}\n isOpen={deleteModalOpen}\n isLoading={deleteImageMutation.isPending}\n isDisabled={!!image.protected}\n onClose={() => setDeleteModalOpen(false)}\n onDelete={handleDelete}\n />\n )}\n\n {activateModalOpen && (\n <ActivateImageModal\n image={image}\n isOpen={activateModalOpen}\n isLoading={reactivateImageMutation.isPending}\n onClose={() => setActivateModalOpen(false)}\n onActivate={handleActivate}\n />\n )}\n\n {deactivateModalOpen && (\n <DeactivateImageModal\n image={image}\n isOpen={deactivateModalOpen}\n isLoading={deactivateImageMutation.isPending}\n onClose={() => setDeactivateModalOpen(false)}\n onDeactivate={handleDeactivate}\n />\n )}\n </>\n )\n}\n"],"mappings":"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"}
1
+ {"version":3,"file":"_imageId-BL0I5_pv.mjs","names":["React","useState","DescriptionList","DescriptionTerm","DescriptionDefinition","Container","ContentHeading","Stack","Message","Box","Button","ButtonRow","SizeDisplay","trpcReact","MEMBER_STATUSES","ImageMembersTable","SharedImageBox","image","myMemberData","canUpdateMember","onStatusChange","isLoading","useLingui","isPending","status","PENDING","isRejected","REJECTED","isAccepted","ACCEPTED","sharedAt","created_at","Date","toLocaleString","t","updatedAt","updated_at","ownerProject","owner","p","className","span","ul","li","onClick","disabled","variant","GeneralImageData","px","py","alignTerms","id","name","size","min_disk","min_ram","disk_format","container_format","toLocaleDateString","SecuritySection","currentProjectId","isSharedWithMe","visibility","undefined","protected","checksum","CustomPropertiesSection","knownFields","Set","customProperties","Object","entries","filter","key","has","sort","a","b","localeCompare","hasProperties","length","map","value","Fragment","JSON","stringify","String","getTabClassName","active","base","SharingDetailsTab","permissions","isAddingMember","setIsAddingMember","message","setMessage","data","imageMembers","isMembersLoading","compute","listImageMembers","useQuery","project_id","imageId","enabled","text","type","onDismiss","canAdd","canCreateMember","canRemove","canDeleteMember","projectId","ImageDetailsView","activeTab","onTabChange","onMemberStatusChange","isMemberStatusChanging","actions","isImageOwner","showTabs","direction","gap","div","button","Button","ButtonRow","Stack","Spinner","PopupMenu","PopupMenuToggle","PopupMenuOptions","PopupMenuItem","Toast","ToastProps","useNavigate","useParams","useSearch","Trans","useLingui","trpcReact","ImageDetailsView","EditImageDetailsModal","EditImageMetadataModal","DeleteImageModal","ActivateImageModal","DeactivateImageModal","IMAGE_STATUSES","IMAGE_VISIBILITY","GlanceImage","MemberStatus","TRPCClientError","InferrableClientTypes","getImageAccessStatusUpdatedToast","getImageAccessStatusErrorToast","useState","ContentHeader","RouteComponent","projectId","imageId","from","tab","navigate","t","data","image","status","error","compute","getImageById","useQuery","project_id","permissionsData","canUser","permission","permissions","canDelete","canUpdate","canCreateMember","canDeleteMember","canUpdateMember","utils","useUtils","editDetailsModalOpen","setEditDetailsModalOpen","editMetadataModalOpen","setEditMetadataModalOpen","deleteModalOpen","setDeleteModalOpen","activateModalOpen","setActivateModalOpen","deactivateModalOpen","setDeactivateModalOpen","toastData","setToastData","updateImageMutation","updateImage","useMutation","onSuccess","updatedImage","setData","listImagesWithPagination","invalidate","deleteImageMutation","deleteImage","deactivateImageMutation","deactivateImage","reactivateImageMutation","reactivateImage","updateImageVisibilityMutation","updateImageVisibility","isSharedWithMe","visibility","SHARED","owner","undefined","myMemberData","getImageMember","memberId","enabled","updateMemberMutation","updateImageMember","listImageMembers","listSharedImagesByMemberStatus","handleMemberStatusChange","newStatus","mutateAsync","onDismiss","errorMessage","message","handleBack","to","params","isLoading","isPending","convertToJsonPatchOperations","updatedProperties","Partial","originalImage","Array","op","path","value","operations","Object","entries","forEach","key","push","propertyExists","handleSaveEdit","Promise","handleDelete","deletedImage","id","handleActivate","img","handleDeactivate","handleUpdateVisibility","newVisibility","isDeactivated","DEACTIVATED","isPrivate","PRIVATE","hasMoreActions","protected","headerActions","String","name","newTab","search","component"],"sources":["../../src/client/routes/_auth/projects/$projectId/compute/-components/Images/-components/ImageDetailsView.tsx","../../src/client/routes/_auth/projects/$projectId/compute/images/$imageId.tsx?tsr-split=component"],"sourcesContent":["import React, { useState } from \"react\"\nimport {\n DescriptionList,\n DescriptionTerm,\n DescriptionDefinition,\n Container,\n ContentHeading,\n Stack,\n Message,\n Box,\n Button,\n ButtonRow,\n} from \"@cloudoperators/juno-ui-components\"\nimport { Trans, useLingui } from \"@lingui/react/macro\"\nimport { GlanceImage, ImageMember, MemberStatus } from \"@/server/Compute/types/image\"\nimport { SizeDisplay } from \"./SizeDisplay\"\nimport { trpcReact } from \"@/client/trpcClient\"\nimport { MEMBER_STATUSES } from \"../../../-constants/filters\"\nimport { ImageMembersTable } from \"./ImageMembersTable\"\n\ninterface ImageDetailsViewProps {\n image: GlanceImage\n currentProjectId?: string\n activeTab?: \"details\" | \"sharing\"\n onTabChange?: (tab: \"details\" | \"sharing\") => void\n permissions?: {\n canCreateMember: boolean\n canDeleteMember: boolean\n canUpdateMember: boolean\n }\n myMemberData?: ImageMember\n onMemberStatusChange?: (status: MemberStatus) => void\n isMemberStatusChanging?: boolean\n actions?: React.ReactNode\n}\n\nconst SharedImageBox: React.FC<{\n image: GlanceImage\n myMemberData: ImageMember\n canUpdateMember: boolean\n onStatusChange: (status: MemberStatus) => void\n isLoading: boolean\n}> = ({ image, myMemberData, canUpdateMember, onStatusChange, isLoading }) => {\n const { t } = useLingui()\n const isPending = myMemberData.status === MEMBER_STATUSES.PENDING\n const isRejected = myMemberData.status === MEMBER_STATUSES.REJECTED\n const isAccepted = myMemberData.status === MEMBER_STATUSES.ACCEPTED\n\n const sharedAt = myMemberData.created_at ? new Date(myMemberData.created_at).toLocaleString() : t`N/A`\n const updatedAt = myMemberData.updated_at ? new Date(myMemberData.updated_at).toLocaleString() : t`N/A`\n const ownerProject = image.owner ?? \"\"\n\n return (\n <Box>\n {isPending && (\n <p className=\"text-theme-highest font-semibold\">\n <Trans>Your action is required</Trans>\n </p>\n )}\n <p>\n <Trans>\n This image was shared with you by <span className=\"font-semibold\">{ownerProject}</span> on {sharedAt}.\n </Trans>\n </p>\n <ul>\n <li>\n <span className=\"font-semibold\">\n <Trans>Access Status:</Trans>\n </span>{\" \"}\n {myMemberData.status}\n </li>\n <li>\n <span className=\"font-semibold\">\n <Trans>Shared:</Trans>\n </span>{\" \"}\n {sharedAt}\n </li>\n <li>\n <span className=\"font-semibold\">\n <Trans>Updated:</Trans>\n </span>{\" \"}\n {updatedAt}\n </li>\n </ul>\n\n {canUpdateMember && (\n <ButtonRow>\n {isPending && (\n <Button onClick={() => onStatusChange(MEMBER_STATUSES.REJECTED)} disabled={isLoading} variant=\"subdued\">\n <Trans>Reject</Trans>\n </Button>\n )}\n {(isPending || isRejected) && (\n <Button onClick={() => onStatusChange(MEMBER_STATUSES.ACCEPTED)} disabled={isLoading} variant=\"primary\">\n <Trans>Accept</Trans>\n </Button>\n )}\n {isAccepted && (\n <Button\n onClick={() => onStatusChange(MEMBER_STATUSES.REJECTED)}\n disabled={isLoading}\n variant=\"primary-danger\"\n >\n <Trans>Reject</Trans>\n </Button>\n )}\n </ButtonRow>\n )}\n </Box>\n )\n}\n\nexport const GeneralImageData: React.FC<{ image: GlanceImage }> = ({ image }) => {\n const { t } = useLingui()\n\n return (\n <Container px={false} py>\n <ContentHeading>{t`General Image Data`}</ContentHeading>\n <DescriptionList alignTerms=\"right\">\n <DescriptionTerm>{t`ID`}</DescriptionTerm>\n <DescriptionDefinition>{image.id}</DescriptionDefinition>\n\n <DescriptionTerm>{t`Name`}</DescriptionTerm>\n <DescriptionDefinition>{image.name}</DescriptionDefinition>\n\n <DescriptionTerm>{t`Status`}</DescriptionTerm>\n <DescriptionDefinition>{image.status}</DescriptionDefinition>\n\n <DescriptionTerm>{t`Size`}</DescriptionTerm>\n <DescriptionDefinition>\n <SizeDisplay size={image.size} />\n </DescriptionDefinition>\n\n <DescriptionTerm>{t`Min. Disk`}</DescriptionTerm>\n <DescriptionDefinition>{image.min_disk} GB</DescriptionDefinition>\n\n <DescriptionTerm>{t`Min. RAM`}</DescriptionTerm>\n <DescriptionDefinition>{image.min_ram} MB</DescriptionDefinition>\n\n <DescriptionTerm>{t`Disk Format`}</DescriptionTerm>\n <DescriptionDefinition>\n <span className=\"uppercase\">{image.disk_format}</span>\n </DescriptionDefinition>\n\n <DescriptionTerm>{t`Container Format`}</DescriptionTerm>\n <DescriptionDefinition>\n <span className=\"uppercase\">{image.container_format}</span>\n </DescriptionDefinition>\n\n <DescriptionTerm>{t`Created At`}</DescriptionTerm>\n <DescriptionDefinition>\n {image.created_at ? new Date(image.created_at).toLocaleDateString() : t`N/A`}\n </DescriptionDefinition>\n\n <DescriptionTerm>{t`Updated At`}</DescriptionTerm>\n <DescriptionDefinition>\n {image.updated_at ? new Date(image.updated_at).toLocaleDateString() : t`N/A`}\n </DescriptionDefinition>\n </DescriptionList>\n </Container>\n )\n}\n\nexport const SecuritySection: React.FC<{ image: GlanceImage; currentProjectId?: string }> = ({\n image,\n currentProjectId,\n}) => {\n const { t } = useLingui()\n\n const isSharedWithMe = image.visibility === \"shared\" && image.owner !== undefined && image.owner !== currentProjectId\n\n return (\n <Container px={false} py>\n <ContentHeading>{t`Security`}</ContentHeading>\n <DescriptionList alignTerms=\"right\">\n <DescriptionTerm>{isSharedWithMe ? t`Shared by Project` : t`Owner Project ID`}</DescriptionTerm>\n <DescriptionDefinition>{image.owner}</DescriptionDefinition>\n\n <DescriptionTerm>{t`Visibility`}</DescriptionTerm>\n <DescriptionDefinition>{image.visibility}</DescriptionDefinition>\n\n <DescriptionTerm>{t`Protected`}</DescriptionTerm>\n <DescriptionDefinition>{image.protected ? t`Yes` : t`No`}</DescriptionDefinition>\n\n <DescriptionTerm>{t`Checksum`}</DescriptionTerm>\n <DescriptionDefinition>{image?.checksum ? image.checksum : \"\"}</DescriptionDefinition>\n </DescriptionList>\n </Container>\n )\n}\n\nexport const CustomPropertiesSection: React.FC<{ image: GlanceImage }> = ({ image }) => {\n const { t } = useLingui()\n\n const knownFields = new Set([\n \"id\",\n \"name\",\n \"status\",\n \"visibility\",\n \"size\",\n \"disk_format\",\n \"container_format\",\n \"min_disk\",\n \"min_ram\",\n \"owner\",\n \"protected\",\n \"created_at\",\n \"updated_at\",\n \"checksum\",\n ])\n\n const customProperties = Object.entries(image)\n .filter(([key]) => !knownFields.has(key))\n .sort(([a], [b]) => a.localeCompare(b))\n\n const hasProperties = customProperties.length > 0\n\n return (\n <Container px={false} py>\n <ContentHeading>{t`Custom Properties / Metadata`}</ContentHeading>\n {hasProperties ? (\n <DescriptionList alignTerms=\"right\" className=\"grid-cols-4\">\n {customProperties.map(([key, value]) => (\n <React.Fragment key={key}>\n <DescriptionTerm className=\"col-span-1\">{key}</DescriptionTerm>\n <DescriptionDefinition className=\"col-span-1\">\n {value === null || value === undefined ? (\n <span>null</span>\n ) : typeof value === \"object\" ? (\n <span className=\"break-all\">{JSON.stringify(value)}</span>\n ) : typeof value === \"boolean\" ? (\n value ? (\n t`True`\n ) : (\n t`False`\n )\n ) : (\n <span className=\"break-all\">{String(value)}</span>\n )}\n </DescriptionDefinition>\n </React.Fragment>\n ))}\n </DescriptionList>\n ) : (\n <p className=\"text-theme-light\">{t`No custom properties defined`}</p>\n )}\n </Container>\n )\n}\n\nconst getTabClassName = (active: boolean) => {\n const base = \"px-6 py-3 font-semibold border-b-2 transition-colors\"\n return active\n ? `${base} border-theme-accent text-theme-highest`\n : `${base} border-transparent text-theme-secondary hover:text-theme-high`\n}\n\nconst SharingDetailsTab: React.FC<ImageDetailsViewProps> = ({ image, permissions, currentProjectId }) => {\n const [isAddingMember, setIsAddingMember] = useState(false)\n const [message, setMessage] = useState<{ text: string; type: \"error\" | \"info\" } | null>(null)\n\n const { data: imageMembers, isLoading: isMembersLoading } = trpcReact.compute.listImageMembers.useQuery(\n { project_id: currentProjectId!, imageId: image.id },\n { enabled: !!image.id && !!currentProjectId }\n )\n\n return (\n <Container px={false} py>\n {message && (\n <Message text={message.text} variant={message.type} onDismiss={() => setMessage(null)} className=\"mb-4\" />\n )}\n <ImageMembersTable\n image={image}\n imageMembers={imageMembers}\n isMembersLoading={isMembersLoading}\n canAdd={permissions?.canCreateMember ?? false}\n canRemove={permissions?.canDeleteMember ?? false}\n isAddingMember={isAddingMember}\n setIsAddingMember={setIsAddingMember}\n setMessage={setMessage}\n projectId={currentProjectId!}\n />\n </Container>\n )\n}\n\nexport const ImageDetailsView: React.FC<ImageDetailsViewProps> = ({\n image,\n currentProjectId,\n activeTab = \"details\",\n onTabChange,\n permissions,\n myMemberData,\n onMemberStatusChange,\n isMemberStatusChanging,\n actions,\n}) => {\n const { t } = useLingui()\n\n const isSharedWithMe = image.visibility === \"shared\" && image.owner !== undefined && image.owner !== currentProjectId\n const isImageOwner = image.owner === currentProjectId\n const showTabs = isImageOwner && image.visibility === \"shared\"\n\n return (\n <Stack direction=\"vertical\" gap=\"6\">\n {isSharedWithMe && myMemberData && onMemberStatusChange && (\n <SharedImageBox\n image={image}\n myMemberData={myMemberData}\n canUpdateMember={permissions?.canUpdateMember ?? false}\n onStatusChange={onMemberStatusChange}\n isLoading={isMemberStatusChanging ?? false}\n />\n )}\n\n {showTabs && (\n <div className=\"border-theme-background-lvl-3 border-b\">\n <Stack direction=\"horizontal\" gap=\"0\">\n <button className={getTabClassName(activeTab === \"details\")} onClick={() => onTabChange?.(\"details\")}>\n {t`Details`}\n </button>\n <button className={getTabClassName(activeTab === \"sharing\")} onClick={() => onTabChange?.(\"sharing\")}>\n {t`Sharing Details`}\n </button>\n </Stack>\n </div>\n )}\n\n {(activeTab === \"details\" || !showTabs) && (\n <>\n {actions}\n <GeneralImageData image={image} />\n <SecuritySection image={image} currentProjectId={currentProjectId} />\n <CustomPropertiesSection image={image} />\n </>\n )}\n\n {activeTab === \"sharing\" && showTabs && <SharingDetailsTab image={image} permissions={permissions} />}\n </Stack>\n )\n}\n","import {\n Button,\n ButtonRow,\n Stack,\n Spinner,\n PopupMenu,\n PopupMenuToggle,\n PopupMenuOptions,\n PopupMenuItem,\n Toast,\n ToastProps,\n} from \"@cloudoperators/juno-ui-components/index\"\nimport { createFileRoute, redirect, useNavigate, useParams, useSearch } from \"@tanstack/react-router\"\nimport { z } from \"zod\"\nimport type { RouteInfo } from \"@/client/routes/routeInfo\"\nimport { Trans, useLingui } from \"@lingui/react/macro\"\nimport { getServiceIndex } from \"@/server/Authentication/helpers\"\nimport { trpcReact } from \"@/client/trpcClient\"\nimport { ImageDetailsView } from \"../-components/Images/-components/ImageDetailsView\"\nimport { EditImageDetailsModal } from \"../-components/Images/-components/EditImageDetailsModal\"\nimport { EditImageMetadataModal } from \"../-components/Images/-components/EditImageMetadataModal\"\nimport { DeleteImageModal } from \"../-components/Images/-components/DeleteImageModal\"\nimport { ActivateImageModal } from \"../-components/Images/-components/ActivateImageModal\"\nimport { DeactivateImageModal } from \"../-components/Images/-components/DeactivateImageModal\"\nimport { IMAGE_STATUSES, IMAGE_VISIBILITY } from \"../-constants/filters\"\nimport { GlanceImage, MemberStatus } from \"@/server/Compute/types/image\"\nimport { TRPCClientError } from \"@trpc/client\"\nimport { InferrableClientTypes } from \"@trpc/server/unstable-core-do-not-import\"\nimport {\n getImageAccessStatusUpdatedToast,\n getImageAccessStatusErrorToast,\n} from \"../-components/Images/-components/ImageToastNotifications\"\nimport { useState } from \"react\"\nimport { ContentHeader } from \"@/client/components/ContentHeader/ContentHeader\"\n\nexport const Route = createFileRoute(\"/_auth/projects/$projectId/compute/images/$imageId\")({\n staticData: {\n section: \"compute\",\n service: \"images\",\n isDetail: true,\n sectionCrumb: { labelKey: \"Compute\" },\n crumb: { labelKey: \"Images\", to: \"/projects/$projectId/compute/images\" },\n } satisfies RouteInfo,\n validateSearch: z.object({\n tab: z.enum([\"details\", \"sharing\"]).optional(),\n }),\n loader: async ({ context, params }) => {\n try {\n const image = await context.trpcClient?.compute.getImageById.query({\n project_id: params.projectId,\n imageId: params.imageId,\n })\n return { imageTitle: (image?.name as string | undefined) ?? image?.id ?? null }\n } catch {\n return { imageTitle: null }\n }\n },\n head: ({ loaderData }) => ({\n meta: [{ title: loaderData?.imageTitle ?? \"Image Details\" }],\n }),\n component: RouteComponent,\n beforeLoad: async ({ context, params }) => {\n const { trpcClient } = context\n const { projectId } = params\n\n const availableServices = (await trpcClient?.auth.getAvailableServices.query()) || []\n\n const serviceIndex = getServiceIndex(availableServices)\n\n // Redirect to the \"Projects Overview\" page if none of compute services available\n if (!serviceIndex[\"image\"] && !serviceIndex[\"compute\"]) {\n throw redirect({\n to: \"/projects/$projectId\",\n params: { projectId },\n })\n }\n\n if (!serviceIndex[\"image\"][\"glance\"]) {\n // Redirect to the \"Compute Services Overview\" page if the \"Glance\" service is not available\n throw redirect({\n to: \"/projects/$projectId\",\n params: { projectId: params.projectId },\n })\n }\n },\n})\n\nfunction RouteComponent() {\n const { projectId, imageId } = useParams({\n from: \"/_auth/projects/$projectId/compute/images/$imageId\",\n })\n const { tab } = useSearch({\n from: \"/_auth/projects/$projectId/compute/images/$imageId\",\n })\n\n const navigate = useNavigate()\n const { t } = useLingui()\n\n const {\n data: image,\n status,\n error,\n } = trpcReact.compute.getImageById.useQuery({ project_id: projectId, imageId: imageId })\n\n const { data: permissionsData } = trpcReact.compute.canUser.useQuery({\n project_id: projectId,\n permission: [\n \"images:delete\",\n \"images:update\",\n \"images:create_member\",\n \"images:delete_member\",\n \"images:update_member\",\n ],\n })\n\n const permissions = {\n canDelete: permissionsData?.[0] ?? false,\n canUpdate: permissionsData?.[1] ?? false,\n canCreateMember: permissionsData?.[2] ?? false,\n canDeleteMember: permissionsData?.[3] ?? false,\n canUpdateMember: permissionsData?.[4] ?? false,\n }\n\n const utils = trpcReact.useUtils()\n\n const [editDetailsModalOpen, setEditDetailsModalOpen] = useState(false)\n const [editMetadataModalOpen, setEditMetadataModalOpen] = useState(false)\n const [deleteModalOpen, setDeleteModalOpen] = useState(false)\n const [activateModalOpen, setActivateModalOpen] = useState(false)\n const [deactivateModalOpen, setDeactivateModalOpen] = useState(false)\n const [toastData, setToastData] = useState<ToastProps | null>(null)\n\n const updateImageMutation = trpcReact.compute.updateImage.useMutation({\n onSuccess: (updatedImage) => {\n utils.compute.getImageById.setData({ project_id: projectId, imageId }, updatedImage)\n utils.compute.listImagesWithPagination.invalidate()\n },\n })\n\n const deleteImageMutation = trpcReact.compute.deleteImage.useMutation({\n onSuccess: () => {\n utils.compute.listImagesWithPagination.invalidate()\n },\n })\n\n const deactivateImageMutation = trpcReact.compute.deactivateImage.useMutation({\n onSuccess: () => {\n utils.compute.getImageById.invalidate({ project_id: projectId, imageId })\n },\n })\n\n const reactivateImageMutation = trpcReact.compute.reactivateImage.useMutation({\n onSuccess: () => {\n utils.compute.getImageById.invalidate({ project_id: projectId, imageId })\n },\n })\n\n const updateImageVisibilityMutation = trpcReact.compute.updateImageVisibility.useMutation({\n onSuccess: (updatedImage) => {\n utils.compute.getImageById.setData({ project_id: projectId, imageId }, updatedImage)\n },\n })\n\n const isSharedWithMe =\n image?.visibility === IMAGE_VISIBILITY.SHARED && image?.owner !== undefined && image?.owner !== projectId\n\n const { data: myMemberData } = trpcReact.compute.getImageMember.useQuery(\n { project_id: projectId, imageId: imageId, memberId: projectId },\n { enabled: isSharedWithMe && !!imageId && !!projectId }\n )\n\n const updateMemberMutation = trpcReact.compute.updateImageMember.useMutation({\n onSuccess: () => {\n utils.compute.getImageMember.invalidate({ project_id: projectId, imageId: imageId, memberId: projectId })\n utils.compute.listImageMembers.invalidate({ project_id: projectId, imageId: imageId })\n utils.compute.listImagesWithPagination.invalidate()\n utils.compute.listSharedImagesByMemberStatus.invalidate()\n },\n })\n\n const handleMemberStatusChange = async (newStatus: MemberStatus) => {\n try {\n await updateMemberMutation.mutateAsync({ project_id: projectId, imageId, memberId: projectId, status: newStatus })\n setToastData(getImageAccessStatusUpdatedToast(newStatus, { onDismiss: () => setToastData(null) }))\n } catch (error) {\n const errorMessage = (error as TRPCClientError<InferrableClientTypes>)?.message\n setToastData(getImageAccessStatusErrorToast(errorMessage, { onDismiss: () => setToastData(null) }))\n }\n }\n\n const handleBack = () => {\n navigate({\n to: \"/projects/$projectId/compute/images\",\n params: { projectId },\n })\n }\n\n const isLoading =\n updateImageMutation.isPending ||\n deleteImageMutation.isPending ||\n deactivateImageMutation.isPending ||\n reactivateImageMutation.isPending ||\n updateImageVisibilityMutation.isPending\n\n const convertToJsonPatchOperations = (\n updatedProperties: Partial<GlanceImage>,\n originalImage: GlanceImage\n ): Array<{ op: \"add\" | \"replace\" | \"remove\"; path: string; value?: unknown }> => {\n const operations: Array<{ op: \"add\" | \"replace\" | \"remove\"; path: string; value?: unknown }> = []\n Object.entries(updatedProperties).forEach(([key, value]) => {\n const path = `/${key}`\n if (value === null || value === undefined) {\n if (key in originalImage) operations.push({ op: \"remove\", path })\n } else {\n const propertyExists = key in originalImage\n operations.push({ op: propertyExists ? \"replace\" : \"add\", path, value })\n }\n })\n return operations\n }\n\n const handleSaveEdit = async (updatedProperties: Partial<GlanceImage>): Promise<boolean> => {\n if (!image) return false\n const operations = convertToJsonPatchOperations(updatedProperties, image)\n try {\n await updateImageMutation.mutateAsync({ project_id: projectId, imageId, operations })\n setEditDetailsModalOpen(false)\n return true\n } catch {\n return false\n }\n }\n\n const handleDelete = async (deletedImage: GlanceImage) => {\n try {\n await deleteImageMutation.mutateAsync({ project_id: projectId, imageId: deletedImage.id })\n setDeleteModalOpen(false)\n handleBack()\n } catch {\n setDeleteModalOpen(false)\n }\n }\n\n const handleActivate = async (img: GlanceImage) => {\n try {\n await reactivateImageMutation.mutateAsync({ project_id: projectId, imageId: img.id })\n setActivateModalOpen(false)\n } catch {\n setActivateModalOpen(false)\n }\n }\n\n const handleDeactivate = async (img: GlanceImage) => {\n try {\n await deactivateImageMutation.mutateAsync({ project_id: projectId, imageId: img.id })\n setDeactivateModalOpen(false)\n } catch {\n setDeactivateModalOpen(false)\n }\n }\n\n const handleUpdateVisibility = async (newVisibility: \"public\" | \"private\" | \"shared\" | \"community\") => {\n if (!image) return\n try {\n await updateImageVisibilityMutation.mutateAsync({\n project_id: projectId,\n imageId: image.id,\n visibility: newVisibility,\n })\n } catch {\n // error handled by mutation state\n }\n }\n\n // Handle loading state\n if (status === \"pending\") {\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 Image Details...</Trans>\n </Stack>\n )\n }\n\n // Handle error state\n if (status === \"error\") {\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 image</Trans>\n </p>\n <p className=\"text-theme-highest\">{errorMessage}</p>\n <Button onClick={handleBack} variant=\"primary\">\n <Trans>Back to Images</Trans>\n </Button>\n </Stack>\n )\n }\n\n // Handle no data state\n if (!image) {\n return (\n <Stack className=\"fixed inset-0\" distribution=\"center\" alignment=\"center\" direction=\"vertical\" gap=\"5\">\n <p className=\"text-theme-highest\">\n <Trans>Image not found</Trans>\n </p>\n <Button onClick={handleBack} variant=\"primary\">\n <Trans>Back to Images</Trans>\n </Button>\n </Stack>\n )\n }\n\n const isDeactivated = image.status === IMAGE_STATUSES.DEACTIVATED\n const isPrivate = image.visibility === IMAGE_VISIBILITY.PRIVATE\n const hasMoreActions = !isSharedWithMe && (permissions.canUpdate || (permissions.canDelete && !image.protected))\n\n const headerActions =\n !isSharedWithMe && (hasMoreActions || permissions.canUpdate) ? (\n <ButtonRow>\n {hasMoreActions && (\n <PopupMenu>\n <PopupMenuToggle as=\"div\">\n <Button icon=\"moreVert\" disabled={isLoading}>\n <Trans>More Actions</Trans>\n </Button>\n </PopupMenuToggle>\n <PopupMenuOptions>\n {permissions.canUpdate && (\n <PopupMenuItem\n label={isDeactivated ? t`Activate` : t`Deactivate`}\n onClick={() => (isDeactivated ? setActivateModalOpen(true) : setDeactivateModalOpen(true))}\n />\n )}\n {permissions.canUpdate && isPrivate && (\n <PopupMenuItem label={t`Set to \"Shared\"`} onClick={() => handleUpdateVisibility(\"shared\")} />\n )}\n {permissions.canDelete && !image.protected && (\n <PopupMenuItem label={t`Delete`} onClick={() => setDeleteModalOpen(true)} />\n )}\n </PopupMenuOptions>\n </PopupMenu>\n )}\n {permissions.canUpdate && (\n <Button onClick={() => setEditMetadataModalOpen(true)} disabled={isLoading}>\n <Trans>Edit Metadata</Trans>\n </Button>\n )}\n {permissions.canUpdate && (\n <Button onClick={() => setEditDetailsModalOpen(true)} variant=\"primary\" disabled={isLoading}>\n <Trans>Edit Details</Trans>\n </Button>\n )}\n </ButtonRow>\n ) : undefined\n\n // Render success state\n return (\n <>\n <ContentHeader title={String(image.name ?? image.id)} projectId={projectId} actions={headerActions} />\n <ImageDetailsView\n key={image.id}\n image={image}\n currentProjectId={projectId}\n activeTab={tab ?? \"details\"}\n onTabChange={(newTab) =>\n navigate({\n search: { tab: newTab === \"details\" ? undefined : newTab } as unknown as true,\n })\n }\n permissions={{\n canCreateMember: permissions.canCreateMember,\n canDeleteMember: permissions.canDeleteMember,\n canUpdateMember: permissions.canUpdateMember,\n }}\n myMemberData={myMemberData}\n onMemberStatusChange={handleMemberStatusChange}\n isMemberStatusChanging={updateMemberMutation.isPending}\n />\n\n {toastData && <Toast {...toastData} />}\n\n {editDetailsModalOpen && (\n <EditImageDetailsModal\n image={image}\n isOpen={editDetailsModalOpen}\n isLoading={updateImageMutation.isPending}\n onClose={() => setEditDetailsModalOpen(false)}\n onSave={handleSaveEdit}\n />\n )}\n\n {editMetadataModalOpen && (\n <EditImageMetadataModal\n image={image}\n isOpen={editMetadataModalOpen}\n isLoading={updateImageMutation.isPending}\n onClose={() => setEditMetadataModalOpen(false)}\n onSave={handleSaveEdit}\n />\n )}\n\n {deleteModalOpen && (\n <DeleteImageModal\n image={image}\n isOpen={deleteModalOpen}\n isLoading={deleteImageMutation.isPending}\n isDisabled={!!image.protected}\n onClose={() => setDeleteModalOpen(false)}\n onDelete={handleDelete}\n />\n )}\n\n {activateModalOpen && (\n <ActivateImageModal\n image={image}\n isOpen={activateModalOpen}\n isLoading={reactivateImageMutation.isPending}\n onClose={() => setActivateModalOpen(false)}\n onActivate={handleActivate}\n />\n )}\n\n {deactivateModalOpen && (\n <DeactivateImageModal\n image={image}\n isOpen={deactivateModalOpen}\n isLoading={deactivateImageMutation.isPending}\n onClose={() => setDeactivateModalOpen(false)}\n onDeactivate={handleDeactivate}\n />\n )}\n </>\n )\n}\n"],"mappings":"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"}
@@ -19,7 +19,7 @@ var i = t("/_auth/projects/$projectId/services/pca/$pcaId/")({
19
19
  return { pcaTitle: n?.configuration?.subject?.common_name || n?.id || null };
20
20
  },
21
21
  head: ({ loaderData: e }) => ({ meta: [{ title: e?.pcaTitle ?? "Certificate Authority" }] }),
22
- component: n(() => import("./_pcaId-CwlH1Kvl.mjs"), "component"),
22
+ component: n(() => import("./_pcaId-CbBhBrX1.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.pca?.["clavis-beta"] || a.pca?.["clavis-dev"])) throw r({
@@ -31,4 +31,4 @@ var i = t("/_auth/projects/$projectId/services/pca/$pcaId/")({
31
31
  //#endregion
32
32
  export { i as t };
33
33
 
34
- //# sourceMappingURL=_pcaId-D1ZEaCdp.mjs.map
34
+ //# sourceMappingURL=_pcaId-BYCoeK6_.mjs.map
@@ -1 +1 @@
1
- {"version":3,"file":"_pcaId-D1ZEaCdp.mjs","names":["createFileRoute","redirect","getServiceIndex","Route","staticData","section","service","isDetail","sectionCrumb","labelKey","crumb","to","RouteInfo","loader","context","params","pca","trpcClient","services","getById","query","project_id","projectId","certificate_authority_id","pcaId","pcaTitle","configuration","subject","common_name","id","head","loaderData","meta","title","component","lazyRouteComponent","$$splitComponentImporter","beforeLoad","availableServices","auth","getAvailableServices","serviceIndex","pcaServices"],"sources":["../../src/client/routes/_auth/projects/$projectId/services/pca/$pcaId/index.tsx"],"sourcesContent":["import { createFileRoute, redirect, useNavigate } from \"@tanstack/react-router\"\nimport { Trans } from \"@lingui/react/macro\"\nimport { Button, Spinner, Stack } from \"@cloudoperators/juno-ui-components/index\"\nimport { getServiceIndex } from \"@/server/Authentication/helpers\"\nimport type { RouteInfo } from \"@/client/routes/routeInfo\"\nimport { trpcReact } from \"@/client/trpcClient\"\nimport { useProjectId } from \"@/client/hooks/useProjectId\"\nimport { PcaDetailsView } from \"./-components/PcaDetailsView\"\n\nexport const Route = createFileRoute(\"/_auth/projects/$projectId/services/pca/$pcaId/\")({\n staticData: {\n section: \"services\",\n service: \"pca\",\n isDetail: true,\n sectionCrumb: { labelKey: \"Services\" },\n crumb: { labelKey: \"PCA (Clavis)\", to: \"/projects/$projectId/services/pca\" },\n } satisfies RouteInfo,\n loader: async ({ context, params }) => {\n const pca = await context.trpcClient?.services.pca.getById.query({\n project_id: params.projectId,\n certificate_authority_id: params.pcaId,\n })\n return { pcaTitle: pca?.configuration?.subject?.common_name || pca?.id || null }\n },\n head: ({ loaderData }) => ({\n meta: [{ title: loaderData?.pcaTitle ?? \"Certificate Authority\" }],\n }),\n component: RouteComponent,\n beforeLoad: async ({ context, params }) => {\n const { trpcClient } = context\n const availableServices = (await trpcClient?.auth.getAvailableServices.query()) || []\n const serviceIndex = getServiceIndex(availableServices)\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\n // Redirect if clavis service not available\n if (!pcaServices) {\n throw redirect({\n to: \"/projects/$projectId/services/pca\",\n params: { projectId: params.projectId },\n })\n }\n },\n})\n\nfunction RouteComponent() {\n const navigate = useNavigate()\n const projectId = useProjectId()\n const { pcaId } = Route.useParams()\n\n const {\n isLoading,\n isError,\n error,\n data: pca,\n } = trpcReact.services.pca.getById.useQuery({\n project_id: projectId,\n certificate_authority_id: pcaId,\n })\n\n // 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 Certificate Authority Details...</Trans>\n </Stack>\n )\n }\n\n const handleBack = () =>\n navigate({\n to: \"/projects/$projectId/services/pca\",\n params: { projectId },\n })\n\n // Error state\n if (isError) {\n const errorMessage = error?.message || \"Unknown error\"\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 Certificate Authority</Trans>\n </p>\n <p className=\"text-theme-highest\">{errorMessage}</p>\n <Button onClick={handleBack} variant=\"primary\">\n <Trans>Back to Certificate Authorities</Trans>\n </Button>\n </Stack>\n )\n }\n\n // No data state\n if (!pca) {\n return (\n <Stack className=\"fixed inset-0\" distribution=\"center\" alignment=\"center\" direction=\"vertical\" gap=\"5\">\n <p className=\"text-theme-secondary\">\n <Trans>Certificate Authority not found</Trans>\n </p>\n <Button onClick={handleBack} variant=\"primary\">\n <Trans>Back to Certificate Authorities</Trans>\n </Button>\n </Stack>\n )\n }\n\n return <PcaDetailsView pca={pca} />\n}\n"],"mappings":";;AASA,IAAaG,IAAQH,EAAgB,kDAAA,CAAmD;CACtFI,YAAY;EACVC,SAAS;EACTC,SAAS;EACTC,UAAU;EACVC,cAAc,EAAEC,UAAU,YAAW;EACrCC,OAAO;GAAED,UAAU;GAAgBE,IAAI;GAAoC;EAC7E;CACAE,QAAQ,OAAO,EAAEC,YAASC,gBAAQ;EAChC,IAAMC,IAAM,MAAMF,EAAQG,YAAYC,SAASF,IAAIG,QAAQC,MAAM;GAC/DC,YAAYN,EAAOO;GACnBC,0BAA0BR,EAAOS;GACnC,CAAA;AACA,SAAO,EAAEC,UAAUT,GAAKU,eAAeC,SAASC,eAAeZ,GAAKa,MAAM,MAAK;;CAEjFC,OAAO,EAAEC,qBAAkB,EACzBC,MAAM,CAAC,EAAEC,OAAOF,GAAYN,YAAY,yBAAwB,CAAC,EACnE;CACAS,WAASC,yCAAA,YAAA;CACTE,YAAY,OAAO,EAAEvB,YAASC,gBAAQ;EACpC,IAAM,EAAEE,kBAAeH,GAEjB2B,IAAevC,EADK,MAAOe,GAAYsB,KAAKC,qBAAqBpB,OAAAA,IAAY,EAAE,CAChDkB;AAKrC,MAAI,EAHgBG,EAAa,MAAS,kBAAkBA,EAAa,MAAS,eAIhF,OAAMxC,EAAS;GACbU,IAAI;GACJI,QAAQ,EAAEO,WAAWP,EAAOO,WAAU;GACxC,CAAA;;CAGN,CAAA"}
1
+ {"version":3,"file":"_pcaId-BYCoeK6_.mjs","names":["createFileRoute","redirect","getServiceIndex","Route","staticData","section","service","isDetail","sectionCrumb","labelKey","crumb","to","RouteInfo","loader","context","params","pca","trpcClient","services","getById","query","project_id","projectId","certificate_authority_id","pcaId","pcaTitle","configuration","subject","common_name","id","head","loaderData","meta","title","component","lazyRouteComponent","$$splitComponentImporter","beforeLoad","availableServices","auth","getAvailableServices","serviceIndex","pcaServices"],"sources":["../../src/client/routes/_auth/projects/$projectId/services/pca/$pcaId/index.tsx"],"sourcesContent":["import { createFileRoute, redirect, useNavigate } from \"@tanstack/react-router\"\nimport { Trans } from \"@lingui/react/macro\"\nimport { Button, Spinner, Stack } from \"@cloudoperators/juno-ui-components/index\"\nimport { getServiceIndex } from \"@/server/Authentication/helpers\"\nimport type { RouteInfo } from \"@/client/routes/routeInfo\"\nimport { trpcReact } from \"@/client/trpcClient\"\nimport { useProjectId } from \"@/client/hooks/useProjectId\"\nimport { PcaDetailsView } from \"./-components/PcaDetailsView\"\n\nexport const Route = createFileRoute(\"/_auth/projects/$projectId/services/pca/$pcaId/\")({\n staticData: {\n section: \"services\",\n service: \"pca\",\n isDetail: true,\n sectionCrumb: { labelKey: \"Services\" },\n crumb: { labelKey: \"PCA (Clavis)\", to: \"/projects/$projectId/services/pca\" },\n } satisfies RouteInfo,\n loader: async ({ context, params }) => {\n const pca = await context.trpcClient?.services.pca.getById.query({\n project_id: params.projectId,\n certificate_authority_id: params.pcaId,\n })\n return { pcaTitle: pca?.configuration?.subject?.common_name || pca?.id || null }\n },\n head: ({ loaderData }) => ({\n meta: [{ title: loaderData?.pcaTitle ?? \"Certificate Authority\" }],\n }),\n component: RouteComponent,\n beforeLoad: async ({ context, params }) => {\n const { trpcClient } = context\n const availableServices = (await trpcClient?.auth.getAvailableServices.query()) || []\n const serviceIndex = getServiceIndex(availableServices)\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\n // Redirect if clavis service not available\n if (!pcaServices) {\n throw redirect({\n to: \"/projects/$projectId/services/pca\",\n params: { projectId: params.projectId },\n })\n }\n },\n})\n\nfunction RouteComponent() {\n const navigate = useNavigate()\n const projectId = useProjectId()\n const { pcaId } = Route.useParams()\n\n const {\n isLoading,\n isError,\n error,\n data: pca,\n } = trpcReact.services.pca.getById.useQuery({\n project_id: projectId,\n certificate_authority_id: pcaId,\n })\n\n // 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 Certificate Authority Details...</Trans>\n </Stack>\n )\n }\n\n const handleBack = () =>\n navigate({\n to: \"/projects/$projectId/services/pca\",\n params: { projectId },\n })\n\n // Error state\n if (isError) {\n const errorMessage = error?.message || \"Unknown error\"\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 Certificate Authority</Trans>\n </p>\n <p className=\"text-theme-highest\">{errorMessage}</p>\n <Button onClick={handleBack} variant=\"primary\">\n <Trans>Back to Certificate Authorities</Trans>\n </Button>\n </Stack>\n )\n }\n\n // No data state\n if (!pca) {\n return (\n <Stack className=\"fixed inset-0\" distribution=\"center\" alignment=\"center\" direction=\"vertical\" gap=\"5\">\n <p className=\"text-theme-secondary\">\n <Trans>Certificate Authority not found</Trans>\n </p>\n <Button onClick={handleBack} variant=\"primary\">\n <Trans>Back to Certificate Authorities</Trans>\n </Button>\n </Stack>\n )\n }\n\n return <PcaDetailsView pca={pca} />\n}\n"],"mappings":";;AASA,IAAaG,IAAQH,EAAgB,kDAAA,CAAmD;CACtFI,YAAY;EACVC,SAAS;EACTC,SAAS;EACTC,UAAU;EACVC,cAAc,EAAEC,UAAU,YAAW;EACrCC,OAAO;GAAED,UAAU;GAAgBE,IAAI;GAAoC;EAC7E;CACAE,QAAQ,OAAO,EAAEC,YAASC,gBAAQ;EAChC,IAAMC,IAAM,MAAMF,EAAQG,YAAYC,SAASF,IAAIG,QAAQC,MAAM;GAC/DC,YAAYN,EAAOO;GACnBC,0BAA0BR,EAAOS;GACnC,CAAA;AACA,SAAO,EAAEC,UAAUT,GAAKU,eAAeC,SAASC,eAAeZ,GAAKa,MAAM,MAAK;;CAEjFC,OAAO,EAAEC,qBAAkB,EACzBC,MAAM,CAAC,EAAEC,OAAOF,GAAYN,YAAY,yBAAwB,CAAC,EACnE;CACAS,WAASC,yCAAA,YAAA;CACTE,YAAY,OAAO,EAAEvB,YAASC,gBAAQ;EACpC,IAAM,EAAEE,kBAAeH,GAEjB2B,IAAevC,EADK,MAAOe,GAAYsB,KAAKC,qBAAqBpB,OAAAA,IAAY,EAAE,CAChDkB;AAKrC,MAAI,EAHgBG,EAAa,MAAS,kBAAkBA,EAAa,MAAS,eAIhF,OAAMxC,EAAS;GACbU,IAAI;GACJI,QAAQ,EAAEO,WAAWP,EAAOO,WAAU;GACxC,CAAA;;CAGN,CAAA"}