@rancher/shell 3.0.12-rc.3 → 3.0.12-rc.5

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 (315) hide show
  1. package/assets/styles/global/_button.scss +1 -1
  2. package/assets/styles/global/_layout.scss +4 -0
  3. package/assets/translations/en-us.yaml +183 -51
  4. package/assets/translations/zh-hans.yaml +1 -7
  5. package/chart/monitoring/ClusterSelector.vue +0 -21
  6. package/chart/monitoring/prometheus/index.vue +6 -3
  7. package/components/ActionDropdownShell.vue +5 -3
  8. package/components/ButtonGroup.vue +26 -1
  9. package/components/CruResource.vue +212 -16
  10. package/components/ExplorerMembers.vue +8 -4
  11. package/components/ExplorerProjectsNamespaces.vue +10 -6
  12. package/components/GrowlManager.vue +4 -0
  13. package/components/MgmtNodeList.vue +184 -0
  14. package/components/PromptRestore.vue +93 -32
  15. package/components/Questions/index.vue +1 -0
  16. package/components/Resource/Detail/Card/StateCard/__tests__/composables.test.ts +90 -1
  17. package/components/Resource/Detail/Card/StateCard/composables.ts +57 -87
  18. package/components/Resource/Detail/Card/StatusCard/__tests__/StatusCard.test.ts +61 -0
  19. package/components/Resource/Detail/Card/StatusCard/index.vue +61 -15
  20. package/components/Resource/Detail/Metadata/IdentifyingInformation/index.vue +2 -0
  21. package/components/Resource/Detail/Metadata/KeyValue.vue +5 -2
  22. package/components/Resource/Detail/Metadata/KeyValueRow.vue +2 -6
  23. package/components/ResourceDetail/index.vue +1 -1
  24. package/components/ResourceList/Masthead.vue +7 -1
  25. package/components/ResourceList/index.vue +82 -1
  26. package/components/ResourceTable.vue +1 -0
  27. package/components/RichTranslation.vue +5 -2
  28. package/components/Setting.vue +1 -0
  29. package/components/SortableTable/index.vue +4 -3
  30. package/components/SubtleLink.vue +31 -6
  31. package/components/Tabbed/Tab.vue +29 -3
  32. package/components/Tabbed/index.vue +25 -3
  33. package/components/TableOfContents/TableOfContents.vue +109 -0
  34. package/components/TableOfContents/composables.ts +258 -0
  35. package/components/Window/ContainerShell.vue +21 -11
  36. package/components/Window/__tests__/ContainerShell.test.ts +107 -37
  37. package/components/Wizard.vue +23 -5
  38. package/components/__tests__/ButtonGroup.test.ts +56 -0
  39. package/components/__tests__/PromptRestore.test.ts +169 -19
  40. package/components/fleet/AppCoChartGrid.vue +401 -0
  41. package/components/fleet/AppCoEmptyState.vue +127 -0
  42. package/components/fleet/AppCoPageHeader.vue +119 -0
  43. package/components/fleet/AppCoVersionSelect.vue +70 -0
  44. package/components/fleet/FleetClusterTargets/ClusterSelectionFields.vue +217 -0
  45. package/components/fleet/FleetClusterTargets/TargetsList.vue +123 -35
  46. package/components/fleet/FleetClusterTargets/index.vue +189 -146
  47. package/components/fleet/FleetIntro.vue +7 -3
  48. package/components/fleet/FleetNoWorkspaces.vue +7 -3
  49. package/components/fleet/FleetSecretSelector.vue +5 -3
  50. package/components/fleet/FleetValuesFrom.vue +8 -2
  51. package/components/fleet/GitRepoAdvancedTab.vue +1 -0
  52. package/components/fleet/GitRepoMetadataTab.vue +5 -0
  53. package/components/fleet/GitRepoTargetTab.vue +0 -2
  54. package/components/fleet/HelmOpAdvancedTab.vue +19 -53
  55. package/components/fleet/HelmOpAppCoConfigTab.vue +597 -0
  56. package/components/fleet/HelmOpAppCoResourcesSection.vue +162 -0
  57. package/components/fleet/HelmOpMetadataTab.vue +5 -0
  58. package/components/fleet/HelmOpResourcesSection.vue +82 -0
  59. package/components/fleet/HelmOpTargetOptionsSection.vue +89 -0
  60. package/components/fleet/HelmOpTargetTab.vue +64 -60
  61. package/components/fleet/HelmOpValuesTab.vue +129 -105
  62. package/components/fleet/__tests__/AppCoEmptyState.test.ts +71 -0
  63. package/components/fleet/__tests__/AppCoVersionSelect.test.ts +36 -0
  64. package/components/fleet/__tests__/ClusterSelectionFields.test.ts +62 -0
  65. package/components/fleet/__tests__/FleetClusterTargets.test.ts +253 -0
  66. package/components/fleet/__tests__/FleetSecretSelector.test.ts +16 -0
  67. package/components/fleet/__tests__/FleetValuesFrom.test.ts +44 -0
  68. package/components/fleet/__tests__/HelmOpAppCoConfigTab.test.ts +59 -0
  69. package/components/fleet/__tests__/HelmOpAppCoResourcesSection.test.ts +62 -0
  70. package/components/fleet/__tests__/HelmOpResourcesSection.test.ts +43 -0
  71. package/components/fleet/__tests__/HelmOpTargetOptionsSection.test.ts +34 -0
  72. package/components/fleet/__tests__/HelmOpValuesTab.test.ts +39 -0
  73. package/components/fleet/__tests__/__snapshots__/AppCoEmptyState.test.ts.snap +97 -0
  74. package/components/fleet/__tests__/__snapshots__/AppCoVersionSelect.test.ts.snap +30 -0
  75. package/components/fleet/__tests__/__snapshots__/ClusterSelectionFields.test.ts.snap +209 -0
  76. package/components/fleet/__tests__/__snapshots__/HelmOpTargetOptionsSection.test.ts.snap +140 -0
  77. package/components/fleet/dashboard/Empty.vue +8 -4
  78. package/components/fleet/dashboard/ResourceCard.vue +28 -0
  79. package/components/fleet/dashboard/ResourceDetails.vue +28 -0
  80. package/components/fleet/dashboard/__tests__/ResourceCard.test.ts +87 -0
  81. package/components/form/ArrayList.vue +61 -4
  82. package/components/form/FileSelector.vue +39 -1
  83. package/components/form/KeyValue.vue +23 -2
  84. package/components/form/LabeledSelect.vue +39 -1
  85. package/components/form/Labels.vue +22 -3
  86. package/components/form/NameNsDescription.vue +13 -5
  87. package/components/form/PrivateRegistry.constants.ts +7 -0
  88. package/components/form/PrivateRegistry.vue +253 -18
  89. package/components/form/ResourceTabs/index.vue +1 -0
  90. package/components/form/SelectOrCreateAuthSecret.vue +140 -17
  91. package/components/form/__tests__/FileSelector.test.ts +23 -0
  92. package/components/form/__tests__/NameNsDescription.test.ts +75 -0
  93. package/components/form/__tests__/PrivateRegistry.test.ts +463 -73
  94. package/components/form/__tests__/SelectOrCreateAuthSecret.test.ts +122 -0
  95. package/components/formatter/EtcdSnapshotName.vue +73 -0
  96. package/components/formatter/InternalExternalIP.vue +10 -4
  97. package/components/formatter/ServiceTargets.vue +26 -7
  98. package/components/formatter/__tests__/InternalExternalIP.test.ts +132 -0
  99. package/components/formatter/__tests__/ServiceTargets.test.ts +412 -0
  100. package/components/nav/Header.vue +12 -1
  101. package/components/nav/TopLevelMenu.vue +7 -2
  102. package/components/nav/__tests__/Header.test.ts +15 -0
  103. package/components/nav/__tests__/TopLevelMenu.test.ts +120 -2
  104. package/components/templates/default.vue +16 -4
  105. package/components/templates/home.vue +9 -4
  106. package/components/templates/plain.vue +9 -4
  107. package/composables/useHelmOpResources.test.ts +56 -0
  108. package/composables/useHelmOpResources.ts +32 -0
  109. package/composables/useStateColor.test.ts +325 -0
  110. package/composables/useStateColor.ts +128 -0
  111. package/config/features.js +1 -0
  112. package/config/home-links.js +1 -1
  113. package/config/labels-annotations.js +3 -0
  114. package/config/product/explorer.js +17 -4
  115. package/config/product/manager.js +8 -0
  116. package/config/router/index.js +16 -0
  117. package/config/router/navigation-guards/__tests__/authentication.test.ts +130 -0
  118. package/config/router/navigation-guards/authentication.js +10 -4
  119. package/config/router/routes.js +20 -6
  120. package/config/secret.ts +10 -0
  121. package/config/settings.ts +6 -4
  122. package/config/table-headers.js +3 -4
  123. package/config/types.js +16 -0
  124. package/core/plugin-products-base.ts +3 -3
  125. package/core/plugin-types.ts +83 -30
  126. package/core/plugin.ts +3 -0
  127. package/core/types-provisioning.ts +34 -1
  128. package/core/types.ts +15 -2
  129. package/detail/__tests__/provisioning.cattle.io.cluster.test.ts +114 -0
  130. package/detail/__tests__/workload.test.ts +3 -152
  131. package/detail/catalog.cattle.io.clusterrepo.vue +1 -1
  132. package/detail/provisioning.cattle.io.cluster.vue +109 -7
  133. package/detail/workload/index.vue +12 -55
  134. package/dialog/RotateEncryptionKeyDialog.vue +33 -9
  135. package/dialog/__tests__/RotateEncryptionKeyDialog.test.ts +78 -0
  136. package/edit/__tests__/catalog.cattle.io.clusterrepo.test.ts +248 -0
  137. package/edit/__tests__/fleet.cattle.io.gitrepo.test.ts +92 -0
  138. package/edit/__tests__/fleet.cattle.io.helmop.test.ts +206 -0
  139. package/edit/__tests__/management.cattle.io.setting.test.ts +2 -1
  140. package/edit/auditlog.cattle.io.auditpolicy/__tests__/__snapshots__/General.test.ts.snap +6 -0
  141. package/edit/auditlog.cattle.io.auditpolicy/__tests__/__snapshots__/index.test.ts.snap +1 -0
  142. package/edit/auth/__tests__/azuread.test.ts +34 -9
  143. package/edit/auth/__tests__/github.test.ts +234 -0
  144. package/edit/auth/__tests__/oidc.test.ts +26 -6
  145. package/edit/auth/__tests__/saml.test.ts +196 -0
  146. package/edit/auth/azuread.vue +128 -95
  147. package/edit/auth/github.vue +72 -13
  148. package/edit/auth/ldap/__tests__/index.test.ts +206 -0
  149. package/edit/auth/ldap/config.vue +8 -0
  150. package/edit/auth/ldap/index.vue +75 -1
  151. package/edit/auth/oidc.vue +119 -73
  152. package/edit/auth/saml.vue +76 -12
  153. package/edit/catalog.cattle.io.clusterrepo.vue +140 -32
  154. package/edit/compliance.cattle.io.clusterscanprofile.vue +39 -41
  155. package/edit/fleet.cattle.io.gitrepo.vue +70 -16
  156. package/edit/fleet.cattle.io.helmop.vue +542 -141
  157. package/edit/helm.cattle.io.projecthelmchart.vue +1 -0
  158. package/edit/{management.cattle.io.setting.vue → management.cattle.io.setting/index.vue} +32 -9
  159. package/edit/management.cattle.io.setting/system-default-registry-pull-secrets.vue +81 -0
  160. package/edit/management.cattle.io.user.vue +5 -2
  161. package/edit/provisioning.cattle.io.cluster/SelectCredential.vue +3 -12
  162. package/edit/provisioning.cattle.io.cluster/__tests__/rke2.test.ts +18 -0
  163. package/edit/provisioning.cattle.io.cluster/rke2.vue +89 -11
  164. package/edit/provisioning.cattle.io.cluster/tabs/MachinePool.vue +11 -0
  165. package/edit/provisioning.cattle.io.cluster/tabs/etcd/index.vue +0 -1
  166. package/edit/provisioning.cattle.io.cluster/tabs/registries/index.vue +14 -55
  167. package/list/group.principal.vue +5 -4
  168. package/list/harvesterhci.io.management.cluster.vue +8 -9
  169. package/list/management.cattle.io.user.vue +12 -9
  170. package/list/provisioning.cattle.io.cluster.vue +16 -10
  171. package/mixins/__tests__/auth-config.test.ts +90 -0
  172. package/mixins/__tests__/chart.test.ts +94 -0
  173. package/mixins/__tests__/resource-fetch-api-pagination.test.ts +48 -0
  174. package/mixins/auth-config.js +7 -0
  175. package/mixins/chart.js +11 -2
  176. package/mixins/child-hook.js +12 -6
  177. package/mixins/create-edit-view/impl.js +5 -3
  178. package/mixins/resource-fetch-api-pagination.js +21 -1
  179. package/models/__tests__/catalog.cattle.io.clusterrepo.test.ts +57 -0
  180. package/models/__tests__/compliance.cattle.io.clusterscan.test.ts +144 -0
  181. package/models/__tests__/fleet-application.test.ts +175 -0
  182. package/models/__tests__/fleet.cattle.io.bundle.test.ts +169 -0
  183. package/models/__tests__/fleet.cattle.io.helmop.test.ts +84 -0
  184. package/models/__tests__/management.cattle.io.node.ts +22 -0
  185. package/models/__tests__/namespace.test.ts +36 -0
  186. package/models/__tests__/provisioning.cattle.io.cluster.test.ts +205 -0
  187. package/models/__tests__/secret.test.ts +68 -1
  188. package/models/__tests__/workload.test.ts +401 -26
  189. package/models/catalog.cattle.io.clusterrepo.js +28 -4
  190. package/models/compliance.cattle.io.clusterscan.js +39 -4
  191. package/models/fleet-application.js +4 -0
  192. package/models/fleet.cattle.io.helmop.js +20 -1
  193. package/models/management.cattle.io.cluster.js +39 -5
  194. package/models/management.cattle.io.node.js +44 -3
  195. package/models/namespace.js +1 -1
  196. package/models/pod.js +46 -3
  197. package/models/provisioning.cattle.io.cluster.js +64 -14
  198. package/models/rke.cattle.io.etcdsnapshot.js +17 -9
  199. package/models/secret.js +19 -0
  200. package/models/workload.js +120 -20
  201. package/models/workload.service.js +5 -0
  202. package/package.json +14 -13
  203. package/pages/about.vue +5 -6
  204. package/pages/auth/login.vue +0 -35
  205. package/pages/auth/setup.vue +11 -0
  206. package/pages/c/_cluster/apps/charts/AppChartCardFooter.vue +2 -2
  207. package/pages/c/_cluster/apps/charts/AppChartCardSubHeader.vue +10 -1
  208. package/pages/c/_cluster/apps/charts/__tests__/index.test.ts +93 -0
  209. package/pages/c/_cluster/apps/charts/__tests__/install.test.ts +485 -107
  210. package/pages/c/_cluster/apps/charts/chart.vue +2 -1
  211. package/pages/c/_cluster/apps/charts/index.vue +48 -10
  212. package/pages/c/_cluster/apps/charts/install.vue +236 -144
  213. package/pages/c/_cluster/auth/roles/index.vue +5 -4
  214. package/pages/c/_cluster/explorer/workload-dashboard/ByNamespaceSection.vue +31 -0
  215. package/pages/c/_cluster/explorer/workload-dashboard/ByStateSection.vue +138 -0
  216. package/pages/c/_cluster/explorer/workload-dashboard/ByTypeSection.vue +30 -0
  217. package/pages/c/_cluster/explorer/workload-dashboard/WorkloadCard.vue +155 -0
  218. package/pages/c/_cluster/explorer/workload-dashboard/WorkloadNamespaceCard.vue +142 -0
  219. package/pages/c/_cluster/explorer/workload-dashboard/WorkloadTypeCard.vue +159 -0
  220. package/pages/c/_cluster/explorer/workload-dashboard/__tests__/composable.test.ts +561 -0
  221. package/pages/c/_cluster/explorer/workload-dashboard/composable.ts +440 -0
  222. package/pages/c/_cluster/explorer/workload-dashboard/index.vue +187 -0
  223. package/pages/c/_cluster/explorer/workload-dashboard/types.ts +80 -0
  224. package/pages/c/_cluster/fleet/application/create.vue +187 -136
  225. package/pages/c/_cluster/fleet/application/index.vue +5 -3
  226. package/pages/c/_cluster/fleet/application/suse-app-collection/ChartDetailBody.vue +338 -0
  227. package/pages/c/_cluster/fleet/application/suse-app-collection/ChartDetailHeader.vue +121 -0
  228. package/pages/c/_cluster/fleet/application/suse-app-collection/chart.vue +369 -0
  229. package/pages/c/_cluster/fleet/application/suse-app-collection/charts.vue +248 -0
  230. package/pages/c/_cluster/fleet/application/suse-app-collection/credentials.vue +310 -0
  231. package/pages/c/_cluster/fleet/index.vue +2 -2
  232. package/pages/c/_cluster/uiplugins/__tests__/index.test.ts +96 -0
  233. package/pages/c/_cluster/uiplugins/index.vue +15 -0
  234. package/pages/fail-whale.vue +16 -11
  235. package/pages/home.vue +16 -46
  236. package/pkg/require-asset.lib.js +25 -0
  237. package/pkg/vue.config.js +7 -0
  238. package/plugins/clean-html.d.ts +9 -0
  239. package/plugins/dashboard-store/__tests__/resource-class.test.ts +177 -0
  240. package/plugins/dashboard-store/getters.js +0 -1
  241. package/plugins/dashboard-store/resource-class.js +114 -19
  242. package/plugins/steve/__tests__/actions.test.ts +212 -0
  243. package/plugins/steve/actions.js +96 -0
  244. package/plugins/steve/steve-pagination-utils.ts +1 -1
  245. package/rancher-components/Accordion/Accordion.vue +53 -9
  246. package/rancher-components/Form/Checkbox/Checkbox.vue +14 -0
  247. package/rancher-components/Form/Radio/RadioButton.vue +17 -1
  248. package/rancher-components/Form/Radio/RadioGroup.vue +10 -0
  249. package/rancher-components/Form/TextArea/TextAreaAutoGrow.vue +30 -0
  250. package/rancher-components/Form/TextArea/__tests__/TextAreaAutoGrow.test.ts +95 -0
  251. package/rancher-components/Pill/RcTag/RcTag.vue +3 -2
  252. package/rancher-components/RcButton/RcButton.test.ts +103 -0
  253. package/rancher-components/RcButton/RcButton.vue +94 -15
  254. package/rancher-components/RcButton/index.ts +1 -1
  255. package/rancher-components/RcButton/types.ts +3 -0
  256. package/rancher-components/RcDropdown/RcDropdownTrigger.vue +6 -1
  257. package/rancher-components/RcItemCard/RcItemCard.test.ts +18 -0
  258. package/rancher-components/RcItemCard/RcItemCard.vue +2 -2
  259. package/rancher-components/RcSection/RcSection.vue +28 -3
  260. package/scripts/extension/helm/package/Dockerfile +1 -1
  261. package/scripts/test-plugins-build.sh +2 -1
  262. package/store/__tests__/features.test.ts +131 -0
  263. package/store/__tests__/growl.test.ts +374 -0
  264. package/store/__tests__/modal.test.ts +131 -0
  265. package/store/__tests__/notifications.test.ts +434 -0
  266. package/store/__tests__/slideInPanel.test.ts +88 -0
  267. package/store/__tests__/type-map.utils.test.ts +433 -0
  268. package/store/catalog.js +57 -0
  269. package/store/features.js +4 -0
  270. package/store/plugins.js +7 -4
  271. package/types/components/buttonGroup.ts +5 -0
  272. package/types/shell/index.d.ts +166 -70
  273. package/utils/__tests__/auth.test.ts +273 -0
  274. package/utils/__tests__/computed.test.ts +193 -0
  275. package/utils/__tests__/cspAdaptor.test.ts +163 -0
  276. package/utils/__tests__/dom.test.ts +81 -0
  277. package/utils/__tests__/duration.test.ts +37 -1
  278. package/utils/__tests__/dynamic-importer.test.ts +102 -0
  279. package/utils/__tests__/fleet-appco.test.ts +312 -0
  280. package/utils/__tests__/monitoring.test.ts +130 -0
  281. package/utils/__tests__/object.test.ts +22 -0
  282. package/utils/__tests__/operation-cr.test.ts +34 -0
  283. package/utils/__tests__/platform.test.ts +91 -0
  284. package/utils/__tests__/position.test.ts +237 -0
  285. package/utils/__tests__/provider.test.ts +51 -1
  286. package/utils/__tests__/queue.test.ts +232 -0
  287. package/utils/__tests__/release-notes.test.ts +221 -0
  288. package/utils/__tests__/router.test.js +254 -1
  289. package/utils/__tests__/select.test.ts +208 -0
  290. package/utils/__tests__/time.test.ts +265 -1
  291. package/utils/__tests__/title.test.ts +47 -0
  292. package/utils/__tests__/width.test.ts +53 -0
  293. package/utils/__tests__/window.test.ts +158 -0
  294. package/utils/__tests__/xccdf.test.ts +126 -6
  295. package/utils/crypto/__tests__/browserHashUtils.test.ts +98 -0
  296. package/utils/crypto/__tests__/index.test.ts +144 -0
  297. package/utils/duration.ts +104 -0
  298. package/utils/dynamic-content/__tests__/notification-handler.test.ts +196 -0
  299. package/utils/dynamic-content/info.ts +2 -1
  300. package/utils/error.js +13 -0
  301. package/utils/fleet-appco.ts +323 -0
  302. package/utils/object.js +22 -2
  303. package/utils/operation-cr.js +19 -0
  304. package/utils/provider.ts +12 -0
  305. package/utils/require-asset.ts +7 -0
  306. package/utils/validators/__tests__/container-images.test.ts +104 -0
  307. package/utils/validators/__tests__/flow-output.test.ts +91 -0
  308. package/utils/validators/__tests__/logging-outputs.test.ts +58 -0
  309. package/utils/validators/__tests__/monitoring-route.test.ts +119 -0
  310. package/utils/validators/__tests__/private-registry.test.ts +27 -15
  311. package/utils/validators/private-registry.ts +15 -4
  312. package/utils/xccdf.ts +39 -42
  313. package/vue.config.js +1 -1
  314. package/pages/support/index.vue +0 -264
  315. package/utils/duration.js +0 -43
