mirror of
https://github.com/optim-enterprises-bv/vault.git
synced 2025-10-30 02:02:43 +00:00
* Client Count Routing Updates (#24733) * updates client count routing for sync and future additions * adds copyright header to clients sync template * adds missing copyright headers * UI: Adds secret_syncs to mirage /activity endpoint (#24846) * add secret_syncs to mirage endpoint * import clients handler * UI: Set up client charts for incoming sync data (#24852) * sum stacked bar values for tooltip total * make tooltip dynamic based on chartLegend * remove redundant helper * add secret_syncs to client count utils * move sum function to helper * update horizontal bar chart to include sync_clients * calculate sum of bars in tooltip * rename color palette const, define chart legends in each parent component instead of token.js * update tooltips * update mirage handler to add sys/ namespace * update mirage handler to add sys/ namespace * use pushObject * update test * UI: Secret sync bar chart (#24926) * install lineal * add ember-style-modifier dep * Add client count types for serialized data * Add sync bar chart component with tests * Chart is responsive * address comments * Clients Counts Parent Route (#24899) * adds interfaces for clients models * moves date formatting logic from clients activity adapter to utils file * adds clients counts route * updates links to clients route to point to top level and updates redirect to counts overview route * removes clients base route and moves overview and sync routes under counts * adds clients counts page component * converts clients route to ts * adds billing start timestamp to clients config mirage response and updates counts route to always attempt to fetch activity * fixes issue with updating namespace and auth mount query params always triggering client counts route model hook * adds tests for clients counts page component * adds missing copyright header to client-counts type file * Update ui/app/components/clients/page/counts.hbs Co-authored-by: claire bontempo <68122737+hellobontempo@users.noreply.github.com> * fixes bad import in sync-bar-chart * updates clients counts route to bypass query if there is not start_time * pins d3-shape to 1.3.7 for now -- makes lineal play nice with old charts * fixes sync bar chart tooltip assertion --------- Co-authored-by: claire bontempo <68122737+hellobontempo@users.noreply.github.com> * UI: convert line-chart to lineal (#24961) * lineal chart alongside svg * Add version-history to sync handler for testing * line chart is TS, test updated * remove d3-shape resolution * fix clients/token-test * use chartHeight in running-total template * use M/yy key instead of timestamp, chart is responsive * Add test for swapping datasets * add more edge case tests * more test * remove untrue assertion * fix weird decimal when between 1.1k and 2k * address feedback * Update line-chart to use timestamp instead of month key * Add timestamp to all places where month is on the clients activity response * Client Counts Overview (#24969) * adds counts base component for use in client counts child routes * adds clients counts overview page component * splits out monthly new chart from clients running total component * adds missing copyright headers * moves running total related assertions from token to overview acceptance test * removes new client assertions from running-total test and adds tests for monthly-new component * updates copy in running-total component * fixes clients overview tests * fixes timestamp stub not being restored in monthly-new test * fixes mfa-login test * renames counts component to activity * removes unused selectedAuthMethod arg from running-total component * adds timestamp back to running-total component * Secrets sync UI: add sync page component (#24982) * adds counts base component for use in client counts child routes * adds clients counts overview page component * splits out monthly new chart from clients running total component * adds missing copyright headers * move sync-bar-chart to charts/ folder * update types and rename chart * rename template file * moves running total related assertions from token to overview acceptance test * removes new client assertions from running-total test and adds tests for monthly-new component * updates copy in running-total component * fixes clients overview tests * fixes timestamp stub not being restored in monthly-new test * fixes mfa-login test * fix 0 values erroring charts * separate timestamp again * address merge conflicts * finish building sync chart component WIP css * renames counts component to activity * update import * revert name to dataKey * update styling for charts without legends * use monthly stat chart component for layout * use monthly chart stats in monthly new * implement stat wrapper; * remove extra grid div * rename component * fix legend css; * update test[ * remove arbitrarily setting max * add single month view * use stat text * update line chart tests * rename line chart * update tests --------- Co-authored-by: Jordan Reimer <zofskeez@gmail.com> * update selectors * add sync page tests * Secrets Sync UI: Add secrets syncs to csv export (#25056) * update mirage and add sync clients to export csv * fix sync legend label * remove word * update copy in modal * update mirage * fix attribution tooltip text * Clients Counts Token Route (#25019) * renames token route and page component back to dashboard * adds client counts token route and page component * updates charts in token page to use ChartContainer component * adds tests for clients token page component * restore clients dashboard test * use var for chart title sync page * updates clients token page to show usage stats when querying single month * updates token page clients averages to only include entity and non-entity clients in calculation * fixes monthly total counts lower than new clients in mirage handler * fixes token test --------- Co-authored-by: clairebontempo@gmail.com <clairebontempo@gmail.com> * Clients Usage Stats/Running Total Updates (#25094) * updates clients usage counts and running totals * updates usage stats total copy * fixes client counts overview tests * Secrets sync UI: cleanup and consolidation of components (#25090) * rename authMethod to mountPath * generalize count template copy * add todo to delete monthly new component * rename to tokenTab * wrap filters in conditional checking for start timestamp * some users may not have access to /config endpoint * fix querying when user has no billing date permissions and clicks current billing period * extend activity component from counts page * Revert "extend activity component from counts page" This reverts commit 1d0e85c82faf88c4385a04b1a5841cdde7fd00e0. * rename to startTimestampISO * remove timestamp from route and just use activity model responseTimestamp * fix chart y domain max * fix typos in usage stat and running totals component * delete backing class for display only template; * updates tests * adds comment for fetching license to get start date for billing * cleans up unused client counts files (#25157) * adds changelog * fix assertion copy * adds changelog description * updates enterprise sidebar nav test --------- Co-authored-by: clairebontempo@gmail.com <clairebontempo@gmail.com> Co-authored-by: claire bontempo <68122737+hellobontempo@users.noreply.github.com> Co-authored-by: Chelsea Shaw <82459713+hashishaw@users.noreply.github.com>
222 lines
9.9 KiB
JavaScript
222 lines
9.9 KiB
JavaScript
/**
|
|
* Copyright (c) HashiCorp, Inc.
|
|
* SPDX-License-Identifier: BUSL-1.1
|
|
*/
|
|
|
|
import Component from '@glimmer/component';
|
|
import { action } from '@ember/object';
|
|
import { tracked } from '@glimmer/tracking';
|
|
import { inject as service } from '@ember/service';
|
|
import { parseAPITimestamp } from 'core/utils/date-formatters';
|
|
import { format, isSameMonth } from 'date-fns';
|
|
|
|
/**
|
|
* @module Attribution
|
|
* Attribution components display the top 10 total client counts for namespaces or auth methods (mounts) during a billing period.
|
|
* A horizontal bar chart shows on the right, with the top namespace/auth method and respective client totals on the left.
|
|
*
|
|
* @example
|
|
* ```js
|
|
* <Clients::Attribution
|
|
* @totalUsageCounts={{this.totalUsageCounts}}
|
|
* @newUsageCounts={{this.newUsageCounts}}
|
|
* @totalClientAttribution={{this.totalClientAttribution}}
|
|
* @newClientAttribution={{this.newClientAttribution}}
|
|
* @selectedNamespace={{this.selectedNamespace}}
|
|
* @startTimestamp={{this.startTime}}
|
|
* @endTimestamp={{this.endTime}}
|
|
* @isHistoricalMonth={{false}}
|
|
* @responseTimestamp={{this.responseTimestamp}}
|
|
* @upgradeExplanation="We added monthly breakdowns and mount level attribution starting in 1.10, so keep that in mind when looking at the data."
|
|
* />
|
|
* ```
|
|
* @param {object} totalUsageCounts - object with total client counts for chart tooltip text
|
|
* @param {object} newUsageCounts - object with new client counts for chart tooltip text
|
|
* @param {array} totalClientAttribution - array of objects containing a label and breakdown of client counts for total clients
|
|
* @param {array} newClientAttribution - array of objects containing a label and breakdown of client counts for new clients
|
|
* @param {string} selectedNamespace - namespace selected from filter bar
|
|
* @param {string} startTimestamp - timestamp string from activity response to render start date for CSV modal and whether copy reads 'month' or 'date range'
|
|
* @param {string} endTimestamp - timestamp string from activity response to render end date for CSV modal and whether copy reads 'month' or 'date range'
|
|
* @param {string} responseTimestamp - ISO timestamp created in serializer to timestamp the response, renders in bottom left corner below attribution chart
|
|
* @param {boolean} isHistoricalMonth - when true data is from a single, historical month so side-by-side charts should display for attribution data
|
|
* @param {boolean} upgradeExplanation - if data contains an upgrade, explanation is generated by the parent to be rendered in the export modal
|
|
*/
|
|
|
|
export default class Attribution extends Component {
|
|
@tracked showCSVDownloadModal = false;
|
|
@service download;
|
|
attributionLegend = [
|
|
{ key: 'entity_clients', label: 'entity clients' },
|
|
{ key: 'non_entity_clients', label: 'non-entity clients' },
|
|
{ key: 'secret_syncs', label: 'secrets sync clients' },
|
|
];
|
|
|
|
get formattedStartDate() {
|
|
if (!this.args.startTimestamp) return null;
|
|
return parseAPITimestamp(this.args.startTimestamp, 'MMMM yyyy');
|
|
}
|
|
get formattedEndDate() {
|
|
if (!this.args.startTimestamp && !this.args.endTimestamp) return null;
|
|
// displays on CSV export modal, no need to display duplicate months and years
|
|
const startDateObject = parseAPITimestamp(this.args.startTimestamp);
|
|
const endDateObject = parseAPITimestamp(this.args.endTimestamp);
|
|
return isSameMonth(startDateObject, endDateObject) ? null : format(endDateObject, 'MMMM yyyy');
|
|
}
|
|
|
|
get hasCsvData() {
|
|
return this.args.totalClientAttribution ? this.args.totalClientAttribution.length > 0 : false;
|
|
}
|
|
|
|
get isSingleNamespace() {
|
|
if (!this.args.totalClientAttribution) {
|
|
return 'no data';
|
|
}
|
|
// if a namespace is selected, then we're viewing top 10 auth methods (mounts)
|
|
return !!this.args.selectedNamespace;
|
|
}
|
|
|
|
// truncate data before sending to chart component
|
|
get barChartTotalClients() {
|
|
return this.args.totalClientAttribution?.slice(0, 10);
|
|
}
|
|
|
|
get barChartNewClients() {
|
|
return this.args.newClientAttribution?.slice(0, 10);
|
|
}
|
|
|
|
get topClientCounts() {
|
|
// get top namespace or auth method
|
|
return this.args.totalClientAttribution ? this.args.totalClientAttribution[0] : null;
|
|
}
|
|
|
|
get attributionBreakdown() {
|
|
// display text for hbs
|
|
return this.isSingleNamespace ? 'auth method' : 'namespace';
|
|
}
|
|
|
|
get chartText() {
|
|
const dateText = this.formattedEndDate ? 'date range' : 'month';
|
|
switch (this.isSingleNamespace) {
|
|
case true:
|
|
return {
|
|
description:
|
|
'This data shows the top ten authentication methods by client count within this namespace, and can be used to understand where clients are originating. Authentication methods are organized by path.',
|
|
newCopy: `The new clients used by the auth method for this ${dateText}. This aids in understanding which auth methods create and use new clients${
|
|
dateText === 'date range' ? ' over time.' : '.'
|
|
}`,
|
|
totalCopy: `The total clients used by the auth method for this ${dateText}. This number is useful for identifying overall usage volume. `,
|
|
};
|
|
case false:
|
|
return {
|
|
description:
|
|
'This data shows the top ten namespaces by client count and can be used to understand where clients are originating. Namespaces are identified by path. To see all namespaces, export this data.',
|
|
newCopy: `The new clients in the namespace for this ${dateText}.
|
|
This aids in understanding which namespaces create and use new clients${
|
|
dateText === 'date range' ? ' over time.' : '.'
|
|
}`,
|
|
totalCopy: `The total clients in the namespace for this ${dateText}. This number is useful for identifying overall usage volume.`,
|
|
};
|
|
case 'no data':
|
|
return {
|
|
description: 'There is a problem gathering data',
|
|
};
|
|
default:
|
|
return '';
|
|
}
|
|
}
|
|
|
|
destructureCountsToArray(object) {
|
|
// destructure the namespace object {label: 'some-namespace', entity_clients: 171, non_entity_clients: 20, secret_syncs: 10, clients: 201}
|
|
// to get integers for CSV file
|
|
const { clients, entity_clients, non_entity_clients, secret_syncs } = object;
|
|
return [clients, entity_clients, non_entity_clients, secret_syncs];
|
|
}
|
|
|
|
constructCsvRow(namespaceColumn, mountColumn = null, totalColumns, newColumns = null) {
|
|
// if namespaceColumn is a string, then we're at mount level attribution, otherwise it is an object
|
|
// if constructing a namespace row, mountColumn=null so the column is blank, otherwise it is an object
|
|
const otherColumns = newColumns ? [...totalColumns, ...newColumns] : [...totalColumns];
|
|
return [
|
|
`${typeof namespaceColumn === 'string' ? namespaceColumn : namespaceColumn.label}`,
|
|
`${mountColumn ? mountColumn.label : '*'}`,
|
|
...otherColumns,
|
|
];
|
|
}
|
|
|
|
generateCsvData() {
|
|
const totalAttribution = this.args.totalClientAttribution;
|
|
const newAttribution = this.barChartNewClients ? this.args.newClientAttribution : null;
|
|
const csvData = [];
|
|
// added to clarify that the row of namespace totals without an auth method (blank) are not additional clients
|
|
// but indicate the total clients for that ns, including its auth methods
|
|
const upgrade = this.args.upgradeExplanation
|
|
? `\n **data contains an upgrade, mount summation may not equal namespace totals`
|
|
: '';
|
|
const descriptionOfBlanks = this.isSingleNamespace
|
|
? ''
|
|
: `\n *namespace totals, inclusive of mount clients ${upgrade}`;
|
|
const csvHeader = [
|
|
'Namespace path',
|
|
`"Mount path ${descriptionOfBlanks}"`,
|
|
'Total clients',
|
|
'Entity clients',
|
|
'Non-entity clients',
|
|
'Secrets sync clients',
|
|
];
|
|
|
|
if (newAttribution) {
|
|
csvHeader.push(
|
|
'Total new clients, New entity clients, New non-entity clients, New secrets sync clients'
|
|
);
|
|
}
|
|
|
|
totalAttribution.forEach((totalClientsObject) => {
|
|
const namespace = this.isSingleNamespace ? this.args.selectedNamespace : totalClientsObject;
|
|
const mount = this.isSingleNamespace ? totalClientsObject : null;
|
|
|
|
// find new client data for namespace/mount object we're iterating over
|
|
const newClientsObject = newAttribution
|
|
? newAttribution.find((d) => d.label === totalClientsObject.label)
|
|
: null;
|
|
|
|
const totalClients = this.destructureCountsToArray(totalClientsObject);
|
|
const newClients = newClientsObject ? this.destructureCountsToArray(newClientsObject) : null;
|
|
|
|
csvData.push(this.constructCsvRow(namespace, mount, totalClients, newClients));
|
|
// constructCsvRow returns an array that corresponds to a row in the csv file:
|
|
// ['ns label', 'mount label', total client #, entity #, non-entity #, ...new client #'s]
|
|
|
|
// only iterate through mounts if NOT viewing a single namespace
|
|
if (!this.isSingleNamespace && namespace.mounts) {
|
|
namespace.mounts.forEach((mount) => {
|
|
const newMountData = newAttribution
|
|
? newClientsObject?.mounts.find((m) => m.label === mount.label)
|
|
: null;
|
|
const mountTotalClients = this.destructureCountsToArray(mount);
|
|
const mountNewClients = newMountData ? this.destructureCountsToArray(newMountData) : null;
|
|
csvData.push(this.constructCsvRow(namespace, mount, mountTotalClients, mountNewClients));
|
|
});
|
|
}
|
|
});
|
|
|
|
csvData.unshift(csvHeader);
|
|
// make each nested array a comma separated string, join each array "row" in csvData with line break (\n)
|
|
return csvData.map((d) => d.join()).join('\n');
|
|
}
|
|
|
|
get formattedCsvFileName() {
|
|
const endRange = this.formattedEndDate ? `-${this.formattedEndDate}` : '';
|
|
const csvDateRange = this.formattedStartDate + endRange;
|
|
return this.isSingleNamespace
|
|
? `clients_by_mount_path_${csvDateRange}`
|
|
: `clients_by_namespace_${csvDateRange}`;
|
|
}
|
|
|
|
@action
|
|
exportChartData(filename) {
|
|
const contents = this.generateCsvData();
|
|
this.download.csv(filename, contents);
|
|
this.showCSVDownloadModal = false;
|
|
}
|
|
}
|