@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
@@ -66,4 +66,126 @@ describe('component: SelectOrCreateAuthSecret', () => {
66
66
 
67
67
  expect(passwordLabeledInput!.props('labelKey')).toBe(expectedLabelKey);
68
68
  });
69
+
70
+ describe('GitHub App auth', () => {
71
+ const githubAppSetup = () => mount(SelectOrCreateAuthSecret, {
72
+ ...requiredSetup(),
73
+ props: {
74
+ mode: _EDIT,
75
+ namespace: 'default',
76
+ value: {},
77
+ allowGithubApp: true,
78
+ registerBeforeHook: () => {},
79
+ },
80
+ data() {
81
+ return { selected: AUTH_TYPE._GITHUB_APP } as any;
82
+ }
83
+ });
84
+
85
+ it('should render the three GitHub App fields when selected', () => {
86
+ const wrapper = githubAppSetup();
87
+
88
+ expect(wrapper.find('[data-testid="auth-secret-github-app-id"]').exists()).toBe(true);
89
+ expect(wrapper.find('[data-testid="auth-secret-github-app-installation-id"]').exists()).toBe(true);
90
+ expect(wrapper.find('[data-testid="auth-secret-github-app-private-key"]').exists()).toBe(true);
91
+ expect(wrapper.find('[data-testid="auth-secret-github-app-private-key-file"]').exists()).toBe(true);
92
+ });
93
+
94
+ it('should not narrow the select column for GitHub App (keeps span-6)', () => {
95
+ const wrapper = githubAppSetup();
96
+
97
+ expect((wrapper.vm as any).firstCol).toBe('col span-6');
98
+ });
99
+
100
+ it('should emit inputauthval with the GitHub App field values', async() => {
101
+ const wrapper = githubAppSetup();
102
+
103
+ await wrapper.setData({
104
+ githubAppId: 'app-id',
105
+ githubAppInstallationId: 'installation-id',
106
+ githubAppPrivateKey: 'private-key',
107
+ });
108
+
109
+ const emitted = wrapper.emitted('inputauthval') as any[];
110
+ const last = emitted[emitted.length - 1][0];
111
+
112
+ expect(last).toStrictEqual({
113
+ selected: AUTH_TYPE._GITHUB_APP,
114
+ publicKey: '',
115
+ privateKey: '',
116
+ githubAppId: 'app-id',
117
+ githubAppInstallationId: 'installation-id',
118
+ githubAppPrivateKey: 'private-key',
119
+ });
120
+ });
121
+
122
+ it('should populate the private key when a file is read', async() => {
123
+ const wrapper = githubAppSetup();
124
+
125
+ const fileSelector = wrapper.findComponent({ name: 'FileSelector' });
126
+
127
+ await fileSelector.vm.$emit('selected', 'key-from-file');
128
+
129
+ expect((wrapper.vm as any).githubAppPrivateKey).toBe('key-from-file');
130
+ });
131
+
132
+ it.each([
133
+ ['offer', true],
134
+ ['not offer', false],
135
+ ])('should %s the GitHub App create option when allowGithubApp is %p', (_, allowGithubApp) => {
136
+ const wrapper = mount(SelectOrCreateAuthSecret, {
137
+ ...requiredSetup(),
138
+ props: {
139
+ mode: _EDIT,
140
+ namespace: 'default',
141
+ value: {},
142
+ allowGithubApp,
143
+ registerBeforeHook: () => {},
144
+ },
145
+ });
146
+
147
+ const hasGithubAppOption = (wrapper.vm as any).options
148
+ .some((o: any) => o.value === AUTH_TYPE._GITHUB_APP);
149
+
150
+ expect(hasGithubAppOption).toBe(allowGithubApp);
151
+ });
152
+
153
+ it('should list existing GitHub App secrets but exclude plain Opaque secrets', () => {
154
+ const githubAppSecret = {
155
+ _type: 'Opaque',
156
+ isGithubApp: true,
157
+ id: 'default/gh-app',
158
+ dataPreview: '3 keys',
159
+ subTypeDisplay: 'Opaque',
160
+ metadata: { name: 'gh-app', namespace: 'default' },
161
+ };
162
+ const plainOpaqueSecret = {
163
+ _type: 'Opaque',
164
+ isGithubApp: false,
165
+ id: 'default/plain',
166
+ dataPreview: '1 key',
167
+ subTypeDisplay: 'Opaque',
168
+ metadata: { name: 'plain', namespace: 'default' },
169
+ };
170
+
171
+ const wrapper = mount(SelectOrCreateAuthSecret, {
172
+ ...requiredSetup(),
173
+ props: {
174
+ mode: _EDIT,
175
+ namespace: 'default',
176
+ value: {},
177
+ allowGithubApp: true,
178
+ registerBeforeHook: () => {},
179
+ },
180
+ data() {
181
+ return { allSecrets: [githubAppSecret, plainOpaqueSecret] } as any;
182
+ }
183
+ });
184
+
185
+ const optionValues = (wrapper.vm as any).options.map((o: any) => o.value);
186
+
187
+ expect(optionValues).toContain('default/gh-app');
188
+ expect(optionValues).not.toContain('default/plain');
189
+ });
190
+ });
69
191
  });
