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

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 (312) 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/global/_form.scss +1 -1
  5. package/assets/styles/global/_labeled-input.scss +1 -1
  6. package/assets/styles/themes/_dark.scss +3 -0
  7. package/assets/styles/themes/_light.scss +3 -0
  8. package/assets/styles/vendor/vue-select.scss +1 -1
  9. package/assets/translations/en-us.yaml +404 -64
  10. package/assets/translations/zh-hans.yaml +3 -4
  11. package/cloud-credential/gcp.vue +9 -1
  12. package/components/AppModal.vue +2 -0
  13. package/components/CodeMirror.vue +1 -1
  14. package/components/ConfigMapSettings/Settings.vue +377 -0
  15. package/components/ConfigMapSettings/index.vue +354 -0
  16. package/components/CruResource.vue +1 -2
  17. package/components/DetailText.vue +61 -11
  18. package/components/Drawer/Chrome.vue +116 -0
  19. package/components/Drawer/ResourceDetailDrawer/ConfigTab.vue +61 -0
  20. package/components/Drawer/ResourceDetailDrawer/YamlTab.vue +48 -0
  21. package/components/Drawer/ResourceDetailDrawer/__tests__/ConfigTab.test.ts +54 -0
  22. package/components/Drawer/ResourceDetailDrawer/__tests__/YamlTab.test.ts +80 -0
  23. package/components/Drawer/ResourceDetailDrawer/__tests__/composables.test.ts +82 -0
  24. package/components/Drawer/ResourceDetailDrawer/__tests__/helpers.test.ts +42 -0
  25. package/components/Drawer/ResourceDetailDrawer/composables.ts +50 -0
  26. package/components/Drawer/ResourceDetailDrawer/helpers.ts +10 -0
  27. package/components/Drawer/ResourceDetailDrawer/index.vue +110 -0
  28. package/components/GrowlManager.vue +16 -15
  29. package/components/IconOrSvg.vue +5 -0
  30. package/components/KeyValueView.vue +1 -1
  31. package/components/LocaleSelector.vue +9 -1
  32. package/components/ProgressBarMulti.vue +1 -0
  33. package/components/PromptModal.vue +6 -1
  34. package/components/RelatedResources.vue +4 -12
  35. package/components/Resource/Detail/Additional.vue +46 -0
  36. package/components/Resource/Detail/Metadata/Annotations/__tests__/index.test.ts +1 -1
  37. package/components/Resource/Detail/Metadata/Annotations/index.vue +5 -0
  38. package/components/Resource/Detail/Metadata/IdentifyingInformation/__tests__/identifying-fields.test.ts +223 -0
  39. package/components/Resource/Detail/Metadata/IdentifyingInformation/composable.ts +37 -254
  40. package/components/Resource/Detail/Metadata/IdentifyingInformation/identifying-fields.ts +298 -0
  41. package/components/Resource/Detail/Metadata/IdentifyingInformation/index.vue +27 -5
  42. package/components/Resource/Detail/Metadata/KeyValue.vue +25 -17
  43. package/components/Resource/Detail/Metadata/Labels/__tests__/index.test.ts +1 -1
  44. package/components/Resource/Detail/Metadata/Labels/index.vue +4 -0
  45. package/components/Resource/Detail/Metadata/__tests__/KeyValue.test.ts +1 -1
  46. package/components/Resource/Detail/Metadata/__tests__/Rectangle.test.ts +1 -1
  47. package/components/Resource/Detail/Metadata/__tests__/composables.test.ts +75 -0
  48. package/components/Resource/Detail/Metadata/composables.ts +60 -11
  49. package/components/Resource/Detail/Metadata/index.vue +12 -5
  50. package/components/Resource/Detail/Page.vue +15 -0
  51. package/components/Resource/Detail/ResourceRow.vue +37 -18
  52. package/components/Resource/Detail/ResourceTabs/ConfigMapDataTab/__tests__/composables.test.ts +29 -0
  53. package/components/Resource/Detail/ResourceTabs/ConfigMapDataTab/__tests__/index.test.ts +48 -0
  54. package/components/Resource/Detail/ResourceTabs/ConfigMapDataTab/composables.ts +31 -0
  55. package/components/Resource/Detail/ResourceTabs/ConfigMapDataTab/index.vue +50 -0
  56. package/components/Resource/Detail/ResourceTabs/KnownHostsTab/__tests__/composables.test.ts +66 -0
  57. package/components/Resource/Detail/ResourceTabs/KnownHostsTab/composables.ts +21 -0
  58. package/components/Resource/Detail/ResourceTabs/KnownHostsTab/index.vue +31 -0
  59. package/components/Resource/Detail/ResourceTabs/SecretDataTab/Basic.vue +45 -0
  60. package/components/Resource/Detail/ResourceTabs/SecretDataTab/BasicAuth.vue +31 -0
  61. package/components/Resource/Detail/ResourceTabs/SecretDataTab/Certificate.vue +31 -0
  62. package/components/Resource/Detail/ResourceTabs/SecretDataTab/Registry.vue +22 -0
  63. package/components/Resource/Detail/ResourceTabs/SecretDataTab/ServiceAccountToken.vue +31 -0
  64. package/components/Resource/Detail/ResourceTabs/SecretDataTab/Ssh.vue +32 -0
  65. package/components/Resource/Detail/ResourceTabs/SecretDataTab/__tests__/Basic.test.ts +40 -0
  66. package/components/Resource/Detail/ResourceTabs/SecretDataTab/__tests__/BasicAuth.test.ts +33 -0
  67. package/components/Resource/Detail/ResourceTabs/SecretDataTab/__tests__/Certificate.test.ts +33 -0
  68. package/components/Resource/Detail/ResourceTabs/SecretDataTab/__tests__/Registry.test.ts +27 -0
  69. package/components/Resource/Detail/ResourceTabs/SecretDataTab/__tests__/ServiceAccountToken.test.ts +33 -0
  70. package/components/Resource/Detail/ResourceTabs/SecretDataTab/__tests__/Ssh.test.ts +33 -0
  71. package/components/Resource/Detail/ResourceTabs/SecretDataTab/__tests__/auth-types.test.ts +186 -0
  72. package/components/Resource/Detail/ResourceTabs/SecretDataTab/__tests__/composables.test.ts +102 -0
  73. package/components/Resource/Detail/ResourceTabs/SecretDataTab/auth-types.ts +109 -0
  74. package/components/Resource/Detail/ResourceTabs/SecretDataTab/composeables.ts +52 -0
  75. package/components/Resource/Detail/ResourceTabs/SecretDataTab/index.vue +71 -0
  76. package/components/Resource/Detail/TitleBar/Title.vue +2 -1
  77. package/components/Resource/Detail/TitleBar/__tests__/Title.test.ts +1 -1
  78. package/components/Resource/Detail/TitleBar/__tests__/Top.test.ts +1 -1
  79. package/components/Resource/Detail/TitleBar/__tests__/composables.test.ts +63 -0
  80. package/components/Resource/Detail/TitleBar/__tests__/index.test.ts +1 -1
  81. package/components/Resource/Detail/TitleBar/composables.ts +44 -0
  82. package/components/Resource/Detail/TitleBar/index.vue +83 -11
  83. package/components/Resource/Detail/composables.ts +45 -0
  84. package/components/ResourceDetail/Masthead/__tests__/index.test.ts +70 -0
  85. package/components/ResourceDetail/{__tests__/Masthead.test.ts → Masthead/__tests__/legacy.test.ts} +3 -3
  86. package/components/ResourceDetail/Masthead/index.vue +65 -0
  87. package/components/ResourceDetail/Masthead/latest.vue +44 -0
  88. package/components/ResourceDetail/__tests__/index.test.ts +26 -5
  89. package/components/ResourceDetail/index.vue +30 -16
  90. package/components/ResourceDetail/legacy.vue +18 -1
  91. package/components/ResourceList/Masthead.vue +6 -0
  92. package/components/ResourceYaml.vue +14 -1
  93. package/components/SlideInPanelManager.vue +46 -7
  94. package/components/StateDot/index.vue +28 -0
  95. package/components/Tabbed/index.vue +11 -15
  96. package/components/Wizard.vue +4 -2
  97. package/components/__tests__/ConfigMapSettings.test.ts +376 -0
  98. package/components/__tests__/GrowlManager.test.ts +0 -25
  99. package/components/auth/login/ldap.vue +1 -1
  100. package/components/fleet/FleetApplications.vue +0 -7
  101. package/components/fleet/FleetClusterTargets/TargetsList.vue +66 -0
  102. package/components/fleet/FleetClusterTargets/index.vue +455 -0
  103. package/components/fleet/FleetClusters.vue +25 -6
  104. package/components/fleet/FleetGitRepoPaths.vue +476 -0
  105. package/components/fleet/FleetHelmOps.vue +8 -0
  106. package/components/fleet/FleetRepos.vue +1 -6
  107. package/components/fleet/FleetResources.vue +4 -5
  108. package/components/fleet/FleetValuesFrom.vue +295 -0
  109. package/components/fleet/__tests__/FleetClusterTargets.test.ts +1224 -0
  110. package/components/fleet/__tests__/FleetGitRepoPaths.test.ts +265 -0
  111. package/components/fleet/__tests__/FleetOCIStorageSecret.test.ts +13 -13
  112. package/components/fleet/__tests__/FleetValuesFrom.test.ts +300 -0
  113. package/components/fleet/dashboard/ResourceCard.vue +1 -0
  114. package/components/fleet/dashboard/ResourceCardSummary.vue +1 -5
  115. package/components/fleet/dashboard/ResourceDetails.vue +8 -10
  116. package/components/fleet/dashboard/ResourcePanel.vue +15 -8
  117. package/components/form/ArrayList.vue +13 -2
  118. package/components/form/ChangePassword.vue +3 -1
  119. package/components/form/Footer.vue +10 -4
  120. package/components/form/KeyValue.vue +81 -43
  121. package/components/form/LabeledSelect.vue +56 -16
  122. package/components/form/Labels.vue +90 -17
  123. package/components/form/MatchExpressions.vue +46 -5
  124. package/components/form/NameNsDescription.vue +1 -1
  125. package/components/form/ResourceSelector.vue +1 -0
  126. package/components/form/ResourceTabs/index.vue +5 -0
  127. package/components/form/SecretSelector.vue +9 -2
  128. package/components/form/Select.vue +57 -19
  129. package/components/form/SimpleSecretSelector.vue +9 -2
  130. package/components/form/Taints.vue +21 -2
  131. package/components/form/UnitInput.vue +8 -0
  132. package/components/form/ValueFromResource.vue +1 -1
  133. package/components/form/__tests__/LabeledSelect.test.ts +8 -4
  134. package/components/form/__tests__/Labels.test.ts +360 -0
  135. package/components/form/__tests__/MatchExpressions.test.ts +16 -13
  136. package/components/form/__tests__/Select.test.ts +5 -2
  137. package/components/formatter/FleetApplicationSource.vue +1 -1
  138. package/components/formatter/WorkloadHealthScale.vue +1 -1
  139. package/components/google/AccountAccess.vue +211 -0
  140. package/components/google/types/gcp.d.ts +136 -0
  141. package/components/google/types/index.d.ts +101 -0
  142. package/components/google/util/__mocks__/gcp.ts +465 -0
  143. package/components/google/util/formatter.ts +82 -0
  144. package/components/google/util/gcp.ts +134 -0
  145. package/components/google/util/index.d.ts +11 -0
  146. package/components/nav/Favorite.vue +1 -1
  147. package/components/nav/Group.vue +70 -47
  148. package/components/nav/Header.vue +5 -1
  149. package/components/nav/NamespaceFilter.vue +13 -1
  150. package/components/nav/NotificationCenter/Notification.vue +510 -0
  151. package/components/nav/NotificationCenter/NotificationHeader.vue +112 -0
  152. package/components/nav/NotificationCenter/index.vue +148 -0
  153. package/composables/drawer.ts +26 -0
  154. package/composables/resources.test.ts +63 -0
  155. package/composables/resources.ts +38 -0
  156. package/composables/useIsNewDetailPageEnabled.ts +17 -0
  157. package/config/labels-annotations.js +6 -0
  158. package/config/product/auth.js +16 -1
  159. package/config/product/{cis.js → compliance.js} +23 -26
  160. package/config/product/explorer.js +5 -1
  161. package/config/product/fleet.js +7 -0
  162. package/config/product/settings.js +22 -11
  163. package/config/query-params.js +3 -0
  164. package/config/roles.ts +1 -1
  165. package/config/router/navigation-guards/authentication.js +51 -2
  166. package/config/router/routes.js +27 -31
  167. package/config/settings.ts +21 -3
  168. package/config/store.js +2 -0
  169. package/config/system-namespaces.js +1 -1
  170. package/config/table-headers.js +2 -2
  171. package/config/types.js +15 -6
  172. package/core/plugin.ts +32 -7
  173. package/core/types.ts +18 -1
  174. package/detail/{cis.cattle.io.clusterscan.vue → compliance.cattle.io.clusterscan.vue} +22 -18
  175. package/detail/management.cattle.io.fleetworkspace.vue +18 -27
  176. package/detail/management.cattle.io.oidcclient.vue +369 -0
  177. package/detail/node.vue +2 -2
  178. package/detail/pod.vue +2 -2
  179. package/detail/service.vue +10 -1
  180. package/detail/workload/index.vue +8 -2
  181. package/dialog/ExtensionCatalogUninstallDialog.vue +7 -4
  182. package/dialog/GenericPrompt.vue +1 -1
  183. package/dialog/ImportDialog.vue +8 -8
  184. package/dialog/OidcClientSecretDialog.vue +117 -0
  185. package/edit/__tests__/cis.cattle.io.clusterscan.test.ts +3 -3
  186. package/edit/__tests__/fleet.cattle.io.gitrepo.test.ts +5 -2
  187. package/edit/autoscaling.horizontalpodautoscaler/index.vue +4 -1
  188. package/edit/{cis.cattle.io.clusterscan.vue → compliance.cattle.io.clusterscan.vue} +30 -31
  189. package/edit/{cis.cattle.io.clusterscanbenchmark.vue → compliance.cattle.io.clusterscanbenchmark.vue} +4 -4
  190. package/edit/{cis.cattle.io.clusterscanprofile.vue → compliance.cattle.io.clusterscanprofile.vue} +5 -5
  191. package/edit/configmap.vue +4 -1
  192. package/edit/constraints.gatekeeper.sh.constraint/index.vue +1 -0
  193. package/edit/fleet.cattle.io.gitrepo.vue +44 -222
  194. package/edit/fleet.cattle.io.helmop.vue +44 -269
  195. package/edit/helm.cattle.io.projecthelmchart.vue +1 -0
  196. package/edit/k8s.cni.cncf.io.networkattachmentdefinition.vue +1 -0
  197. package/edit/logging-flow/index.vue +1 -0
  198. package/edit/logging.banzaicloud.io.output/index.vue +1 -0
  199. package/edit/management.cattle.io.fleetworkspace.vue +1 -0
  200. package/edit/management.cattle.io.oidcclient.vue +162 -0
  201. package/edit/management.cattle.io.project.vue +4 -1
  202. package/edit/monitoring.coreos.com.alertmanagerconfig/index.vue +1 -1
  203. package/edit/monitoring.coreos.com.alertmanagerconfig/receiverConfig.vue +5 -0
  204. package/edit/monitoring.coreos.com.prometheusrule/index.vue +1 -0
  205. package/edit/monitoring.coreos.com.receiver/auth.vue +30 -30
  206. package/edit/monitoring.coreos.com.receiver/index.vue +1 -0
  207. package/edit/monitoring.coreos.com.receiver/types/email.vue +1 -1
  208. package/edit/monitoring.coreos.com.route.vue +1 -0
  209. package/edit/namespace.vue +1 -0
  210. package/edit/networking.istio.io.destinationrule/index.vue +4 -1
  211. package/edit/networking.k8s.io.ingress/index.vue +4 -1
  212. package/edit/networking.k8s.io.networkpolicy/PolicyRules.vue +7 -2
  213. package/edit/networking.k8s.io.networkpolicy/index.vue +6 -2
  214. package/edit/node.vue +1 -0
  215. package/edit/persistentvolume/index.vue +4 -1
  216. package/edit/provisioning.cattle.io.cluster/rke2.vue +418 -382
  217. package/edit/provisioning.cattle.io.cluster/tabs/Basics.vue +27 -27
  218. package/edit/provisioning.cattle.io.cluster/tabs/MachinePool.vue +5 -0
  219. package/edit/resources.cattle.io.restore.vue +1 -1
  220. package/edit/secret/index.vue +1 -0
  221. package/edit/service.vue +4 -1
  222. package/edit/serviceaccount.vue +4 -1
  223. package/edit/storage.k8s.io.storageclass/index.vue +4 -1
  224. package/edit/workload/index.vue +5 -0
  225. package/list/{cis.cattle.io.clusterscan.vue → compliance.cattle.io.clusterscan.vue} +2 -2
  226. package/list/management.cattle.io.oidcclient.vue +108 -0
  227. package/list/node.vue +2 -0
  228. package/machine-config/amazonec2.vue +3 -24
  229. package/machine-config/components/GCEImage.vue +374 -0
  230. package/machine-config/google.vue +617 -0
  231. package/mixins/__tests__/brand.spec.ts +170 -0
  232. package/mixins/brand.js +16 -17
  233. package/mixins/create-edit-view/index.js +5 -0
  234. package/mixins/resource-fetch-api-pagination.js +16 -0
  235. package/mixins/vue-select-overrides.js +1 -0
  236. package/models/{cis.cattle.io.clusterscan.js → compliance.cattle.io.clusterscan.js} +8 -8
  237. package/models/{cis.cattle.io.clusterscanbenchmark.js → compliance.cattle.io.clusterscanbenchmark.js} +1 -1
  238. package/models/{cis.cattle.io.clusterscanprofile.js → compliance.cattle.io.clusterscanprofile.js} +5 -5
  239. package/models/{cis.cattle.io.clusterscanreport.js → compliance.cattle.io.clusterscanreport.js} +1 -1
  240. package/models/fleet-application.js +8 -79
  241. package/models/fleet.cattle.io.cluster.js +11 -0
  242. package/models/fleet.cattle.io.gitrepo.js +2 -2
  243. package/models/fleet.cattle.io.helmop.js +9 -39
  244. package/models/management.cattle.io.fleetworkspace.js +2 -1
  245. package/models/management.cattle.io.oidcclient.js +18 -0
  246. package/models/management.cattle.io.registration.js +3 -0
  247. package/models/provisioning.cattle.io.cluster.js +5 -5
  248. package/models/service.js +4 -0
  249. package/models/workload.js +5 -0
  250. package/package.json +1 -1
  251. package/pages/about.vue +4 -58
  252. package/pages/auth/login.vue +1 -1
  253. package/pages/c/_cluster/apps/charts/AddRepoLink.vue +0 -1
  254. package/pages/c/_cluster/apps/charts/index.vue +285 -81
  255. package/pages/c/_cluster/auth/user.retention/index.vue +87 -78
  256. package/pages/c/_cluster/explorer/index.vue +3 -3
  257. package/pages/c/_cluster/explorer/tools/pages/_page.vue +0 -1
  258. package/pages/c/_cluster/fleet/application/create.vue +3 -2
  259. package/pages/c/_cluster/fleet/index.vue +94 -56
  260. package/pages/c/_cluster/fleet/settings/index.vue +229 -0
  261. package/pages/c/_cluster/longhorn/index.vue +5 -2
  262. package/pages/c/_cluster/uiplugins/CatalogList/index.vue +16 -1
  263. package/pages/c/_cluster/uiplugins/PluginInfoPanel.vue +2 -2
  264. package/pages/explorer/resource/detail/configmap.vue +30 -7
  265. package/pages/explorer/resource/detail/secret.vue +50 -0
  266. package/pages/home.vue +9 -55
  267. package/pages/support/index.vue +4 -6
  268. package/plugins/dashboard-store/actions.js +19 -5
  269. package/plugins/dashboard-store/getters.js +4 -0
  270. package/plugins/dashboard-store/resource-class.js +16 -2
  271. package/plugins/steve/steve-pagination-utils.ts +26 -18
  272. package/plugins/steve/subscribe.js +6 -1
  273. package/rancher-components/Banner/Banner.vue +13 -0
  274. package/rancher-components/Form/Checkbox/Checkbox.vue +9 -4
  275. package/rancher-components/Form/LabeledInput/LabeledInput.vue +1 -1
  276. package/rancher-components/LabeledTooltip/LabeledTooltip.vue +1 -0
  277. package/rancher-components/RcItemCard/RcItemCard.vue +8 -3
  278. package/store/auth.js +2 -0
  279. package/store/catalog.js +23 -1
  280. package/store/growl.js +97 -8
  281. package/store/index.js +6 -0
  282. package/store/notifications.ts +426 -0
  283. package/store/prefs.js +0 -1
  284. package/store/type-map.js +19 -16
  285. package/store/uiplugins.ts +15 -1
  286. package/types/fleet.d.ts +24 -0
  287. package/types/notifications/index.ts +74 -0
  288. package/types/shell/index.d.ts +46 -32
  289. package/types/store/dashboard-store.types.ts +16 -0
  290. package/utils/__tests__/fleet.test.ts +148 -0
  291. package/utils/__tests__/object.test.ts +54 -1
  292. package/utils/__tests__/string.test.ts +273 -1
  293. package/utils/__tests__/time.test.ts +31 -0
  294. package/utils/auth.js +9 -2
  295. package/utils/crypto/encryption.ts +103 -0
  296. package/utils/cspAdaptor.ts +51 -0
  297. package/utils/fleet.ts +54 -65
  298. package/utils/object.js +36 -0
  299. package/utils/pagination-utils.ts +1 -1
  300. package/utils/release-notes.ts +48 -0
  301. package/utils/selector-typed.ts +7 -2
  302. package/utils/string.js +24 -0
  303. package/utils/{time.js → time.ts} +25 -6
  304. package/utils/uiplugins.ts +22 -0
  305. package/utils/validators/formRules/index.ts +3 -0
  306. package/components/Resource/Detail/TitleBar/composable.ts +0 -31
  307. package/config/product/legacy.js +0 -62
  308. package/pages/c/_cluster/legacy/pages/_page.vue +0 -29
  309. package/pages/c/_cluster/legacy/project/_page.vue +0 -57
  310. package/pages/c/_cluster/legacy/project/index.vue +0 -32
  311. package/pages/c/_cluster/legacy/project/pipelines.vue +0 -96
  312. /package/components/ResourceDetail/{Masthead.vue → Masthead/legacy.vue} +0 -0
