/** * Copyright (c) HashiCorp, Inc. * SPDX-License-Identifier: BUSL-1.1 */ import { module, test } from 'qunit'; import { setupRenderingTest } from 'ember-qunit'; import { setupMirage } from 'ember-cli-mirage/test-support'; import { render } from '@ember/test-helpers'; import { hbs } from 'ember-cli-htmlbars'; import clientsHandler, { LICENSE_START, STATIC_NOW } from 'vault/mirage/handlers/clients'; import sinon from 'sinon'; import { formatRFC3339, getUnixTime } from 'date-fns'; import { findAll } from '@ember/test-helpers'; import { formatNumber } from 'core/helpers/format-number'; import timestamp from 'core/utils/timestamp'; import { setRunOptions } from 'ember-a11y-testing/test-support'; import { CLIENT_COUNT, CHARTS } from 'vault/tests/helpers/clients/client-count-selectors'; const START_TIME = getUnixTime(LICENSE_START); module('Integration | Component | clients/running-total', function (hooks) { setupRenderingTest(hooks); setupMirage(hooks); hooks.beforeEach(async function () { sinon.replace(timestamp, 'now', sinon.fake.returns(STATIC_NOW)); clientsHandler(this.server); const store = this.owner.lookup('service:store'); const activityQuery = { start_time: { timestamp: START_TIME }, end_time: { timestamp: getUnixTime(timestamp.now()) }, }; const activity = await store.queryRecord('clients/activity', activityQuery); this.byMonthActivity = activity.byMonth; this.newActivity = this.byMonthActivity.map((d) => d.new_clients); this.totalUsageCounts = activity.total; this.set('timestamp', formatRFC3339(timestamp.now())); this.set('chartLegend', [ { label: 'entity clients', key: 'entity_clients' }, { label: 'non-entity clients', key: 'non_entity_clients' }, ]); this.isSecretsSyncActivated = true; this.isHistoricalMonth = false; this.renderComponent = async () => { await render(hbs` `); }; // Fails on #ember-testing-container setRunOptions({ rules: { 'scrollable-region-focusable': { enabled: false }, }, }); }); test('it renders with full monthly activity data', async function (assert) { await this.renderComponent(); assert.dom(CHARTS.container('Vault client counts')).exists('running total component renders'); assert.dom(CHARTS.chart('Vault client counts')).exists('bar chart renders'); const expectedValues = { 'Running client total': formatNumber([this.totalUsageCounts.clients]), Entity: formatNumber([this.totalUsageCounts.entity_clients]), 'Non-entity': formatNumber([this.totalUsageCounts.non_entity_clients]), ACME: formatNumber([this.totalUsageCounts.acme_clients]), 'Secret sync': formatNumber([this.totalUsageCounts.secret_syncs]), }; for (const label in expectedValues) { assert .dom(CLIENT_COUNT.statTextValue(label)) .hasText( `${expectedValues[label]}`, `stat label: ${label} renders correct total: ${expectedValues[label]}` ); } // assert grouped bar chart is correct findAll(CHARTS.xAxisLabel).forEach((e, i) => { assert .dom(e) .hasText( `${this.byMonthActivity[i].month}`, `renders x-axis labels for bar chart: ${this.byMonthActivity[i].month}` ); }); assert .dom(CHARTS.verticalBar) .exists({ count: this.byMonthActivity.length * 2 }, 'renders correct number of bars '); }); test('it renders with no new monthly data', async function (assert) { this.byMonthActivity = this.byMonthActivity.map((d) => ({ ...d, new_clients: { month: d.month }, })); await this.renderComponent(); assert.dom(CHARTS.container('Vault client counts')).exists('running total component renders'); assert.dom(CHARTS.chart('Vault client counts')).exists('bar chart renders'); const expectedValues = { Entity: formatNumber([this.totalUsageCounts.entity_clients]), 'Non-entity': formatNumber([this.totalUsageCounts.non_entity_clients]), ACME: formatNumber([this.totalUsageCounts.acme_clients]), 'Secret sync': formatNumber([this.totalUsageCounts.secret_syncs]), }; for (const label in expectedValues) { assert .dom(CLIENT_COUNT.statTextValue(label)) .hasText( `${expectedValues[label]}`, `stat label: ${label} renders correct total: ${expectedValues[label]}` ); } }); test('it renders with single historical month data', async function (assert) { const singleMonth = this.byMonthActivity[this.byMonthActivity.length - 1]; const singleMonthNew = this.newActivity[this.newActivity.length - 1]; this.byMonthActivity = [singleMonth]; this.isHistoricalMonth = true; await this.renderComponent(); let expectedStats = { 'Total monthly clients': formatNumber([singleMonth.clients]), Entity: formatNumber([singleMonth.entity_clients]), 'Non-entity': formatNumber([singleMonth.non_entity_clients]), ACME: formatNumber([singleMonth.acme_clients]), 'Secret sync': formatNumber([singleMonth.secret_syncs]), }; for (const label in expectedStats) { assert .dom(`[data-test-total] ${CLIENT_COUNT.statTextValue(label)}`) .hasText( `${expectedStats[label]}`, `stat label: ${label} renders single month total: ${expectedStats[label]}` ); } expectedStats = { 'New clients': formatNumber([singleMonthNew.clients]), Entity: formatNumber([singleMonthNew.entity_clients]), 'Non-entity': formatNumber([singleMonthNew.non_entity_clients]), ACME: formatNumber([singleMonthNew.acme_clients]), 'Secret sync': formatNumber([singleMonthNew.secret_syncs]), }; for (const label in expectedStats) { assert .dom(`[data-test-new] ${CLIENT_COUNT.statTextValue(label)}`) .hasText( `${expectedStats[label]}`, `stat label: ${label} renders single month new clients: ${expectedStats[label]}` ); } assert.dom(CHARTS.chart('Vault client counts')).doesNotExist('bar chart does not render'); assert.dom(CLIENT_COUNT.statTextValue()).exists({ count: 10 }, 'renders 10 stat text containers'); }); test('it hides secret sync totals when feature is not activated', async function (assert) { this.isSecretsSyncActivated = false; await this.renderComponent(); assert.dom(CHARTS.container('Vault client counts')).exists('running total component renders'); assert.dom(CHARTS.chart('Vault client counts')).exists('bar chart renders'); assert.dom(CLIENT_COUNT.statTextValue('Entity')).exists(); assert.dom(CLIENT_COUNT.statTextValue('Non-entity')).exists(); assert.dom(CLIENT_COUNT.statTextValue('Secret sync')).doesNotExist('does not render secret syncs'); }); });