@@ -0,0 +1,73 @@
1
+ <script>
2
+ import { get } from '@shell/utils/object';
3
+
4
+ export default {
5
+ props: {
6
+ value: {
7
+ type: String,
8
+ default: ''
9
+ },
10
+ row: {
11
+ type: Object,
12
+ required: true
13
+ },
14
+ col: {
15
+ type: Object,
16
+ default: () => ({})
17
+ },
18
+ reference: {
19
+ type: String,
20
+ default: null,
21
+ },
22
+ getCustomDetailLink: {
23
+ type: Function,
24
+ default: null
25
+ }
26
+ },
27
+
28
+ computed: {
29
+ to() {
30
+ if (this.getCustomDetailLink) {
31
+ return this.getCustomDetailLink(this.row);
32
+ }
33
+ if ( this.row && this.reference ) {
34
+ return get(this.row, this.reference);
35
+ }
36
+
37
+ return this.row?.detailLocation;
38
+ },
39
+
40
+ showPredatesImportIcon() {
41
+ return !!this.row?.isSnapshotTooOld;
42
+ },
43
+
44
+ predatesImportMessage() {
45
+ return this.t('cluster.snapshot.predatesImportTooltip');
46
+ },
47
+ }
48
+ };
49
+ </script>
50
+
51
+ <template>
52
+ <span>
53
+ <router-link
54
+ v-if="to"
55
+ :to="to"
56
+ >
57
+ {{ value }}
58
+ </router-link>
59
+ <span v-else>
60
+ {{ value }}
61
+ <template v-if="!value && col.dashIfEmpty">
62
+ <span class="text-muted">&mdash;</span>
63
+ </template>
64
+ </span>
65
+ <i
66
+ v-if="showPredatesImportIcon"
67
+ v-clean-tooltip="{ content: predatesImportMessage, triggers: ['hover', 'touch', 'focus'] }"
68
+ v-stripped-aria-label="predatesImportMessage"
69
+ tabindex="0"
70
+ class="icon icon-error text-error ml-5"
71
+ />
72
+ </span>
73
+ </template>
@@ -1,5 +1,5 @@
1
1
  <script>
2
- import { isV4Format, isV6Format } from 'ip';
2
+ import ipaddr from 'ipaddr.js';
3
3
  import CopyToClipboard from '@shell/components/CopyToClipboard';
4
4
  import { mapGetters } from 'vuex';
5
5
  import RcStatusBadge from '@components/Pill/RcStatusBadge/RcStatusBadge';
