@rancher/shell 3.0.5-rc.5 → 3.0.5-rc.7

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 (361) hide show
  1. package/assets/data/aws-regions.json +1 -0
  2. package/assets/images/key.svg +17 -0
  3. package/assets/styles/base/_spacing.scss +2 -2
  4. package/assets/styles/base/_variables.scss +17 -11
  5. package/assets/styles/global/_form.scss +1 -1
  6. package/assets/styles/global/_labeled-input.scss +1 -1
  7. package/assets/styles/themes/_dark.scss +5 -0
  8. package/assets/styles/themes/_light.scss +11 -2
  9. package/assets/styles/vendor/vue-select.scss +1 -1
  10. package/assets/translations/en-us.yaml +426 -64
  11. package/assets/translations/zh-hans.yaml +3 -4
  12. package/cloud-credential/gcp.vue +9 -1
  13. package/components/AppModal.vue +2 -0
  14. package/components/CodeMirror.vue +2 -2
  15. package/components/ConfigMapSettings/Settings.vue +377 -0
  16. package/components/ConfigMapSettings/index.vue +354 -0
  17. package/components/CruResource.vue +1 -2
  18. package/components/DetailText.vue +61 -11
  19. package/components/Drawer/Chrome.vue +115 -0
  20. package/components/Drawer/ResourceDetailDrawer/ConfigTab.vue +61 -0
  21. package/components/Drawer/ResourceDetailDrawer/YamlTab.vue +48 -0
  22. package/components/Drawer/ResourceDetailDrawer/__tests__/ConfigTab.test.ts +54 -0
  23. package/components/Drawer/ResourceDetailDrawer/__tests__/YamlTab.test.ts +80 -0
  24. package/components/Drawer/ResourceDetailDrawer/__tests__/composables.test.ts +106 -0
  25. package/components/Drawer/ResourceDetailDrawer/__tests__/helpers.test.ts +42 -0
  26. package/components/Drawer/ResourceDetailDrawer/composables.ts +53 -0
  27. package/components/Drawer/ResourceDetailDrawer/helpers.ts +10 -0
  28. package/components/Drawer/ResourceDetailDrawer/index.vue +111 -0
  29. package/components/GrowlManager.vue +16 -15
  30. package/components/IconOrSvg.vue +5 -0
  31. package/components/KeyValueView.vue +1 -1
  32. package/components/Loading.vue +1 -1
  33. package/components/LocaleSelector.vue +9 -1
  34. package/components/PaginatedResourceTable.vue +46 -1
  35. package/components/ProgressBarMulti.vue +1 -0
  36. package/components/PromptModal.vue +6 -1
  37. package/components/PromptRestore.vue +22 -44
  38. package/components/RelatedResources.vue +4 -12
  39. package/components/Resource/Detail/Additional.vue +46 -0
  40. package/components/Resource/Detail/Metadata/Annotations/__tests__/index.test.ts +1 -1
  41. package/components/Resource/Detail/Metadata/Annotations/index.vue +5 -0
  42. package/components/Resource/Detail/Metadata/IdentifyingInformation/__tests__/identifying-fields.test.ts +223 -0
  43. package/components/Resource/Detail/Metadata/IdentifyingInformation/composable.ts +47 -256
  44. package/components/Resource/Detail/Metadata/IdentifyingInformation/identifying-fields.ts +317 -0
  45. package/components/Resource/Detail/Metadata/IdentifyingInformation/index.vue +34 -5
  46. package/components/Resource/Detail/Metadata/KeyValue.vue +32 -22
  47. package/components/Resource/Detail/Metadata/Labels/__tests__/index.test.ts +1 -1
  48. package/components/Resource/Detail/Metadata/Labels/index.vue +4 -0
  49. package/components/Resource/Detail/Metadata/Rectangle.vue +3 -1
  50. package/components/Resource/Detail/Metadata/__tests__/KeyValue.test.ts +1 -1
  51. package/components/Resource/Detail/Metadata/__tests__/Rectangle.test.ts +1 -1
  52. package/components/Resource/Detail/Metadata/__tests__/composables.test.ts +75 -0
  53. package/components/Resource/Detail/Metadata/composables.ts +60 -11
  54. package/components/Resource/Detail/Metadata/index.vue +12 -5
  55. package/components/Resource/Detail/Page.vue +15 -0
  56. package/components/Resource/Detail/ResourceRow.vue +37 -18
  57. package/components/Resource/Detail/ResourceTabs/ConfigMapDataTab/__tests__/composables.test.ts +29 -0
  58. package/components/Resource/Detail/ResourceTabs/ConfigMapDataTab/__tests__/index.test.ts +48 -0
  59. package/components/Resource/Detail/ResourceTabs/ConfigMapDataTab/composables.ts +31 -0
  60. package/components/Resource/Detail/ResourceTabs/ConfigMapDataTab/index.vue +50 -0
  61. package/components/Resource/Detail/ResourceTabs/KnownHostsTab/__tests__/composables.test.ts +66 -0
  62. package/components/Resource/Detail/ResourceTabs/KnownHostsTab/composables.ts +21 -0
  63. package/components/Resource/Detail/ResourceTabs/KnownHostsTab/index.vue +31 -0
  64. package/components/Resource/Detail/ResourceTabs/SecretDataTab/Basic.vue +45 -0
  65. package/components/Resource/Detail/ResourceTabs/SecretDataTab/BasicAuth.vue +31 -0
  66. package/components/Resource/Detail/ResourceTabs/SecretDataTab/Certificate.vue +31 -0
  67. package/components/Resource/Detail/ResourceTabs/SecretDataTab/Registry.vue +22 -0
  68. package/components/Resource/Detail/ResourceTabs/SecretDataTab/ServiceAccountToken.vue +31 -0
  69. package/components/Resource/Detail/ResourceTabs/SecretDataTab/Ssh.vue +32 -0
  70. package/components/Resource/Detail/ResourceTabs/SecretDataTab/__tests__/Basic.test.ts +40 -0
  71. package/components/Resource/Detail/ResourceTabs/SecretDataTab/__tests__/BasicAuth.test.ts +33 -0
  72. package/components/Resource/Detail/ResourceTabs/SecretDataTab/__tests__/Certificate.test.ts +33 -0
  73. package/components/Resource/Detail/ResourceTabs/SecretDataTab/__tests__/Registry.test.ts +27 -0
  74. package/components/Resource/Detail/ResourceTabs/SecretDataTab/__tests__/ServiceAccountToken.test.ts +33 -0
  75. package/components/Resource/Detail/ResourceTabs/SecretDataTab/__tests__/Ssh.test.ts +33 -0
  76. package/components/Resource/Detail/ResourceTabs/SecretDataTab/__tests__/auth-types.test.ts +186 -0
  77. package/components/Resource/Detail/ResourceTabs/SecretDataTab/__tests__/composables.test.ts +102 -0
  78. package/components/Resource/Detail/ResourceTabs/SecretDataTab/auth-types.ts +109 -0
  79. package/components/Resource/Detail/ResourceTabs/SecretDataTab/composeables.ts +52 -0
  80. package/components/Resource/Detail/ResourceTabs/SecretDataTab/index.vue +71 -0
  81. package/components/Resource/Detail/SpacedRow.vue +1 -1
  82. package/components/Resource/Detail/TitleBar/Title.vue +2 -1
  83. package/components/Resource/Detail/TitleBar/__tests__/Title.test.ts +1 -1
  84. package/components/Resource/Detail/TitleBar/__tests__/Top.test.ts +1 -1
  85. package/components/Resource/Detail/TitleBar/__tests__/composables.test.ts +63 -0
  86. package/components/Resource/Detail/TitleBar/__tests__/index.test.ts +1 -1
  87. package/components/Resource/Detail/TitleBar/composables.ts +45 -0
  88. package/components/Resource/Detail/TitleBar/index.vue +85 -13
  89. package/components/Resource/Detail/composables.ts +45 -0
  90. package/components/ResourceDetail/Masthead/__tests__/index.test.ts +70 -0
  91. package/components/ResourceDetail/{__tests__/Masthead.test.ts → Masthead/__tests__/legacy.test.ts} +3 -3
  92. package/components/ResourceDetail/Masthead/index.vue +65 -0
  93. package/components/ResourceDetail/Masthead/latest.vue +44 -0
  94. package/components/ResourceDetail/{Masthead.vue → Masthead/legacy.vue} +1 -1
  95. package/components/ResourceDetail/__tests__/index.test.ts +26 -5
  96. package/components/ResourceDetail/index.vue +33 -17
  97. package/components/ResourceDetail/legacy.vue +18 -1
  98. package/components/ResourceList/Masthead.vue +6 -0
  99. package/components/ResourceList/index.vue +1 -0
  100. package/components/ResourceTable.vue +6 -1
  101. package/components/ResourceYaml.vue +15 -2
  102. package/components/RichTranslation.vue +106 -0
  103. package/components/SlideInPanelManager.vue +46 -11
  104. package/components/SortableTable/index.vue +1 -1
  105. package/components/SortableTable/selection.js +0 -1
  106. package/components/StateDot/index.vue +28 -0
  107. package/components/Tabbed/index.vue +17 -16
  108. package/components/Wizard.vue +4 -2
  109. package/components/__tests__/ConfigMapSettings.test.ts +376 -0
  110. package/components/__tests__/GrowlManager.test.ts +0 -25
  111. package/components/__tests__/PromptRestore.test.ts +1 -65
  112. package/components/__tests__/RichTranslation.test.ts +115 -0
  113. package/components/auth/login/ldap.vue +1 -1
  114. package/components/fleet/FleetApplications.vue +0 -7
  115. package/components/fleet/FleetClusterTargets/TargetsList.vue +66 -0
  116. package/components/fleet/FleetClusterTargets/index.vue +455 -0
  117. package/components/fleet/FleetClusters.vue +25 -6
  118. package/components/fleet/FleetGitRepoPaths.vue +476 -0
  119. package/components/fleet/FleetHelmOps.vue +8 -0
  120. package/components/fleet/FleetRepos.vue +1 -6
  121. package/components/fleet/FleetResources.vue +4 -5
  122. package/components/fleet/FleetValuesFrom.vue +295 -0
  123. package/components/fleet/__tests__/FleetClusterTargets.test.ts +1224 -0
  124. package/components/fleet/__tests__/FleetGitRepoPaths.test.ts +265 -0
  125. package/components/fleet/__tests__/FleetOCIStorageSecret.test.ts +13 -13
  126. package/components/fleet/__tests__/FleetValuesFrom.test.ts +300 -0
  127. package/components/fleet/dashboard/ResourceCard.vue +1 -0
  128. package/components/fleet/dashboard/ResourceCardSummary.vue +1 -5
  129. package/components/fleet/dashboard/ResourceDetails.vue +8 -10
  130. package/components/fleet/dashboard/ResourcePanel.vue +17 -9
  131. package/components/form/ArrayList.vue +13 -2
  132. package/components/form/ChangePassword.vue +3 -1
  133. package/components/form/FileImageSelector.vue +1 -1
  134. package/components/form/Footer.vue +10 -4
  135. package/components/form/KeyValue.vue +81 -43
  136. package/components/form/LabeledSelect.vue +56 -16
  137. package/components/form/Labels.vue +90 -17
  138. package/components/form/MatchExpressions.vue +46 -5
  139. package/components/form/NameNsDescription.vue +2 -1
  140. package/components/form/Networking.vue +24 -19
  141. package/components/form/ResourceLabeledSelect.vue +4 -3
  142. package/components/form/ResourceSelector.vue +1 -0
  143. package/components/form/ResourceTabs/index.vue +5 -0
  144. package/components/form/SecretSelector.vue +9 -2
  145. package/components/form/Select.vue +57 -19
  146. package/components/form/SelectOrCreateAuthSecret.vue +6 -3
  147. package/components/form/SimpleSecretSelector.vue +9 -2
  148. package/components/form/Taints.vue +21 -2
  149. package/components/form/UnitInput.vue +8 -0
  150. package/components/form/ValueFromResource.vue +1 -1
  151. package/components/form/__tests__/LabeledSelect.test.ts +8 -4
  152. package/components/form/__tests__/Labels.test.ts +360 -0
  153. package/components/form/__tests__/MatchExpressions.test.ts +16 -13
  154. package/components/form/__tests__/Networking.test.ts +116 -0
  155. package/components/form/__tests__/Select.test.ts +5 -2
  156. package/components/formatter/FleetApplicationSource.vue +1 -1
  157. package/components/formatter/PodImages.vue +1 -1
  158. package/components/formatter/WorkloadHealthScale.vue +1 -1
  159. package/components/formatter/__tests__/LiveDate.test.ts +10 -2
  160. package/components/google/AccountAccess.vue +209 -0
  161. package/components/google/types/gcp.d.ts +136 -0
  162. package/components/google/types/index.d.ts +101 -0
  163. package/components/google/util/__mocks__/gcp.ts +465 -0
  164. package/components/google/util/formatter.ts +82 -0
  165. package/components/google/util/gcp.ts +134 -0
  166. package/components/google/util/index.d.ts +11 -0
  167. package/components/nav/Favorite.vue +1 -1
  168. package/components/nav/Group.vue +71 -45
  169. package/components/nav/Header.vue +5 -1
  170. package/components/nav/NamespaceFilter.vue +13 -1
  171. package/components/nav/NotificationCenter/Notification.vue +510 -0
  172. package/components/nav/NotificationCenter/NotificationHeader.vue +112 -0
  173. package/components/nav/NotificationCenter/index.vue +148 -0
  174. package/composables/drawer.ts +26 -0
  175. package/composables/resources.test.ts +63 -0
  176. package/composables/resources.ts +38 -0
  177. package/composables/useIsNewDetailPageEnabled.ts +17 -0
  178. package/config/labels-annotations.js +8 -0
  179. package/config/pagination-table-headers.js +8 -1
  180. package/config/product/auth.js +16 -1
  181. package/config/product/{cis.js → compliance.js} +23 -26
  182. package/config/product/explorer.js +32 -3
  183. package/config/product/fleet.js +7 -0
  184. package/config/product/manager.js +0 -1
  185. package/config/product/settings.js +22 -11
  186. package/config/query-params.js +13 -0
  187. package/config/roles.ts +1 -1
  188. package/config/router/navigation-guards/authentication.js +51 -2
  189. package/config/router/routes.js +47 -31
  190. package/config/settings.ts +21 -3
  191. package/config/store.js +2 -0
  192. package/config/system-namespaces.js +1 -1
  193. package/config/table-headers.js +32 -3
  194. package/config/types.js +16 -7
  195. package/config/version.js +1 -1
  196. package/core/plugin.ts +32 -7
  197. package/core/types.ts +18 -1
  198. package/detail/{cis.cattle.io.clusterscan.vue → compliance.cattle.io.clusterscan.vue} +22 -18
  199. package/detail/management.cattle.io.fleetworkspace.vue +18 -27
  200. package/detail/management.cattle.io.oidcclient.vue +369 -0
  201. package/detail/node.vue +2 -2
  202. package/detail/pod.vue +2 -2
  203. package/detail/provisioning.cattle.io.cluster.vue +3 -47
  204. package/detail/service.vue +10 -1
  205. package/detail/workload/index.vue +8 -2
  206. package/dialog/ExtensionCatalogUninstallDialog.vue +7 -4
  207. package/dialog/GenericPrompt.vue +1 -1
  208. package/dialog/ImportDialog.vue +8 -8
  209. package/dialog/OidcClientSecretDialog.vue +117 -0
  210. package/dialog/RotateEncryptionKeyDialog.vue +10 -30
  211. package/edit/__tests__/cis.cattle.io.clusterscan.test.ts +3 -3
  212. package/edit/__tests__/fleet.cattle.io.gitrepo.test.ts +5 -2
  213. package/edit/auth/ldap/__tests__/config.test.ts +14 -0
  214. package/edit/auth/ldap/config.vue +24 -0
  215. package/edit/autoscaling.horizontalpodautoscaler/index.vue +4 -1
  216. package/edit/{cis.cattle.io.clusterscan.vue → compliance.cattle.io.clusterscan.vue} +30 -31
  217. package/edit/{cis.cattle.io.clusterscanbenchmark.vue → compliance.cattle.io.clusterscanbenchmark.vue} +4 -4
  218. package/edit/{cis.cattle.io.clusterscanprofile.vue → compliance.cattle.io.clusterscanprofile.vue} +5 -5
  219. package/edit/configmap.vue +8 -2
  220. package/edit/constraints.gatekeeper.sh.constraint/index.vue +1 -0
  221. package/edit/fleet.cattle.io.gitrepo.vue +44 -222
  222. package/edit/fleet.cattle.io.helmop.vue +44 -269
  223. package/edit/helm.cattle.io.projecthelmchart.vue +1 -0
  224. package/edit/k8s.cni.cncf.io.networkattachmentdefinition.vue +1 -0
  225. package/edit/logging-flow/index.vue +1 -0
  226. package/edit/logging.banzaicloud.io.output/index.vue +1 -0
  227. package/edit/management.cattle.io.fleetworkspace.vue +1 -0
  228. package/edit/management.cattle.io.oidcclient.vue +162 -0
  229. package/edit/management.cattle.io.project.vue +4 -1
  230. package/edit/monitoring.coreos.com.alertmanagerconfig/index.vue +1 -1
  231. package/edit/monitoring.coreos.com.alertmanagerconfig/receiverConfig.vue +5 -0
  232. package/edit/monitoring.coreos.com.prometheusrule/index.vue +1 -0
  233. package/edit/monitoring.coreos.com.receiver/auth.vue +30 -30
  234. package/edit/monitoring.coreos.com.receiver/index.vue +1 -0
  235. package/edit/monitoring.coreos.com.receiver/types/email.vue +1 -1
  236. package/edit/monitoring.coreos.com.route.vue +1 -0
  237. package/edit/namespace.vue +1 -0
  238. package/edit/networking.istio.io.destinationrule/index.vue +4 -1
  239. package/edit/networking.k8s.io.ingress/Certificate.vue +12 -12
  240. package/edit/networking.k8s.io.ingress/__tests__/Certificate.test.ts +165 -0
  241. package/edit/networking.k8s.io.ingress/index.vue +4 -1
  242. package/edit/networking.k8s.io.networkpolicy/PolicyRules.vue +7 -2
  243. package/edit/networking.k8s.io.networkpolicy/index.vue +6 -2
  244. package/edit/node.vue +1 -0
  245. package/edit/persistentvolume/index.vue +4 -1
  246. package/edit/provisioning.cattle.io.cluster/__tests__/rke2.test.ts +3 -2
  247. package/edit/provisioning.cattle.io.cluster/rke2.vue +516 -426
  248. package/edit/provisioning.cattle.io.cluster/tabs/Basics.vue +48 -39
  249. package/edit/provisioning.cattle.io.cluster/tabs/MachinePool.vue +5 -0
  250. package/edit/provisioning.cattle.io.cluster/tabs/registries/RegistryConfigs.vue +2 -2
  251. package/edit/resources.cattle.io.restore.vue +1 -1
  252. package/edit/secret/basic.vue +1 -0
  253. package/edit/secret/index.vue +127 -15
  254. package/edit/service.vue +4 -1
  255. package/edit/serviceaccount.vue +4 -1
  256. package/edit/storage.k8s.io.storageclass/index.vue +4 -1
  257. package/edit/workload/index.vue +5 -0
  258. package/list/{cis.cattle.io.clusterscan.vue → compliance.cattle.io.clusterscan.vue} +2 -2
  259. package/list/management.cattle.io.oidcclient.vue +108 -0
  260. package/list/node.vue +2 -0
  261. package/list/projectsecret.vue +345 -0
  262. package/list/secret.vue +109 -0
  263. package/machine-config/amazonec2.vue +3 -24
  264. package/machine-config/components/GCEImage.vue +374 -0
  265. package/machine-config/google.vue +617 -0
  266. package/mixins/__tests__/brand.spec.ts +170 -0
  267. package/mixins/brand.js +16 -17
  268. package/mixins/create-edit-view/impl.js +10 -1
  269. package/mixins/create-edit-view/index.js +5 -0
  270. package/mixins/resource-fetch-api-pagination.js +24 -8
  271. package/mixins/resource-fetch.js +3 -1
  272. package/mixins/vue-select-overrides.js +1 -0
  273. package/models/cluster.x-k8s.io.machinedeployment.js +11 -2
  274. package/models/{cis.cattle.io.clusterscan.js → compliance.cattle.io.clusterscan.js} +8 -8
  275. package/models/{cis.cattle.io.clusterscanbenchmark.js → compliance.cattle.io.clusterscanbenchmark.js} +1 -1
  276. package/models/{cis.cattle.io.clusterscanprofile.js → compliance.cattle.io.clusterscanprofile.js} +5 -5
  277. package/models/{cis.cattle.io.clusterscanreport.js → compliance.cattle.io.clusterscanreport.js} +1 -1
  278. package/models/fleet-application.js +8 -79
  279. package/models/fleet.cattle.io.cluster.js +13 -2
  280. package/models/fleet.cattle.io.gitrepo.js +2 -2
  281. package/models/fleet.cattle.io.helmop.js +9 -39
  282. package/models/management.cattle.io.fleetworkspace.js +2 -1
  283. package/models/management.cattle.io.oidcclient.js +18 -0
  284. package/models/management.cattle.io.registration.js +3 -0
  285. package/models/provisioning.cattle.io.cluster.js +29 -33
  286. package/models/secret.js +157 -2
  287. package/models/service.js +4 -0
  288. package/models/workload.js +5 -0
  289. package/package.json +2 -2
  290. package/pages/about.vue +4 -58
  291. package/pages/auth/login.vue +1 -1
  292. package/pages/c/_cluster/apps/charts/AddRepoLink.vue +0 -1
  293. package/pages/c/_cluster/apps/charts/index.vue +296 -81
  294. package/pages/c/_cluster/auth/user.retention/index.vue +87 -78
  295. package/pages/c/_cluster/explorer/index.vue +3 -3
  296. package/pages/c/_cluster/explorer/projectsecret.vue +34 -0
  297. package/pages/c/_cluster/explorer/tools/pages/_page.vue +0 -1
  298. package/pages/c/_cluster/fleet/application/create.vue +3 -2
  299. package/pages/c/_cluster/fleet/index.vue +94 -57
  300. package/pages/c/_cluster/fleet/settings/index.vue +229 -0
  301. package/pages/c/_cluster/longhorn/index.vue +5 -2
  302. package/pages/c/_cluster/uiplugins/CatalogList/index.vue +16 -1
  303. package/pages/c/_cluster/uiplugins/PluginInfoPanel.vue +2 -2
  304. package/pages/explorer/resource/detail/configmap.vue +30 -7
  305. package/pages/explorer/resource/detail/projectsecret.vue +9 -0
  306. package/pages/explorer/resource/detail/secret.vue +63 -0
  307. package/pages/home.vue +9 -55
  308. package/pages/support/index.vue +4 -6
  309. package/plugins/dashboard-store/__tests__/normalize.test.ts +223 -0
  310. package/plugins/dashboard-store/__tests__/resource-class.test.ts +191 -0
  311. package/plugins/dashboard-store/__tests__/utils/normalize-usecases.ts +1526 -0
  312. package/plugins/dashboard-store/actions.js +19 -5
  313. package/plugins/dashboard-store/getters.js +4 -0
  314. package/plugins/dashboard-store/normalize.js +29 -17
  315. package/plugins/dashboard-store/resource-class.js +68 -19
  316. package/plugins/steve/steve-pagination-utils.ts +38 -19
  317. package/plugins/steve/subscribe.js +6 -1
  318. package/rancher-components/Banner/Banner.vue +13 -0
  319. package/rancher-components/Form/Checkbox/Checkbox.vue +9 -4
  320. package/rancher-components/Form/LabeledInput/LabeledInput.vue +1 -1
  321. package/rancher-components/LabeledTooltip/LabeledTooltip.vue +1 -0
  322. package/rancher-components/RcItemCard/RcItemCard.vue +8 -3
  323. package/store/auth.js +2 -0
  324. package/store/catalog.js +23 -1
  325. package/store/growl.js +97 -8
  326. package/store/index.js +6 -0
  327. package/store/notifications.ts +426 -0
  328. package/store/prefs.js +0 -1
  329. package/store/type-map.js +19 -16
  330. package/store/uiplugins.ts +15 -1
  331. package/types/fleet.d.ts +24 -0
  332. package/types/kube/kube-api.ts +12 -0
  333. package/types/notifications/index.ts +74 -0
  334. package/types/shell/index.d.ts +661 -589
  335. package/types/store/dashboard-store.types.ts +16 -0
  336. package/types/store/pagination.types.ts +16 -6
  337. package/utils/__tests__/create-yaml.test.ts +235 -0
  338. package/utils/__tests__/fleet.test.ts +148 -0
  339. package/utils/__tests__/object.test.ts +54 -1
  340. package/utils/__tests__/string.test.ts +273 -1
  341. package/utils/__tests__/time.test.ts +31 -0
  342. package/utils/auth.js +9 -2
  343. package/utils/create-yaml.js +103 -9
  344. package/utils/crypto/encryption.ts +103 -0
  345. package/utils/cspAdaptor.ts +51 -0
  346. package/utils/fleet.ts +54 -65
  347. package/utils/object.js +36 -0
  348. package/utils/pagination-utils.ts +19 -1
  349. package/utils/release-notes.ts +48 -0
  350. package/utils/selector-typed.ts +7 -2
  351. package/utils/string.js +24 -0
  352. package/utils/{time.js → time.ts} +25 -6
  353. package/utils/uiplugins.ts +22 -0
  354. package/utils/validators/formRules/index.ts +3 -0
  355. package/components/Resource/Detail/TitleBar/composable.ts +0 -31
  356. package/config/product/legacy.js +0 -62
  357. package/models/etcdbackup.js +0 -45
  358. package/pages/c/_cluster/legacy/pages/_page.vue +0 -29
  359. package/pages/c/_cluster/legacy/project/_page.vue +0 -57
  360. package/pages/c/_cluster/legacy/project/index.vue +0 -32
  361. package/pages/c/_cluster/legacy/project/pipelines.vue +0 -96
