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

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 (272) 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/translations/en-us.yaml +22 -5
  56. package/chart/monitoring/index.vue +10 -1
  57. package/components/ActionDropdownShell.vue +2 -1
  58. package/components/CruResourceFooter.vue +9 -5
  59. package/components/ExplorerProjectsNamespaces.vue +1 -1
  60. package/components/InstallHelmCharts.vue +2 -2
  61. package/components/LandingPagePreference.vue +14 -5
  62. package/components/Resource/Detail/Metadata/IdentifyingInformation/index.vue +15 -1
  63. package/components/Resource/Detail/Metadata/index.vue +6 -0
  64. package/components/Resource/Detail/ResourcePopover/index.vue +12 -1
  65. package/components/Resource/Detail/SpacedRow.vue +3 -1
  66. package/components/Resource/Detail/TitleBar/index.vue +10 -11
  67. package/components/ResourceList/Masthead.vue +12 -8
  68. package/components/SelectIconGrid.vue +0 -10
  69. package/components/SingleClusterInfo.vue +1 -0
  70. package/components/SortableTable/__tests__/sorting.test.ts +126 -0
  71. package/components/SortableTable/index.vue +6 -9
  72. package/components/SortableTable/selection.js +23 -5
  73. package/components/SortableTable/sorting.js +6 -3
  74. package/components/Wizard.vue +14 -13
  75. package/components/fleet/FleetBundles.vue +100 -12
  76. package/components/fleet/FleetClusterTargets/index.vue +37 -15
  77. package/components/fleet/__tests__/FleetClusterTargets.test.ts +149 -115
  78. package/components/fleet/__tests__/FleetClusters.test.ts +12 -12
  79. package/components/form/LabeledSelect.vue +20 -3
  80. package/components/form/NameNsDescription.vue +11 -0
  81. package/components/form/Security.vue +6 -2
  82. package/components/form/WorkloadPorts.vue +2 -7
  83. package/components/form/__tests__/Security.test.ts +76 -0
  84. package/components/formatter/Autoscaler.vue +4 -4
  85. package/components/formatter/ClusterKubeVersion.vue +27 -0
  86. package/components/formatter/ClusterLink.vue +1 -7
  87. package/components/formatter/ClusterProvider.vue +6 -10
  88. package/components/formatter/FleetSummaryGraph.vue +0 -3
  89. package/components/formatter/MachineSummaryGraph.vue +1 -1
  90. package/components/formatter/PodsUsage.vue +2 -2
  91. package/components/formatter/__tests__/Autoscaler.test.ts +19 -22
  92. package/components/formatter/__tests__/FleetSummaryGraph.test.ts +216 -0
  93. package/components/formatter/__tests__/PodsUsage.test.ts +6 -10
  94. package/components/nav/NamespaceFilter.vue +2 -2
  95. package/components/nav/TopLevelMenu.helper.ts +15 -3
  96. package/components/nav/TopLevelMenu.vue +16 -5
  97. package/components/nav/__tests__/TopLevelMenu.test.ts +145 -21
  98. package/components/templates/home.vue +18 -0
  99. package/components/templates/plain.vue +18 -0
  100. package/components/templates/standalone.vue +17 -0
  101. package/composables/useFormValidation.ts +93 -0
  102. package/composables/useVeeValidateField.test.ts +159 -0
  103. package/composables/useVeeValidateField.ts +67 -0
  104. package/config/pagination-table-headers.js +18 -1
  105. package/config/product/manager.js +82 -21
  106. package/config/router/routes.js +6 -0
  107. package/config/table-headers.js +20 -1
  108. package/config/types.js +2 -1
  109. package/core/__tests__/plugin-products.test.ts +904 -20
  110. package/core/plugin-products-base.ts +107 -7
  111. package/core/plugin-products.ts +4 -0
  112. package/core/plugin-types.ts +111 -1
  113. package/core/plugin.ts +15 -7
  114. package/core/productDebugger.js +9 -4
  115. package/core/types-provisioning.ts +43 -30
  116. package/core/types.ts +57 -20
  117. package/detail/__tests__/pod.test.ts +41 -0
  118. package/detail/harvesterhci.io.management.cluster.vue +6 -2
  119. package/detail/pod.vue +1 -1
  120. package/detail/provisioning.cattle.io.cluster.vue +4 -10
  121. package/edit/auth/__tests__/azuread.test.ts +217 -34
  122. package/edit/auth/azuread.vue +122 -14
  123. package/edit/auth/oidc.vue +2 -2
  124. package/edit/networking.k8s.io.ingress/DefaultBackend.vue +13 -4
  125. package/edit/networking.k8s.io.ingress/RulePath.vue +8 -4
  126. package/edit/networking.k8s.io.ingress/index.vue +75 -20
  127. package/edit/provisioning.cattle.io.cluster/__tests__/MachinePool.test.ts +104 -0
  128. package/edit/provisioning.cattle.io.cluster/index.vue +11 -7
  129. package/edit/provisioning.cattle.io.cluster/rke2.vue +8 -4
  130. package/edit/provisioning.cattle.io.cluster/tabs/MachinePool.vue +11 -0
  131. package/edit/provisioning.cattle.io.cluster/tabs/registries/RegistryConfigs.vue +37 -4
  132. package/edit/provisioning.cattle.io.cluster/tabs/registries/__tests__/RegistryConfigs.test.ts +132 -7
  133. package/edit/provisioning.cattle.io.cluster/tabs/registries/index.vue +2 -1
  134. package/edit/secret/__tests__/ssh.test.ts +5 -6
  135. package/edit/secret/basic.vue +31 -0
  136. package/edit/secret/index.vue +68 -17
  137. package/edit/secret/registry.vue +38 -0
  138. package/edit/secret/ssh.vue +29 -0
  139. package/edit/secret/tls.vue +30 -0
  140. package/edit/service.vue +4 -4
  141. package/edit/workload/Upgrading.vue +3 -3
  142. package/edit/workload/__tests__/Upgrading.test.ts +6 -9
  143. package/edit/workload/mixins/workload.js +2 -1
  144. package/list/fleet.cattle.io.bundle.vue +7 -104
  145. package/list/fleet.cattle.io.clusterregistrationtoken.vue +20 -0
  146. package/list/provisioning.cattle.io.cluster.vue +262 -180
  147. package/list/utils/management.cattle.io.cluster.utils.ts +128 -0
  148. package/mixins/__tests__/chart.test.ts +112 -0
  149. package/mixins/brand.js +2 -1
  150. package/mixins/chart.js +12 -8
  151. package/mixins/resource-fetch-api-pagination.js +41 -5
  152. package/models/__tests__/ext.cattle.io.kubeconfig.test.ts +67 -67
  153. package/models/__tests__/management.cattle.io.cluster.test.ts +1 -1
  154. package/models/__tests__/management.cattle.io.node.ts +6 -5
  155. package/models/__tests__/management.cattle.io.nodepool.ts +5 -4
  156. package/models/__tests__/provisioning.cattle.io.cluster.test.ts +32 -11
  157. package/models/base-cluster.x-k8s.io.js +26 -0
  158. package/models/cluster.js +1 -1
  159. package/models/cluster.x-k8s.io.machine.js +4 -22
  160. package/models/cluster.x-k8s.io.machinedeployment.js +2 -20
  161. package/models/cluster.x-k8s.io.machineset.js +2 -20
  162. package/models/compliance.cattle.io.clusterscan.js +130 -2
  163. package/models/ext.cattle.io.kubeconfig.ts +4 -7
  164. package/models/fleet-application.js +3 -1
  165. package/models/management.cattle.io.cluster.js +417 -40
  166. package/models/management.cattle.io.node.js +6 -4
  167. package/models/management.cattle.io.nodepool.js +1 -1
  168. package/models/networking.k8s.io.ingress.js +12 -4
  169. package/models/provisioning.cattle.io.cluster.js +47 -330
  170. package/models/rke.cattle.io.etcdsnapshot.js +1 -2
  171. package/package.json +11 -29
  172. package/pages/__tests__/readme.test.ts +49 -0
  173. package/pages/auth/setup.vue +2 -3
  174. package/pages/c/_cluster/apps/charts/__tests__/chart.test.ts +76 -0
  175. package/pages/c/_cluster/apps/charts/chart.vue +60 -8
  176. package/pages/c/_cluster/apps/charts/install.vue +10 -7
  177. package/pages/c/_cluster/explorer/__tests__/index.test.ts +23 -25
  178. package/pages/c/_cluster/explorer/index.vue +5 -49
  179. package/pages/c/_cluster/istio/__tests__/istio.index.test.ts +194 -0
  180. package/pages/c/_cluster/istio/index.vue +21 -6
  181. package/pages/c/_cluster/uiplugins/PluginInfoPanel.vue +1 -0
  182. package/pages/c/_cluster/uiplugins/__tests__/index.test.ts +719 -2
  183. package/pages/c/_cluster/uiplugins/index.vue +203 -197
  184. package/pages/diagnostic.vue +13 -17
  185. package/pages/fail-whale.vue +18 -0
  186. package/pages/home.vue +77 -260
  187. package/pages/readme.vue +88 -0
  188. package/plugins/dashboard-store/__tests__/resource-class.test.ts +88 -0
  189. package/plugins/dashboard-store/actions.js +40 -18
  190. package/plugins/dashboard-store/resource-class.js +5 -2
  191. package/plugins/steve/__tests__/subscribe.spec.ts +6 -3
  192. package/plugins/steve/steve-pagination-utils.ts +11 -3
  193. package/plugins/steve/subscribe.js +35 -5
  194. package/rancher-components/Form/LabeledInput/LabeledInput.test.ts +10 -4
  195. package/rancher-components/Form/LabeledInput/LabeledInput.vue +7 -52
  196. package/rancher-components/RcButton/RcButton.test.ts +37 -1
  197. package/rancher-components/RcButton/RcButton.vue +38 -8
  198. package/rancher-components/RcDropdown/RcDropdownTrigger.vue +10 -8
  199. package/store/__tests__/catalog.test.ts +115 -1
  200. package/store/__tests__/type-map.test.ts +556 -1
  201. package/store/action-menu.js +8 -3
  202. package/store/auth.js +1 -1
  203. package/store/aws.js +27 -16
  204. package/store/catalog.js +27 -3
  205. package/store/digitalocean.js +20 -38
  206. package/store/index.js +2 -0
  207. package/store/linode.js +25 -40
  208. package/store/pnap.js +1 -0
  209. package/store/type-map.js +111 -29
  210. package/tsconfig.paths.json +8 -8
  211. package/types/kube/kube-api.ts +14 -1
  212. package/types/rancher/steve.api.ts +12 -12
  213. package/types/resources/settings.d.ts +2 -1
  214. package/types/shell/index.d.ts +102 -2
  215. package/types/store/dashboard-store.types.ts +108 -11
  216. package/types/store/pagination.types.ts +6 -3
  217. package/utils/__tests__/alertmanagerconfig.test.ts +117 -0
  218. package/utils/__tests__/async.test.ts +87 -0
  219. package/utils/__tests__/aws.test.ts +140 -0
  220. package/utils/__tests__/banners.test.ts +176 -0
  221. package/utils/__tests__/chart.test.ts +64 -1
  222. package/utils/__tests__/color.test.ts +226 -0
  223. package/utils/__tests__/duration.test.ts +140 -0
  224. package/utils/__tests__/fleet.test.ts +340 -0
  225. package/utils/__tests__/ingress.test.ts +553 -0
  226. package/utils/__tests__/kube.test.ts +68 -0
  227. package/utils/__tests__/namespace-filter.test.ts +109 -0
  228. package/utils/__tests__/pagination-utils.test.ts +361 -0
  229. package/utils/__tests__/parse-externalid.test.ts +137 -0
  230. package/utils/__tests__/perf-setting.utils.test.ts +98 -0
  231. package/utils/__tests__/poller-sequential.test.ts +177 -0
  232. package/utils/__tests__/poller.test.ts +170 -0
  233. package/utils/__tests__/promise.test.ts +346 -0
  234. package/utils/__tests__/settings.test.ts +140 -0
  235. package/utils/__tests__/sort-utils.test.ts +301 -0
  236. package/utils/__tests__/string-utils.test.ts +798 -0
  237. package/utils/__tests__/string.test.ts +23 -1
  238. package/utils/__tests__/style.test.ts +154 -0
  239. package/utils/__tests__/svg-filter.test.ts +184 -0
  240. package/utils/__tests__/units.test.ts +417 -0
  241. package/utils/__tests__/versions.test.ts +128 -0
  242. package/utils/__tests__/xccdf.test.ts +391 -0
  243. package/utils/chart.js +36 -0
  244. package/utils/fleet.ts +13 -3
  245. package/utils/gatekeeper/__tests__/util.test.ts +174 -0
  246. package/utils/gc/__tests__/gc-interval.test.ts +119 -0
  247. package/utils/gc/__tests__/gc-root-store.test.ts +225 -0
  248. package/utils/gc/__tests__/gc-route-changed.test.ts +96 -0
  249. package/utils/gc/__tests__/gc.test.ts +487 -0
  250. package/utils/ingress.ts +9 -1
  251. package/utils/pagination-utils.ts +2 -1
  252. package/utils/string.js +25 -2
  253. package/utils/uiplugins.ts +5 -5
  254. package/utils/validators/__tests__/cluster-name.test.ts +110 -0
  255. package/utils/validators/__tests__/cron-schedule.test.ts +79 -0
  256. package/utils/validators/__tests__/index.test.ts +481 -0
  257. package/utils/validators/__tests__/kubernetes-name.test.ts +163 -0
  258. package/utils/validators/__tests__/misc-validators.test.ts +246 -0
  259. package/utils/validators/__tests__/pod-affinity.test.ts +382 -0
  260. package/utils/validators/__tests__/prometheusrule.test.ts +211 -0
  261. package/utils/validators/__tests__/role-template.test.ts +149 -0
  262. package/utils/validators/__tests__/service.test.ts +283 -0
  263. package/utils/validators/__tests__/setting.test.js +32 -0
  264. package/utils/validators/formRules/__tests__/index.test.ts +50 -0
  265. package/utils/validators/formRules/index.ts +5 -5
  266. package/utils/validators/machine-pool.ts +1 -1
  267. package/utils/validators/setting.js +18 -3
  268. package/utils/xccdf.ts +418 -0
  269. package/assets/fonts/lato/lato-v17-latin-700.woff +0 -0
  270. package/assets/fonts/lato/lato-v17-latin-700.woff2 +0 -0
  271. package/assets/fonts/lato/lato-v17-latin-regular.woff +0 -0
  272. package/assets/fonts/lato/lato-v17-latin-regular.woff2 +0 -0
