@rancher/shell 3.0.12-rc.2 → 3.0.12-rc.4

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 (473) hide show
  1. package/apis/impl/apis.ts +6 -0
  2. package/apis/index.ts +26 -0
  3. package/apis/intf/resources-api/cluster-api.ts +18 -0
  4. package/apis/intf/resources-api/mgmt-api.ts +15 -0
  5. package/apis/intf/resources-api/resource-base.ts +107 -0
  6. package/apis/intf/resources-api/resource-constants.ts +147 -0
  7. package/apis/intf/resources-api/resources-api.ts +143 -0
  8. package/apis/intf/resources.ts +49 -0
  9. package/apis/intf/{modal.ts → shell-api/modal.ts} +21 -26
  10. package/apis/intf/shell-api/proxy.ts +216 -0
  11. package/apis/intf/{slide-in.ts → shell-api/slide-in.ts} +4 -3
  12. package/apis/intf/{system.ts → shell-api/system.ts} +4 -1
  13. package/apis/intf/shell.ts +12 -6
  14. package/apis/resources/__tests__/resources-api-class.test.ts +550 -0
  15. package/apis/resources/index.ts +22 -0
  16. package/apis/resources/resources-api-class.ts +187 -0
  17. package/apis/shell/__tests__/proxy.test.ts +369 -0
  18. package/apis/shell/index.ts +8 -1
  19. package/apis/shell/modal.ts +4 -1
  20. package/apis/shell/notifications.ts +9 -6
  21. package/apis/shell/proxy.ts +256 -0
  22. package/apis/shell/slide-in.ts +4 -1
  23. package/apis/vue-shim.d.ts +2 -1
  24. package/assets/data/aws-regions.json +4 -0
  25. package/assets/fonts/lato/LatoLatin-Black.woff +0 -0
  26. package/assets/fonts/lato/LatoLatin-Black.woff2 +0 -0
  27. package/assets/fonts/lato/LatoLatin-BlackItalic.woff +0 -0
  28. package/assets/fonts/lato/LatoLatin-BlackItalic.woff2 +0 -0
  29. package/assets/fonts/lato/LatoLatin-Bold.woff +0 -0
  30. package/assets/fonts/lato/LatoLatin-Bold.woff2 +0 -0
  31. package/assets/fonts/lato/LatoLatin-BoldItalic.woff +0 -0
  32. package/assets/fonts/lato/LatoLatin-BoldItalic.woff2 +0 -0
  33. package/assets/fonts/lato/LatoLatin-Heavy.woff +0 -0
  34. package/assets/fonts/lato/LatoLatin-Heavy.woff2 +0 -0
  35. package/assets/fonts/lato/LatoLatin-HeavyItalic.woff +0 -0
  36. package/assets/fonts/lato/LatoLatin-HeavyItalic.woff2 +0 -0
  37. package/assets/fonts/lato/LatoLatin-Italic.woff +0 -0
  38. package/assets/fonts/lato/LatoLatin-Italic.woff2 +0 -0
  39. package/assets/fonts/lato/LatoLatin-Light.woff +0 -0
  40. package/assets/fonts/lato/LatoLatin-Light.woff2 +0 -0
  41. package/assets/fonts/lato/LatoLatin-LightItalic.woff +0 -0
  42. package/assets/fonts/lato/LatoLatin-LightItalic.woff2 +0 -0
  43. package/assets/fonts/lato/LatoLatin-Medium.woff +0 -0
  44. package/assets/fonts/lato/LatoLatin-Medium.woff2 +0 -0
  45. package/assets/fonts/lato/LatoLatin-MediumItalic.woff +0 -0
  46. package/assets/fonts/lato/LatoLatin-MediumItalic.woff2 +0 -0
  47. package/assets/fonts/lato/LatoLatin-Regular.woff +0 -0
  48. package/assets/fonts/lato/LatoLatin-Regular.woff2 +0 -0
  49. package/assets/fonts/lato/LatoLatin-Semibold.woff +0 -0
  50. package/assets/fonts/lato/LatoLatin-Semibold.woff2 +0 -0
  51. package/assets/fonts/lato/LatoLatin-SemiboldItalic.woff +0 -0
  52. package/assets/fonts/lato/LatoLatin-SemiboldItalic.woff2 +0 -0
  53. package/assets/styles/base/_variables.scss +2 -0
  54. package/assets/styles/fonts/_fontstack.scss +132 -8
  55. package/assets/styles/global/_layout.scss +4 -0
  56. package/assets/translations/en-us.yaml +165 -45
  57. package/assets/translations/zh-hans.yaml +1 -7
  58. package/chart/monitoring/ClusterSelector.vue +0 -21
  59. package/chart/monitoring/index.vue +10 -1
  60. package/chart/monitoring/prometheus/index.vue +6 -3
  61. package/components/ActionDropdownShell.vue +2 -1
  62. package/components/CruResource.vue +161 -14
  63. package/components/CruResourceFooter.vue +9 -5
  64. package/components/ExplorerMembers.vue +8 -4
  65. package/components/ExplorerProjectsNamespaces.vue +11 -7
  66. package/components/GrowlManager.vue +4 -0
  67. package/components/InstallHelmCharts.vue +2 -2
  68. package/components/LandingPagePreference.vue +14 -5
  69. package/components/MgmtNodeList.vue +184 -0
  70. package/components/Resource/Detail/Card/StateCard/__tests__/composables.test.ts +90 -1
  71. package/components/Resource/Detail/Card/StateCard/composables.ts +57 -87
  72. package/components/Resource/Detail/Card/StatusCard/__tests__/StatusCard.test.ts +61 -0
  73. package/components/Resource/Detail/Card/StatusCard/index.vue +61 -15
  74. package/components/Resource/Detail/Metadata/IdentifyingInformation/index.vue +17 -1
  75. package/components/Resource/Detail/Metadata/KeyValue.vue +5 -2
  76. package/components/Resource/Detail/Metadata/KeyValueRow.vue +2 -6
  77. package/components/Resource/Detail/Metadata/index.vue +6 -0
  78. package/components/Resource/Detail/ResourcePopover/index.vue +12 -1
  79. package/components/Resource/Detail/SpacedRow.vue +3 -1
  80. package/components/Resource/Detail/TitleBar/index.vue +10 -11
  81. package/components/ResourceDetail/index.vue +1 -1
  82. package/components/ResourceList/Masthead.vue +19 -9
  83. package/components/ResourceList/index.vue +82 -1
  84. package/components/RichTranslation.vue +5 -2
  85. package/components/SelectIconGrid.vue +0 -10
  86. package/components/Setting.vue +1 -0
  87. package/components/SingleClusterInfo.vue +1 -0
  88. package/components/SortableTable/__tests__/sorting.test.ts +126 -0
  89. package/components/SortableTable/index.vue +6 -9
  90. package/components/SortableTable/selection.js +23 -5
  91. package/components/SortableTable/sorting.js +6 -3
  92. package/components/SubtleLink.vue +31 -6
  93. package/components/Tabbed/Tab.vue +29 -3
  94. package/components/Tabbed/index.vue +25 -3
  95. package/components/TableOfContents/TableOfContents.vue +109 -0
  96. package/components/TableOfContents/composables.ts +258 -0
  97. package/components/Window/ContainerShell.vue +21 -11
  98. package/components/Window/__tests__/ContainerShell.test.ts +107 -37
  99. package/components/Wizard.vue +23 -17
  100. package/components/fleet/AppCoChartGrid.vue +401 -0
  101. package/components/fleet/AppCoEmptyState.vue +127 -0
  102. package/components/fleet/AppCoPageHeader.vue +119 -0
  103. package/components/fleet/AppCoVersionSelect.vue +70 -0
  104. package/components/fleet/FleetBundles.vue +100 -12
  105. package/components/fleet/FleetClusterTargets/ClusterSelectionFields.vue +217 -0
  106. package/components/fleet/FleetClusterTargets/TargetsList.vue +123 -35
  107. package/components/fleet/FleetClusterTargets/index.vue +226 -161
  108. package/components/fleet/FleetIntro.vue +7 -3
  109. package/components/fleet/FleetNoWorkspaces.vue +7 -3
  110. package/components/fleet/FleetSecretSelector.vue +5 -3
  111. package/components/fleet/FleetValuesFrom.vue +8 -2
  112. package/components/fleet/GitRepoTargetTab.vue +0 -2
  113. package/components/fleet/HelmOpAdvancedTab.vue +19 -53
  114. package/components/fleet/HelmOpAppCoConfigTab.vue +593 -0
  115. package/components/fleet/HelmOpAppCoResourcesSection.vue +162 -0
  116. package/components/fleet/HelmOpResourcesSection.vue +82 -0
  117. package/components/fleet/HelmOpTargetOptionsSection.vue +89 -0
  118. package/components/fleet/HelmOpTargetTab.vue +64 -60
  119. package/components/fleet/HelmOpValuesTab.vue +129 -105
  120. package/components/fleet/__tests__/AppCoEmptyState.test.ts +71 -0
  121. package/components/fleet/__tests__/AppCoVersionSelect.test.ts +36 -0
  122. package/components/fleet/__tests__/ClusterSelectionFields.test.ts +62 -0
  123. package/components/fleet/__tests__/FleetClusterTargets.test.ts +402 -115
  124. package/components/fleet/__tests__/FleetClusters.test.ts +12 -12
  125. package/components/fleet/__tests__/FleetSecretSelector.test.ts +16 -0
  126. package/components/fleet/__tests__/FleetValuesFrom.test.ts +44 -0
  127. package/components/fleet/__tests__/HelmOpAppCoConfigTab.test.ts +59 -0
  128. package/components/fleet/__tests__/HelmOpAppCoResourcesSection.test.ts +62 -0
  129. package/components/fleet/__tests__/HelmOpResourcesSection.test.ts +43 -0
  130. package/components/fleet/__tests__/HelmOpTargetOptionsSection.test.ts +34 -0
  131. package/components/fleet/__tests__/HelmOpValuesTab.test.ts +39 -0
  132. package/components/fleet/__tests__/__snapshots__/AppCoEmptyState.test.ts.snap +97 -0
  133. package/components/fleet/__tests__/__snapshots__/AppCoVersionSelect.test.ts.snap +30 -0
  134. package/components/fleet/__tests__/__snapshots__/ClusterSelectionFields.test.ts.snap +209 -0
  135. package/components/fleet/__tests__/__snapshots__/HelmOpTargetOptionsSection.test.ts.snap +140 -0
  136. package/components/fleet/dashboard/Empty.vue +8 -4
  137. package/components/fleet/dashboard/ResourceCard.vue +28 -0
  138. package/components/fleet/dashboard/ResourceDetails.vue +28 -0
  139. package/components/fleet/dashboard/__tests__/ResourceCard.test.ts +87 -0
  140. package/components/form/ArrayList.vue +61 -4
  141. package/components/form/KeyValue.vue +23 -2
  142. package/components/form/LabeledSelect.vue +59 -4
  143. package/components/form/Labels.vue +22 -3
  144. package/components/form/NameNsDescription.vue +24 -5
  145. package/components/form/ResourceTabs/index.vue +1 -0
  146. package/components/form/Security.vue +6 -2
  147. package/components/form/WorkloadPorts.vue +2 -7
  148. package/components/form/__tests__/NameNsDescription.test.ts +75 -0
  149. package/components/form/__tests__/Security.test.ts +76 -0
  150. package/components/formatter/Autoscaler.vue +4 -4
  151. package/components/formatter/ClusterKubeVersion.vue +27 -0
  152. package/components/formatter/ClusterLink.vue +1 -7
  153. package/components/formatter/ClusterProvider.vue +6 -10
  154. package/components/formatter/FleetSummaryGraph.vue +0 -3
  155. package/components/formatter/InternalExternalIP.vue +10 -4
  156. package/components/formatter/MachineSummaryGraph.vue +1 -1
  157. package/components/formatter/PodsUsage.vue +2 -2
  158. package/components/formatter/ServiceTargets.vue +26 -7
  159. package/components/formatter/__tests__/Autoscaler.test.ts +19 -22
  160. package/components/formatter/__tests__/FleetSummaryGraph.test.ts +216 -0
  161. package/components/formatter/__tests__/InternalExternalIP.test.ts +132 -0
  162. package/components/formatter/__tests__/PodsUsage.test.ts +6 -10
  163. package/components/formatter/__tests__/ServiceTargets.test.ts +412 -0
  164. package/components/nav/Header.vue +4 -0
  165. package/components/nav/NamespaceFilter.vue +2 -2
  166. package/components/nav/TopLevelMenu.helper.ts +15 -3
  167. package/components/nav/TopLevelMenu.vue +22 -6
  168. package/components/nav/__tests__/Header.test.ts +15 -0
  169. package/components/nav/__tests__/TopLevelMenu.test.ts +263 -21
  170. package/components/templates/default.vue +9 -4
  171. package/components/templates/home.vue +23 -0
  172. package/components/templates/plain.vue +23 -0
  173. package/components/templates/standalone.vue +17 -0
  174. package/composables/useFormValidation.ts +93 -0
  175. package/composables/useHelmOpResources.test.ts +56 -0
  176. package/composables/useHelmOpResources.ts +32 -0
  177. package/composables/useStateColor.test.ts +325 -0
  178. package/composables/useStateColor.ts +128 -0
  179. package/composables/useVeeValidateField.test.ts +159 -0
  180. package/composables/useVeeValidateField.ts +67 -0
  181. package/config/home-links.js +1 -1
  182. package/config/labels-annotations.js +1 -0
  183. package/config/pagination-table-headers.js +18 -1
  184. package/config/product/explorer.js +17 -4
  185. package/config/product/manager.js +84 -21
  186. package/config/router/index.js +16 -0
  187. package/config/router/navigation-guards/__tests__/authentication.test.ts +130 -0
  188. package/config/router/navigation-guards/authentication.js +10 -4
  189. package/config/router/routes.js +26 -6
  190. package/config/settings.ts +0 -2
  191. package/config/table-headers.js +23 -5
  192. package/config/types.js +11 -1
  193. package/core/__tests__/plugin-products.test.ts +904 -20
  194. package/core/plugin-products-base.ts +110 -10
  195. package/core/plugin-products.ts +4 -0
  196. package/core/plugin-types.ts +194 -31
  197. package/core/plugin.ts +18 -7
  198. package/core/productDebugger.js +9 -4
  199. package/core/types-provisioning.ts +77 -31
  200. package/core/types.ts +72 -22
  201. package/detail/__tests__/pod.test.ts +41 -0
  202. package/detail/__tests__/provisioning.cattle.io.cluster.test.ts +114 -0
  203. package/detail/__tests__/workload.test.ts +3 -152
  204. package/detail/catalog.cattle.io.clusterrepo.vue +1 -1
  205. package/detail/harvesterhci.io.management.cluster.vue +6 -2
  206. package/detail/pod.vue +1 -1
  207. package/detail/provisioning.cattle.io.cluster.vue +34 -14
  208. package/detail/workload/index.vue +12 -55
  209. package/edit/__tests__/catalog.cattle.io.clusterrepo.test.ts +248 -0
  210. package/edit/__tests__/fleet.cattle.io.helmop.test.ts +105 -0
  211. package/edit/auditlog.cattle.io.auditpolicy/__tests__/__snapshots__/General.test.ts.snap +6 -0
  212. package/edit/auditlog.cattle.io.auditpolicy/__tests__/__snapshots__/index.test.ts.snap +1 -0
  213. package/edit/auth/__tests__/azuread.test.ts +247 -39
  214. package/edit/auth/__tests__/github.test.ts +234 -0
  215. package/edit/auth/__tests__/oidc.test.ts +26 -6
  216. package/edit/auth/__tests__/saml.test.ts +196 -0
  217. package/edit/auth/azuread.vue +197 -56
  218. package/edit/auth/github.vue +72 -13
  219. package/edit/auth/ldap/__tests__/index.test.ts +206 -0
  220. package/edit/auth/ldap/config.vue +8 -0
  221. package/edit/auth/ldap/index.vue +75 -1
  222. package/edit/auth/oidc.vue +119 -73
  223. package/edit/auth/saml.vue +76 -12
  224. package/edit/catalog.cattle.io.clusterrepo.vue +140 -32
  225. package/edit/fleet.cattle.io.helmop.vue +491 -136
  226. package/edit/management.cattle.io.user.vue +5 -2
  227. package/edit/networking.k8s.io.ingress/DefaultBackend.vue +13 -4
  228. package/edit/networking.k8s.io.ingress/RulePath.vue +8 -4
  229. package/edit/networking.k8s.io.ingress/index.vue +75 -20
  230. package/edit/provisioning.cattle.io.cluster/__tests__/MachinePool.test.ts +104 -0
  231. package/edit/provisioning.cattle.io.cluster/index.vue +11 -7
  232. package/edit/provisioning.cattle.io.cluster/rke2.vue +92 -14
  233. package/edit/provisioning.cattle.io.cluster/tabs/MachinePool.vue +22 -0
  234. package/edit/provisioning.cattle.io.cluster/tabs/registries/RegistryConfigs.vue +37 -4
  235. package/edit/provisioning.cattle.io.cluster/tabs/registries/__tests__/RegistryConfigs.test.ts +132 -7
  236. package/edit/provisioning.cattle.io.cluster/tabs/registries/index.vue +2 -1
  237. package/edit/secret/__tests__/ssh.test.ts +5 -6
  238. package/edit/secret/basic.vue +31 -0
  239. package/edit/secret/index.vue +68 -17
  240. package/edit/secret/registry.vue +38 -0
  241. package/edit/secret/ssh.vue +29 -0
  242. package/edit/secret/tls.vue +30 -0
  243. package/edit/service.vue +4 -4
  244. package/edit/workload/Upgrading.vue +3 -3
  245. package/edit/workload/__tests__/Upgrading.test.ts +6 -9
  246. package/edit/workload/mixins/workload.js +2 -1
  247. package/list/fleet.cattle.io.bundle.vue +7 -104
  248. package/list/fleet.cattle.io.clusterregistrationtoken.vue +20 -0
  249. package/list/group.principal.vue +5 -4
  250. package/list/harvesterhci.io.management.cluster.vue +8 -9
  251. package/list/management.cattle.io.user.vue +12 -9
  252. package/list/provisioning.cattle.io.cluster.vue +268 -180
  253. package/list/utils/management.cattle.io.cluster.utils.ts +128 -0
  254. package/mixins/__tests__/auth-config.test.ts +90 -0
  255. package/mixins/__tests__/chart.test.ts +206 -0
  256. package/mixins/__tests__/resource-fetch-api-pagination.test.ts +48 -0
  257. package/mixins/auth-config.js +7 -0
  258. package/mixins/brand.js +2 -1
  259. package/mixins/chart.js +22 -9
  260. package/mixins/child-hook.js +12 -6
  261. package/mixins/create-edit-view/impl.js +5 -3
  262. package/mixins/resource-fetch-api-pagination.js +62 -6
  263. package/models/__tests__/catalog.cattle.io.clusterrepo.test.ts +57 -0
  264. package/models/__tests__/compliance.cattle.io.clusterscan.test.ts +144 -0
  265. package/models/__tests__/ext.cattle.io.kubeconfig.test.ts +67 -67
  266. package/models/__tests__/fleet-application.test.ts +175 -0
  267. package/models/__tests__/fleet.cattle.io.bundle.test.ts +169 -0
  268. package/models/__tests__/fleet.cattle.io.helmop.test.ts +84 -0
  269. package/models/__tests__/management.cattle.io.cluster.test.ts +1 -1
  270. package/models/__tests__/management.cattle.io.node.ts +28 -5
  271. package/models/__tests__/management.cattle.io.nodepool.ts +5 -4
  272. package/models/__tests__/namespace.test.ts +36 -0
  273. package/models/__tests__/provisioning.cattle.io.cluster.test.ts +81 -11
  274. package/models/__tests__/workload.test.ts +401 -26
  275. package/models/base-cluster.x-k8s.io.js +26 -0
  276. package/models/catalog.cattle.io.clusterrepo.js +28 -4
  277. package/models/cluster.js +1 -1
  278. package/models/cluster.x-k8s.io.machine.js +4 -22
  279. package/models/cluster.x-k8s.io.machinedeployment.js +2 -20
  280. package/models/cluster.x-k8s.io.machineset.js +2 -20
  281. package/models/compliance.cattle.io.clusterscan.js +165 -2
  282. package/models/ext.cattle.io.kubeconfig.ts +4 -7
  283. package/models/fleet-application.js +7 -1
  284. package/models/fleet.cattle.io.helmop.js +20 -1
  285. package/models/management.cattle.io.cluster.js +434 -41
  286. package/models/management.cattle.io.node.js +50 -7
  287. package/models/management.cattle.io.nodepool.js +1 -1
  288. package/models/namespace.js +1 -1
  289. package/models/networking.k8s.io.ingress.js +12 -4
  290. package/models/pod.js +33 -1
  291. package/models/provisioning.cattle.io.cluster.js +51 -334
  292. package/models/rke.cattle.io.etcdsnapshot.js +1 -2
  293. package/models/workload.js +108 -13
  294. package/models/workload.service.js +5 -0
  295. package/package.json +22 -39
  296. package/pages/__tests__/readme.test.ts +49 -0
  297. package/pages/about.vue +5 -6
  298. package/pages/auth/login.vue +0 -35
  299. package/pages/auth/setup.vue +13 -3
  300. package/pages/c/_cluster/apps/charts/AppChartCardFooter.vue +2 -2
  301. package/pages/c/_cluster/apps/charts/AppChartCardSubHeader.vue +10 -1
  302. package/pages/c/_cluster/apps/charts/__tests__/chart.test.ts +76 -0
  303. package/pages/c/_cluster/apps/charts/__tests__/index.test.ts +93 -0
  304. package/pages/c/_cluster/apps/charts/chart.vue +62 -9
  305. package/pages/c/_cluster/apps/charts/index.vue +48 -10
  306. package/pages/c/_cluster/apps/charts/install.vue +122 -113
  307. package/pages/c/_cluster/auth/roles/index.vue +5 -4
  308. package/pages/c/_cluster/explorer/__tests__/index.test.ts +23 -25
  309. package/pages/c/_cluster/explorer/index.vue +5 -49
  310. package/pages/c/_cluster/explorer/workload-dashboard/ByNamespaceSection.vue +31 -0
  311. package/pages/c/_cluster/explorer/workload-dashboard/ByStateSection.vue +138 -0
  312. package/pages/c/_cluster/explorer/workload-dashboard/ByTypeSection.vue +30 -0
  313. package/pages/c/_cluster/explorer/workload-dashboard/WorkloadCard.vue +155 -0
  314. package/pages/c/_cluster/explorer/workload-dashboard/WorkloadNamespaceCard.vue +142 -0
  315. package/pages/c/_cluster/explorer/workload-dashboard/WorkloadTypeCard.vue +159 -0
  316. package/pages/c/_cluster/explorer/workload-dashboard/__tests__/composable.test.ts +561 -0
  317. package/pages/c/_cluster/explorer/workload-dashboard/composable.ts +440 -0
  318. package/pages/c/_cluster/explorer/workload-dashboard/index.vue +187 -0
  319. package/pages/c/_cluster/explorer/workload-dashboard/types.ts +80 -0
  320. package/pages/c/_cluster/fleet/application/create.vue +187 -136
  321. package/pages/c/_cluster/fleet/application/index.vue +5 -3
  322. package/pages/c/_cluster/fleet/application/suse-app-collection/ChartDetailBody.vue +338 -0
  323. package/pages/c/_cluster/fleet/application/suse-app-collection/ChartDetailHeader.vue +121 -0
  324. package/pages/c/_cluster/fleet/application/suse-app-collection/chart.vue +369 -0
  325. package/pages/c/_cluster/fleet/application/suse-app-collection/charts.vue +248 -0
  326. package/pages/c/_cluster/fleet/application/suse-app-collection/credentials.vue +310 -0
  327. package/pages/c/_cluster/fleet/index.vue +2 -2
  328. package/pages/c/_cluster/istio/__tests__/istio.index.test.ts +194 -0
  329. package/pages/c/_cluster/istio/index.vue +21 -6
  330. package/pages/c/_cluster/uiplugins/PluginInfoPanel.vue +1 -0
  331. package/pages/c/_cluster/uiplugins/__tests__/index.test.ts +815 -2
  332. package/pages/c/_cluster/uiplugins/index.vue +218 -197
  333. package/pages/diagnostic.vue +13 -17
  334. package/pages/fail-whale.vue +30 -7
  335. package/pages/home.vue +93 -306
  336. package/pages/readme.vue +88 -0
  337. package/plugins/clean-html.d.ts +9 -0
  338. package/plugins/dashboard-store/__tests__/resource-class.test.ts +181 -0
  339. package/plugins/dashboard-store/actions.js +40 -18
  340. package/plugins/dashboard-store/resource-class.js +67 -9
  341. package/plugins/steve/__tests__/actions.test.ts +212 -0
  342. package/plugins/steve/__tests__/subscribe.spec.ts +6 -3
  343. package/plugins/steve/actions.js +96 -0
  344. package/plugins/steve/steve-pagination-utils.ts +12 -4
  345. package/plugins/steve/subscribe.js +35 -5
  346. package/rancher-components/Accordion/Accordion.vue +53 -9
  347. package/rancher-components/Form/Checkbox/Checkbox.vue +14 -0
  348. package/rancher-components/Form/LabeledInput/LabeledInput.test.ts +10 -4
  349. package/rancher-components/Form/LabeledInput/LabeledInput.vue +7 -52
  350. package/rancher-components/Form/Radio/RadioButton.vue +17 -1
  351. package/rancher-components/Form/Radio/RadioGroup.vue +10 -0
  352. package/rancher-components/Pill/RcTag/RcTag.vue +3 -2
  353. package/rancher-components/RcButton/RcButton.test.ts +140 -1
  354. package/rancher-components/RcButton/RcButton.vue +126 -17
  355. package/rancher-components/RcButton/types.ts +3 -0
  356. package/rancher-components/RcDropdown/RcDropdownTrigger.vue +10 -8
  357. package/rancher-components/RcItemCard/RcItemCard.test.ts +18 -0
  358. package/rancher-components/RcItemCard/RcItemCard.vue +2 -2
  359. package/rancher-components/RcSection/RcSection.vue +28 -3
  360. package/scripts/extension/helm/package/Dockerfile +1 -1
  361. package/scripts/test-plugins-build.sh +2 -1
  362. package/store/__tests__/catalog.test.ts +115 -1
  363. package/store/__tests__/notifications.test.ts +434 -0
  364. package/store/__tests__/type-map.test.ts +556 -1
  365. package/store/action-menu.js +8 -3
  366. package/store/auth.js +1 -1
  367. package/store/aws.js +27 -16
  368. package/store/catalog.js +84 -3
  369. package/store/digitalocean.js +20 -38
  370. package/store/index.js +2 -0
  371. package/store/linode.js +25 -40
  372. package/store/plugins.js +7 -4
  373. package/store/pnap.js +1 -0
  374. package/store/type-map.js +111 -29
  375. package/tsconfig.paths.json +8 -8
  376. package/types/components/buttonGroup.ts +5 -0
  377. package/types/kube/kube-api.ts +14 -1
  378. package/types/rancher/steve.api.ts +12 -12
  379. package/types/resources/settings.d.ts +2 -1
  380. package/types/shell/index.d.ts +206 -72
  381. package/types/store/dashboard-store.types.ts +108 -11
  382. package/types/store/pagination.types.ts +6 -3
  383. package/utils/__tests__/alertmanagerconfig.test.ts +117 -0
  384. package/utils/__tests__/async.test.ts +87 -0
  385. package/utils/__tests__/auth.test.ts +273 -0
  386. package/utils/__tests__/aws.test.ts +140 -0
  387. package/utils/__tests__/banners.test.ts +176 -0
  388. package/utils/__tests__/chart.test.ts +64 -1
  389. package/utils/__tests__/color.test.ts +226 -0
  390. package/utils/__tests__/computed.test.ts +193 -0
  391. package/utils/__tests__/cspAdaptor.test.ts +163 -0
  392. package/utils/__tests__/dom.test.ts +81 -0
  393. package/utils/__tests__/duration.test.ts +176 -0
  394. package/utils/__tests__/dynamic-importer.test.ts +102 -0
  395. package/utils/__tests__/fleet-appco.test.ts +312 -0
  396. package/utils/__tests__/fleet.test.ts +340 -0
  397. package/utils/__tests__/ingress.test.ts +553 -0
  398. package/utils/__tests__/kube.test.ts +68 -0
  399. package/utils/__tests__/monitoring.test.ts +130 -0
  400. package/utils/__tests__/namespace-filter.test.ts +109 -0
  401. package/utils/__tests__/object.test.ts +22 -0
  402. package/utils/__tests__/pagination-utils.test.ts +361 -0
  403. package/utils/__tests__/parse-externalid.test.ts +137 -0
  404. package/utils/__tests__/perf-setting.utils.test.ts +98 -0
  405. package/utils/__tests__/platform.test.ts +91 -0
  406. package/utils/__tests__/poller-sequential.test.ts +177 -0
  407. package/utils/__tests__/poller.test.ts +170 -0
  408. package/utils/__tests__/position.test.ts +237 -0
  409. package/utils/__tests__/promise.test.ts +346 -0
  410. package/utils/__tests__/provider.test.ts +51 -1
  411. package/utils/__tests__/queue.test.ts +232 -0
  412. package/utils/__tests__/release-notes.test.ts +221 -0
  413. package/utils/__tests__/router.test.js +254 -1
  414. package/utils/__tests__/select.test.ts +208 -0
  415. package/utils/__tests__/settings.test.ts +140 -0
  416. package/utils/__tests__/sort-utils.test.ts +301 -0
  417. package/utils/__tests__/string-utils.test.ts +798 -0
  418. package/utils/__tests__/string.test.ts +23 -1
  419. package/utils/__tests__/style.test.ts +154 -0
  420. package/utils/__tests__/svg-filter.test.ts +184 -0
  421. package/utils/__tests__/time.test.ts +265 -1
  422. package/utils/__tests__/title.test.ts +47 -0
  423. package/utils/__tests__/units.test.ts +417 -0
  424. package/utils/__tests__/versions.test.ts +128 -0
  425. package/utils/__tests__/width.test.ts +53 -0
  426. package/utils/__tests__/window.test.ts +158 -0
  427. package/utils/__tests__/xccdf.test.ts +511 -0
  428. package/utils/chart.js +36 -0
  429. package/utils/crypto/__tests__/browserHashUtils.test.ts +98 -0
  430. package/utils/crypto/__tests__/index.test.ts +144 -0
  431. package/utils/duration.ts +104 -0
  432. package/utils/dynamic-content/__tests__/notification-handler.test.ts +196 -0
  433. package/utils/dynamic-content/info.ts +2 -1
  434. package/utils/error.js +13 -0
  435. package/utils/fleet-appco.ts +323 -0
  436. package/utils/fleet.ts +13 -3
  437. package/utils/gatekeeper/__tests__/util.test.ts +174 -0
  438. package/utils/gc/__tests__/gc-interval.test.ts +119 -0
  439. package/utils/gc/__tests__/gc-root-store.test.ts +225 -0
  440. package/utils/gc/__tests__/gc-route-changed.test.ts +96 -0
  441. package/utils/gc/__tests__/gc.test.ts +487 -0
  442. package/utils/ingress.ts +9 -1
  443. package/utils/object.js +22 -2
  444. package/utils/pagination-utils.ts +2 -1
  445. package/utils/provider.ts +12 -0
  446. package/utils/string.js +25 -2
  447. package/utils/uiplugins.ts +5 -5
  448. package/utils/validators/__tests__/cluster-name.test.ts +110 -0
  449. package/utils/validators/__tests__/container-images.test.ts +104 -0
  450. package/utils/validators/__tests__/cron-schedule.test.ts +79 -0
  451. package/utils/validators/__tests__/flow-output.test.ts +91 -0
  452. package/utils/validators/__tests__/index.test.ts +481 -0
  453. package/utils/validators/__tests__/kubernetes-name.test.ts +163 -0
  454. package/utils/validators/__tests__/logging-outputs.test.ts +58 -0
  455. package/utils/validators/__tests__/misc-validators.test.ts +246 -0
  456. package/utils/validators/__tests__/monitoring-route.test.ts +119 -0
  457. package/utils/validators/__tests__/pod-affinity.test.ts +382 -0
  458. package/utils/validators/__tests__/prometheusrule.test.ts +211 -0
  459. package/utils/validators/__tests__/role-template.test.ts +149 -0
  460. package/utils/validators/__tests__/service.test.ts +283 -0
  461. package/utils/validators/__tests__/setting.test.js +32 -0
  462. package/utils/validators/formRules/__tests__/index.test.ts +50 -0
  463. package/utils/validators/formRules/index.ts +5 -5
  464. package/utils/validators/machine-pool.ts +1 -1
  465. package/utils/validators/setting.js +18 -3
  466. package/utils/xccdf.ts +415 -0
  467. package/vue.config.js +1 -1
  468. package/assets/fonts/lato/lato-v17-latin-700.woff +0 -0
  469. package/assets/fonts/lato/lato-v17-latin-700.woff2 +0 -0
  470. package/assets/fonts/lato/lato-v17-latin-regular.woff +0 -0
  471. package/assets/fonts/lato/lato-v17-latin-regular.woff2 +0 -0
  472. package/pages/support/index.vue +0 -264
  473. package/utils/duration.js +0 -43