@@ -0,0 +1,186 @@
1
+ import * as authTypes from '@shell/components/Resource/Detail/ResourceTabs/SecretDataTab/auth-types';
2
+ import * as crypto from '@shell/utils/crypto';
3
+
4
+ jest.mock('@shell/utils/crypto');
5
+ jest.mock('vuex', () => ({ useStore: jest.fn() }));
6
+
7
+ describe('composables: SecretDataTab/auth-types', () => {
8
+ const resource = { _type: 'RESOURCE', data: { type: 'data' } };
9
+ const base64 = '64 decoded string';
10
+ const base64DecodeSpy = jest.spyOn(crypto, 'base64Decode');
11
+
12
+ beforeEach(() => {
13
+ jest.clearAllMocks();
14
+ });
15
+
16
+ describe('useSecretInfo', () => {
17
+ it('should return the secret type and data set in resource', () => {
18
+ const info = authTypes.useSecretInfo(resource);
19
+
20
+ expect(info.value.secretType).toStrictEqual(resource._type);
21
+ expect(info.value.secretData).toStrictEqual(resource.data);
22
+ });
23
+
24
+ it('should return the an empty object for secretData if not set on resource', () => {
25
+ const info = authTypes.useSecretInfo({ _type: 'RESOURCE' });
26
+
27
+ expect(info.value.secretType).toStrictEqual(resource._type);
28
+ expect(info.value.secretData).toStrictEqual({});
29
+ });
30
+ });
31
+
32
+ describe('useSecretRows', () => {
33
+ it('should return empty array if data is not set', () => {
34
+ const rows = authTypes.useSecretRows({});
35
+
36
+ expect(rows.value).toStrictEqual([]);
37
+ });
38
+
39
+ it('should return a base64Decoded key/value if data is set', () => {
40
+ base64DecodeSpy.mockReturnValue(base64);
41
+ const row = { mockKey: 'value' };
42
+ const rows = authTypes.useSecretRows({ data: row });
43
+
44
+ expect(rows.value[0].key).toStrictEqual('mockKey');
45
+ expect(rows.value[0].value).toStrictEqual(base64);
46
+ expect(base64DecodeSpy).toHaveBeenCalledWith(row.mockKey);
47
+ });
48
+ });
49
+
50
+ describe('useDockerAuths', () => {
51
+ it('should return the auths field from a base64decoded json object in the `.dockerconfigjson` data field', () => {
52
+ const data = { '.dockerconfigjson': 'base64Json' };
53
+ const json = { auths: 'AUTHS' };
54
+
55
+ base64DecodeSpy.mockReturnValue(JSON.stringify(json));
56
+ const dockerAuths = authTypes.useDockerAuths({ data });
57
+
58
+ expect(dockerAuths.value).toStrictEqual(json.auths);
59
+ expect(base64DecodeSpy).toHaveBeenCalledWith(data['.dockerconfigjson']);
60
+ });
61
+ });
62
+
63
+ describe('useDockerRegistry', () => {
64
+ it('should return the first value from the docker auths', () => {
65
+ const data = { '.dockerconfigjson': 'base64Json' };
66
+ const registryUrl = 'registry-url.com';
67
+ const json = { auths: { [registryUrl]: { something: 'test' } } };
68
+
69
+ base64DecodeSpy.mockReturnValue(JSON.stringify(json));
70
+ const registry = authTypes.useDockerRegistry({ data });
71
+
72
+ expect(registry.value.registryUrl).toStrictEqual(registryUrl);
73
+ expect(base64DecodeSpy).toHaveBeenCalledWith(data['.dockerconfigjson']);
74
+ });
75
+ });
76
+
77
+ describe('useDockerBasic', () => {
78
+ it('should return the username and password from the docker auths', () => {
79
+ const data = { '.dockerconfigjson': 'base64Json' };
80
+ const registryUrl = 'registry-url.com';
81
+ const username = 'USERNAME';
82
+ const password = 'PASSWORD';
83
+ const json = { auths: { [registryUrl]: { username, password } } };
84
+
85
+ base64DecodeSpy.mockReturnValue(JSON.stringify(json));
86
+ const dockerBasic = authTypes.useDockerBasic({ data });
87
+
88
+ expect(dockerBasic.value.username).toStrictEqual(username);
89
+ expect(dockerBasic.value.password).toStrictEqual(password);
90
+ expect(base64DecodeSpy).toHaveBeenCalledWith(data['.dockerconfigjson']);
91
+ });
92
+ });
93
+
94
+ describe('useBasic', () => {
95
+ it('should return the username, password, rows from the docker auths', () => {
96
+ const username = 'USERNAME';
97
+ const password = 'PASSWORD';
98
+ const data = { username, password };
99
+
100
+ base64DecodeSpy
101
+ .mockReturnValueOnce(username)
102
+ .mockReturnValueOnce(password)
103
+ .mockReturnValueOnce(username)
104
+ .mockReturnValueOnce(password);
105
+
106
+ const basic = authTypes.useBasic({ data });
107
+
108
+ expect(basic.value.username).toStrictEqual(username);
109
+ expect(basic.value.password).toStrictEqual(password);
110
+ expect(basic.value.rows).toStrictEqual([{ key: 'username', value: username }, { key: 'password', value: password }]);
111
+ expect(base64DecodeSpy).toHaveBeenCalledWith(username);
112
+ expect(base64DecodeSpy).toHaveBeenCalledWith(password);
113
+ expect(base64DecodeSpy).toHaveBeenCalledWith(username);
114
+ expect(base64DecodeSpy).toHaveBeenCalledWith(password);
115
+ });
116
+ });
117
+
118
+ describe('useSsh', () => {
119
+ it('should return the username, password from data fields ssh-publickey/privatekey', () => {
120
+ const username = 'USERNAME';
121
+ const password = 'PASSWORD';
122
+ const data = {
123
+ 'ssh-publickey': username,
124
+ 'ssh-privatekey': password
125
+ };
126
+
127
+ base64DecodeSpy
128
+ .mockReturnValueOnce(username)
129
+ .mockReturnValueOnce(password);
130
+
131
+ const ssh = authTypes.useSsh({ data });
132
+
133
+ expect(ssh.value.username).toStrictEqual(username);
134
+ expect(ssh.value.password).toStrictEqual(password);
135
+
136
+ expect(base64DecodeSpy).toHaveBeenCalledWith(username);
137
+ expect(base64DecodeSpy).toHaveBeenCalledWith(password);
138
+ });
139
+ });
140
+
141
+ describe('useServiceAccount', () => {
142
+ it('should return the token, crt from data fields token, ca.crt', () => {
143
+ const token = 'TOKEN';
144
+ const crt = 'CRT';
145
+ const data = {
146
+ token,
147
+ 'ca.crt': crt
148
+ };
149
+
150
+ base64DecodeSpy
151
+ .mockReturnValueOnce(token)
152
+ .mockReturnValueOnce(crt);
153
+
154
+ const serviceAccount = authTypes.useServiceAccount({ data });
155
+
156
+ expect(serviceAccount.value.token).toStrictEqual(token);
157
+ expect(serviceAccount.value.crt).toStrictEqual(crt);
158
+
159
+ expect(base64DecodeSpy).toHaveBeenCalledWith(token);
160
+ expect(base64DecodeSpy).toHaveBeenCalledWith(crt);
161
+ });
162
+ });
163
+
164
+ describe('useTls', () => {
165
+ it('should return the token, crt from data fields token, ca.crt', () => {
166
+ const token = 'TOKEN';
167
+ const crt = 'CRT';
168
+ const data = {
169
+ 'tls.key': token,
170
+ 'tls.crt': crt
171
+ };
172
+
173
+ base64DecodeSpy
174
+ .mockReturnValueOnce(token)
175
+ .mockReturnValueOnce(crt);
176
+
177
+ const serviceAccount = authTypes.useTls({ data });
178
+
179
+ expect(serviceAccount.value.token).toStrictEqual(token);
180
+ expect(serviceAccount.value.crt).toStrictEqual(crt);
181
+
182
+ expect(base64DecodeSpy).toHaveBeenCalledWith(token);
183
+ expect(base64DecodeSpy).toHaveBeenCalledWith(crt);
184
+ });
185
+ });
186
+ });
@@ -0,0 +1,102 @@
1
+ import * as authTypes from '@shell/components/Resource/Detail/ResourceTabs/SecretDataTab/auth-types';
2
+ import { useSecretDataTabDefaultProps } from '@shell/components/Resource/Detail/ResourceTabs/SecretDataTab/composeables';
3
+ import { SECRET_TYPES } from '@shell/config/secret';
4
+
5
+ jest.mock('@shell/components/Resource/Detail/ResourceTabs/SecretDataTab/auth-types');
6
+ jest.mock('vuex', () => ({ useStore: jest.fn() }));
7
+
8
+ describe('composables: SecretDataTab/composables', () => {
9
+ describe('useSecretDataTabDefaultProps', () => {
10
+ const useSecretInfoSpy = jest.spyOn(authTypes, 'useSecretInfo');
11
+ const resource = { type: 'Resource' };
12
+
13
+ it('should return the appropriate props when secret type is DOCKER_JSON', () => {
14
+ const dockerRegistry = { type: 'DockerRegistry' };
15
+ const dockerBasic = { type: 'DockerBasic' };
16
+ const useDockerRegistrySpy = jest.spyOn(authTypes, 'useDockerRegistry');
17
+ const useDockerBasic = jest.spyOn(authTypes, 'useDockerBasic');
18
+
19
+ useSecretInfoSpy.mockImplementation((): any => ({ value: { secretType: SECRET_TYPES.DOCKER_JSON } }));
20
+ useDockerRegistrySpy.mockImplementation((): any => ({ value: dockerRegistry }));
21
+ useDockerBasic.mockImplementation((): any => ({ value: dockerBasic }));
22
+
23
+ const props = useSecretDataTabDefaultProps(resource);
24
+
25
+ expect(props.value.tabLabel).toStrictEqual('secret.data');
26
+ expect(useDockerRegistrySpy).toHaveBeenCalledWith(resource);
27
+ expect(useDockerBasic).toHaveBeenCalledWith(resource);
28
+ expect(props.value.secretData.registry).toStrictEqual(dockerRegistry);
29
+ expect(props.value.secretData.basicAuth).toStrictEqual(dockerBasic);
30
+ });
31
+
32
+ it('should return the appropriate props when secret type is TLS', () => {
33
+ const tls = { type: 'TLS' };
34
+ const useTlsSpy = jest.spyOn(authTypes, 'useTls');
35
+
36
+ useSecretInfoSpy.mockImplementation((): any => ({ value: { secretType: SECRET_TYPES.TLS } }));
37
+ useTlsSpy.mockImplementation((): any => ({ value: tls }));
38
+
39
+ const props = useSecretDataTabDefaultProps(resource);
40
+
41
+ expect(props.value.tabLabel).toStrictEqual('secret.certificate.certificate');
42
+ expect(useTlsSpy).toHaveBeenCalledWith(resource);
43
+ expect(props.value.secretData.certificate).toStrictEqual(tls);
44
+ });
45
+
46
+ it('should return the appropriate props when secret type is SERVICE_ACCT', () => {
47
+ const serviceAccount = { type: 'serviceAccount' };
48
+ const useServiceAccountSpy = jest.spyOn(authTypes, 'useServiceAccount');
49
+
50
+ useSecretInfoSpy.mockImplementation((): any => ({ value: { secretType: SECRET_TYPES.SERVICE_ACCT } }));
51
+ useServiceAccountSpy.mockImplementation((): any => ({ value: serviceAccount }));
52
+
53
+ const props = useSecretDataTabDefaultProps(resource);
54
+
55
+ expect(props.value.tabLabel).toStrictEqual('secret.data');
56
+ expect(useServiceAccountSpy).toHaveBeenCalledWith(resource);
57
+ expect(props.value.secretData.serviceAccount).toStrictEqual(serviceAccount);
58
+ });
59
+
60
+ it('should return the appropriate props when secret type is SSH', () => {
61
+ const ssh = { type: 'SSH' };
62
+ const useSshSpy = jest.spyOn(authTypes, 'useSsh');
63
+
64
+ useSecretInfoSpy.mockImplementation((): any => ({ value: { secretType: SECRET_TYPES.SSH } }));
65
+ useSshSpy.mockImplementation((): any => ({ value: ssh }));
66
+
67
+ const props = useSecretDataTabDefaultProps(resource);
68
+
69
+ expect(props.value.tabLabel).toStrictEqual('secret.ssh.keys');
70
+ expect(useSshSpy).toHaveBeenCalledWith(resource);
71
+ expect(props.value.secretData.ssh).toStrictEqual(ssh);
72
+ });
73
+
74
+ it('should return the appropriate props when secret type is BASIC', () => {
75
+ const basic = { type: 'basic' };
76
+ const useBasicSpy = jest.spyOn(authTypes, 'useBasic');
77
+
78
+ useSecretInfoSpy.mockImplementation((): any => ({ value: { secretType: SECRET_TYPES.BASIC } }));
79
+ useBasicSpy.mockImplementation((): any => ({ value: basic }));
80
+
81
+ const props = useSecretDataTabDefaultProps(resource);
82
+
83
+ expect(props.value.tabLabel).toStrictEqual('secret.data');
84
+ expect(useBasicSpy).toHaveBeenCalledWith(resource);
85
+ expect(props.value.secretData.basicAuth).toStrictEqual(basic);
86
+ });
87
+
88
+ it('should return the appropriate props when secret type is anything else', () => {
89
+ const basic = { type: 'basic' };
90
+ const useBasicSpy = jest.spyOn(authTypes, 'useBasic');
91
+
92
+ useSecretInfoSpy.mockImplementation((): any => ({ value: { secretType: SECRET_TYPES.FLEET_CLUSTER } }));
93
+ useBasicSpy.mockImplementation((): any => ({ value: basic }));
94
+
95
+ const props = useSecretDataTabDefaultProps(resource);
96
+
97
+ expect(props.value.tabLabel).toStrictEqual('secret.data');
98
+ expect(useBasicSpy).toHaveBeenCalledWith(resource);
99
+ expect(props.value.secretData.basic).toStrictEqual(basic);
100
+ });
101
+ });
102
+ });
@@ -0,0 +1,109 @@
1
+ import { computed, toValue } from 'vue';
2
+ import { base64Decode } from '@shell/utils/crypto';
3
+
4
+ export const useSecretInfo = (resource: any) => {
5
+ return computed(() => {
6
+ const resourceValue = toValue(resource);
7
+
8
+ return {
9
+ secretType: resourceValue._type,
10
+ secretData: resourceValue.data || {}
11
+ };
12
+ });
13
+ };
14
+
15
+ export const useSecretRows = (resource: any) => {
16
+ return computed(() => {
17
+ const resourceValue = toValue(resource);
18
+
19
+ const rows: any[] = [];
20
+ const { data = {} } = resourceValue;
21
+
22
+ Object.keys(data).forEach((key) => {
23
+ const value = base64Decode(data[key]);
24
+
25
+ rows.push({
26
+ key,
27
+ value
28
+ });
29
+ });
30
+
31
+ return rows;
32
+ });
33
+ };
34
+
35
+ export const useDockerAuths = (resource: any) => {
36
+ const secretInfo = useSecretInfo(resource);
37
+
38
+ return computed(() => {
39
+ const json = base64Decode(secretInfo.value.secretData['.dockerconfigjson']);
40
+
41
+ return JSON.parse(json).auths;
42
+ });
43
+ };
44
+
45
+ export const useDockerRegistry = (resource: any) => {
46
+ const dockerAuths = useDockerAuths(resource);
47
+
48
+ return computed(() => {
49
+ return { registryUrl: Object.keys(dockerAuths.value)[0] };
50
+ });
51
+ };
52
+
53
+ export const useDockerBasic = (resource: any) => {
54
+ const dockerAuths = useDockerAuths(resource);
55
+ const dockerRegistry = useDockerRegistry(resource);
56
+
57
+ return computed(() => {
58
+ return {
59
+ username: dockerAuths.value[dockerRegistry.value.registryUrl].username,
60
+ password: dockerAuths.value[dockerRegistry.value.registryUrl].password,
61
+ };
62
+ });
63
+ };
64
+
65
+ export const useBasic = (resource: any) => {
66
+ const rows = useSecretRows(resource);
67
+ const secretInfo = useSecretInfo(resource);
68
+
69
+ return computed(() => {
70
+ return {
71
+ username: base64Decode(secretInfo.value.secretData.username || ''),
72
+ password: base64Decode(secretInfo.value.secretData.password || ''),
73
+ rows: rows.value
74
+ };
75
+ });
76
+ };
77
+
78
+ export const useSsh = (resource: any) => {
79
+ const secretInfo = useSecretInfo(resource);
80
+
81
+ return computed(() => {
82
+ return {
83
+ username: base64Decode(secretInfo.value.secretData['ssh-publickey'] || ''),
84
+ password: base64Decode(secretInfo.value.secretData['ssh-privatekey'] || ''),
85
+ };
86
+ });
87
+ };
88
+
89
+ export const useServiceAccount = (resource: any) => {
90
+ const secretInfo = useSecretInfo(resource);
91
+
92
+ return computed(() => {
93
+ return {
94
+ token: base64Decode(secretInfo.value.secretData['token']),
95
+ crt: base64Decode(secretInfo.value.secretData['ca.crt']),
96
+ };
97
+ });
98
+ };
99
+
100
+ export const useTls = (resource: any) => {
101
+ const secretInfo = useSecretInfo(resource);
102
+
103
+ return computed(() => {
104
+ return {
105
+ token: base64Decode(secretInfo.value.secretData['tls.key']),
106
+ crt: base64Decode(secretInfo.value.secretData['tls.crt']),
107
+ };
108
+ });
109
+ };
@@ -0,0 +1,52 @@
1
+ import { computed, ComputedRef } from 'vue';
2
+ import { Props } from '@shell/components/Resource/Detail/ResourceTabs/SecretDataTab/index.vue';
3
+ import { SECRET_TYPES } from '@shell/config/secret';
4
+ import { useStore } from 'vuex';
5
+ import { useI18n } from '@shell/composables/useI18n';
6
+ import {
7
+ useBasic, useSsh, useTls, useSecretInfo, useDockerRegistry, useServiceAccount, useDockerBasic
8
+ } from '@shell/components/Resource/Detail/ResourceTabs/SecretDataTab/auth-types';
9
+
10
+ export const useSecretDataTabDefaultProps = (resource: any): ComputedRef<Props> => {
11
+ const store = useStore();
12
+ const i18n = useI18n(store);
13
+ const secretInfo = useSecretInfo(resource);
14
+
15
+ return computed(() => {
16
+ switch (secretInfo.value.secretType) {
17
+ case SECRET_TYPES.DOCKER_JSON:
18
+ return {
19
+ tabLabel: i18n.t('secret.data'),
20
+ secretData: {
21
+ registry: useDockerRegistry(resource).value,
22
+ basicAuth: useDockerBasic(resource).value
23
+ }
24
+ };
25
+ case SECRET_TYPES.TLS:
26
+ return {
27
+ tabLabel: i18n.t('secret.certificate.certificate'),
28
+ secretData: { certificate: useTls(resource).value }
29
+ };
30
+ case SECRET_TYPES.SERVICE_ACCT:
31
+ return {
32
+ tabLabel: i18n.t('secret.data'),
33
+ secretData: { serviceAccount: useServiceAccount(resource).value }
34
+ };
35
+ case SECRET_TYPES.SSH:
36
+ return {
37
+ tabLabel: i18n.t('secret.ssh.keys'),
38
+ secretData: { ssh: useSsh(resource).value }
39
+ };
40
+ case SECRET_TYPES.BASIC:
41
+ return {
42
+ tabLabel: i18n.t('secret.data'),
43
+ secretData: { basicAuth: useBasic(resource).value }
44
+ };
45
+ default:
46
+ return {
47
+ tabLabel: i18n.t('secret.data'),
48
+ secretData: { basic: useBasic(resource).value }
49
+ };
50
+ }
51
+ });
52
+ };
@@ -0,0 +1,71 @@
1
+ <script lang="ts">
2
+ import Tab from '@shell/components/Tabbed/Tab.vue';
3
+ import Basic, { Props as BasicProps } from '@shell/components/Resource/Detail/ResourceTabs/SecretDataTab/Basic.vue';
4
+ import Ssh, { Props as SshProps } from '@shell/components/Resource/Detail/ResourceTabs/SecretDataTab/Ssh.vue';
5
+ import ServiceAccountToken, { Props as ServiceAccountTokenProps } from '@shell/components/Resource/Detail/ResourceTabs/SecretDataTab/ServiceAccountToken.vue';
6
+ import Certificate, { Props as CertificateProps } from '@shell/components/Resource/Detail/ResourceTabs/SecretDataTab/Certificate.vue';
7
+ import BasicAuth, { Props as BasicAuthProps } from '@shell/components/Resource/Detail/ResourceTabs/SecretDataTab/BasicAuth.vue';
8
+ import Registry, { Props as RegistryProps } from '@shell/components/Resource/Detail/ResourceTabs/SecretDataTab/Registry.vue';
9
+
10
+ export interface SecretData {
11
+ basic?: BasicProps;
12
+ basicAuth?: BasicAuthProps;
13
+ ssh?: SshProps;
14
+ serviceAccount?: ServiceAccountTokenProps;
15
+ certificate?: CertificateProps;
16
+ registry?: RegistryProps;
17
+ }
18
+
19
+ export interface Props {
20
+ tabLabel: string;
21
+ secretData: SecretData;
22
+
23
+ weight?: number;
24
+ }
25
+ </script>
26
+
27
+ <script lang="ts" setup>
28
+ const props = defineProps<Props>();
29
+ </script>
30
+
31
+ <template>
32
+ <Tab
33
+ class="secret-data-tab"
34
+ name="data"
35
+ :label="props.tabLabel"
36
+ :weight="props.weight"
37
+ >
38
+ <Registry
39
+ v-if="props.secretData.registry"
40
+ v-bind="props.secretData.registry"
41
+ />
42
+ <BasicAuth
43
+ v-if="props.secretData.basicAuth"
44
+ v-bind="props.secretData.basicAuth"
45
+ />
46
+ <Certificate
47
+ v-else-if="props.secretData.certificate"
48
+ v-bind="props.secretData.certificate"
49
+ />
50
+ <ServiceAccountToken
51
+ v-if="props.secretData.serviceAccount"
52
+ v-bind="props.secretData.serviceAccount"
53
+ />
54
+ <Ssh
55
+ v-if="props.secretData.ssh"
56
+ v-bind="props.secretData.ssh"
57
+ />
58
+ <Basic
59
+ v-if="props.secretData.basic"
60
+ v-bind="props.secretData.basic"
61
+ />
62
+ </Tab>
63
+ </template>
64
+
65
+ <style lang="scss" scoped>
66
+ .secret-data-tab {
67
+ :deep() .entry:not(:first-of-type) {
68
+ margin-top: 16px;
69
+ }
70
+ }
71
+ </style>
@@ -7,7 +7,7 @@
7
7
  <style lang="scss" scoped>