@@ -1,9 +1,13 @@
1
1
  import { shallowMount, VueWrapper } from '@vue/test-utils';
2
2
  import UiPluginsPage from '@shell/pages/c/_cluster/uiplugins/index.vue';
3
- import { UI_PLUGIN_NAMESPACE } from '@shell/config/uiplugins';
3
+ import {
4
+ UI_PLUGIN_NAMESPACE,
5
+ UI_PLUGIN_CHART_ANNOTATIONS,
6
+ EXTENSIONS_INCOMPATIBILITY_TYPES
7
+ } from '@shell/config/uiplugins';
4
8
  import { CATALOG as CATALOG_ANNOTATIONS } from '@shell/config/labels-annotations';
5
9
 
6
- const t = (key: string, args: Object) => {
10
+ const t = (key: string, args?: object, raw?: boolean) => {
7
11
  if (args) {
8
12
  return `${ key } with ${ JSON.stringify(args) }`;
9
13
  }
@@ -611,4 +615,717 @@ describe('page: UI plugins/Extensions', () => {
611
615
  expect(updatePluginInstallStatusMock).toHaveBeenCalledWith('repo/plugin4', false);
612
616
  });
613
617
  });
618
+
619
+ describe('extractCertificationFlags', () => {
620
+ it('should extract all flags as true when annotations match', () => {
621
+ const annotations = {
622
+ [CATALOG_ANNOTATIONS.PRIME_ONLY]: 'true',
623
+ [CATALOG_ANNOTATIONS.EXPERIMENTAL]: 'true',
624
+ [CATALOG_ANNOTATIONS.CERTIFIED]: CATALOG_ANNOTATIONS._RANCHER,
625
+ };
626
+ const result = wrapper.vm.extractCertificationFlags(annotations);
627
+
628
+ expect(result).toStrictEqual({
629
+ primeOnly: true,
630
+ experimental: true,
631
+ certified: true,
632
+ });
633
+ });
634
+
635
+ it('should return all flags as false when annotations are empty', () => {
636
+ const result = wrapper.vm.extractCertificationFlags({});
637
+
638
+ expect(result).toStrictEqual({
639
+ primeOnly: false,
640
+ experimental: false,
641
+ certified: false,
642
+ });
643
+ });
644
+
645
+ it('should return all flags as false when annotations are undefined', () => {
646
+ const result = wrapper.vm.extractCertificationFlags(undefined);
647
+
648
+ expect(result).toStrictEqual({
649
+ primeOnly: false,
650
+ experimental: false,
651
+ certified: false,
652
+ });
653
+ });
654
+
655
+ it('should not treat non-"true" values as truthy for primeOnly and experimental', () => {
656
+ const annotations = {
657
+ [CATALOG_ANNOTATIONS.PRIME_ONLY]: 'false',
658
+ [CATALOG_ANNOTATIONS.EXPERIMENTAL]: '1',
659
+ [CATALOG_ANNOTATIONS.CERTIFIED]: 'community',
660
+ };
661
+ const result = wrapper.vm.extractCertificationFlags(annotations);
662
+
663
+ expect(result).toStrictEqual({
664
+ primeOnly: false,
665
+ experimental: false,
666
+ certified: false,
667
+ });
668
+ });
669
+
670
+ it('should handle partial annotations', () => {
671
+ const annotations = { [CATALOG_ANNOTATIONS.PRIME_ONLY]: 'true' };
672
+ const result = wrapper.vm.extractCertificationFlags(annotations);
673
+
674
+ expect(result).toStrictEqual({
675
+ primeOnly: true,
676
+ experimental: false,
677
+ certified: false,
678
+ });
679
+ });
680
+ });
681
+
682
+ describe('buildIncompatibilityMessage', () => {
683
+ it('should build a message for HOST incompatibility type', () => {
684
+ const versionData = {
685
+ version: '2.0.0',
686
+ versionIncompatibilityData: {
687
+ type: EXTENSIONS_INCOMPATIBILITY_TYPES.HOST,
688
+ cardMessageKey: 'plugins.incompatible.host',
689
+ required: 'rancher-prime',
690
+ mainHost: 'rancher-manager',
691
+ }
692
+ };
693
+ const result = wrapper.vm.buildIncompatibilityMessage(versionData);
694
+
695
+ expect(result).toBe(`plugins.incompatible.host with ${ JSON.stringify({
696
+ version: '2.0.0',
697
+ required: 'rancher-prime',
698
+ mainHost: 'rancher-manager',
699
+ }) }`);
700
+ });
701
+
702
+ it('should build a message for non-HOST incompatibility type without mainHost', () => {
703
+ const versionData = {
704
+ version: '2.0.0',
705
+ versionIncompatibilityData: {
706
+ type: EXTENSIONS_INCOMPATIBILITY_TYPES.KUBE,
707
+ cardMessageKey: 'plugins.incompatible.kube',
708
+ required: '1.25.0',
709
+ }
710
+ };
711
+ const result = wrapper.vm.buildIncompatibilityMessage(versionData);
712
+
713
+ expect(result).toBe(`plugins.incompatible.kube with ${ JSON.stringify({
714
+ version: '2.0.0',
715
+ required: '1.25.0',
716
+ }) }`);
717
+ });
718
+
719
+ it('should handle missing required field', () => {
720
+ const versionData = {
721
+ version: '2.0.0',
722
+ versionIncompatibilityData: {
723
+ type: EXTENSIONS_INCOMPATIBILITY_TYPES.UI,
724
+ cardMessageKey: 'plugins.incompatible.ui',
725
+ }
726
+ };
727
+ const result = wrapper.vm.buildIncompatibilityMessage(versionData);
728
+
729
+ expect(result).toBe(`plugins.incompatible.ui with ${ JSON.stringify({
730
+ version: '2.0.0',
731
+ required: undefined,
732
+ }) }`);
733
+ });
734
+ });
735
+
736
+ describe('mapChartToPluginItem', () => {
737
+ const createChartWrapper = (overrides = {}) => {
738
+ const store = {
739
+ getters: {
740
+ 'prefs/get': jest.fn(),
741
+ 'catalog/rawCharts': [],
742
+ 'uiplugins/plugins': [],
743
+ 'uiplugins/errors': {},
744
+ },
745
+ dispatch: () => Promise.resolve(),
746
+ };
747
+
748
+ const w = shallowMount(UiPluginsPage, {
749
+ global: {
750
+ mocks: {
751
+ $store: store,
752
+ t,
753
+ ...overrides,
754
+ },
755
+ stubs: { ActionMenu: { template: '<div />' } }
756
+ }
757
+ });
758
+
759
+ w.vm.rancherVersion = '2.9.0';
760
+ w.vm.kubeVersion = '1.28.0';
761
+ w.vm.installing = {};
762
+
763
+ return w;
764
+ };
765
+
766
+ it('should map a chart with compatible versions correctly', () => {
767
+ const w = createChartWrapper();
768
+
769
+ const chart = {
770
+ chartName: 'my-extension',
771
+ chartNameDisplay: 'My Extension',
772
+ chartDescription: 'A test extension',
773
+ id: 'repo/my-extension',
774
+ icon: 'chart-icon.svg',
775
+ versions: [{
776
+ version: '1.0.0',
777
+ appVersion: '1.0.0',
778
+ created: '2024-01-01',
779
+ annotations: {
780
+ [CATALOG_ANNOTATIONS.EXPERIMENTAL]: 'true',
781
+ [CATALOG_ANNOTATIONS.CERTIFIED]: CATALOG_ANNOTATIONS._RANCHER,
782
+ }
783
+ }]
784
+ };
785
+
786
+ const result = w.vm.mapChartToPluginItem(chart);
787
+
788
+ expect(result.name).toBe('my-extension');
789
+ expect(result.label).toBe('My Extension');
790
+ expect(result.description).toBe('A test extension');
791
+ expect(result.id).toBe('repo/my-extension');
792
+ expect(result.installed).toBe(false);
793
+ expect(result.builtin).toBe(false);
794
+ expect(result.chart).toBe(chart);
795
+ expect(result.versions).toHaveLength(1);
796
+ expect(result.displayVersion).toBe('1.0.0');
797
+ expect(result.displayVersionLabel).toBe('1.0.0');
798
+ });
799
+
800
+ it('should use chart annotation DISPLAY_NAME as label when present', () => {
801
+ const w = createChartWrapper();
802
+
803
+ const chart = {
804
+ chartName: 'my-extension',
805
+ chartNameDisplay: 'My Extension',
806
+ chartDescription: 'desc',
807
+ id: 'repo/my-extension',
808
+ versions: [{
809
+ version: '1.0.0',
810
+ annotations: { [UI_PLUGIN_CHART_ANNOTATIONS.DISPLAY_NAME]: 'Custom Display Name' }
811
+ }],
812
+ };
813
+
814
+ const result = w.vm.mapChartToPluginItem(chart);
815
+
816
+ expect(result.label).toBe('Custom Display Name');
817
+ });
818
+
819
+ it('should set installing status when plugin is being installed', () => {
820
+ const w = createChartWrapper();
821
+
822
+ w.vm.installing = { 'repo/my-extension': 'install' };
823
+
824
+ const chart = {
825
+ chartName: 'my-extension',
826
+ chartNameDisplay: 'My Extension',
827
+ chartDescription: 'desc',
828
+ id: 'repo/my-extension',
829
+ versions: [{ version: '1.0.0', annotations: {} }]
830
+ };
831
+
832
+ const result = w.vm.mapChartToPluginItem(chart);
833
+
834
+ expect(result.installing).toBe('install');
835
+ });
836
+
837
+ it('should extract certification flags from compatible version', () => {
838
+ const w = createChartWrapper();
839
+
840
+ const chart = {
841
+ chartName: 'certified-extension',
842
+ chartNameDisplay: 'Certified Extension',
843
+ chartDescription: 'desc',
844
+ id: 'repo/certified',
845
+ versions: [{
846
+ version: '1.0.0',
847
+ annotations: {
848
+ [CATALOG_ANNOTATIONS.PRIME_ONLY]: 'true',
849
+ [CATALOG_ANNOTATIONS.EXPERIMENTAL]: 'false',
850
+ [CATALOG_ANNOTATIONS.CERTIFIED]: CATALOG_ANNOTATIONS._RANCHER,
851
+ }
852
+ }]
853
+ };
854
+
855
+ const result = w.vm.mapChartToPluginItem(chart);
856
+
857
+ expect(result.primeOnly).toBe(true);
858
+ expect(result.experimental).toBe(false);
859
+ expect(result.certified).toBe(true);
860
+ });
861
+ });
862
+
863
+ describe('buildLoadedPluginItem', () => {
864
+ it('should build an item for a non-builtin loaded plugin', () => {
865
+ const plugin = {
866
+ name: 'my-plugin',
867
+ id: 'my-plugin-id',
868
+ builtin: false,
869
+ metadata: {
870
+ version: '1.2.3',
871
+ description: 'Plugin description',
872
+ icon: 'plugin-icon.svg',
873
+ rancher: { annotations: {} },
874
+ }
875
+ };
876
+
877
+ const result = wrapper.vm.buildLoadedPluginItem(plugin);
878
+
879
+ expect(result).toStrictEqual({
880
+ name: 'my-plugin',
881
+ label: 'my-plugin',
882
+ description: 'Plugin description',
883
+ icon: 'plugin-icon.svg',
884
+ id: 'my-plugin-id',
885
+ versions: [],
886
+ displayVersion: '1.2.3',
887
+ displayVersionLabel: '1.2.3',
888
+ installed: true,
889
+ installedVersion: '1.2.3',
890
+ builtin: false,
891
+ primeOnly: false,
892
+ experimental: false,
893
+ certified: false,
894
+ });
895
+ });
896
+
897
+ it('should use DISPLAY_NAME annotation as label when available', () => {
898
+ const plugin = {
899
+ name: 'my-plugin',
900
+ id: 'my-plugin-id',
901
+ builtin: false,
902
+ metadata: {
903
+ version: '1.0.0',
904
+ rancher: { annotations: { [UI_PLUGIN_CHART_ANNOTATIONS.DISPLAY_NAME]: 'Pretty Name' } }
905
+ }
906
+ };
907
+
908
+ const result = wrapper.vm.buildLoadedPluginItem(plugin);
909
+
910
+ expect(result?.label).toBe('Pretty Name');
911
+ });
912
+
913
+ it('should return null for hidden builtin plugins', () => {
914
+ const plugin = {
915
+ name: 'hidden-builtin',
916
+ id: 'hidden-id',
917
+ builtin: true,
918
+ metadata: {
919
+ version: '1.0.0',
920
+ rancher: { [UI_PLUGIN_CHART_ANNOTATIONS.HIDDEN_BUILTIN]: true, annotations: {} }
921
+ }
922
+ };
923
+
924
+ const result = wrapper.vm.buildLoadedPluginItem(plugin);
925
+
926
+ expect(result).toBeNull();
927
+ });
928
+
929
+ it('should NOT return null for visible builtin plugins', () => {
930
+ const plugin = {
931
+ name: 'visible-builtin',
932
+ id: 'visible-id',
933
+ builtin: true,
934
+ metadata: {
935
+ version: '1.0.0',
936
+ rancher: { annotations: {} }
937
+ }
938
+ };
939
+
940
+ const result = wrapper.vm.buildLoadedPluginItem(plugin);
941
+
942
+ expect(result).not.toBeNull();
943
+ expect(result?.builtin).toBe(true);
944
+ });
945
+
946
+ it('should default displayVersionLabel to "-" when version is missing', () => {
947
+ const plugin = {
948
+ name: 'no-version',
949
+ id: 'no-version-id',
950
+ builtin: false,
951
+ metadata: { rancher: { annotations: {} } }
952
+ };
953
+
954
+ const result = wrapper.vm.buildLoadedPluginItem(plugin);
955
+
956
+ expect(result?.displayVersionLabel).toBe('-');
957
+ });
958
+
959
+ it('should extract certification flags from rancher annotations', () => {
960
+ const plugin = {
961
+ name: 'certified-plugin',
962
+ id: 'cert-id',
963
+ builtin: false,
964
+ metadata: {
965
+ version: '1.0.0',
966
+ rancher: {
967
+ annotations: {
968
+ [CATALOG_ANNOTATIONS.PRIME_ONLY]: 'true',
969
+ [CATALOG_ANNOTATIONS.EXPERIMENTAL]: 'true',
970
+ [CATALOG_ANNOTATIONS.CERTIFIED]: CATALOG_ANNOTATIONS._RANCHER,
971
+ }
972
+ }
973
+ }
974
+ };
975
+
976
+ const result = wrapper.vm.buildLoadedPluginItem(plugin);
977
+
978
+ expect(result?.primeOnly).toBe(true);
979
+ expect(result?.experimental).toBe(true);
980
+ expect(result?.certified).toBe(true);
981
+ });
982
+
983
+ it('should handle metadata.rancher being a non-object', () => {
984
+ const plugin = {
985
+ name: 'legacy-plugin',
986
+ id: 'legacy-id',
987
+ builtin: false,
988
+ metadata: {
989
+ version: '1.0.0',
990
+ rancher: 'not-an-object'
991
+ }
992
+ };
993
+
994
+ const result = wrapper.vm.buildLoadedPluginItem(plugin);
995
+
996
+ expect(result?.label).toBe('legacy-plugin');
997
+ expect(result?.primeOnly).toBe(false);
998
+ });
999
+ });
1000
+
1001
+ describe('wirePluginCRToChart', () => {
1002
+ let wireWrapper: VueWrapper<any>;
1003
+
1004
+ const createWireWrapper = (apps: any[] = []) => {
1005
+ const store = {
1006
+ getters: {
1007
+ 'prefs/get': jest.fn(),
1008
+ 'catalog/rawCharts': {},
1009
+ 'uiplugins/plugins': [],
1010
+ 'uiplugins/errors': {},
1011
+ 'management/all': () => apps,
1012
+ 'i18n/withFallback': (_key: string, _fallback: null, repoName: string) => repoName,
1013
+ },
1014
+ dispatch: () => Promise.resolve(),
1015
+ };
1016
+
1017
+ const w = shallowMount(UiPluginsPage, {
1018
+ global: {
1019
+ mocks: {
1020
+ $store: store,
1021
+ t,
1022
+ },
1023
+ stubs: { ActionMenu: { template: '<div />' } }
1024
+ }
1025
+ });
1026
+
1027
+ w.vm.installing = {};
1028
+
1029
+ return w;
1030
+ };
1031
+
1032
+ beforeEach(() => {
1033
+ wireWrapper = createWireWrapper();
1034
+ });
1035
+
1036
+ it('should wire CR to matching chart when found by repo name', () => {
1037
+ const apps = [{
1038
+ metadata: {
1039
+ name: 'my-plugin',
1040
+ namespace: UI_PLUGIN_NAMESPACE,
1041
+ labels: { [CATALOG_ANNOTATIONS.CLUSTER_REPO_NAME]: 'rancher-charts' }
1042
+ },
1043
+ spec: { chart: { metadata: { annotations: {} } } }
1044
+ }];
1045
+
1046
+ wireWrapper = createWireWrapper(apps);
1047
+
1048
+ const all: any[] = [{
1049
+ name: 'my-plugin',
1050
+ chart: { repoName: 'rancher-charts' },
1051
+ installableVersions: [],
1052
+ }];
1053
+
1054
+ const pluginCR = { name: 'my-plugin', version: '1.0.0' };
1055
+
1056
+ wireWrapper.vm.wirePluginCRToChart(pluginCR, all);
1057
+
1058
+ expect(all).toHaveLength(1);
1059
+ expect(all[0].installed).toBe(true);
1060
+ expect(all[0].uiplugin).toBe(pluginCR);
1061
+ expect(all[0].installedVersion).toBe('1.0.0');
1062
+ });
1063
+
1064
+ it('should set upgrade when a newer installable version exists', () => {
1065
+ const apps = [{
1066
+ metadata: {
1067
+ name: 'my-plugin',
1068
+ namespace: UI_PLUGIN_NAMESPACE,
1069
+ labels: { [CATALOG_ANNOTATIONS.CLUSTER_REPO_NAME]: 'rancher-charts' }
1070
+ },
1071
+ spec: { chart: { metadata: { annotations: {} } } }
1072
+ }];
1073
+
1074
+ wireWrapper = createWireWrapper(apps);
1075
+
1076
+ const all: any[] = [{
1077
+ name: 'my-plugin',
1078
+ chart: { repoName: 'rancher-charts' },
1079
+ installableVersions: [{
1080
+ version: '2.0.0', appVersion: '2.0.0', annotations: {}
1081
+ }],
1082
+ }];
1083
+
1084
+ const pluginCR = { name: 'my-plugin', version: '1.0.0' };
1085
+
1086
+ wireWrapper.vm.wirePluginCRToChart(pluginCR, all);
1087
+
1088
+ expect(all[0].upgrade).toBe('2.0.0');
1089
+ });
1090
+
1091
+ it('should NOT set upgrade when installed version matches latest', () => {
1092
+ const apps = [{
1093
+ metadata: {
1094
+ name: 'my-plugin',
1095
+ namespace: UI_PLUGIN_NAMESPACE,
1096
+ labels: { [CATALOG_ANNOTATIONS.CLUSTER_REPO_NAME]: 'rancher-charts' }
1097
+ },
1098
+ spec: { chart: { metadata: { annotations: {} } } }
1099
+ }];
1100
+
1101
+ wireWrapper = createWireWrapper(apps);
1102
+
1103
+ const all: any[] = [{
1104
+ name: 'my-plugin',
1105
+ chart: { repoName: 'rancher-charts' },
1106
+ installableVersions: [{ version: '1.0.0', annotations: {} }],
1107
+ }];
1108
+
1109
+ const pluginCR = { name: 'my-plugin', version: '1.0.0' };
1110
+
1111
+ wireWrapper.vm.wirePluginCRToChart(pluginCR, all);
1112
+
1113
+ expect(all[0].upgrade).toBeUndefined();
1114
+ });
1115
+
1116
+ it('should push new item when no matching chart is found', () => {
1117
+ const apps = [{
1118
+ metadata: {
1119
+ name: 'orphan-plugin',
1120
+ namespace: UI_PLUGIN_NAMESPACE,
1121
+ labels: { [CATALOG_ANNOTATIONS.CLUSTER_REPO_NAME]: 'removed-repo' }
1122
+ },
1123
+ spec: {
1124
+ chart: {
1125
+ metadata: {
1126
+ name: 'Orphan Plugin',
1127
+ description: 'Plugin from removed repo',
1128
+ icon: 'orphan-icon.svg',
1129
+ annotations: { [UI_PLUGIN_CHART_ANNOTATIONS.DISPLAY_NAME]: 'Orphan Display Name' }
1130
+ }
1131
+ }
1132
+ }
1133
+ }];
1134
+
1135
+ wireWrapper = createWireWrapper(apps);
1136
+
1137
+ const all: any[] = [];
1138
+ const pluginCR = {
1139
+ name: 'orphan-plugin',
1140
+ version: '1.0.0',
1141
+ description: 'fallback desc',
1142
+ isDeveloper: false,
1143
+ };
1144
+
1145
+ wireWrapper.vm.wirePluginCRToChart(pluginCR, all);
1146
+
1147
+ expect(all).toHaveLength(1);
1148
+ expect(all[0].name).toBe('orphan-plugin');
1149
+ expect(all[0].label).toBe('Orphan Display Name');
1150
+ expect(all[0].description).toBe('Plugin from removed repo');
1151
+ expect(all[0].icon).toBe('orphan-icon.svg');
1152
+ expect(all[0].installed).toBe(true);
1153
+ expect(all[0].uiplugin).toBe(pluginCR);
1154
+ expect(all[0].originalRepoNameDisplay).toBe('removed-repo');
1155
+ });
1156
+
1157
+ it('should push new item with fallback values when app has no chart metadata', () => {
1158
+ wireWrapper = createWireWrapper([]);
1159
+
1160
+ const all: any[] = [];
1161
+ const pluginCR = {
1162
+ name: 'no-app-plugin',
1163
+ version: '1.0.0',
1164
+ description: 'CR description',
1165
+ isDeveloper: true,
1166
+ };
1167
+
1168
+ wireWrapper.vm.wirePluginCRToChart(pluginCR, all);
1169
+
1170
+ expect(all).toHaveLength(1);
1171
+ expect(all[0].label).toBe('no-app-plugin');
1172
+ expect(all[0].description).toBe('CR description');
1173
+ expect(all[0].isDeveloper).toBe(true);
1174
+ expect(all[0].originalRepoNameDisplay).toBeNull();
1175
+ });
1176
+
1177
+ it('should not match chart from a different repo', () => {
1178
+ const apps = [{
1179
+ metadata: {
1180
+ name: 'my-plugin',
1181
+ namespace: UI_PLUGIN_NAMESPACE,
1182
+ labels: { [CATALOG_ANNOTATIONS.CLUSTER_REPO_NAME]: 'repo-a' }
1183
+ },
1184
+ spec: { chart: { metadata: { annotations: {} } } }
1185
+ }];
1186
+
1187
+ wireWrapper = createWireWrapper(apps);
1188
+
1189
+ const all: any[] = [{
1190
+ name: 'my-plugin',
1191
+ chart: { repoName: 'repo-b' },
1192
+ installableVersions: [],
1193
+ }];
1194
+
1195
+ const pluginCR = { name: 'my-plugin', version: '1.0.0' };
1196
+
1197
+ wireWrapper.vm.wirePluginCRToChart(pluginCR, all);
1198
+
1199
+ expect(all).toHaveLength(2);
1200
+ expect(all[0].installed).toBeUndefined();
1201
+ expect(all[1].installed).toBe(true);
1202
+ expect(all[1].uiplugin).toBe(pluginCR);
1203
+ });
1204
+ });
1205
+
1206
+ describe('mergePluginErrors', () => {
1207
+ it('should set installedError for string UI errors', () => {
1208
+ const store = {
1209
+ getters: {
1210
+ 'prefs/get': jest.fn(),
1211
+ 'catalog/rawCharts': [],
1212
+ 'uiplugins/plugins': [],
1213
+ 'uiplugins/errors': { 'my-plugin': 'plugins.error.load' },
1214
+ },
1215
+ dispatch: () => Promise.resolve(),
1216
+ };
1217
+
1218
+ const w = shallowMount(UiPluginsPage, {
1219
+ global: {
1220
+ mocks: { $store: store, t },
1221
+ stubs: { ActionMenu: { template: '<div />' } }
1222
+ }
1223
+ });
1224
+
1225
+ w.vm.errors = {};
1226
+
1227
+ const all = [{ name: 'my-plugin', id: 'repo/my-plugin' }];
1228
+
1229
+ w.vm.mergePluginErrors(all);
1230
+
1231
+ expect(all[0]).toHaveProperty('installedError', 'plugins.error.load');
1232
+ });
1233
+
1234
+ it('should set empty installedError for non-string UI errors', () => {
1235
+ const store = {
1236
+ getters: {
1237
+ 'prefs/get': jest.fn(),
1238
+ 'catalog/rawCharts': [],
1239
+ 'uiplugins/plugins': [],
1240
+ 'uiplugins/errors': { 'my-plugin': true },
1241
+ },
1242
+ dispatch: () => Promise.resolve(),
1243
+ };
1244
+
1245
+ const w = shallowMount(UiPluginsPage, {
1246
+ global: {
1247
+ mocks: { $store: store, t },
1248
+ stubs: { ActionMenu: { template: '<div />' } }
1249
+ }
1250
+ });
1251
+
1252
+ w.vm.errors = {};
1253
+
1254
+ const all = [{ name: 'my-plugin', id: 'repo/my-plugin' }];
1255
+
1256
+ w.vm.mergePluginErrors(all);
1257
+
1258
+ expect(all[0]).toHaveProperty('installedError', '');
1259
+ });
1260
+
1261
+ it('should set helmError from component errors', () => {
1262
+ const all = [{ name: 'my-plugin', id: 'repo/my-plugin' }];
1263
+
1264
+ wrapper.vm.errors = { 'repo/my-plugin': 'helm failed' };
1265
+ wrapper.vm.mergePluginErrors(all);
1266
+
1267
+ expect(all[0]).toHaveProperty('helmError', true);
1268
+ });
1269
+
1270
+ it('should not modify plugins that have no matching errors', () => {
1271
+ const store = {
1272
+ getters: {
1273
+ 'prefs/get': jest.fn(),
1274
+ 'catalog/rawCharts': [],
1275
+ 'uiplugins/plugins': [],
1276
+ 'uiplugins/errors': { 'other-plugin': 'some error' },
1277
+ },
1278
+ dispatch: () => Promise.resolve(),
1279
+ };
1280
+
1281
+ const w = shallowMount(UiPluginsPage, {
1282
+ global: {
1283
+ mocks: { $store: store, t },
1284
+ stubs: { ActionMenu: { template: '<div />' } }
1285
+ }
1286
+ });
1287
+
1288
+ w.vm.errors = { 'other-id': 'error' };
1289
+
1290
+ const all = [{ name: 'my-plugin', id: 'repo/my-plugin' }];
1291
+
1292
+ w.vm.mergePluginErrors(all);
1293
+
1294
+ expect(all[0]).not.toHaveProperty('installedError');
1295
+ expect(all[0]).not.toHaveProperty('helmError');
1296
+ });
1297
+
1298
+ it('should handle both UI errors and helm errors for different plugins', () => {
1299
+ const store = {
1300
+ getters: {
1301
+ 'prefs/get': jest.fn(),
1302
+ 'catalog/rawCharts': [],
1303
+ 'uiplugins/plugins': [],
1304
+ 'uiplugins/errors': { 'plugin-a': 'plugins.error.load' },
1305
+ },
1306
+ dispatch: () => Promise.resolve(),
1307
+ };
1308
+
1309
+ const w = shallowMount(UiPluginsPage, {
1310
+ global: {
1311
+ mocks: { $store: store, t },
1312
+ stubs: { ActionMenu: { template: '<div />' } }
1313
+ }
1314
+ });
1315
+
1316
+ w.vm.errors = { 'repo/plugin-b': 'helm failure' };
1317
+
1318
+ const all = [
1319
+ { name: 'plugin-a', id: 'repo/plugin-a' },
1320
+ { name: 'plugin-b', id: 'repo/plugin-b' },
1321
+ ];
1322
+
1323
+ w.vm.mergePluginErrors(all);
1324
+
1325
+ expect(all[0]).toHaveProperty('installedError', 'plugins.error.load');
1326
+ expect(all[0]).not.toHaveProperty('helmError');
1327
+ expect(all[1]).not.toHaveProperty('installedError');
1328
+ expect(all[1]).toHaveProperty('helmError', true);
1329
+ });
1330
+ });
614
1331
  });