From b2e16c881cc25c044bd157520a8f675640732026 Mon Sep 17 00:00:00 2001
From: Chelsea Shaw <82459713+hashishaw@users.noreply.github.com>
Date: Wed, 10 Apr 2024 09:44:57 -0500
Subject: [PATCH] UI: Replication page navigation fix (#26325)
* Add replication mirage handler
* Add test with skipped failed assertion
* Use component-calculated attrsForCurrentMode instead of cluster.replicationAttrs which wasn't triggering component updates
* assert previously-skipped assertion
* Add changelog
---
changelog/26325.txt | 3 +
.../addon/components/replication-summary.js | 8 +-
.../components/replication-summary.hbs | 14 +--
ui/mirage/handlers/index.js | 2 +
ui/mirage/handlers/replication.js | 107 ++++++++++++++++++
ui/tests/acceptance/replication-nav-test.js | 55 +++++++++
6 files changed, 179 insertions(+), 10 deletions(-)
create mode 100644 changelog/26325.txt
create mode 100644 ui/mirage/handlers/replication.js
create mode 100644 ui/tests/acceptance/replication-nav-test.js
diff --git a/changelog/26325.txt b/changelog/26325.txt
new file mode 100644
index 000000000000..cbfc6c1f9c64
--- /dev/null
+++ b/changelog/26325.txt
@@ -0,0 +1,3 @@
+```release-note:bug
+ui: fixed a bug where the replication pages did not update display when navigating between DR and performance
+```
diff --git a/ui/lib/replication/addon/components/replication-summary.js b/ui/lib/replication/addon/components/replication-summary.js
index 0812b99a3434..5a862e465e3d 100644
--- a/ui/lib/replication/addon/components/replication-summary.js
+++ b/ui/lib/replication/addon/components/replication-summary.js
@@ -3,8 +3,7 @@
* SPDX-License-Identifier: BUSL-1.1
*/
-import { inject as service } from '@ember/service';
-import { alias } from '@ember/object/computed';
+import { service } from '@ember/service';
import { computed } from '@ember/object';
import Component from '@ember/component';
import decodeConfigFromJWT from 'replication/utils/decode-config-from-jwt';
@@ -27,6 +26,7 @@ export default Component.extend(ReplicationActions, DEFAULTS, {
replicationMode: 'dr',
mode: 'primary',
version: service(),
+ rm: service('replication-mode'),
didReceiveAttrs() {
this._super(...arguments);
const initialReplicationMode = this.initialReplicationMode;
@@ -38,7 +38,9 @@ export default Component.extend(ReplicationActions, DEFAULTS, {
initialReplicationMode: null,
cluster: null,
- replicationAttrs: alias('cluster.replicationAttrs'),
+ attrsForCurrentMode: computed('cluster', 'rm.mode', function () {
+ return this.cluster[this.rm.mode];
+ }),
tokenIncludesAPIAddr: computed('token', function () {
const config = decodeConfigFromJWT(this.token);
diff --git a/ui/lib/replication/addon/templates/components/replication-summary.hbs b/ui/lib/replication/addon/templates/components/replication-summary.hbs
index c32a365a53e6..b42ed5dd5859 100644
--- a/ui/lib/replication/addon/templates/components/replication-summary.hbs
+++ b/ui/lib/replication/addon/templates/components/replication-summary.hbs
@@ -336,7 +336,7 @@
{{/if}}
{{/if}}
{{else}}
- {{#if (eq this.replicationAttrs.mode "initializing")}}
+ {{#if (eq this.attrsForCurrentMode.mode "initializing")}}
The cluster is initializing replication. This may take some time.
{{else}}
{{this.cluster.replicationModeStatus.cluster_id}}
@@ -345,28 +345,28 @@
- {{#if (eq this.replicationAttrs.mode "secondary")}}
+ {{#if (eq this.attrsForCurrentMode.mode "secondary")}}
{{else}}
-
+
{{/if}}
diff --git a/ui/mirage/handlers/index.js b/ui/mirage/handlers/index.js
index b736c6a8af5e..17c094e64b9f 100644
--- a/ui/mirage/handlers/index.js
+++ b/ui/mirage/handlers/index.js
@@ -16,6 +16,7 @@ import oidcConfig from './oidc-config';
import hcpLink from './hcp-link';
import kubernetes from './kubernetes';
import ldap from './ldap';
+import replication from './replication';
export {
base,
@@ -29,4 +30,5 @@ export {
mfaConfig,
mfaLogin,
oidcConfig,
+ replication,
};
diff --git a/ui/mirage/handlers/replication.js b/ui/mirage/handlers/replication.js
new file mode 100644
index 000000000000..4a1d2ff6c52a
--- /dev/null
+++ b/ui/mirage/handlers/replication.js
@@ -0,0 +1,107 @@
+/**
+ * Copyright (c) HashiCorp, Inc.
+ * SPDX-License-Identifier: BUSL-1.1
+ */
+
+export default function (server) {
+ server.get('sys/replication/status', function () {
+ return {
+ data: {
+ dr: {
+ cluster_id: 'dr-cluster-id',
+ corrupted_merkle_tree: false,
+ known_secondaries: ['foobar'],
+ last_corruption_check_epoch: '-62135596800',
+ last_dr_wal: 98,
+ last_reindex_epoch: '0',
+ last_wal: 98,
+ merkle_root: 'ad721f32ed6789a1e5824841f358a517340a4585',
+ mode: 'primary',
+ primary_cluster_addr: 'dr-foobar',
+ secondaries: [
+ {
+ api_address: 'http://127.0.0.1:8202',
+ cluster_address: 'https://127.0.0.1:8203',
+ connection_status: 'disconnected',
+ last_heartbeat: '2024-04-09T09:04:22-05:00',
+ node_id: 'foobar',
+ },
+ ],
+ ssct_generation_counter: 0,
+ state: 'running',
+ },
+ performance: {
+ cluster_id: 'perf-cluster-id',
+ corrupted_merkle_tree: false,
+ known_secondaries: [],
+ last_corruption_check_epoch: '-62135596800',
+ last_performance_wal: 98,
+ last_reindex_epoch: '0',
+ last_wal: 98,
+ merkle_root: '618c9136bb443aa584f5d6b90755d42888c9c54a',
+ mode: 'primary',
+ primary_cluster_addr: 'perf-foobar',
+ secondaries: [],
+ ssct_generation_counter: 0,
+ state: 'running',
+ },
+ },
+ };
+ });
+
+ server.get('sys/replication/performance/status', function () {
+ return {
+ data: {
+ cluster_id: 'perf-cluster-id',
+ corrupted_merkle_tree: false,
+ known_secondaries: ['foobar'],
+ last_corruption_check_epoch: '-62135596800',
+ last_dr_wal: 98,
+ last_reindex_epoch: '0',
+ last_wal: 98,
+ merkle_root: 'ad721f32ed6789a1e5824841f358a517340a4585',
+ mode: 'primary',
+ primary_cluster_addr: 'perf-foobar',
+ secondaries: [
+ {
+ api_address: 'http://127.0.0.1:8202',
+ cluster_address: 'https://127.0.0.1:8203',
+ connection_status: 'disconnected',
+ last_heartbeat: '2024-04-09T09:04:22-05:00',
+ node_id: 'foobar',
+ },
+ ],
+ ssct_generation_counter: 0,
+ state: 'running',
+ },
+ };
+ });
+
+ server.get('sys/replication/dr/status', function () {
+ return {
+ data: {
+ cluster_id: 'dr-cluster-id',
+ corrupted_merkle_tree: false,
+ known_secondaries: ['foobar'],
+ last_corruption_check_epoch: '-62135596800',
+ last_dr_wal: 98,
+ last_reindex_epoch: '0',
+ last_wal: 98,
+ merkle_root: 'ad721f32ed6789a1e5824841f358a517340a4585',
+ mode: 'primary',
+ primary_cluster_addr: 'dr-foobar',
+ secondaries: [
+ {
+ api_address: 'http://127.0.0.1:8202',
+ cluster_address: 'https://127.0.0.1:8203',
+ connection_status: 'disconnected',
+ last_heartbeat: '2024-04-09T09:04:22-05:00',
+ node_id: 'foobar',
+ },
+ ],
+ ssct_generation_counter: 0,
+ state: 'running',
+ },
+ };
+ });
+}
diff --git a/ui/tests/acceptance/replication-nav-test.js b/ui/tests/acceptance/replication-nav-test.js
new file mode 100644
index 000000000000..5edd37230a71
--- /dev/null
+++ b/ui/tests/acceptance/replication-nav-test.js
@@ -0,0 +1,55 @@
+/**
+ * Copyright (c) HashiCorp, Inc.
+ * SPDX-License-Identifier: BUSL-1.1
+ */
+
+import { module, test } from 'qunit';
+import { setupApplicationTest } from 'ember-qunit';
+import { setupMirage } from 'ember-cli-mirage/test-support';
+import authPage from 'vault/tests/pages/auth';
+import ENV from 'vault/config/environment';
+import { click } from '@ember/test-helpers';
+
+const SELECTORS = {
+ navReplication: '[data-test-sidebar-nav-link="Replication"]',
+ navPerformance: '[data-test-sidebar-nav-link="Performance"]',
+ navDR: '[data-test-sidebar-nav-link="Disaster Recovery"]',
+ title: '[data-test-replication-title]',
+ primaryCluster: '[data-test-value-div="primary_cluster_addr"]',
+ replicationSet: '[data-test-row-value="Replication set"]',
+ knownSecondariesTitle: '.known-secondaries-card h3',
+};
+module('Acceptance | Enterprise | replication navigation', function (hooks) {
+ setupApplicationTest(hooks);
+ setupMirage(hooks);
+
+ hooks.before(function () {
+ ENV['ember-cli-mirage'].handler = 'replication';
+ });
+ hooks.after(function () {
+ ENV['ember-cli-mirage'].handler = null;
+ });
+
+ hooks.beforeEach(function () {
+ return authPage.login();
+ });
+
+ test('navigate between replication types updates page', async function (assert) {
+ await click(SELECTORS.navReplication);
+ assert.dom(SELECTORS.title).hasText('Disaster Recovery & Performance primary');
+ await click(SELECTORS.navPerformance);
+
+ // Ensure data is expected for performance
+ assert.dom(SELECTORS.title).hasText('Performance primary');
+ assert.dom(SELECTORS.primaryCluster).hasText('perf-foobar');
+ assert.dom(SELECTORS.replicationSet).hasText('perf-cluster-id');
+ assert.dom(SELECTORS.knownSecondariesTitle).hasText('0 Known secondaries');
+
+ // Nav to DR and see updated data
+ await click(SELECTORS.navDR);
+ assert.dom(SELECTORS.title).hasText('Disaster Recovery primary');
+ assert.dom(SELECTORS.primaryCluster).hasText('dr-foobar');
+ assert.dom(SELECTORS.replicationSet).hasText('dr-cluster-id');
+ assert.dom(SELECTORS.knownSecondariesTitle).hasText('1 Known secondaries');
+ });
+});