8
8
  .spaced-row {
9
9
  display: grid;
10
- grid-template-columns: 1fr 1fr 1fr;
10
+ grid-template-columns: repeat(3, minmax(0, 1fr));;
11
11
  grid-auto-flow: dense;
12
12
  grid-gap: 24px;
13
13
  }
@@ -8,6 +8,7 @@
8
8
  h1.title {
9
9
  display: inline-block;
10
10
  align-items: center;
11
- line-height: 18px;
11
+ display: flex;
12
+ flex-direction: row;
12
13
  }
13
14
  </style>
@@ -2,7 +2,7 @@ import { mount } from '@vue/test-utils';
2
2
  import Title from '@shell/components/Resource/Detail/TitleBar/Title.vue';
3
3
 
4
4
  describe('component: TitleBar/Title', () => {
5
- it('shoulder render container with class title', async() => {
5
+ it('should render container with class title', async() => {
6
6
  const wrapper = mount(Title);
7
7
 
8
8
  expect(wrapper.find('.title').exists()).toBeTruthy();
@@ -2,7 +2,7 @@ import { mount } from '@vue/test-utils';
2
2
  import Top from '@shell/components/Resource/Detail/TitleBar/Top.vue';
3
3
 
4
4
  describe('component: TitleBar/Top', () => {
5
- it('shoulder render container with class top', async() => {
5
+ it('should render container with class top', async() => {
6
6
  const wrapper = mount(Top);
7
7
 
8
8
  expect(wrapper.find('.top').exists()).toBeTruthy();
@@ -0,0 +1,63 @@
1
+ import { useDefaultTitleBarProps } from '@shell/components/Resource/Detail/TitleBar/composables';
2
+ import { ref } from 'vue';
3
+ import { useRoute } from 'vue-router';
4
+
5
+ const mockStore = {
6
+ getters: {
7
+ 'type-map/labelFor': jest.fn(),
8
+ 'type-map/hasGraph': jest.fn(),
9
+ currentStore: jest.fn(),
10
+ 'cluster/schemaFor': jest.fn()
11
+ }
12
+ };
13
+ const mockRoute = { params: { cluster: 'CLUSTER' } };
14
+ const mockDrawer = { openResourceDetailDrawer: jest.fn() };
15
+
16
+ jest.mock('vuex', () => ({ useStore: () => mockStore }));
17
+ jest.mock('vue-router', () => ({ useRoute: () => mockRoute }));
18
+ jest.mock('@shell/components/Drawer/ResourceDetailDrawer/composables', () => ({ useResourceDetailDrawer: () => mockDrawer }));
19
+
20
+ describe('composables: TitleBar', () => {
21
+ const resource = {
22
+ nameDisplay: 'RESOURCE_NAME',
23
+ namespace: 'RESOURCE_NAMESPACE',
24
+ type: 'RESOURCE_TYPE',
25
+ stateBackground: 'RESOURCE_STATE_BACKGROUND',
26
+ stateDisplay: 'RESOURCE_STATE_DISPLAY',
27
+ description: 'RESOURCE_DESCRIPTION',
28
+ };
29
+ const labelFor = 'LABEL_FOR';
30
+ const schema = { type: 'SCHEMA' };
31
+ const hasGraph = true;
32
+
33
+ it('should return the appropriate values based on input', async() => {
34
+ const route = useRoute();
35
+
36
+ mockStore.getters['currentStore'].mockImplementation(() => 'cluster');
37
+ mockStore.getters['cluster/schemaFor'].mockImplementation(() => schema);
38
+ mockStore.getters['type-map/labelFor'].mockImplementation(() => labelFor);
39
+ mockStore.getters['type-map/hasGraph'].mockImplementation(() => hasGraph);
40
+
41
+ const props = useDefaultTitleBarProps(resource, ref(undefined));
42
+
43
+ expect(props.value.resourceTypeLabel).toStrictEqual(labelFor);
44
+ expect(mockStore.getters['type-map/labelFor']).toHaveBeenLastCalledWith(schema);
45
+ expect(mockStore.getters['type-map/hasGraph']).toHaveBeenLastCalledWith(resource.type);
46
+ expect(mockStore.getters['currentStore']).toHaveBeenLastCalledWith(resource.type);
47
+ expect(mockStore.getters['cluster/schemaFor']).toHaveBeenLastCalledWith(resource.type);
48
+ expect(props.value.resourceTo?.params.product).toStrictEqual('explorer');
49
+ expect(props.value.resourceTo?.params.cluster).toStrictEqual(route.params.cluster);
50
+ expect(props.value.resourceTo?.params.namespace).toStrictEqual(resource.namespace);
51
+ expect(props.value.resourceTo?.params.resource).toStrictEqual(resource.type);
52
+ expect(props.value.resourceName).toStrictEqual(resource.nameDisplay);
53
+
54
+ expect(props.value.actionMenuResource).toStrictEqual(resource);
55
+ expect(props.value.badge?.color).toStrictEqual(resource.stateBackground);
56
+ expect(props.value.badge?.label).toStrictEqual(resource.stateDisplay);
57
+ expect(props.value.description).toStrictEqual(resource.description);
58
+ expect(props.value.showViewOptions).toStrictEqual(hasGraph);
59
+
60
+ props.value.onShowConfiguration?.('callback');
61
+ expect(mockDrawer.openResourceDetailDrawer).toHaveBeenCalledTimes(1);
62
+ });
63
+ });
@@ -11,7 +11,7 @@ describe('component: TitleBar/index', () => {
11
11
  const resourceName = 'RESOURCE_NAME';
12
12
  const store = createStore({});
13
13
 
14
- it('shoulder render container with class title-bar', async() => {
14
+ it('should render container with class title-bar', async() => {
15
15
  const wrapper = mount(TitleBar, {
16
16
  props: {
17
17
  resourceTypeLabel,
@@ -0,0 +1,45 @@
1
+ import { useResourceDetailDrawer } from '@shell/components/Drawer/ResourceDetailDrawer/composables';
2
+ import { TitleBarProps } from '@shell/components/Resource/Detail/TitleBar/index.vue';
3
+ import { computed, Ref, toValue } from 'vue';
4
+ import { useRoute } from 'vue-router';
5
+ import { useStore } from 'vuex';
6
+
7
+ export const useDefaultTitleBarProps = (resource: any, resourceSubtype?: Ref<string | undefined>): Ref<TitleBarProps> => {
8
+ const route = useRoute();
9
+ const store = useStore();
10
+ const { openResourceDetailDrawer } = useResourceDetailDrawer();
11
+ const resourceValue = toValue(resource);
12
+
13
+ return computed(() => {
14
+ const resourceSubtypeValue = toValue(resourceSubtype);
15
+ const currentStore = store.getters['currentStore'](resourceValue.type);
16
+ const schema = store.getters[`${ currentStore }/schemaFor`](resourceValue.type);
17
+ const resourceTypeLabel = resourceValue.parentNameOverride || store.getters['type-map/labelFor'](schema);
18
+ const resourceName = resourceSubtypeValue ? `${ resourceSubtypeValue } - ${ resourceValue.nameDisplay }` : resourceValue.nameDisplay;
19
+ const resourceTo = resourceValue.listLocation || {
20
+ name: 'c-cluster-product-resource',
21
+ params: {
22
+ product: 'explorer',
23
+ cluster: route?.params.cluster,
24
+ namespace: resourceValue.namespace,
25
+ resource: resourceValue.type
26
+ }
27
+ };
28
+ const hasGraph = !!store.getters['type-map/hasGraph'](resourceValue.type);
29
+ const onShowConfiguration = resourceValue.disableResourceDetailDrawer ? undefined : (returnFocusSelector: string) => openResourceDetailDrawer(resourceValue, returnFocusSelector);
30
+
31
+ return {
32
+ resourceTypeLabel,
33
+ resourceTo,
34
+ resourceName,
35
+ actionMenuResource: resourceValue,
36
+ badge: {
37
+ color: resourceValue.stateBackground,
38
+ label: resourceValue.stateDisplay
39
+ },
40
+ description: resourceValue.description,
41
+ showViewOptions: hasGraph,
42
+ onShowConfiguration
43
+ };
44
+ });
45
+ };