@@ -0,0 +1,159 @@
1
+ import {
2
+ computed, defineComponent, nextTick, provide, ref
3
+ } from 'vue';
4
+ import { mount, flushPromises } from '@vue/test-utils';
5
+ import { useVeeValidateField } from './useVeeValidateField';
6
+
7
+ function createHarness(opts: {
8
+ name?: string | null;
9
+ rules?: Array<(v: unknown) => string | undefined>;
10
+ value?: unknown;
11
+ validationMessage?: string;
12
+ showAllErrors?: boolean;
13
+ }) {
14
+ return defineComponent({
15
+ setup() {
16
+ const nameRef = ref(opts.name ?? null);
17
+ const rulesRef = ref(opts.rules ?? []);
18
+ const valueRef = ref(opts.value ?? '');
19
+ const validationMessageRef = computed(() => opts.validationMessage);
20
+
21
+ if (opts.showAllErrors) {
22
+ provide('vee-show-all-errors', ref(true));
23
+ }
24
+
25
+ const result = useVeeValidateField({
26
+ name: nameRef,
27
+ rules: rulesRef,
28
+ value: valueRef,
29
+ validationMessage: validationMessageRef,
30
+ });
31
+
32
+ return {
33
+ ...result,
34
+ nameRef,
35
+ valueRef,
36
+ };
37
+ },
38
+ template: '<div />',
39
+ });
40
+ }
41
+
42
+ describe('useVeeValidateField', () => {
43
+ describe('without a name prop', () => {
44
+ it('falls back to the passed validationMessage', async() => {
45
+ const errorMessage = 'This field is required';
46
+ const wrapper = mount(createHarness({
47
+ name: null,
48
+ validationMessage: errorMessage,
49
+ }));
50
+
51
+ await flushPromises();
52
+
53
+ expect(wrapper.vm.effectiveValidationMessage).toBe(errorMessage);
54
+ });
55
+
56
+ it('does not run rules through vee-validate', async() => {
57
+ const rule = jest.fn(() => 'error');
58
+ const wrapper = mount(createHarness({
59
+ name: null,
60
+ rules: [rule],
61
+ value: '',
62
+ }));
63
+
64
+ await flushPromises();
65
+
66
+ expect(wrapper.vm.effectiveValidationMessage).toBeUndefined();
67
+ });
68
+ });
69
+
70
+ describe('with a name prop', () => {
71
+ it('does not show error before the field is touched', async() => {
72
+ const errorMessage = 'Cannot be empty';
73
+ const wrapper = mount(createHarness({
74
+ name: 'testField',
75
+ rules: [(v) => (!v ? errorMessage : undefined)],
76
+ value: '',
77
+ }));
78
+
79
+ await flushPromises();
80
+
81
+ expect(wrapper.vm.effectiveValidationMessage).toBeUndefined();
82
+ });
83
+
84
+ it('shows vee-validate error after veeHandleBlur + veeValidate', async() => {
85
+ const errorMessage = 'Cannot be empty';
86
+ const wrapper = mount(createHarness({
87
+ name: 'testField',
88
+ rules: [(v) => (!v ? errorMessage : undefined)],
89
+ value: '',
90
+ }));
91
+
92
+ wrapper.vm.veeHandleBlur(undefined, false);
93
+ await wrapper.vm.veeValidate();
94
+ await flushPromises();
95
+
96
+ expect(wrapper.vm.effectiveValidationMessage).toBe(errorMessage);
97
+ });
98
+
99
+ it('shows error without touch when showAllErrors is injected as true', async() => {
100
+ const errorMessage = 'Cannot be empty';
101
+ const showAllErrors = ref(false);
102
+
103
+ const inner = defineComponent({
104
+ setup() {
105
+ const nameRef = ref('testField');
106
+ const rulesRef = ref([(v: unknown) => (!v ? errorMessage : undefined)]);
107
+ const valueRef = ref('');
108
+ const validationMessageRef = computed(() => undefined as string | undefined);
109
+
110
+ return useVeeValidateField({
111
+ name: nameRef,
112
+ rules: rulesRef,
113
+ value: valueRef,
114
+ validationMessage: validationMessageRef,
115
+ });
116
+ },
117
+ template: '<div />',
118
+ });
119
+
120
+ const outer = defineComponent({
121
+ components: { inner },
122
+ setup() {
123
+ provide('vee-show-all-errors', showAllErrors);
124
+ },
125
+ template: '<inner ref="innerRef" />',
126
+ });
127
+
128
+ const wrapper = mount(outer);
129
+ const innerVm = wrapper.getComponent(inner);
130
+
131
+ await flushPromises();
132
+ expect(innerVm.vm.effectiveValidationMessage).toBeUndefined();
133
+
134
+ showAllErrors.value = true;
135
+ await nextTick();
136
+
137
+ expect(innerVm.vm.effectiveValidationMessage).toBe(errorMessage);
138
+ });
139
+
140
+ it('clears vee-validate error when value becomes valid', async() => {
141
+ const errorMessage = 'Cannot be empty';
142
+ const wrapper = mount(createHarness({
143
+ name: 'testField',
144
+ rules: [(v) => (!v ? errorMessage : undefined)],
145
+ value: '',
146
+ }));
147
+
148
+ wrapper.vm.veeHandleBlur(undefined, false);
149
+ await wrapper.vm.veeValidate();
150
+ await flushPromises();
151
+ expect(wrapper.vm.effectiveValidationMessage).toBe(errorMessage);
152
+
153
+ wrapper.vm.valueRef = 'some value';
154
+ await flushPromises();
155
+
156
+ expect(wrapper.vm.effectiveValidationMessage).toBeUndefined();
157
+ });
158
+ });
159
+ });
@@ -0,0 +1,67 @@
1
+ import { computed, inject, ref, watch } from 'vue';
2
+ import type { Ref } from 'vue';
3
+ import { useField } from 'vee-validate';
4
+ import { generateRandomAlphaString } from '@shell/utils/string';
5
+
6
+ interface UseVeeValidateFieldOptions {
7
+ name: Ref<string | null | undefined>;
8
+ rules: Ref<Array<any>>;
9
+ value: Ref<unknown>;
10
+ validationMessage: Ref<unknown>;
11
+ }
12
+
13
+ export function useVeeValidateField({
14
+ name,
15
+ rules,
16
+ value,
17
+ validationMessage,
18
+ }: UseVeeValidateFieldOptions) {
19
+ const showAllErrors = inject<Ref<boolean>>('vee-show-all-errors', ref(false));
20
+ const standaloneFieldId = `__field__${ generateRandomAlphaString(12) }`;
21
+ const veeFieldName = computed(() => name.value || standaloneFieldId);
22
+
23
+ const veeValidator = (v: unknown): boolean | string => {
24
+ if (!name.value) return true;
25
+ for (const rule of rules.value as Array<(v: unknown) => string | undefined>) {
26
+ const msg = rule(v);
27
+
28
+ if (msg) return msg;
29
+ }
30
+
31
+ return true;
32
+ };
33
+
34
+ const {
35
+ errorMessage: veeError,
36
+ handleBlur: veeHandleBlur,
37
+ validate: veeValidate,
38
+ value: veeValue,
39
+ meta: veeMeta,
40
+ } = useField<unknown>(veeFieldName, veeValidator, {
41
+ initialValue: value.value,
42
+ validateOnValueUpdate: true,
43
+ validateOnMount: true,
44
+ });
45
+
46
+ // Keep vee-validate's internal value in sync with the controlled prop value.
47
+ watch(value, (v) => {
48
+ if (veeValue.value !== v) {
49
+ veeValue.value = v;
50
+ }
51
+ });
52
+
53
+ const effectiveValidationMessage = computed(() => {
54
+ if (name.value && veeError.value && (veeMeta.touched || showAllErrors.value)) {
55
+ return veeError.value;
56
+ }
57
+
58
+ return validationMessage.value;
59
+ });
60
+
61
+ return {
62
+ effectiveValidationMessage,
63
+ veeHandleBlur,
64
+ veeValidate,
65
+ showAllErrors,
66
+ };
67
+ }
@@ -44,7 +44,7 @@ const APP_COLLECTION_LINK = {
44
44
 
45
45
  const SUPPORT_LINK = {
46
46
  key: 'commercialSupport',
47
- value: '/support',
47
+ value: 'https://www.suse.com/products/rancher/',
48
48
  enabled: true,
49
49
  readonly: true
50
50
  };
@@ -19,6 +19,7 @@ export const NODE_ARCHITECTURE = 'kubernetes.io/arch';
19
19
  export const IMPORTED_CLUSTER_VERSION_MANAGEMENT = 'rancher.io/imported-cluster-version-management';
20
20
  export const UI_PROJECT_SECRET = 'management.cattle.io/project-scoped-secret';
21
21
  export const UI_PROJECT_SECRET_COPY = 'management.cattle.io/project-scoped-secret-copy';
22
+ export const SERVICE_LINKS = 'ui.rancher/service-links';
22
23
 
23
24
  export const KUBERNETES = {
24
25
  SERVICE_ACCOUNT_UID: 'kubernetes.io/service-account.uid',
@@ -5,7 +5,10 @@ import {
5
5
  EVENT_TYPE,
6
6
  SECRET_ORIGIN,
7
7
  EVENT_FIRST_SEEN_TIME,
8
- WORKLOAD_HEALTH_SCALE
8
+ WORKLOAD_HEALTH_SCALE,
9
+ MGMT_CLUSTER_PROVIDER,
10
+ MGMT_CLUSTER_KUBE_VERSION,
11
+ AUTOSCALER_ENABLED
9
12
  } from '@shell/config/table-headers';
10
13
 
11
14
  // This file contains table headers
@@ -102,3 +105,17 @@ export const STEVE_WORKLOAD_HEALTH_SCALE = {
102
105
  sort: false,
103
106
  search: false,
104
107
  };
108
+
109
+ export const STEVE_MGMT_CLUSTER_PROVIDER = { ...MGMT_CLUSTER_PROVIDER };
110
+
111
+ export const STEVE_MGMT_CLUSTER_KUBE_VERSION = {
112
+ ...MGMT_CLUSTER_KUBE_VERSION,
113
+ sort: 'status.info.kubernetesVersion',
114
+ search: 'status.info.kubernetesVersion',
115
+ };
116
+
117
+ export const STEVE_AUTOSCALER_ENABLED = {
118
+ ...AUTOSCALER_ENABLED,
119
+ sort: false,
120
+ search: false,
121
+ };
@@ -10,6 +10,7 @@ import {
10
10
  SNAPSHOT,
11
11
  VIRTUAL_TYPES,
12
12
  CAPI,
13
+ WORKLOAD_DASHBOARD,
13
14
  } from '@shell/config/types';
14
15
 
15
16
  import {
@@ -103,6 +104,7 @@ export function init(store) {
103
104
  CONFIG_MAP
104
105
  ], 'storage');
105
106
  basicType([
107
+ WORKLOAD_DASHBOARD,
106
108
  WORKLOAD,
107
109
  WORKLOAD_TYPES.DEPLOYMENT,
108
110
  WORKLOAD_TYPES.DAEMON_SET,
@@ -112,10 +114,6 @@ export function init(store) {
112
114
  POD,
113
115
  ], 'workload');
114
116
 
115
- setGroupDefaultType('workload', () => {
116
- return store.getters['features/get'](STEVE_CACHE) ? WORKLOAD_TYPES.DEPLOYMENT : undefined;
117
- });
118
-
119
117
  weightGroup('cluster', 99, true);
120
118
  weightGroup('workload', 98, true);
121
119
  weightGroup('serviceDiscovery', 96, true);
@@ -586,6 +584,21 @@ export function init(store) {
586
584
  overview: true,
587
585
  });
588
586
 
587
+ // Workload Dashboard - overview page using the Resource Summary API
588
+ virtualType({
589
+ label: store.getters['i18n/t'](`typeLabel.${ WORKLOAD }`, { count: 2 }),
590
+ group: 'Root',
591
+ namespaced: true,
592
+ name: WORKLOAD_DASHBOARD,
593
+ weight: 100,
594
+ icon: 'folder',
595
+ ifHaveSubTypes: Object.values(WORKLOAD_TYPES),
596
+ ifFeature: STEVE_CACHE,
597
+ route: { name: 'c-cluster-explorer-workload-dashboard' },
598
+ exact: true,
599
+ overview: true,
600
+ });
601
+
589
602
  virtualType({
590
603
  labelKey: 'members.clusterAndProject',
591
604
  group: 'cluster',
@@ -1,4 +1,6 @@
1
- import { AGE, NAME as NAME_COL, STATE } from '@shell/config/table-headers';
1
+ import {
2
+ AGE, MGMT_CLUSTER_KUBE_VERSION, MGMT_CLUSTER_PROVIDER, NAME as NAME_COL, STATE
3
+ } from '@shell/config/table-headers';
2
4
  import {
3
5
  CAPI,
4
6
  CATALOG,
@@ -16,6 +18,9 @@ import { MULTI_CLUSTER } from '@shell/store/features';
16
18
  import { DSL } from '@shell/store/type-map';
17
19
  import { BLANK_CLUSTER } from '@shell/store/store-types.js';
18
20
  import { markRaw } from 'vue';
21
+ import {
22
+ STEVE_AGE_COL, STEVE_MGMT_CLUSTER_KUBE_VERSION, STEVE_MGMT_CLUSTER_PROVIDER, STEVE_NAMESPACE_COL, STEVE_STATE_COL
23
+ } from '@shell/config/pagination-table-headers';
19
24
 
20
25
  export const NAME = 'manager';
21
26
 
@@ -73,6 +78,7 @@ export function init(store) {
73
78
  ]);
74
79
 
75
80
  configureType(SNAPSHOT, { depaginate: true });
81
+ configureType(CATALOG.CLUSTER_REPO, { listCreateButtonLabelKey: 'catalog.repo.add' });
76
82
 
77
83
  configureType(CAPI.RANCHER_CLUSTER, {
78
84
  showListMasthead: false, namespaced: false, alias: [HCI.CLUSTER]
@@ -124,10 +130,12 @@ export function init(store) {
124
130
 
125
131
  basicType([
126
132
  HOSTED_PROVIDER,
133
+ CAPI.CAPI_PROVIDER,
127
134
  'rke-kontainer-providers',
128
135
  'rke-node-providers',
129
136
  ], 'providers');
130
137
 
138
+ weightType(CAPI.CAPI_PROVIDER, 4, true);
131
139
  weightType(CAPI.MACHINE_DEPLOYMENT, 4, true);
132
140
  weightType(CAPI.MACHINE_SET, 3, true);
133
141
  weightType(CAPI.MACHINE, 2, true);
@@ -159,7 +167,16 @@ export function init(store) {
159
167
  width: 100,
160
168
  };
161
169
 
162
- headers(CAPI.RANCHER_CLUSTER, [
170
+ const EXPLORER = {
171
+ name: 'explorer',
172
+ label: ' ',
173
+ align: 'right',
174
+ width: 65,
175
+ sort: false,
176
+ search: false
177
+ };
178
+
179
+ headers(MANAGEMENT.CLUSTER, [
163
180
  STATE,
164
181
  {
165
182
  name: 'name',
@@ -169,35 +186,61 @@ export function init(store) {
169
186
  formatter: 'ClusterLink',
170
187
  canBeVariable: true,
171
188
  },
189
+ MGMT_CLUSTER_PROVIDER,
190
+ MGMT_CLUSTER_KUBE_VERSION,
172
191
  {
173
- name: 'kubernetesVersion',
174
- labelKey: 'tableHeaders.version',
175
- subLabel: 'Architecture',
176
- value: 'kubernetesVersion',
177
- sort: 'kubernetesVersion',
178
- search: 'kubernetesVersion',
192
+ ...MACHINE_SUMMARY,
193
+ sort: 'statusInfo.nodeCount'
179
194
  },
195
+ AGE,
196
+ EXPLORER,
197
+ ], [
198
+ STEVE_STATE_COL,
180
199
  {
181
- name: 'provider',
182
- labelKey: 'tableHeaders.provider',
183
- subLabel: 'Distro',
184
- value: 'machineProvider',
185
- sort: ['machineProvider', 'provisioner'],
186
- formatter: 'ClusterProvider',
200
+ name: 'name',
201
+ labelKey: 'tableHeaders.name',
202
+ value: 'spec.displayName',
203
+ sort: ['spec.displayName'],
204
+ search: ['spec.displayName'],
205
+ formatter: 'ClusterLink',
206
+ canBeVariable: true,
187
207
  },
188
- MACHINE_SUMMARY,
189
- AGE,
208
+ STEVE_MGMT_CLUSTER_PROVIDER,
209
+ STEVE_MGMT_CLUSTER_KUBE_VERSION,
190
210
  {
191
- name: 'explorer',
192
- label: ' ',
193
- align: 'right',
194
- width: 65,
211
+ ...MACHINE_SUMMARY,
212
+ sort: 'status.info.nodeCount'
195
213
  },
214
+ STEVE_AGE_COL,
215
+ EXPLORER
196
216
  ]);
197
217
 
218
+ configureType(MANAGEMENT.CLUSTER, {
219
+ listGroups: [{
220
+ tooltipKey: 'resourceTable.groupBy.none',
221
+ icon: 'icon-list-flat',
222
+ value: 'none',
223
+ }, {
224
+ icon: 'icon-folder',
225
+ // Given management.cattle.io.cluster is not namespaced we group by fleet workspace
226
+ value: 'spec.fleetWorkspaceName',
227
+ field: 'spec.fleetWorkspaceName',
228
+ hideColumn: STEVE_NAMESPACE_COL.name,
229
+ tooltipKey: 'resourceTable.groupBy.workspace',
230
+ groupLabelKey: 'groupByLabel',
231
+ }],
232
+ listGroupsWillOverride: true,
233
+ });
234
+
198
235
  headers(CAPI.MACHINE_DEPLOYMENT, [
199
236
  STATE,
200
- NAME_COL,
237
+ NAME_COL, {
238
+ name: 'cluster',
239
+ labelKey: 'tableHeaders.cluster',
240
+ value: 'clusterName',
241
+ getValue: (row) => row.clusterName,
242
+ sort: ['clusterName'],
243
+ },
201
244
  MACHINE_SUMMARY,
202
245
  AGE
203
246
  ]);
@@ -236,4 +279,24 @@ export function init(store) {
236
279
  })
237
280
  }
238
281
  });
282
+
283
+ const clusterGroupConfig = {
284
+ listGroups: [{
285
+ tooltipKey: 'resourceTable.groupBy.none',
286
+ icon: 'icon-list-flat',
287
+ value: 'none',
288
+ }, {
289
+ icon: 'icon-folder',
290
+ value: 'clusterName',
291
+ field: 'clusterName',
292
+ hideColumn: 'cluster',
293
+ tooltipKey: 'resourceTable.groupBy.cluster',
294
+ groupLabelKey: 'groupByLabel',
295
+ }],
296
+ listGroupsWillOverride: true,
297
+ };
298
+
299
+ configureType(CAPI.MACHINE_DEPLOYMENT, { ...clusterGroupConfig });
300
+ configureType(CAPI.MACHINE_SET, { ...clusterGroupConfig });
301
+ configureType(CAPI.MACHINE, { ...clusterGroupConfig });
239
302
  }
@@ -9,6 +9,22 @@ export const routerOptions = {
9
9
  base: process.env.routerBase || '/',
10
10
  routes: Routes,
11
11
  fallback: false,
12
+ scrollBehavior(to, from, savedPosition) {
13
+ // Returning the savedPosition will result in a native-like behavior when
14
+ // navigating with back/forward buttons
15
+ if (savedPosition) {
16
+ return savedPosition;
17
+ }
18
+
19
+ // Handle the "skip to main content" link
20
+ if (to.hash === '#main-content') {
21
+ const el = document.getElementById('main-content');
22
+
23
+ el?.focus();
24
+
25
+ return { el: to.hash };
26
+ }
27
+ },
12
28
  };
13
29
 
14
30
  export function extendRouter(config, context) {
@@ -0,0 +1,130 @@
1
+ import { authenticate } from '@shell/config/router/navigation-guards/authentication';
2
+
3
+ jest.mock('@shell/utils/router', () => ({ routeRequiresAuthentication: () => true }));
4
+
5
+ const isLoggedInMock = jest.fn();
6
+ const findMeMock = jest.fn();
7
+ const notLoggedInMock = jest.fn();
8
+ const noAuthMock = jest.fn();
9
+
10
+ jest.mock('@shell/utils/auth', () => {
11
+ const actual = jest.requireActual('@shell/utils/auth');
12
+
13
+ return {
14
+ ...actual,
15
+ isLoggedIn: (...args: any[]) => isLoggedInMock(...args),
16
+ findMe: (...args: any[]) => findMeMock(...args),
17
+ notLoggedIn: (...args: any[]) => notLoggedInMock(...args),
18
+ noAuth: (...args: any[]) => noAuthMock(...args),
19
+ };
20
+ });
21
+
22
+ function makeStore({ user, fromHeader }: { user: any, fromHeader: string }) {
23
+ const getters: Record<string, any> = {
24
+ 'auth/enabled': true,
25
+ 'auth/loggedIn': false,
26
+ 'auth/user': user,
27
+ 'auth/fromHeader': fromHeader,
28
+ };
29
+
30
+ return {
31
+ dispatch: jest.fn().mockResolvedValue(undefined),
32
+ commit: jest.fn(),
33
+ getters,
34
+ };
35
+ }
36
+
37
+ const to = { name: 'c-cluster-explorer', query: {} };
38
+
39
+ describe('navigation-guards/authentication: mustChangePassword', () => {
40
+ beforeEach(() => {
41
+ isLoggedInMock.mockReset().mockResolvedValue(undefined);
42
+ findMeMock.mockReset();
43
+ notLoggedInMock.mockReset();
44
+ noAuthMock.mockReset();
45
+ });
46
+
47
+ it('redirects local users with mustChangePassword to auth-setup', async() => {
48
+ const store = makeStore({
49
+ user: { mustChangePassword: true, principalIds: ['local://user-1'] },
50
+ fromHeader: 'true',
51
+ });
52
+
53
+ findMeMock.mockResolvedValue({ id: 'local://user-1', provider: 'local' });
54
+
55
+ const next = jest.fn();
56
+
57
+ await authenticate(to as any, {} as any, next, { store } as any);
58
+
59
+ expect(next).toHaveBeenCalledWith({ name: 'auth-setup' });
60
+ expect(isLoggedInMock).not.toHaveBeenCalled();
61
+ });
62
+
63
+ it('lets SSO users with mustChangePassword reach the requested route', async() => {
64
+ const me = { id: 'oidc_user://user@example.com', provider: 'genericoidc' };
65
+ const store = makeStore({
66
+ user: { mustChangePassword: true, principalIds: ['local://user-1', me.id] },
67
+ fromHeader: 'true',
68
+ });
69
+
70
+ findMeMock.mockResolvedValue(me);
71
+
72
+ const next = jest.fn();
73
+
74
+ await authenticate(to as any, {} as any, next, { store } as any);
75
+
76
+ expect(next).toHaveBeenCalledWith();
77
+ expect(next).not.toHaveBeenCalledWith({ name: 'auth-setup' });
78
+ expect(isLoggedInMock).toHaveBeenCalledTimes(1);
79
+ });
80
+
81
+ it('does not redirect when mustChangePassword is false, regardless of provider', async() => {
82
+ const store = makeStore({
83
+ user: { mustChangePassword: false, principalIds: ['local://user-1'] },
84
+ fromHeader: 'true',
85
+ });
86
+
87
+ findMeMock.mockResolvedValue({ id: 'local://user-1', provider: 'local' });
88
+
89
+ const next = jest.fn();
90
+
91
+ await authenticate(to as any, {} as any, next, { store } as any);
92
+
93
+ expect(next).toHaveBeenCalledWith();
94
+ expect(isLoggedInMock).toHaveBeenCalledTimes(1);
95
+ });
96
+
97
+ it('redirects local users with mustChangePassword to auth-setup for older-style fromHeader', async() => {
98
+ const store = makeStore({
99
+ user: { mustChangePassword: true, principalIds: ['local://user-1'] },
100
+ fromHeader: 'unknown',
101
+ });
102
+
103
+ findMeMock.mockResolvedValue({ id: 'local://user-1', provider: 'local' });
104
+
105
+ const next = jest.fn();
106
+
107
+ await authenticate(to as any, {} as any, next, { store } as any);
108
+
109
+ expect(next).toHaveBeenCalledWith({ name: 'auth-setup' });
110
+ expect(isLoggedInMock).not.toHaveBeenCalled();
111
+ });
112
+
113
+ it('lets SSO users with mustChangePassword through for older-style fromHeader', async() => {
114
+ const me = { id: 'saml_user://user', provider: 'saml' };
115
+ const store = makeStore({
116
+ user: { mustChangePassword: true, principalIds: ['local://user-1', me.id] },
117
+ fromHeader: 'unknown',
118
+ });
119
+
120
+ findMeMock.mockResolvedValue(me);
121
+
122
+ const next = jest.fn();
123
+
124
+ await authenticate(to as any, {} as any, next, { store } as any);
125
+
126
+ expect(next).toHaveBeenCalledWith();
127
+ expect(next).not.toHaveBeenCalledWith({ name: 'auth-setup' });
128
+ expect(isLoggedInMock).toHaveBeenCalledTimes(1);
129
+ });
130
+ });
@@ -61,19 +61,21 @@ export async function authenticate(to, from, next, { store }) {
61
61
  await store.dispatch('auth/getUser');
62
62
  const user = store.getters['auth/user'] || {};
63
63
 
64
- if (user?.mustChangePassword) {
65
- return next({ name: 'auth-setup' });
66
- }
67
-
68
64
  // In newer versions the API calls return the auth state instead of having to make a new call all the time.
69
65
  const fromHeader = store.getters['auth/fromHeader'];
70
66
 
67
+ const mustChangePasswordFor = (me) => user?.mustChangePassword && me?.provider === 'local';
68
+
71
69
  if ( fromHeader === 'none' ) {
72
70
  noAuth(store);
73
71
  handleOidcRedirectToCallbackUrl();
74
72
  } else if ( fromHeader === 'true' ) {
75
73
  const me = await findMe(store);
76
74
 
75
+ if (mustChangePasswordFor(me)) {
76
+ return next({ name: 'auth-setup' });
77
+ }
78
+
77
79
  await isLoggedIn(store, getUserObject(user, me));
78
80
  handleOidcRedirectToCallbackUrl();
79
81
  } else if ( fromHeader === 'false' ) {
@@ -85,6 +87,10 @@ export async function authenticate(to, from, next, { store }) {
85
87
  try {
86
88
  const me = await findMe(store);
87
89
 
90
+ if (mustChangePasswordFor(me)) {
91
+ return next({ name: 'auth-setup' });
92
+ }
93
+
88
94
  await isLoggedIn(store, getUserObject(user, me));
89
95
  handleOidcRedirectToCallbackUrl();
90
96
  } catch (e) {