@@ -3,78 +3,95 @@ import { mount } from '@vue/test-utils';
3
3
  import Install from '@shell/pages/c/_cluster/apps/charts/install.vue';
4
4
  import { CATALOG as CATALOG_ANNOTATIONS } from '@shell/config/labels-annotations';
5
5
 
6
- describe('page: Install', () => {
7
- it('should use version annotations for target namespace and name', async() => {
8
- const mockStore = {
9
- dispatch: jest.fn((action) => {
10
- if (action === 'cluster/create') {
11
- return Promise.resolve({ metadata: { namespace: '', name: '' } });
12
- }
13
-
14
- return Promise.resolve();
15
- }),
16
- getters: {
17
- 'catalog/inStore': 'cluster',
18
- 'catalog/repo': () => ({ metadata: { name: 'test-repo' } }),
19
- 'features/get': () => false,
20
- defaultNamespace: 'default',
21
- 'i18n/withFallback': (key: string) => key,
22
- 'type-map/hasCustomChart': () => false,
23
- 'cluster/all': () => [],
24
- 'cluster/byId': () => null,
25
- 'management/all': () => [],
26
- 'prefs/get': () => {},
27
- 'catalog/charts': [],
28
- 'wm/byId': () => null,
29
- 'i18n/t': (key: string) => key,
6
+ const defaultStubs = {
7
+ Loading: true,
8
+ Wizard: true,
9
+ Banner: true,
10
+ Checkbox: true,
11
+ LabeledInput: true,
12
+ LabeledSelect: true,
13
+ NameNsDescription: true,
14
+ Tabbed: true,
15
+ Questions: true,
16
+ YamlEditor: true,
17
+ ResourceCancelModal: true,
18
+ UnitInput: true,
19
+ TypeDescription: true,
20
+ LazyImage: true,
21
+ ChartReadme: true,
22
+ ButtonGroup: true,
23
+ PrivateRegistry: true,
24
+ };
25
+
26
+ const defaultGetters = {
27
+ 'catalog/inStore': 'cluster',
28
+ 'catalog/repo': () => ({ metadata: { name: 'test-repo' } }),
29
+ 'features/get': () => false,
30
+ defaultNamespace: 'default',
31
+ 'i18n/withFallback': (key: string) => key,
32
+ 'type-map/hasCustomChart': () => false,
33
+ 'cluster/all': () => [],
34
+ 'cluster/byId': () => null,
35
+ 'management/all': () => [],
36
+ 'prefs/get': () => {},
37
+ 'catalog/charts': [],
38
+ 'wm/byId': () => null,
39
+ 'i18n/t': (key: string) => key,
40
+ };
41
+
42
+ const mountInstall = (options: {
43
+ data?: () => Record<string, any>;
44
+ getters?: Record<string, any>;
45
+ mocks?: Record<string, any>;
46
+ } = {}) => {
47
+ const mockStore = {
48
+ dispatch: jest.fn((action) => {
49
+ if (action === 'cluster/create') {
50
+ return Promise.resolve({ metadata: { namespace: '', name: '' } });
30
51
  }
31
- };
32
52
 
33
- const wrapper = mount(Install, {
34
- global: {
35
- mocks: {
36
- $store: mockStore,
37
- $route: { query: {} },
38
- $fetchState: { pending: false },
39
- t: (key: string) => key,
40
- },
41
- stubs: {
42
- Loading: true,
43
- Wizard: true,
44
- Banner: true,
45
- Checkbox: true,
46
- LabeledInput: true,
47
- LabeledSelect: true,
48
- NameNsDescription: true,
49
- Tabbed: true,
50
- Questions: true,
51
- YamlEditor: true,
52
- ResourceCancelModal: true,
53
- UnitInput: true,
54
- TypeDescription: true,
55
- LazyImage: true,
56
- ChartReadme: true,
57
- ButtonGroup: true,
58
- }
53
+ return Promise.resolve();
54
+ }),
55
+ getters: {
56
+ ...defaultGetters,
57
+ ...options.getters
58
+ }
59
+ };
60
+
61
+ return mount(Install, {
62
+ global: {
63
+ mocks: {
64
+ $store: mockStore,
65
+ $route: { query: {} },
66
+ $fetchState: { pending: false },
67
+ t: (key: string) => key,
68
+ ...options.mocks
59
69
  },
60
- data() {
61
- return {
62
- version: {
63
- annotations: {
64
- [CATALOG_ANNOTATIONS.NAMESPACE]: 'custom-ns',
65
- [CATALOG_ANNOTATIONS.RELEASE_NAME]: 'custom-name',
66
- }
67
- },
68
- chart: {
69
- targetNamespace: 'wrong-ns',
70
- targetName: 'wrong-name',
71
- versions: []
72
- },
73
- query: { versionName: '1.0.0' },
74
- chartValues: { global: { imagePullSecrets: [] } },
75
- repo: { spec: { clientSecret: { name: 'test-secret' } } }
76
- };
77
- }
70
+ stubs: defaultStubs
71
+ },
72
+ data: options.data
73
+ });
74
+ };
75
+
76
+ describe('page: Install', () => {
77
+ it('should use version annotations for target namespace and name', async() => {
78
+ const wrapper = mountInstall({
79
+ data: () => ({
80
+ version: {
81
+ annotations: {
82
+ [CATALOG_ANNOTATIONS.NAMESPACE]: 'custom-ns',
83
+ [CATALOG_ANNOTATIONS.RELEASE_NAME]: 'custom-name',
84
+ }
85
+ },
86
+ chart: {
87
+ targetNamespace: 'wrong-ns',
88
+ targetName: 'wrong-name',
89
+ versions: []
90
+ },
91
+ query: { versionName: '1.0.0' },
92
+ chartValues: { global: { imagePullSecrets: [] } },
93
+ repo: { spec: { clientSecret: { name: 'test-secret' } } }
94
+ })
78
95
  });
79
96
 
80
97
  // Mock methods from mixins
@@ -97,44 +114,12 @@ describe('page: Install', () => {
97
114
  const mockReplace = jest.fn();
98
115
  const expectedLocation = { name: 'app-location' };
99
116
 
100
- const wrapper = mount(Install, {
101
- global: {
102
- mocks: {
103
- $store: {
104
- getters: {
105
- 'i18n/t': (key: string) => key,
106
- 'cluster/id': 'cluster-id',
107
- }
108
- },
109
- $route: { query: {} },
110
- $router: { replace: mockReplace },
111
- $fetchState: { pending: false },
112
- },
113
- stubs: {
114
- Loading: true,
115
- Wizard: true,
116
- Banner: true,
117
- Checkbox: true,
118
- LabeledInput: true,
119
- LabeledSelect: true,
120
- NameNsDescription: true,
121
- Tabbed: true,
122
- Questions: true,
123
- YamlEditor: true,
124
- ResourceCancelModal: true,
125
- UnitInput: true,
126
- TypeDescription: true,
127
- LazyImage: true,
128
- ChartReadme: true,
129
- ButtonGroup: true,
130
- }
131
- },
132
- data() {
133
- return {
134
- existing: false,
135
- chart: null,
136
- };
137
- }
117
+ const wrapper = mountInstall({
118
+ data: () => ({
119
+ existing: false,
120
+ chart: null,
121
+ }),
122
+ mocks: { $router: { replace: mockReplace } }
138
123
  });
139
124
 
140
125
  jest.spyOn((wrapper.vm as any), 'appLocation').mockReturnValue(expectedLocation);
@@ -144,4 +129,397 @@ describe('page: Install', () => {
144
129
  expect(mockReplace).toHaveBeenCalledWith(expectedLocation);
145
130
  });
146
131
  });
132
+
133
+ describe('showRegistryPullSecrets', () => {
134
+ it('should return true when repo has defaultImagePullSecrets', () => {
135
+ const wrapper = mountInstall({ data: () => ({ repo: { spec: { defaultImagePullSecrets: [{ name: 'my-secret' }] } } }) });
136
+
137
+ expect(wrapper.vm.showRegistryPullSecrets).toBe(true);
138
+ });
139
+
140
+ it('should return false when repo has no defaultImagePullSecrets', () => {
141
+ const wrapper = mountInstall({ data: () => ({ repo: { spec: {} } }) });
142
+
143
+ expect(wrapper.vm.showRegistryPullSecrets).toBe(false);
144
+ });
145
+
146
+ it('should return false when defaultImagePullSecrets is empty', () => {
147
+ const wrapper = mountInstall({ data: () => ({ repo: { spec: { defaultImagePullSecrets: [] } } }) });
148
+
149
+ expect(wrapper.vm.showRegistryPullSecrets).toBe(false);
150
+ });
151
+ });
152
+
153
+ describe('repoDefaultPullSecretNames', () => {
154
+ it('should map defaultImagePullSecrets to names', () => {
155
+ const wrapper = mountInstall({
156
+ data: () => ({
157
+ repo: {
158
+ spec: {
159
+ defaultImagePullSecrets: [
160
+ { name: 'secret-a' },
161
+ { name: 'secret-b' }
162
+ ]
163
+ }
164
+ }
165
+ })
166
+ });
167
+
168
+ expect(wrapper.vm.repoDefaultPullSecretNames).toStrictEqual(['secret-a', 'secret-b']);
169
+ });
170
+
171
+ it('should filter out entries without a name', () => {
172
+ const wrapper = mountInstall({
173
+ data: () => ({
174
+ repo: {
175
+ spec: {
176
+ defaultImagePullSecrets: [
177
+ { name: 'valid' },
178
+ { name: '' },
179
+ {}
180
+ ]
181
+ }
182
+ }
183
+ })
184
+ });
185
+
186
+ expect(wrapper.vm.repoDefaultPullSecretNames).toStrictEqual(['valid']);
187
+ });
188
+
189
+ it('should return empty array when no defaultImagePullSecrets', () => {
190
+ const wrapper = mountInstall({ data: () => ({ repo: { spec: {} } }) });
191
+
192
+ expect(wrapper.vm.repoDefaultPullSecretNames).toStrictEqual([]);
193
+ });
194
+ });
195
+
196
+ describe('existingValuesPullSecrets', () => {
197
+ it('should return empty array on fresh install', () => {
198
+ const wrapper = mountInstall({
199
+ data: () => ({
200
+ existing: false,
201
+ chartValues: { global: { imagePullSecrets: ['secret-1', 'secret-2'] } }
202
+ })
203
+ });
204
+
205
+ expect(wrapper.vm.existingValuesPullSecrets).toStrictEqual([]);
206
+ });
207
+
208
+ it('should return imagePullSecrets from chart values on upgrade', () => {
209
+ const wrapper = mountInstall({
210
+ data: () => ({
211
+ existing: { metadata: { name: 'existing-release' } },
212
+ chartValues: { global: { imagePullSecrets: ['secret-1', 'secret-2'] } }
213
+ })
214
+ });
215
+
216
+ expect(wrapper.vm.existingValuesPullSecrets).toStrictEqual(['secret-1', 'secret-2']);
217
+ });
218
+
219
+ it('should filter out falsy entries', () => {
220
+ const wrapper = mountInstall({
221
+ data: () => ({
222
+ existing: { metadata: { name: 'existing-release' } },
223
+ chartValues: { global: { imagePullSecrets: ['secret-1', null, '', 'secret-2'] } }
224
+ })
225
+ });
226
+
227
+ expect(wrapper.vm.existingValuesPullSecrets).toStrictEqual(['secret-1', 'secret-2']);
228
+ });
229
+
230
+ it('should return empty array when imagePullSecrets is not an array', () => {
231
+ const wrapper = mountInstall({
232
+ data: () => ({
233
+ existing: { metadata: { name: 'existing-release' } },
234
+ chartValues: { global: { imagePullSecrets: 'not-an-array' } }
235
+ })
236
+ });
237
+
238
+ expect(wrapper.vm.existingValuesPullSecrets).toStrictEqual([]);
239
+ });
240
+
241
+ it('should return empty array when global has no imagePullSecrets', () => {
242
+ const wrapper = mountInstall({
243
+ data: () => ({
244
+ existing: { metadata: { name: 'existing-release' } },
245
+ chartValues: { global: {} }
246
+ })
247
+ });
248
+
249
+ expect(wrapper.vm.existingValuesPullSecrets).toStrictEqual([]);
250
+ });
251
+ });
252
+
253
+ describe('addGlobalValuesTo', () => {
254
+ it('should set imagePullSecrets when registryPullSecret is selected', () => {
255
+ const wrapper = mountInstall({
256
+ data: () => ({
257
+ repo: { spec: { defaultImagePullSecrets: [{ name: 'default-secret' }] } },
258
+ registryPullSecret: 'my-selected-secret',
259
+ customRegistrySetting: null,
260
+ currentCluster: null,
261
+ serverUrlSetting: { value: '' },
262
+ })
263
+ });
264
+
265
+ const values = { global: {} };
266
+
267
+ wrapper.vm.addGlobalValuesTo(values);
268
+
269
+ expect(values.global.imagePullSecrets).toStrictEqual(['my-selected-secret']);
270
+ });
271
+
272
+ it('should delete imagePullSecrets when showRegistryPullSecrets is true but no pullSecret selected', () => {
273
+ const wrapper = mountInstall({
274
+ data: () => ({
275
+ repo: { spec: { defaultImagePullSecrets: [{ name: 'default-secret' }] } },
276
+ registryPullSecret: null,
277
+ customRegistrySetting: null,
278
+ currentCluster: null,
279
+ serverUrlSetting: { value: '' },
280
+ })
281
+ });
282
+
283
+ const values = { global: { imagePullSecrets: ['old-secret'] } };
284
+
285
+ wrapper.vm.addGlobalValuesTo(values);
286
+
287
+ expect(values.global.imagePullSecrets).toBeUndefined();
288
+ });
289
+
290
+ it('should not modify imagePullSecrets when showRegistryPullSecrets is false', () => {
291
+ const wrapper = mountInstall({
292
+ data: () => ({
293
+ repo: { spec: {} },
294
+ registryPullSecret: null,
295
+ customRegistrySetting: null,
296
+ currentCluster: null,
297
+ serverUrlSetting: { value: '' },
298
+ })
299
+ });
300
+
301
+ const values = { global: { imagePullSecrets: ['existing-secret'] } };
302
+
303
+ wrapper.vm.addGlobalValuesTo(values);
304
+
305
+ expect(values.global.imagePullSecrets).toStrictEqual(['existing-secret']);
306
+ });
307
+ });
308
+
309
+ describe('finish()', () => {
310
+ const createFinishWrapper = (overrides: Record<string, any> = {}) => {
311
+ const mockDoAction = jest.fn().mockResolvedValue({
312
+ operationNamespace: 'ns',
313
+ operationName: 'op'
314
+ });
315
+ const mockActionLinkFor = jest.fn().mockReturnValue('https://rancher/v1/catalog.cattle.io.clusterrepos/rancher-charts?action=install');
316
+
317
+ const wrapper = mountInstall({
318
+ data: () => ({
319
+ repo: {
320
+ spec: { defaultImagePullSecrets: overrides.defaultImagePullSecrets || [] },
321
+ doAction: mockDoAction,
322
+ actionLinkFor: mockActionLinkFor,
323
+ waitForOperation: jest.fn().mockResolvedValue(undefined),
324
+ },
325
+ existing: overrides.existing || null,
326
+ skipPullSecrets: overrides.skipPullSecrets ?? false,
327
+ registryPullSecret: overrides.registryPullSecret ?? null,
328
+ customRegistrySetting: null,
329
+ currentCluster: null,
330
+ serverUrlSetting: { value: '' },
331
+ errors: [],
332
+ chart: { versions: [] },
333
+ version: { annotations: {} },
334
+ chartValues: overrides.chartValues || { global: {} },
335
+ value: { metadata: { namespace: 'default', name: 'test' } },
336
+ ...overrides
337
+ })
338
+ });
339
+
340
+ // Mock mixin methods
341
+ jest.spyOn(wrapper.vm as any, 'createNamespaceIfNeeded').mockResolvedValue(undefined);
342
+ jest.spyOn(wrapper.vm as any, 'applyHooks').mockResolvedValue(overrides.hookResults || {});
343
+ jest.spyOn(wrapper.vm as any, 'actionInput').mockReturnValue({ errors: [], input: {} });
344
+ jest.spyOn(wrapper.vm as any, 'done').mockImplementation();
345
+
346
+ // Mock store dispatch for operation find
347
+ const store = (wrapper.vm as any).$store;
348
+
349
+ store.dispatch = jest.fn().mockResolvedValue({
350
+ waitForLink: jest.fn().mockResolvedValue(undefined),
351
+ openLogs: jest.fn(),
352
+ });
353
+
354
+ return {
355
+ wrapper, mockDoAction, mockActionLinkFor
356
+ };
357
+ };
358
+
359
+ it('should read created secret name from hookResults when registryPullSecret is null', async() => {
360
+ const { wrapper } = createFinishWrapper({
361
+ defaultImagePullSecrets: [{ name: 'repo-default' }],
362
+ hookResults: { registerAuthSecret: { metadata: { name: 'hook-created-secret' } } },
363
+ });
364
+ const btnCb = jest.fn();
365
+
366
+ await wrapper.vm.finish(btnCb);
367
+
368
+ expect(wrapper.vm.registryPullSecret).toBe('hook-created-secret');
369
+ });
370
+
371
+ it('should not overwrite registryPullSecret from hookResults when already set', async() => {
372
+ const { wrapper } = createFinishWrapper({
373
+ defaultImagePullSecrets: [{ name: 'repo-default' }],
374
+ registryPullSecret: 'already-selected',
375
+ hookResults: { registerAuthSecret: { metadata: { name: 'hook-created-secret' } } },
376
+ });
377
+ const btnCb = jest.fn();
378
+
379
+ await wrapper.vm.finish(btnCb);
380
+
381
+ expect(wrapper.vm.registryPullSecret).toBe('already-selected');
382
+ });
383
+
384
+ it('should not read hookResults when skipPullSecrets is true', async() => {
385
+ const { wrapper } = createFinishWrapper({
386
+ defaultImagePullSecrets: [{ name: 'repo-default' }],
387
+ skipPullSecrets: true,
388
+ hookResults: { registerAuthSecret: { metadata: { name: 'hook-created-secret' } } },
389
+ });
390
+ const btnCb = jest.fn();
391
+
392
+ await wrapper.vm.finish(btnCb);
393
+
394
+ expect(wrapper.vm.registryPullSecret).toBeNull();
395
+ });
396
+
397
+ it('should add skipPullSecrets query param when skipPullSecrets is true', async() => {
398
+ const { wrapper, mockDoAction } = createFinishWrapper({ skipPullSecrets: true });
399
+ const btnCb = jest.fn();
400
+
401
+ await wrapper.vm.finish(btnCb);
402
+
403
+ expect(mockDoAction).toHaveBeenCalledWith(
404
+ 'install',
405
+ expect.anything(),
406
+ expect.objectContaining({ url: expect.stringContaining('skipPullSecrets=true') })
407
+ );
408
+ });
409
+
410
+ it('should not add skipPullSecrets query param when skipPullSecrets is false', async() => {
411
+ const { wrapper, mockDoAction } = createFinishWrapper({ skipPullSecrets: false });
412
+ const btnCb = jest.fn();
413
+
414
+ await wrapper.vm.finish(btnCb);
415
+
416
+ expect(mockDoAction).toHaveBeenCalledWith(
417
+ 'install',
418
+ expect.anything(),
419
+ {}
420
+ );
421
+ });
422
+
423
+ it('should use upgrade action name when existing release is present', async() => {
424
+ const { wrapper, mockDoAction } = createFinishWrapper({
425
+ existing: { metadata: { name: 'existing-release' } },
426
+ skipPullSecrets: true,
427
+ });
428
+ const btnCb = jest.fn();
429
+
430
+ await wrapper.vm.finish(btnCb);
431
+
432
+ expect(mockDoAction).toHaveBeenCalledWith(
433
+ 'upgrade',
434
+ expect.anything(),
435
+ expect.objectContaining({ url: expect.stringContaining('skipPullSecrets=true') })
436
+ );
437
+ });
438
+ });
439
+
440
+ describe('fetch() pull secret pre-selection', () => {
441
+ const createFetchWrapper = (overrides: Record<string, any> = {}) => {
442
+ const existingMock = overrides.existing ? {
443
+ metadata: overrides.existing.metadata || { namespace: 'default', name: 'release' },
444
+ fetchValues: jest.fn().mockResolvedValue(undefined),
445
+ deployedAsLegacy: jest.fn().mockResolvedValue(false),
446
+ deployedAsMultiCluster: jest.fn().mockResolvedValue(false),
447
+ values: overrides.chartValues?.global ? overrides.chartValues : {},
448
+ } : null;
449
+
450
+ const wrapper = mountInstall({
451
+ data: () => ({
452
+ repo: {
453
+ spec: {
454
+ defaultImagePullSecrets: overrides.defaultImagePullSecrets || [],
455
+ clientSecret: { name: 'test-secret' }
456
+ }
457
+ },
458
+ existing: existingMock,
459
+ chartValues: overrides.chartValues || { global: {} },
460
+ chart: { versions: [] },
461
+ version: { annotations: {} },
462
+ query: { versionName: '1.0.0' },
463
+ })
464
+ });
465
+
466
+ // Mock mixin methods used during fetch
467
+ jest.spyOn(wrapper.vm as any, 'fetchChart').mockResolvedValue(undefined);
468
+ jest.spyOn(wrapper.vm as any, 'fetchAutoInstallInfo').mockResolvedValue(undefined);
469
+ jest.spyOn(wrapper.vm as any, 'getClusterRegistry').mockResolvedValue(undefined);
470
+ jest.spyOn(wrapper.vm as any, 'getGlobalRegistry').mockResolvedValue(undefined);
471
+ jest.spyOn(wrapper.vm as any, 'loadValuesComponent').mockResolvedValue(undefined);
472
+ jest.spyOn(wrapper.vm as any, 'updateStepOneReady').mockImplementation();
473
+
474
+ return wrapper;
475
+ };
476
+
477
+ it('should pre-select single existing pull secret on upgrade', async() => {
478
+ const wrapper = createFetchWrapper({
479
+ existing: { metadata: { namespace: 'default', name: 'release' } },
480
+ defaultImagePullSecrets: [{ name: 'repo-default' }],
481
+ chartValues: { global: { imagePullSecrets: ['existing-secret'] } },
482
+ });
483
+
484
+ await Install.fetch.call(wrapper.vm);
485
+
486
+ expect(wrapper.vm.registryPullSecret).toBe('existing-secret');
487
+ });
488
+
489
+ it('should not pre-select when multiple existing pull secrets exist', async() => {
490
+ const wrapper = createFetchWrapper({
491
+ existing: { metadata: { namespace: 'default', name: 'release' } },
492
+ defaultImagePullSecrets: [{ name: 'repo-default' }],
493
+ chartValues: { global: { imagePullSecrets: ['secret-1', 'secret-2'] } },
494
+ });
495
+
496
+ await Install.fetch.call(wrapper.vm);
497
+
498
+ expect(wrapper.vm.registryPullSecret).toBeNull();
499
+ });
500
+
501
+ it('should not pre-select on fresh install', async() => {
502
+ const wrapper = createFetchWrapper({
503
+ existing: null,
504
+ defaultImagePullSecrets: [{ name: 'repo-default' }],
505
+ chartValues: { global: { imagePullSecrets: ['existing-secret'] } },
506
+ });
507
+
508
+ await Install.fetch.call(wrapper.vm);
509
+
510
+ expect(wrapper.vm.registryPullSecret).toBeNull();
511
+ });
512
+
513
+ it('should not pre-select when no imagePullSecrets in chart values', async() => {
514
+ const wrapper = createFetchWrapper({
515
+ existing: { metadata: { namespace: 'default', name: 'release' } },
516
+ defaultImagePullSecrets: [{ name: 'repo-default' }],
517
+ chartValues: { global: {} },
518
+ });
519
+
520
+ await Install.fetch.call(wrapper.vm);
521
+
522
+ expect(wrapper.vm.registryPullSecret).toBeNull();
523
+ });
524
+ });
147
525
  });
@@ -859,6 +859,7 @@ export default {
859
859
  .readme-wrapper {
860
860
  position: relative;
861
861
  flex: 1;
862
+ min-width: 0;
862
863
 
863
864
  .open-readme-button {
864
865
  display: flex;
@@ -881,7 +882,7 @@ export default {
881
882
 
882
883
  &__readme {
883
884
  flex: 1;
884
- min-width: 400px;
885
+ min-width: 0;
885
886
  padding: 12px 24px;
886
887
  }
887
888
  &__info {