@@ -15,10 +15,16 @@ export default {
15
15
  computed: {
16
16
  ...mapGetters({ t: 'i18n/t' }),
17
17
  filteredExternalIps() {
18
- return this.row.externalIps?.filter((ip) => this.isIp(ip)) || [];
18
+ const ips = this.row.externalIps ||
19
+ (this.row.externalIp && this.row.externalIp !== this.t('generic.none') ? [this.row.externalIp] : []);
20
+
21
+ return ips.filter((ip) => this.isIp(ip));
19
22
  },
20
23
  filteredInternalIps() {
21
- return this.row.internalIps?.filter((ip) => this.isIp(ip)) || [];
24
+ const ips = this.row.internalIps ||
25
+ (this.row.internalIp ? [this.row.internalIp] : []);
26
+
27
+ return ips.filter((ip) => this.isIp(ip));
22
28
  },
23
29
  internalSameAsExternal() {
24
30
  return this.externalIp && this.internalIp && this.externalIp === this.internalIp;
@@ -59,7 +65,7 @@ export default {
59
65
  },
60
66
  methods: {
61
67
  isIp(ip) {
62
- return ip && (isV4Format(ip) || isV6Format(ip));
68
+ return ip && (ipaddr.IPv4.isValidFourPartDecimal(ip) || ipaddr.IPv6.isValid(ip));
63
69
  }
64
70
  }
65
71
  };
@@ -1,6 +1,6 @@
1
1
  <script>
2
2
  import isEmpty from 'lodash/isEmpty';
3
- import { CATTLE_PUBLIC_ENDPOINTS } from '@shell/config/labels-annotations';
3
+ import { CATTLE_PUBLIC_ENDPOINTS, SERVICE_LINKS } from '@shell/config/labels-annotations';
4
4
  import Endpoints from '@shell/components/formatter/Endpoints';
5
5
  import has from 'lodash/has';
6
6
  import { isMaybeSecure } from '@shell/utils/url';
@@ -52,6 +52,23 @@ export default {
52
52
  return annotations[CATTLE_PUBLIC_ENDPOINTS];
53
53
  }
54
54
 
55
+ const serviceLinksAnnotation = annotations[SERVICE_LINKS];
56
+ const serviceLinksMap = new Map();
57
+
58
+ if (serviceLinksAnnotation) {
59
+ serviceLinksAnnotation.split(',').forEach((entry) => {
60
+ const [portStr, scheme] = entry.trim().split('/');
61
+ const port = Number(portStr);
62
+
63
+ const validSchemes = ['http', 'https'];
64
+ const normalizedScheme = scheme?.toLowerCase();
65
+
66
+ if (port > 0 && !isNaN(port)) {
67
+ serviceLinksMap.set(port, validSchemes.includes(normalizedScheme) ? normalizedScheme : null);
68
+ }
69
+ });
70
+ }
71
+
55
72
  // <CLUSTER_IP>:<PORT>/<PROTOCOL> > <TARGET PORT>
56
73
  if (isEmpty(ports)) {
57
74
  if (!isEmpty(parsedClusterIp)) {
@@ -70,12 +87,14 @@ export default {
70
87
 
71
88
  const stringPort = p.port.toString();
72
89
 
73
- if (p?.protocol === 'TCP' && (stringPort.endsWith('80') || stringPort.endsWith('443'))) {
74
- if (isMaybeSecure(p.port, p?.protocol)) {
75
- proxyUrl = row.proxyUrl('https', p.port);
76
- } else {
77
- proxyUrl = row.proxyUrl('http', p.port);
78
- }
90
+ const isDefaultPort = stringPort.endsWith('80') || stringPort.endsWith('443');
91
+ const hasServiceLink = serviceLinksMap.has(p.port);
92
+
93
+ if (p?.protocol === 'TCP' && (isDefaultPort || hasServiceLink)) {
94
+ const explicitScheme = hasServiceLink ? serviceLinksMap.get(p.port) : null;
95
+ const scheme = explicitScheme || (isMaybeSecure(p.port, p?.protocol) ? 'https' : 'http');
96
+
97
+ proxyUrl = row.proxyUrl(scheme, p.port);
79
98
  }
80
99
 
81
100
  const clusterIpAndPort = proxyUrl ? `<a href="${ proxyUrl }" target="_blank" rel="noopener noreferrer nofollow">${ p?.name ? p.name : `${ parsedClusterIp }${ p.port }` }</a>` : `${ parsedClusterIp }${ p.port }`;
@@ -113,6 +113,39 @@ describe('component: InternalExternalIP', () => {
113
113
  });
114
114
  });
115
115
 
116
+ describe('fallback to singular IPs', () => {
117
+ it('should use singular externalIp and internalIp when arrays are not present', () => {
118
+ const wrapper = mountComponent({ row: { externalIp: '1.1.1.1', internalIp: '2.2.2.2' } });
119
+
120
+ expect(wrapper.vm.filteredExternalIps).toStrictEqual(['1.1.1.1']);
121
+ expect(wrapper.vm.filteredInternalIps).toStrictEqual(['2.2.2.2']);
122
+ expect(wrapper.find('[data-testid="external-ip"]').text()).toStrictEqual('1.1.1.1');
123
+ expect(wrapper.find('[data-testid="internal-ip"]').text()).toStrictEqual('2.2.2.2');
124
+ });
125
+
126
+ it('should ignore generic.none value for externalIp', () => {
127
+ // Assuming the mock translation for 'generic.none' returns 'generic.none'
128
+ const wrapper = mountComponent({ row: { externalIp: 'generic.none', internalIp: '2.2.2.2' } });
129
+
130
+ expect(wrapper.vm.filteredExternalIps).toStrictEqual([]);
131
+ expect(wrapper.vm.filteredInternalIps).toStrictEqual(['2.2.2.2']);
132
+ });
133
+
134
+ it('should prioritize array properties over singular strings', () => {
135
+ const wrapper = mountComponent({
136
+ row: {
137
+ externalIps: ['3.3.3.3'],
138
+ externalIp: '1.1.1.1',
139
+ internalIps: ['4.4.4.4'],
140
+ internalIp: '2.2.2.2'
141
+ }
142
+ });
143
+
144
+ expect(wrapper.vm.filteredExternalIps).toStrictEqual(['3.3.3.3']);
145
+ expect(wrapper.vm.filteredInternalIps).toStrictEqual(['4.4.4.4']);
146
+ });
147
+ });
148
+
116
149
  describe('invalid IPs', () => {
117
150
  it('should filter invalid IPs', () => {
118
151
  const wrapper = mountComponent({ row: { externalIps: ['1.1.1.1', 'not-an-ip'], internalIps: ['2.2.2.2'] } });
@@ -122,6 +155,105 @@ describe('component: InternalExternalIP', () => {
122
155
  });
123
156
  });
124
157
 
158
+ describe('isIp validation', () => {
159
+ describe('valid IPv4 addresses', () => {
160
+ it.each([
161
+ ['0.0.0.0'],
162
+ ['1.1.1.1'],
163
+ ['127.0.0.1'],
164
+ ['192.168.1.1'],
165
+ ['10.0.0.1'],
166
+ ['255.255.255.255'],
167
+ ['172.16.0.1'],
168
+ ])('should accept %s as a valid IP', (ip) => {
169
+ const wrapper = mountComponent({ row: { externalIps: [ip], internalIps: [] } });
170
+
171
+ expect(wrapper.vm.filteredExternalIps).toStrictEqual([ip]);
172
+ expect(wrapper.find('[data-testid="external-ip"]').text()).toStrictEqual(ip);
173
+ });
174
+ });
175
+
176
+ describe('valid IPv6 addresses', () => {
177
+ it.each([
178
+ ['::1'],
179
+ ['::'],
180
+ ['fe80::1'],
181
+ ['2001:0db8:85a3:0000:0000:8a2e:0370:7334'],
182
+ ['2001:db8::1'],
183
+ ['::ffff:192.168.1.1'],
184
+ ])('should accept %s as a valid IP', (ip) => {
185
+ const wrapper = mountComponent({ row: { externalIps: [ip], internalIps: [] } });
186
+
187
+ expect(wrapper.vm.filteredExternalIps).toStrictEqual([ip]);
188
+ expect(wrapper.find('[data-testid="external-ip"]').text()).toStrictEqual(ip);
189
+ });
190
+ });
191
+
192
+ describe('invalid values', () => {
193
+ it.each([
194
+ ['not-an-ip'],
195
+ ['abc.def.ghi.jkl'],
196
+ ['999.999.999.999'],
197
+ ['1.2.3'],
198
+ ['1.2.3.4.5'],
199
+ [''],
200
+ ['hello'],
201
+ ['192.168.1'],
202
+ ['192.168.1.1.1'],
203
+ ['1.2.3.4/24'],
204
+ ])('should reject %s as an invalid IP', (ip) => {
205
+ const wrapper = mountComponent({ row: { externalIps: [ip], internalIps: [] } });
206
+
207
+ expect(wrapper.vm.filteredExternalIps).toStrictEqual([]);
208
+ expect(wrapper.find('[data-testid="external-ip"]').exists()).toBe(false);
209
+ });
210
+ });
211
+
212
+ it('should filter invalid IPs from internal IPs list', () => {
213
+ const wrapper = mountComponent({ row: { externalIps: [], internalIps: ['10.0.0.1', 'bad-ip', '192.168.1.1'] } });
214
+
215
+ expect(wrapper.vm.filteredInternalIps).toStrictEqual(['10.0.0.1', '192.168.1.1']);
216
+ expect(wrapper.find('[data-testid="internal-ip"]').text()).toStrictEqual('10.0.0.1');
217
+ expect(wrapper.vm.remainingIpCount).toStrictEqual(1);
218
+ });
219
+
220
+ it('should filter all invalid IPs leaving no results', () => {
221
+ const wrapper = mountComponent({ row: { externalIps: ['not-valid', 'also-bad'], internalIps: ['nope'] } });
222
+
223
+ expect(wrapper.vm.filteredExternalIps).toStrictEqual([]);
224
+ expect(wrapper.vm.filteredInternalIps).toStrictEqual([]);
225
+ expect(wrapper.find('[data-testid="external-ip"]').exists()).toBe(false);
226
+ expect(wrapper.find('[data-testid="internal-ip"]').exists()).toBe(false);
227
+ });
228
+
229
+ it('should handle mixed valid and invalid IPs preserving only valid ones', () => {
230
+ const wrapper = mountComponent({
231
+ row: {
232
+ externalIps: ['1.1.1.1', 'garbage', '8.8.8.8', '999.0.0.1'],
233
+ internalIps: ['10.0.0.1', '', '172.16.0.1', 'abc']
234
+ }
235
+ });
236
+
237
+ expect(wrapper.vm.filteredExternalIps).toStrictEqual(['1.1.1.1', '8.8.8.8']);
238
+ expect(wrapper.vm.filteredInternalIps).toStrictEqual(['10.0.0.1', '172.16.0.1']);
239
+ expect(wrapper.find('[data-testid="external-ip"]').text()).toStrictEqual('1.1.1.1');
240
+ expect(wrapper.find('[data-testid="internal-ip"]').text()).toStrictEqual('10.0.0.1');
241
+ expect(wrapper.vm.remainingIpCount).toStrictEqual(2);
242
+ });
243
+
244
+ it('should handle undefined externalIps gracefully', () => {
245
+ const wrapper = mountComponent({ row: { internalIps: ['1.1.1.1'] } });
246
+
247
+ expect(wrapper.vm.filteredExternalIps).toStrictEqual([]);
248
+ });
249
+
250
+ it('should handle undefined internalIps gracefully', () => {
251
+ const wrapper = mountComponent({ row: { externalIps: ['1.1.1.1'] } });
252
+
253
+ expect(wrapper.vm.filteredInternalIps).toStrictEqual([]);
254
+ });
255
+ });
256
+
125
257
  describe('tooltip', () => {
126
258
  it('should display the correct tooltip text', () => {
127
259
  const wrapper = mountComponent({ row: { externalIps: ['1.1.1.1', '3.3.3.3'], internalIps: ['2.2.2.2', '4.4.4.4'] } });