@@ -0,0 +1,426 @@
1
+ import { md5 } from '@shell/utils/crypto';
2
+ import { randomStr } from '@shell/utils/string';
3
+ import { EncryptedNotification, Notification, StoredNotification } from '@shell/types/notifications';
4
+ import { encrypt, decrypt, deriveKey } from '@shell/utils/crypto/encryption';
5
+
6
+ /**
7
+ * Key used to store notifications in the browser's local storage
8
+ */
9
+ const LOCAL_STORAGE_KEY_PREFIX = 'rancher-notifications-';
10
+
11
+ /**
12
+ * Expire notifications in seconds (14 days)
13
+ */
14
+ const EXPIRY = 14 * 24 * 60 * 60;
15
+
16
+ /**
17
+ * Maximum number of notifications that will be kept
18
+ */
19
+ const MAX_NOTIFICATIONS = 50;
20
+
21
+ /**
22
+ * Broadcast channel name to send changes across tabs
23
+ */
24
+ const NOTIFICATION_CHANNEL_NAME = 'rancher-notification-sync';
25
+
26
+ /**
27
+ * Store for the UI Notification Centre
28
+ */
29
+ interface NotificationsStore {
30
+ localStorageKey: string,
31
+ userId: string;
32
+ notifications: StoredNotification[],
33
+ encryptionKey: CryptoKey | undefined,
34
+ }
35
+
36
+ export const state = function(): NotificationsStore {
37
+ const notifications: StoredNotification[] = [];
38
+
39
+ return {
40
+ localStorageKey: '',
41
+ userId: '',
42
+ encryptionKey: undefined,
43
+ notifications,
44
+ };
45
+ };
46
+
47
+ let bc: BroadcastChannel;
48
+
49
+ /**
50
+ * Sync notifications to other tabs using the broadcast channel. Send the user id, to cover corner case
51
+ * where a stale login exists for a different user in another tab.
52
+ */
53
+ function sync(userId: string, operation: string, param?: any) {
54
+ bc?.postMessage({
55
+ userId,
56
+ operation,
57
+ param
58
+ });
59
+ }
60
+
61
+ async function saveEncryptedNotification(getters: any, notification: Notification) {
62
+ const toEncrypt: EncryptedNotification = {
63
+ title: notification.title,
64
+ message: notification.message,
65
+ level: notification.level,
66
+ primaryAction: notification.primaryAction,
67
+ secondaryAction: notification.secondaryAction,
68
+ };
69
+
70
+ const localStorageKey = getters['localStorageKey'];
71
+ const encryptionKey = getters['encryptionKey'];
72
+
73
+ try {
74
+ const data = JSON.stringify(toEncrypt);
75
+ const enc = await encrypt(data, encryptionKey);
76
+
77
+ window.localStorage.setItem(`${ localStorageKey }-${ notification.id }`, JSON.stringify(enc));
78
+ } catch (e) {
79
+ console.error('Unable to save notification to local storage', e); // eslint-disable-line no-console
80
+ }
81
+ }
82
+
83
+ /**
84
+ * Sync the notifications index to local storage.
85
+ *
86
+ * We only store the non-sensitive data about the notifications in the index - the other data is stored in individual entries which are encrypted.
87
+ */
88
+ function syncIndex(state: NotificationsStore) {
89
+ const localStorageKey = state.localStorageKey;
90
+
91
+ // We just want the id, created, read and progress properties for the index
92
+ const index = state.notifications.map((n) => ({
93
+ id: n.id,
94
+ created: n.created,
95
+ read: n.read,
96
+ progress: n.progress,
97
+ }));
98
+
99
+ try {
100
+ window.localStorage.setItem(localStorageKey, JSON.stringify(index));
101
+ } catch (e) {
102
+ console.error('Unable to save notifications index to local storage', e); // eslint-disable-line no-console
103
+ }
104
+ }
105
+
106
+ export const getters = {
107
+ all: (state: NotificationsStore) => {
108
+ return state.notifications;
109
+ },
110
+
111
+ item: (state: NotificationsStore) => {
112
+ return (id: string) => {
113
+ return state.notifications.find((i) => i.id === id);
114
+ };
115
+ },
116
+
117
+ // Count of unread notifications
118
+ unreadCount: (state: NotificationsStore) => {
119
+ return state.notifications.filter((n) => !n.read).length;
120
+ },
121
+
122
+ /**
123
+ * Local storage key includes the user key
124
+ */
125
+ localStorageKey: (state: NotificationsStore) => {
126
+ return state.localStorageKey;
127
+ },
128
+
129
+ userId: (state: NotificationsStore) => {
130
+ return state.userId;
131
+ },
132
+
133
+ encryptionKey: (state: NotificationsStore) => {
134
+ return state.encryptionKey;
135
+ },
136
+ };
137
+
138
+ export const mutations = {
139
+ add(state: NotificationsStore, notification: Notification) {
140
+ if (!notification.id) {
141
+ notification.id = randomStr();
142
+ } else {
143
+ // Check that there is not already a notification with this id
144
+ const index = state.notifications.findIndex((n) => n.id === notification.id);
145
+
146
+ if (index !== -1) {
147
+ console.error(`Can not add a notification with the same id as an existing notification (${ notification.id })`); // eslint-disable-line no-console
148
+
149
+ return;
150
+ }
151
+ }
152
+
153
+ const stored = {
154
+ ...notification,
155
+ read: false,
156
+ created: new Date()
157
+ };
158
+
159
+ // Add to top of list
160
+ state.notifications.unshift(stored);
161
+
162
+ // Check that we have not exceeded the maximum number of notifications
163
+ if (state.notifications.length > MAX_NOTIFICATIONS) {
164
+ const removed = state.notifications.pop();
165
+
166
+ if (removed) {
167
+ // Remove the encrypted data for the notification that we just removed
168
+ window.localStorage.removeItem(`${ state.localStorageKey }-${ removed.id }`);
169
+ }
170
+ }
171
+
172
+ syncIndex(state);
173
+ },
174
+
175
+ markRead(state: NotificationsStore, id: string) {
176
+ const notification = state.notifications.find((n) => n.id === id);
177
+
178
+ if (notification && !notification.read) {
179
+ notification.read = true;
180
+ }
181
+
182
+ syncIndex(state);
183
+ },
184
+
185
+ markUnread(state: NotificationsStore, id: string) {
186
+ const notification = state.notifications.find((n) => n.id === id);
187
+
188
+ if (notification && notification.read) {
189
+ notification.read = false;
190
+ }
191
+
192
+ syncIndex(state);
193
+ },
194
+
195
+ markAllRead(state: NotificationsStore) {
196
+ state.notifications.forEach((notification) => {
197
+ if (!notification.read) {
198
+ notification.read = true;
199
+ }
200
+ });
201
+
202
+ syncIndex(state);
203
+ },
204
+
205
+ update(state: NotificationsStore, notification: Partial<Notification>) {
206
+ if (notification?.id) {
207
+ const index = state.notifications.findIndex((n) => n.id === notification.id);
208
+
209
+ if (index >= 0) {
210
+ state.notifications[index] = {
211
+ ...state.notifications[index],
212
+ ...notification
213
+ };
214
+ }
215
+ }
216
+
217
+ syncIndex(state);
218
+ },
219
+
220
+ clearAll(state: NotificationsStore) {
221
+ // Remove the encrypted data for each notification
222
+ state.notifications.forEach((n) => {
223
+ window.localStorage.removeItem(`${ state.localStorageKey }-${ n.id }`);
224
+ });
225
+
226
+ state.notifications = [];
227
+ syncIndex(state);
228
+ },
229
+
230
+ remove(state: NotificationsStore, id: string) {
231
+ // Remove the encrypted data for the notification
232
+ window.localStorage.removeItem(`${ state.localStorageKey }-${ id }`);
233
+
234
+ state.notifications = state.notifications.filter((n) => n.id !== id);
235
+ syncIndex(state);
236
+ },
237
+
238
+ load(state: NotificationsStore, notifications: StoredNotification[]) {
239
+ state.notifications = notifications;
240
+ },
241
+
242
+ localStorageKey(state: NotificationsStore, userKey: string) {
243
+ state.localStorageKey = `${ LOCAL_STORAGE_KEY_PREFIX }${ userKey }`;
244
+ },
245
+
246
+ userId(state: NotificationsStore, userId: string) {
247
+ state.userId = userId;
248
+ },
249
+
250
+ encryptionKey(state: NotificationsStore, encryptionKey: CryptoKey) {
251
+ state.encryptionKey = encryptionKey;
252
+ },
253
+ };
254
+
255
+ export const actions = {
256
+ async add( { commit, dispatch, getters }: any, notification: Notification) {
257
+ // We encrypt the notification on add - this is the only time we will encrypt it
258
+ if (!notification.id) {
259
+ notification.id = randomStr();
260
+ }
261
+
262
+ // Need to save the encrypted notification to local storage
263
+ await saveEncryptedNotification(getters, notification);
264
+
265
+ commit('add', notification);
266
+ sync(getters['userId'], 'add', notification);
267
+
268
+ // Show a growl for the notification if necessary
269
+ dispatch('growl/notification', notification, { root: true });
270
+ },
271
+
272
+ async fromGrowl( { commit, getters }: any, notification: Notification) {
273
+ notification.id = randomStr();
274
+
275
+ // Need to save the encrypted notification to local storage
276
+ await saveEncryptedNotification(getters, notification);
277
+
278
+ commit('add', notification);
279
+ sync(getters['userId'], 'add', notification);
280
+
281
+ return notification.id;
282
+ },
283
+
284
+ update({ commit, getters }: any, notification: Notification) {
285
+ commit('update', notification);
286
+ sync(getters['userId'], 'update', notification);
287
+ },
288
+
289
+ async markRead({ commit, dispatch, getters }: any, id: string) {
290
+ commit('markRead', id);
291
+ sync(getters['userId'], 'markRead', id);
292
+
293
+ const notification = getters.item(id);
294
+
295
+ if (notification?.preference) {
296
+ await dispatch('prefs/set', notification.preference, { root: true });
297
+ }
298
+ },
299
+
300
+ async markUnread({ commit, dispatch, getters }: any, id: string) {
301
+ commit('markUnread', id);
302
+ sync(getters['userId'], 'markUnread', id);
303
+
304
+ const notification = getters.item(id) as Notification;
305
+
306
+ if (notification?.preference) {
307
+ await dispatch('prefs/set', {
308
+ key: notification.preference.key,
309
+ value: notification.preference.unsetValue || '',
310
+ }, { root: true });
311
+ }
312
+ },
313
+
314
+ async markAllRead({ commit, dispatch, getters }: any) {
315
+ commit('markAllRead');
316
+ sync(getters['userId'], 'markAllRead');
317
+
318
+ // For all notifications that have a preference, set the preference, since they are now read
319
+ const withPreference = getters.all.filter((n: Notification) => !!n.preference);
320
+
321
+ for (let i = 0; i < withPreference.length; i++) {
322
+ await dispatch('prefs/set', withPreference[i].preference, { root: true });
323
+ }
324
+ },
325
+
326
+ remove({ commit, getters }: any, id: string) {
327
+ commit('remove', id);
328
+ sync(getters['userId'], 'remove', id);
329
+ },
330
+
331
+ clearAll({ commit, getters }: any) {
332
+ commit('clearAll');
333
+ sync(getters['userId'], 'clearAll');
334
+ },
335
+
336
+ /**
337
+ * Initialize the notifications store and load the notifications from local storage
338
+ */
339
+ async init({ commit, getters } : any, userData: any) {
340
+ const userKey = userData.id;
341
+ const userId = userData.v3User?.uuid;
342
+
343
+ if (!userKey || !userId) {
344
+ console.error('Unable to initialize notifications - required user info not available'); // eslint-disable-line no-console
345
+
346
+ return;
347
+ }
348
+
349
+ // Notifications are stored under a key for the current user, so set the local storage key based on the user id
350
+ commit('localStorageKey', md5(userKey, 'hex'));
351
+ commit('userId', userId);
352
+
353
+ let index: StoredNotification[] = [];
354
+ let notifications: StoredNotification[] = [];
355
+ const localStorageKey = getters['localStorageKey'];
356
+
357
+ let encryptionKey;
358
+
359
+ try {
360
+ encryptionKey = await deriveKey(userId);
361
+ } catch (e) {
362
+ console.error('Unable to generate encryption key for notifications', e); // eslint-disable-line no-console
363
+
364
+ return;
365
+ }
366
+
367
+ // Store the encryption key
368
+ commit('encryptionKey', encryptionKey);
369
+
370
+ // Load the notifications from local storage
371
+ // We store the index of notifications in local storage, and the actual notification data is stored in individual entries which are encrypted
372
+ try {
373
+ const data = window.localStorage.getItem(localStorageKey) || '[]';
374
+
375
+ index = JSON.parse(data) as StoredNotification[];
376
+ } catch (e) {
377
+ console.error('Unable to read notifications from local storage', e); // eslint-disable-line no-console
378
+ }
379
+
380
+ for (let i = 0; i < index.length; i++) {
381
+ const n = index[i];
382
+
383
+ try {
384
+ const data = window.localStorage.getItem(`${ localStorageKey }-${ n.id }`);
385
+ const parsedData = data ? JSON.parse(data) : '{}';
386
+ const decryptedString = await decrypt(parsedData, encryptionKey);
387
+ const decrypted = JSON.parse(decryptedString) as EncryptedNotification;
388
+
389
+ // Overlay the decrypted data onto the notification
390
+ notifications.push({
391
+ ...n,
392
+ ...decrypted
393
+ });
394
+ } catch (e) {
395
+ console.error('Unable to decrypt notification data', e); // eslint-disable-line no-console
396
+ }
397
+ }
398
+
399
+ // Expire old notifications
400
+ const now = new Date();
401
+
402
+ notifications = notifications.filter((n: StoredNotification) => {
403
+ // Try ... catch in case the date parsing fails
404
+ try {
405
+ const created = new Date(n.created);
406
+ const diff = (now.getTime() - created.getTime()) / 1000; // Diff in seconds
407
+
408
+ return diff < EXPIRY;
409
+ } catch (e) {}
410
+
411
+ return true;
412
+ });
413
+
414
+ commit('load', notifications);
415
+
416
+ // Set up broadcast listener to listen for updates from other tabs
417
+ bc = new BroadcastChannel(NOTIFICATION_CHANNEL_NAME);
418
+
419
+ bc.onmessage = (msgEvent: any) => {
420
+ // Ignore events where the user id does not match (corner case of stale login in another tab)
421
+ if (msgEvent?.data?.operation && msgEvent?.data?.userId === userId) {
422
+ commit(msgEvent.data.operation, msgEvent.data.param);
423
+ }
424
+ };
425
+ }
426
+ };
package/store/prefs.js CHANGED
@@ -74,7 +74,6 @@ export const HIDE_REPOS = create('hide-repos', [], { parseJSON });
74
74
  export const HIDE_DESC = create('hide-desc', [], { parseJSON });
75
75
  export const HIDE_SENSITIVE = create('hide-sensitive', true, { options: [true, false], parseJSON });
76
76
  export const SHOW_PRE_RELEASE = create('show-pre-release', false, { options: [false, true], parseJSON });
77
- export const SHOW_CHART_MODE = create('chart-mode', 'featured', { parseJSON });
78
77
 
79
78
  export const DATE_FORMAT = create('date-format', 'ddd, MMM D YYYY', {
80
79
  options: [
package/store/type-map.js CHANGED
@@ -512,21 +512,22 @@ export const getters = {
512
512
 
513
513
  optionsFor(state, getters, rootState, rootGetters) {
514
514
  const def = {
515
- isCreatable: true,
516
- isEditable: true,
517
- isRemovable: true,
518
- showState: true,
519
- showAge: true,
520
- canYaml: true,
521
- namespaced: null,
522
- listGroups: [],
523
- listGroupsWillOverride: false,
524
- listMandatorySort: null,
525
- depaginate: false,
526
- customRoute: undefined,
527
- resourceEditMasthead: true,
528
- custom: {},
529
- subTypes: [],
515
+ listCreateButtonLabelKey: undefined,
516
+ isCreatable: true,
517
+ isEditable: true,
518
+ isRemovable: true,
519
+ showState: true,
520
+ showAge: true,
521
+ canYaml: true,
522
+ namespaced: null,
523
+ listGroups: [],
524
+ listGroupsWillOverride: false,
525
+ listMandatorySort: null,
526
+ depaginate: false,
527
+ customRoute: undefined,
528
+ resourceEditMasthead: true,
529
+ custom: {},
530
+ subTypes: [],
530
531
  };
531
532
 
532
533
  return (schemaOrType, pagination) => {
@@ -754,11 +755,13 @@ export const getters = {
754
755
  let group = findBy(tree.children, 'name', name);
755
756
 
756
757
  if ( !group ) {
758
+ const groupDefaultTypeFor = getters.groupDefaultTypeFor(name);
759
+
757
760
  group = {
758
761
  name,
759
762
  label,
760
763
  weight: getters.groupWeightFor(name, forBasic),
761
- defaultType: getters.groupDefaultTypeFor(name),
764
+ defaultType: typeof groupDefaultTypeFor === 'function' ? groupDefaultTypeFor() : groupDefaultTypeFor,
762
765
  };
763
766
 
764
767
  tree.children.push(group);
@@ -9,6 +9,7 @@ import { Plugin } from '@shell/core/plugin';
9
9
  interface UIPluginState {
10
10
  plugins: Plugin[],
11
11
  errors: any,
12
+ ready: boolean,
12
13
  }
13
14
 
14
15
  interface LoadError {
@@ -20,6 +21,7 @@ export const state = function(): UIPluginState {
20
21
  return {
21
22
  plugins: [],
22
23
  errors: {},
24
+ ready: false,
23
25
  };
24
26
  };
25
27
 
@@ -31,6 +33,10 @@ export const getters = {
31
33
  errors: (state: any) => {
32
34
  return state.errors;
33
35
  },
36
+
37
+ ready: (state: any) => {
38
+ return state.ready;
39
+ },
34
40
  };
35
41
 
36
42
  export const mutations = {
@@ -50,6 +56,10 @@ export const mutations = {
50
56
  state.plugins.splice(index, 1);
51
57
  }
52
58
  },
59
+
60
+ setReady(state: UIPluginState, ready: boolean) {
61
+ state.ready = ready;
62
+ },
53
63
  };
54
64
 
55
65
  export const actions = {
@@ -63,5 +73,9 @@ export const actions = {
63
73
 
64
74
  removePlugin({ commit }: any, pluginName: string) {
65
75
  commit('removePlugin', pluginName);
66
- }
76
+ },
77
+
78
+ setReady( { commit }: any, ready: boolean ) {
79
+ commit('setReady', ready);
80
+ },
67
81
  };
package/types/fleet.d.ts CHANGED
@@ -34,3 +34,27 @@ export interface Application {
34
34
  name: string
35
35
  }
36
36
  }
37
+
38
+ export type TargetMode = 'none' | 'all' | 'clusters' | 'local' | 'advanced';
39
+
40
+ export type MatchLabels = Record<string, string>;
41
+
42
+ export interface Expression {
43
+ key: string,
44
+ operator: string,
45
+ values: string[]
46
+ }
47
+
48
+ export interface Selector {
49
+ key?: number,
50
+ matchLabels?: MatchLabels,
51
+ matchExpressions?: Expression[]
52
+ }
53
+
54
+ export interface Target {
55
+ name?: string,
56
+ clusterName?: string,
57
+ clusterSelector?: Selector,
58
+ clusterGroup?: string,
59
+ clusterGroupSelector?: Selector
60
+ }
@@ -0,0 +1,74 @@
1
+ import { RouteLocationRaw } from 'vue-router';
2
+
3
+ /**
4
+ * Type definitions for the Notification Center
5
+ */
6
+
7
+ /**
8
+ * Notification Level for a notification in the Notification Center
9
+ */
10
+ export enum NotificationLevel {
11
+ Announcement = 0, // eslint-disable-line no-unused-vars
12
+ Task, // eslint-disable-line no-unused-vars
13
+ Info, // eslint-disable-line no-unused-vars
14
+ Success, // eslint-disable-line no-unused-vars
15
+ Warning, // eslint-disable-line no-unused-vars
16
+ Error, // eslint-disable-line no-unused-vars
17
+ }
18
+
19
+ /**
20
+ * An action that is shown as a button in the Notification Center
21
+ */
22
+ export type NotificationAction = {
23
+ label: string; // Button label for the action
24
+ target?: string; // HREF target when the button is clicked
25
+ route?: RouteLocationRaw; // Route to navigate to when the button is clicked
26
+ };
27
+
28
+ /**
29
+ * Defines the User Preference linked to a notification
30
+ */
31
+ export type NotificationPreference = {
32
+ key: string; // User preference key to use when setting the preference when the notification is marked as read
33
+ value: string; // User preference value to use when setting the preference when the notification is marked as read
34
+ unsetValue?: string; // User preference value to use when setting the preference when the notification is marked as unread - defaults to empty string
35
+ };
36
+
37
+ /**
38
+ * Type for Encrypted Notification data that is stored in local storage
39
+ */
40
+ export type EncryptedNotification = {
41
+ title: string;
42
+ // Message to be shown in the notification (optional)
43
+ message?: string;
44
+ // Notification Level
45
+ level: NotificationLevel;
46
+ // Primary action to be shown in the notification (optional)
47
+ primaryAction?: NotificationAction;
48
+ // Secondary to be shown in the notification (optional)
49
+ secondaryAction?: NotificationAction;
50
+ };
51
+
52
+ /**
53
+ * Type for Notification that is sent
54
+ */
55
+ export type Notification = {
56
+ // Unique ID for the notification
57
+ id: string;
58
+ // Progress (0-100) for notifications of type `Task` (optional)
59
+ progress?: number;
60
+ // User Preference tied to the notification (optional) (the preference will be updated when the notification is marked read)
61
+ preference?: NotificationPreference;
62
+ } & EncryptedNotification;
63
+
64
+ /**
65
+ * Type for notification that is stored
66
+ *
67
+ * This should not be used outside of this store or the Notification Center UI components
68
+ *
69
+ * Includes extra fields managed by the notification center
70
+ */
71
+ export type StoredNotification = {
72
+ created: Date;
73
+ read: Boolean;
74
+ } & Notification;