From fbdf60ce38627e4895ab68ff1412f156a6c6c698 Mon Sep 17 00:00:00 2001 From: Elwyn Benson <ebenson@gitlab.com> Date: Tue, 28 Nov 2023 12:38:04 +0000 Subject: [PATCH] Modify product analytics usage quota GraphQL endpoint - accept array of month selections as parameter instead of single year+month pair - return array of monthly usage data instead of single count - return year+month as part of response --- ...duct_analytics_usage_quota_annual_data.yml | 8 + doc/api/graphql/reference/index.md | 34 +++- ....vue => product_analytics_group_usage.vue} | 76 ++++---- .../product_analytics_usage_quota_app.vue | 6 +- .../product_analytics_projects_usage.vue | 17 +- ...product_analytics_projects_usage_chart.vue | 10 +- ...product_analytics_projects_usage_table.vue | 17 +- .../product_analytics/components/utils.js | 66 ++++++- ...prev_product_analytics_usage.query.graphql | 32 ---- ...roup_product_analytics_usage.query.graphql | 18 ++ .../product_analytics/graphql/utils.js | 40 ---- .../usage_quotas/product_analytics/utils.js | 5 + .../ee/groups/usage_quotas_controller.rb | 1 + ee/app/graphql/ee/types/project_type.rb | 5 +- .../project_usage_data_resolver.rb | 20 +- .../month_selection_input_type.rb | 18 ++ .../product_analytics/monthly_usage_type.rb | 14 ++ ... => product_analytics_group_usage_spec.js} | 165 ++++++++++++---- .../product_analytics_usage_quota_app_spec.js | 9 +- ...uct_analytics_projects_usage_chart_spec.js | 51 ++--- .../product_analytics_projects_usage_spec.js | 41 ++-- ...uct_analytics_projects_usage_table_spec.js | 31 +-- .../components/utils_spec.js | 180 ++++++++++++++---- .../product_analytics/graphql/mock_data.js | 117 +++++++++--- .../product_analytics/graphql/utils_spec.js | 166 ---------------- .../product_analytics/utils_spec.js | 18 ++ .../product_analytics/events_stored_spec.rb | 49 ++--- 27 files changed, 706 insertions(+), 508 deletions(-) create mode 100644 config/feature_flags/development/product_analytics_usage_quota_annual_data.yml rename ee/app/assets/javascripts/usage_quotas/product_analytics/components/{product_analytics_group_usage_chart.vue => product_analytics_group_usage.vue} (55%) delete mode 100644 ee/app/assets/javascripts/usage_quotas/product_analytics/graphql/queries/get_group_current_and_prev_product_analytics_usage.query.graphql create mode 100644 ee/app/assets/javascripts/usage_quotas/product_analytics/graphql/queries/get_group_product_analytics_usage.query.graphql delete mode 100644 ee/app/assets/javascripts/usage_quotas/product_analytics/graphql/utils.js create mode 100644 ee/app/assets/javascripts/usage_quotas/product_analytics/utils.js create mode 100644 ee/app/graphql/types/product_analytics/month_selection_input_type.rb create mode 100644 ee/app/graphql/types/product_analytics/monthly_usage_type.rb rename ee/spec/frontend/usage_quotas/product_analytics/components/{product_analytics_group_usage_chart_spec.js => product_analytics_group_usage_spec.js} (55%) delete mode 100644 ee/spec/frontend/usage_quotas/product_analytics/graphql/utils_spec.js create mode 100644 ee/spec/frontend/usage_quotas/product_analytics/utils_spec.js diff --git a/config/feature_flags/development/product_analytics_usage_quota_annual_data.yml b/config/feature_flags/development/product_analytics_usage_quota_annual_data.yml new file mode 100644 index 000000000000..9c269f8ce0e3 --- /dev/null +++ b/config/feature_flags/development/product_analytics_usage_quota_annual_data.yml @@ -0,0 +1,8 @@ +--- +name: product_analytics_usage_quota_annual_data +introduced_by_url: https://gitlab.com/gitlab-org/gitlab/-/merge_requests/136932 +rollout_issue_url: https://gitlab.com/gitlab-org/gitlab/-/issues/432734 +milestone: '16.7' +type: development +group: group::product analytics +default_enabled: false diff --git a/doc/api/graphql/reference/index.md b/doc/api/graphql/reference/index.md index a5f3be5ea19b..ca3aa1172780 100644 --- a/doc/api/graphql/reference/index.md +++ b/doc/api/graphql/reference/index.md @@ -22511,6 +22511,18 @@ Version of a machine learning model. | <a id="mlmodelversionid"></a>`id` | [`MlModelVersionID!`](#mlmodelversionid) | ID of the model version. | | <a id="mlmodelversionversion"></a>`version` | [`String!`](#string) | Name of the version. | +### `MonthlyUsage` + +Product analytics events for a specific month and year. + +#### Fields + +| Name | Type | Description | +| ---- | ---- | ----------- | +| <a id="monthlyusagecount"></a>`count` | [`Int`](#int) | Count of product analytics events. | +| <a id="monthlyusagemonth"></a>`month` | [`Int!`](#int) | Month of the data. | +| <a id="monthlyusageyear"></a>`year` | [`Int!`](#int) | Year of the data. | + ### `Namespace` #### Fields @@ -24780,16 +24792,19 @@ four standard [pagination arguments](#connection-pagination-arguments): ##### `Project.productAnalyticsEventsStored` -Count of all events used, filtered optionally by month. +Count of all events used, broken down by month. -Returns [`Int`](#int). +WARNING: +**Introduced** in 16.7. +This feature is an Experiment. It can be changed or removed at any time. + +Returns [`[MonthlyUsage!]`](#monthlyusage). ###### Arguments | Name | Type | Description | | ---- | ---- | ----------- | -| <a id="projectproductanalyticseventsstoredmonth"></a>`month` | [`Int`](#int) | Month for the period to return. | -| <a id="projectproductanalyticseventsstoredyear"></a>`year` | [`Int`](#int) | Year for the period to return. | +| <a id="projectproductanalyticseventsstoredmonthselection"></a>`monthSelection` | [`[MonthSelectionInput!]!`](#monthselectioninput) | Selection for the period to return. | ##### `Project.projectMembers` @@ -33418,6 +33433,17 @@ Represents an escalation rule. | <a id="mergerequestsresolvernegatedparamslabels"></a>`labels` | [`[String!]`](#string) | Array of label names. All resolved merge requests will not have these labels. | | <a id="mergerequestsresolvernegatedparamsmilestonetitle"></a>`milestoneTitle` | [`String`](#string) | Title of the milestone. | +### `MonthSelectionInput` + +A year and month input for querying product analytics usage data. + +#### Arguments + +| Name | Type | Description | +| ---- | ---- | ----------- | +| <a id="monthselectioninputmonth"></a>`month` | [`Int!`](#int) | Month of the period to return. | +| <a id="monthselectioninputyear"></a>`year` | [`Int!`](#int) | Year of the period to return. | + ### `NegatedBoardIssueInput` #### Arguments diff --git a/ee/app/assets/javascripts/usage_quotas/product_analytics/components/product_analytics_group_usage_chart.vue b/ee/app/assets/javascripts/usage_quotas/product_analytics/components/product_analytics_group_usage.vue similarity index 55% rename from ee/app/assets/javascripts/usage_quotas/product_analytics/components/product_analytics_group_usage_chart.vue rename to ee/app/assets/javascripts/usage_quotas/product_analytics/components/product_analytics_group_usage.vue index ac7d38824abc..6de9c86a3096 100644 --- a/ee/app/assets/javascripts/usage_quotas/product_analytics/components/product_analytics_group_usage_chart.vue +++ b/ee/app/assets/javascripts/usage_quotas/product_analytics/components/product_analytics_group_usage.vue @@ -1,19 +1,19 @@ <script> -import { GlAlert, GlLink, GlSprintf, GlSkeletonLoader } from '@gitlab/ui'; +import { GlAlert, GlLink, GlSkeletonLoader, GlSprintf } from '@gitlab/ui'; import { GlAreaChart } from '@gitlab/ui/dist/charts'; + import * as Sentry from '~/sentry/sentry_browser_wrapper'; -import { - dateAtFirstDayOfMonth, - nMonthsBefore, -} from '~/lib/utils/datetime/date_calculation_utility'; -import { formatDate } from '~/lib/utils/datetime/date_format_utility'; +import { nMonthsBefore } from '~/lib/utils/datetime/date_calculation_utility'; import { s__ } from '~/locale'; import { helpPagePath } from '~/helpers/help_page_helper'; -import { projectHasProductAnalyticsEnabled } from 'ee/usage_quotas/product_analytics/graphql/utils'; -import getGroupCurrentAndPrevProductAnalyticsUsageQuery from '../graphql/queries/get_group_current_and_prev_product_analytics_usage.query.graphql'; +import glFeatureFlagsMixin from '~/vue_shared/mixins/gl_feature_flags_mixin'; + +import { projectHasProductAnalyticsEnabled } from '../utils'; +import getGroupProductAnalyticsUsage from '../graphql/queries/get_group_product_analytics_usage.query.graphql'; +import { getCurrentMonth, mapMonthlyTotals } from './utils'; export default { - name: 'ProductAnalyticsGroupUsageChart', + name: 'ProductAnalyticsGroupUsage', components: { GlAlert, GlAreaChart, @@ -21,20 +21,16 @@ export default { GlSkeletonLoader, GlSprintf, }, + mixins: [glFeatureFlagsMixin()], inject: { namespacePath: { type: String, }, }, data() { - const currentMonth = dateAtFirstDayOfMonth(new Date()); - const previousMonth = nMonthsBefore(currentMonth, 1); - return { error: null, projectsUsageData: null, - currentMonth, - previousMonth, }; }, computed: { @@ -45,46 +41,29 @@ export default { return [ { name: s__('ProductAnalytics|Analytics events by month'), - data: this.projectsUsageData.map(([date, usageData]) => [ - formatDate(date, 'mmm yyyy'), - usageData, - ]), + data: this.projectsUsageData, }, ]; }, }, apollo: { projectsUsageData: { - // TODO refactor this component to fetch a years worth of data at a time and display arbitrary months of data - // instead of using explicit "previous" and "current" months: https://gitlab.com/gitlab-org/gitlab/-/issues/429312 - query: getGroupCurrentAndPrevProductAnalyticsUsageQuery, + query: getGroupProductAnalyticsUsage, variables() { return { namespacePath: this.namespacePath, - currentYear: this.currentMonth.getFullYear(), - previousYear: this.previousMonth.getFullYear(), - - // JS `getMonth()` is 0 based - currentMonth: this.currentMonth.getMonth() + 1, - previousMonth: this.previousMonth.getMonth() + 1, + monthSelection: this.getMonthsToQuery(), }; }, update(data) { - const months = [ - [this.previousMonth, this.sumProjectEvents(data.previous.projects.nodes)], - [this.currentMonth, this.sumProjectEvents(data.current.projects.nodes)], - ]; + const projects = data.group.projects.nodes.filter(projectHasProductAnalyticsEnabled); - const hasNoProjects = [data.previous.projects.nodes, data.current.projects.nodes] - .flat() - .filter(projectHasProductAnalyticsEnabled) - .every((projects) => projects.length === 0); - - if (hasNoProjects) { + if (projects.length === 0) { this.$emit('no-projects'); + return []; } - return months; + return mapMonthlyTotals(projects); }, error(error) { this.error = error; @@ -93,11 +72,22 @@ export default { }, }, methods: { - sumProjectEvents(projects) { - return projects.reduce( - (sum, project) => sum + (project.productAnalyticsEventsStored || 0), - 0, - ); + getMonthsToQuery() { + // 12 months data will cause backend performance issues for some large groups. So we can toggle + // this when needed until performance is improved in https://gitlab.com/gitlab-org/gitlab/-/issues/430865 + const ONE_YEAR = 12; + const TWO_MONTHS = 2; + const numMonthsDataToFetch = this.glFeatures.productAnalyticsUsageQuotaAnnualData + ? ONE_YEAR + : TWO_MONTHS; + + const currentMonth = getCurrentMonth(); + return Array.from({ length: numMonthsDataToFetch }).map((_, index) => { + const date = nMonthsBefore(currentMonth, index); + + // note: JS `getMonth()` is 0 based, so add 1 + return { year: date.getFullYear(), month: date.getMonth() + 1 }; + }); }, }, CHART_OPTIONS: { diff --git a/ee/app/assets/javascripts/usage_quotas/product_analytics/components/product_analytics_usage_quota_app.vue b/ee/app/assets/javascripts/usage_quotas/product_analytics/components/product_analytics_usage_quota_app.vue index 919ad8ad3cbe..054b7576fde0 100644 --- a/ee/app/assets/javascripts/usage_quotas/product_analytics/components/product_analytics_usage_quota_app.vue +++ b/ee/app/assets/javascripts/usage_quotas/product_analytics/components/product_analytics_usage_quota_app.vue @@ -3,14 +3,14 @@ import { GlEmptyState } from '@gitlab/ui'; import { helpPagePath } from '~/helpers/help_page_helper'; -import ProductAnalyticsGroupUsageChart from './product_analytics_group_usage_chart.vue'; +import ProductAnalyticsGroupUsage from './product_analytics_group_usage.vue'; import ProductAnalyticsProjectsUsage from './projects_usage/product_analytics_projects_usage.vue'; export default { name: 'ProductAnalyticsUsageQuotaApp', components: { GlEmptyState, - ProductAnalyticsGroupUsageChart, + ProductAnalyticsGroupUsage, ProductAnalyticsProjectsUsage, }, inject: { @@ -70,7 +70,7 @@ export default { /> </template> <template v-else> - <product-analytics-group-usage-chart @no-projects="handleNoProjects" /> + <product-analytics-group-usage @no-projects="handleNoProjects" /> <product-analytics-projects-usage /> </template> </section> diff --git a/ee/app/assets/javascripts/usage_quotas/product_analytics/components/projects_usage/product_analytics_projects_usage.vue b/ee/app/assets/javascripts/usage_quotas/product_analytics/components/projects_usage/product_analytics_projects_usage.vue index 9a74bab0037d..a401ad1e1b9b 100644 --- a/ee/app/assets/javascripts/usage_quotas/product_analytics/components/projects_usage/product_analytics_projects_usage.vue +++ b/ee/app/assets/javascripts/usage_quotas/product_analytics/components/projects_usage/product_analytics_projects_usage.vue @@ -6,8 +6,8 @@ import { nMonthsBefore, } from '~/lib/utils/datetime/date_calculation_utility'; -import getGroupCurrentAndPrevProductAnalyticsUsageQuery from '../../graphql/queries/get_group_current_and_prev_product_analytics_usage.query.graphql'; -import { mapProjectsUsageResponse } from '../../graphql/utils'; +import getGroupCurrentAndPrevProductAnalyticsUsageQuery from '../../graphql/queries/get_group_product_analytics_usage.query.graphql'; +import { projectHasProductAnalyticsEnabled } from '../../utils'; import ProductAnalyticsProjectsUsageChart from './product_analytics_projects_usage_chart.vue'; import ProductAnalyticsProjectsUsageTable from './product_analytics_projects_usage_table.vue'; @@ -43,16 +43,15 @@ export default { return { namespacePath: this.namespacePath, - currentYear: current.getFullYear(), - previousYear: previous.getFullYear(), - - // JS `getMonth()` is 0 based - currentMonth: current.getMonth() + 1, - previousMonth: previous.getMonth() + 1, + monthSelection: [ + // note: JS `getMonth()` is 0 based, so add 1 + { year: current.getFullYear(), month: current.getMonth() + 1 }, + { year: previous.getFullYear(), month: previous.getMonth() + 1 }, + ], }; }, update(data) { - return mapProjectsUsageResponse(data); + return data.group.projects.nodes.filter(projectHasProductAnalyticsEnabled); }, error(error) { this.error = error; diff --git a/ee/app/assets/javascripts/usage_quotas/product_analytics/components/projects_usage/product_analytics_projects_usage_chart.vue b/ee/app/assets/javascripts/usage_quotas/product_analytics/components/projects_usage/product_analytics_projects_usage_chart.vue index b2bd9f91779e..5a95303f748b 100644 --- a/ee/app/assets/javascripts/usage_quotas/product_analytics/components/projects_usage/product_analytics_projects_usage_chart.vue +++ b/ee/app/assets/javascripts/usage_quotas/product_analytics/components/projects_usage/product_analytics_projects_usage_chart.vue @@ -2,7 +2,11 @@ import { GlSkeletonLoader, GlTooltipDirective } from '@gitlab/ui'; import { GlColumnChart } from '@gitlab/ui/dist/charts'; import { s__, sprintf } from '~/locale'; -import { projectsUsageDataValidator } from '../utils'; +import { + findCurrentMonthUsage, + findPreviousMonthUsage, + projectsUsageDataValidator, +} from '../utils'; // Trying to show more than this many projects on a single chart starts to // get illegible, so we only render this many projects if there's many @@ -50,7 +54,7 @@ export default { stack: 'previous', data: this.projectsUsageData ?.map((project) => { - return [project.name, project.previousEvents]; + return [project.name, findPreviousMonthUsage(project).count]; }) .slice(0, MAX_PROJECTS_TO_CHART), }, @@ -59,7 +63,7 @@ export default { stack: 'current', data: this.projectsUsageData ?.map((project) => { - return [project.name, project.currentEvents]; + return [project.name, findCurrentMonthUsage(project).count]; }) .slice(0, MAX_PROJECTS_TO_CHART), }, diff --git a/ee/app/assets/javascripts/usage_quotas/product_analytics/components/projects_usage/product_analytics_projects_usage_table.vue b/ee/app/assets/javascripts/usage_quotas/product_analytics/components/projects_usage/product_analytics_projects_usage_table.vue index 466e2fdebf2d..b980b813b536 100644 --- a/ee/app/assets/javascripts/usage_quotas/product_analytics/components/projects_usage/product_analytics_projects_usage_table.vue +++ b/ee/app/assets/javascripts/usage_quotas/product_analytics/components/projects_usage/product_analytics_projects_usage_table.vue @@ -2,7 +2,11 @@ import { GlLink, GlSkeletonLoader, GlTableLite, GlTooltipDirective } from '@gitlab/ui'; import { __, s__ } from '~/locale'; import ProjectAvatar from '~/vue_shared/components/project_avatar.vue'; -import { projectsUsageDataValidator } from '../utils'; +import { + findCurrentMonthUsage, + findPreviousMonthUsage, + projectsUsageDataValidator, +} from '../utils'; export default { name: 'ProductAnalyticsProjectsUsageTable', @@ -31,6 +35,15 @@ export default { hasProjects() { return this.projectsUsageData?.length > 0; }, + tableData() { + return this.projectsUsageData.map((project) => { + return { + ...project, + currentEvents: findCurrentMonthUsage(project).count, + previousEvents: findPreviousMonthUsage(project).count, + }; + }); + }, }, TABLE_FIELDS: [ { key: 'name', label: __('Project') }, @@ -43,7 +56,7 @@ export default { <div> <gl-skeleton-loader v-if="isLoading" :lines="3" :equal-width-lines="true" /> <div v-else-if="hasProjects" data-testid="projects-usage-table"> - <gl-table-lite :items="projectsUsageData" :fields="$options.TABLE_FIELDS"> + <gl-table-lite :items="tableData" :fields="$options.TABLE_FIELDS"> <template #cell(name)="{ item: { id, name, avatarUrl, webUrl } }"> <project-avatar :project-id="id" diff --git a/ee/app/assets/javascripts/usage_quotas/product_analytics/components/utils.js b/ee/app/assets/javascripts/usage_quotas/product_analytics/components/utils.js index 7ccc47b1815d..77181eeb5c1a 100644 --- a/ee/app/assets/javascripts/usage_quotas/product_analytics/components/utils.js +++ b/ee/app/assets/javascripts/usage_quotas/product_analytics/components/utils.js @@ -1,14 +1,64 @@ +import { + dateAtFirstDayOfMonth, + nMonthsBefore, +} from '~/lib/utils/datetime/date_calculation_utility'; +import { formatDate } from '~/lib/utils/datetime/date_format_utility'; + +const isValidUsage = ({ year, month, count }) => + typeof year === 'number' && + typeof month === 'number' && + (count === null || typeof count === 'number'); + +const isValidProject = ({ id, name, avatarUrl, webUrl, productAnalyticsEventsStored }) => + typeof id === 'string' && + typeof name === 'string' && + (avatarUrl === null || typeof avatarUrl === 'string') && + typeof webUrl === 'string' && + Array.isArray(productAnalyticsEventsStored) && + productAnalyticsEventsStored.every(isValidUsage); + /** * Validator for the projectsUsageData property */ export const projectsUsageDataValidator = (items) => { - return ( - Array.isArray(items) && - items.every( - ({ name, currentEvents, previousEvents }) => - typeof name === 'string' && - typeof currentEvents === 'number' && - typeof previousEvents === 'number', - ) + return Array.isArray(items) && items.every(isValidProject); +}; + +export const getCurrentMonth = () => { + return dateAtFirstDayOfMonth(new Date()); +}; + +const findMonthsUsage = (project, monthOffset) => { + const month = nMonthsBefore(getCurrentMonth(), monthOffset); + + return project.productAnalyticsEventsStored.find( + (usage) => usage.year === month.getFullYear() && usage.month === month.getMonth() + 1, + ); +}; + +export const findCurrentMonthUsage = (project) => { + return findMonthsUsage(project, 0); +}; + +export const findPreviousMonthUsage = (project) => { + return findMonthsUsage(project, 1); +}; + +/** + * Maps projects into an array of monthLabel, numEvents pairs. + */ +export const mapMonthlyTotals = (projects) => { + const monthlyTotals = {}; + + projects.forEach(({ productAnalyticsEventsStored }) => + productAnalyticsEventsStored.forEach(({ year, month, count }) => { + const timestamp = Date.UTC(year, month - 1); + monthlyTotals[timestamp] = (monthlyTotals[timestamp] || 0) + count; + }), ); + + return Array.from(Object.entries(monthlyTotals)) + .map(([timestampString, count]) => [Number(timestampString), count]) + .sort(([timestampA], [timestampB]) => timestampA - timestampB) + .map(([timestamp, count]) => [formatDate(new Date(timestamp), 'mmm yyyy'), count]); }; diff --git a/ee/app/assets/javascripts/usage_quotas/product_analytics/graphql/queries/get_group_current_and_prev_product_analytics_usage.query.graphql b/ee/app/assets/javascripts/usage_quotas/product_analytics/graphql/queries/get_group_current_and_prev_product_analytics_usage.query.graphql deleted file mode 100644 index f2ff90047e4d..000000000000 --- a/ee/app/assets/javascripts/usage_quotas/product_analytics/graphql/queries/get_group_current_and_prev_product_analytics_usage.query.graphql +++ /dev/null @@ -1,32 +0,0 @@ -query getGroupCurrentAndPreviousProductAnalyticsUsage( - $namespacePath: ID! - $currentYear: Int - $currentMonth: Int - $previousYear: Int - $previousMonth: Int -) { - previous: group(fullPath: $namespacePath) { - id - projects(includeSubgroups: true) { - nodes { - id - name - avatarUrl - webUrl - productAnalyticsEventsStored(year: $previousYear, month: $previousMonth) - } - } - } - current: group(fullPath: $namespacePath) { - id - projects(includeSubgroups: true) { - nodes { - id - name - avatarUrl - webUrl - productAnalyticsEventsStored(year: $currentYear, month: $currentMonth) - } - } - } -} diff --git a/ee/app/assets/javascripts/usage_quotas/product_analytics/graphql/queries/get_group_product_analytics_usage.query.graphql b/ee/app/assets/javascripts/usage_quotas/product_analytics/graphql/queries/get_group_product_analytics_usage.query.graphql new file mode 100644 index 000000000000..0eaba32c8ca5 --- /dev/null +++ b/ee/app/assets/javascripts/usage_quotas/product_analytics/graphql/queries/get_group_product_analytics_usage.query.graphql @@ -0,0 +1,18 @@ +query getGroupProductAnalyticsUsage($namespacePath: ID!, $monthSelection: [MonthSelectionInput!]!) { + group(fullPath: $namespacePath) { + id + projects(includeSubgroups: true) { + nodes { + id + name + avatarUrl + webUrl + productAnalyticsEventsStored(monthSelection: $monthSelection) { + year + month + count + } + } + } + } +} diff --git a/ee/app/assets/javascripts/usage_quotas/product_analytics/graphql/utils.js b/ee/app/assets/javascripts/usage_quotas/product_analytics/graphql/utils.js deleted file mode 100644 index a677eb18138a..000000000000 --- a/ee/app/assets/javascripts/usage_quotas/product_analytics/graphql/utils.js +++ /dev/null @@ -1,40 +0,0 @@ -/** - * Determines if a project has been onboarded with product analytics based on its usage data - */ -export const projectHasProductAnalyticsEnabled = (project) => - project.productAnalyticsEventsStored !== null; - -/** - * Maps a GraphQL response containing two sets of projects (for current / prev months) - * into a single array of projects, with the current/prev counts mapped onto each project. - * Fills in a `0` count if a project exists in one set but not the other. - */ -export const mapProjectsUsageResponse = (data) => { - const currentUsage = data.current.projects.nodes.filter(projectHasProductAnalyticsEnabled); - const previousUsage = data.previous.projects.nodes.filter(projectHasProductAnalyticsEnabled); - - const combinedUsage = new Map( - currentUsage.map((project) => { - const { __typename, productAnalyticsEventsStored, ...projectWithoutCount } = project; - return [ - project.id, - { - ...projectWithoutCount, - currentEvents: productAnalyticsEventsStored, - previousEvents: 0, - }, - ]; - }), - ); - - previousUsage.forEach((project) => { - const { __typename, productAnalyticsEventsStored, ...projectWithoutCount } = project; - combinedUsage.set(project.id, { - ...projectWithoutCount, - currentEvents: combinedUsage.get(project.id)?.currentEvents || 0, - previousEvents: productAnalyticsEventsStored, - }); - }); - - return Array.from(combinedUsage, ([, project]) => project); -}; diff --git a/ee/app/assets/javascripts/usage_quotas/product_analytics/utils.js b/ee/app/assets/javascripts/usage_quotas/product_analytics/utils.js new file mode 100644 index 000000000000..019a3dbb2cb9 --- /dev/null +++ b/ee/app/assets/javascripts/usage_quotas/product_analytics/utils.js @@ -0,0 +1,5 @@ +/** + * Determines if a project has been onboarded with product analytics based on its usage data + */ +export const projectHasProductAnalyticsEnabled = (project) => + project.productAnalyticsEventsStored?.some((usage) => usage.count !== null); diff --git a/ee/app/controllers/ee/groups/usage_quotas_controller.rb b/ee/app/controllers/ee/groups/usage_quotas_controller.rb index 4d9c58976d6d..f1b5891b290c 100644 --- a/ee/app/controllers/ee/groups/usage_quotas_controller.rb +++ b/ee/app/controllers/ee/groups/usage_quotas_controller.rb @@ -15,6 +15,7 @@ module UsageQuotasController push_frontend_feature_flag(:data_transfer_monitoring, group) push_frontend_feature_flag(:limited_access_modal) push_frontend_feature_flag(:enable_add_on_users_filtering, group) + push_frontend_feature_flag(:product_analytics_usage_quota_annual_data, group) end end diff --git a/ee/app/graphql/ee/types/project_type.rb b/ee/app/graphql/ee/types/project_type.rb index cd4b32118013..9c7a1246fff2 100644 --- a/ee/app/graphql/ee/types/project_type.rb +++ b/ee/app/graphql/ee/types/project_type.rb @@ -289,10 +289,11 @@ module ProjectType method: :prevent_merge_without_jira_issue?, description: 'Indicates if an associated issue from Jira is required.' - field :product_analytics_events_stored, GraphQL::Types::Int, + field :product_analytics_events_stored, [::Types::ProductAnalytics::MonthlyUsageType], null: true, resolver: ::Resolvers::ProductAnalytics::ProjectUsageDataResolver, - description: 'Count of all events used, filtered optionally by month.' + description: 'Count of all events used, broken down by month', + alpha: { milestone: '16.7' } field :dependency_proxy_packages_setting, ::Types::DependencyProxy::Packages::SettingType, diff --git a/ee/app/graphql/resolvers/product_analytics/project_usage_data_resolver.rb b/ee/app/graphql/resolvers/product_analytics/project_usage_data_resolver.rb index 6c9d1e706878..8e4d8c47b0c7 100644 --- a/ee/app/graphql/resolvers/product_analytics/project_usage_data_resolver.rb +++ b/ee/app/graphql/resolvers/product_analytics/project_usage_data_resolver.rb @@ -7,15 +7,21 @@ class ProjectUsageDataResolver < BaseResolver authorizes_object! authorize :maintainer_access - type GraphQL::Types::Int, null: true + type [::Types::ProductAnalytics::MonthlyUsageType], null: true - argument :year, GraphQL::Types::Int, - required: false, description: 'Year for the period to return.' + argument :month_selection, [::Types::ProductAnalytics::MonthSelectionInputType], + required: true, description: 'Selection for the period to return.' - argument :month, GraphQL::Types::Int, - required: false, description: 'Month for the period to return.' - def resolve(year: Time.current.year, month: Time.current.month) - object.product_analytics_events_used(year: year, month: month) + def resolve(month_selection: []) + month_selection.map do |selection| + year = selection[:year] + month = selection[:month] + { + year: year, + month: month, + count: object.product_analytics_events_used(year: year, month: month) + } + end end end end diff --git a/ee/app/graphql/types/product_analytics/month_selection_input_type.rb b/ee/app/graphql/types/product_analytics/month_selection_input_type.rb new file mode 100644 index 000000000000..0a21ab61c906 --- /dev/null +++ b/ee/app/graphql/types/product_analytics/month_selection_input_type.rb @@ -0,0 +1,18 @@ +# frozen_string_literal: true + +module Types + module ProductAnalytics + class MonthSelectionInputType < BaseInputObject + graphql_name 'MonthSelectionInput' + description "A year and month input for querying product analytics usage data." + + argument :month, GraphQL::Types::Int, + required: true, + description: 'Month of the period to return.' + + argument :year, GraphQL::Types::Int, + required: true, + description: 'Year of the period to return.' + end + end +end diff --git a/ee/app/graphql/types/product_analytics/monthly_usage_type.rb b/ee/app/graphql/types/product_analytics/monthly_usage_type.rb new file mode 100644 index 000000000000..b570c44f59d7 --- /dev/null +++ b/ee/app/graphql/types/product_analytics/monthly_usage_type.rb @@ -0,0 +1,14 @@ +# frozen_string_literal: true + +module Types + module ProductAnalytics + class MonthlyUsageType < BaseObject # rubocop:disable Graphql/AuthorizeTypes -- Authorization is done in resolver layer + graphql_name 'MonthlyUsage' + description 'Product analytics events for a specific month and year.' + + field :count, GraphQL::Types::Int, null: true, description: 'Count of product analytics events.' + field :month, GraphQL::Types::Int, null: false, description: 'Month of the data.' + field :year, GraphQL::Types::Int, null: false, description: 'Year of the data.' + end + end +end diff --git a/ee/spec/frontend/usage_quotas/product_analytics/components/product_analytics_group_usage_chart_spec.js b/ee/spec/frontend/usage_quotas/product_analytics/components/product_analytics_group_usage_spec.js similarity index 55% rename from ee/spec/frontend/usage_quotas/product_analytics/components/product_analytics_group_usage_chart_spec.js rename to ee/spec/frontend/usage_quotas/product_analytics/components/product_analytics_group_usage_spec.js index a6ef1e4165b7..876186d26968 100644 --- a/ee/spec/frontend/usage_quotas/product_analytics/components/product_analytics_group_usage_chart_spec.js +++ b/ee/spec/frontend/usage_quotas/product_analytics/components/product_analytics_group_usage_spec.js @@ -8,22 +8,26 @@ import createMockApollo from 'helpers/mock_apollo_helper'; import { shallowMountExtended } from 'helpers/vue_test_utils_helper'; import waitForPromises from 'helpers/wait_for_promises'; import { + getProjectWithYearsUsage, getProjectsUsageDataResponse, getProjectUsage, } from 'ee_jest/usage_quotas/product_analytics/graphql/mock_data'; import { useFakeDate } from 'helpers/fake_date'; -import getGroupCurrentAndPrevProductAnalyticsUsage from 'ee/usage_quotas/product_analytics/graphql/queries/get_group_current_and_prev_product_analytics_usage.query.graphql'; -import ProductAnalyticsGroupUsageChart from 'ee/usage_quotas/product_analytics/components/product_analytics_group_usage_chart.vue'; +import getGroupCurrentAndPrevProductAnalyticsUsage from 'ee/usage_quotas/product_analytics/graphql/queries/get_group_product_analytics_usage.query.graphql'; +import ProductAnalyticsGroupUsage from 'ee/usage_quotas/product_analytics/components/product_analytics_group_usage.vue'; Vue.use(VueApollo); jest.mock('~/sentry/sentry_browser_wrapper'); -describe('ProductAnalyticsGroupUsageChart', () => { +describe('ProductAnalyticsGroupUsage', () => { /** @type {import('helpers/vue_test_utils_helper').ExtendedWrapper} */ let wrapper; + const mockNow = '2023-01-15T12:00:00Z'; + useFakeDate(mockNow); + const findError = () => wrapper.findComponent(GlAlert); const findSkeletonLoader = () => wrapper.findComponent(GlSkeletonLoader); const findChart = () => wrapper.findComponent(GlAreaChart); @@ -31,15 +35,19 @@ describe('ProductAnalyticsGroupUsageChart', () => { const mockProjectsUsageDataHandler = jest.fn(); - const createComponent = () => { + const createComponent = ({ glFeatures } = {}) => { const mockApollo = createMockApollo([ [getGroupCurrentAndPrevProductAnalyticsUsage, mockProjectsUsageDataHandler], ]); - wrapper = shallowMountExtended(ProductAnalyticsGroupUsageChart, { + wrapper = shallowMountExtended(ProductAnalyticsGroupUsage, { apolloProvider: mockApollo, provide: { namespacePath: 'some-group', + glFeatures: { + productAnalyticsUsageQuotaAnnualData: true, + ...glFeatures, + }, }, stubs: { GlSprintf, @@ -61,18 +69,83 @@ describe('ProductAnalyticsGroupUsageChart', () => { }); describe('when fetching data', () => { - const mockNow = '2023-01-15T12:00:00Z'; - useFakeDate(mockNow); - - it('requests data from the current and previous months', () => { - createComponent(); + describe('when "product_analytics_usage_quota_annual_data" feature flag is enabled', () => { + it('requests data from the last 12 months', () => { + createComponent({ glFeatures: { productAnalyticsUsageQuotaAnnualData: true } }); + + expect(mockProjectsUsageDataHandler).toHaveBeenCalledWith({ + namespacePath: 'some-group', + monthSelection: [ + { + month: 1, + year: 2023, + }, + { + month: 12, + year: 2022, + }, + { + month: 11, + year: 2022, + }, + { + month: 10, + year: 2022, + }, + { + month: 9, + year: 2022, + }, + { + month: 8, + year: 2022, + }, + { + month: 7, + year: 2022, + }, + { + month: 6, + year: 2022, + }, + { + month: 5, + year: 2022, + }, + { + month: 4, + year: 2022, + }, + { + month: 3, + year: 2022, + }, + { + month: 2, + year: 2022, + }, + ], + }); + }); + }); - expect(mockProjectsUsageDataHandler).toHaveBeenCalledWith({ - namespacePath: 'some-group', - currentMonth: 1, - currentYear: 2023, - previousMonth: 12, - previousYear: 2022, + describe('when "product_analytics_usage_quota_annual_data" feature flag is disabled', () => { + it('requests data from the last 2 months', () => { + createComponent({ glFeatures: { productAnalyticsUsageQuotaAnnualData: false } }); + + expect(mockProjectsUsageDataHandler).toHaveBeenCalledWith({ + namespacePath: 'some-group', + monthSelection: [ + { + month: 1, + year: 2023, + }, + { + month: 12, + year: 2022, + }, + ], + }); }); }); @@ -125,13 +198,13 @@ describe('ProductAnalyticsGroupUsageChart', () => { describe('and the data has loaded', () => { describe.each` - scenario | currentProjects | previousProjects - ${'with no projects'} | ${[]} | ${[]} - ${'with no product analytics enabled projects'} | ${[getProjectUsage({ id: 1, name: 'not onboarded', numEvents: null })]} | ${[getProjectUsage({ id: 1, name: 'not onboarded', numEvents: null })]} - `('$scenario', ({ currentProjects, previousProjects }) => { + scenario | projects + ${'with no projects'} | ${[]} + ${'with no product analytics enabled projects'} | ${[getProjectUsage({ id: 1, name: 'not onboarded', usage: [{ year: 2023, month: 1, count: null }] })]} + `('$scenario', ({ projects }) => { beforeEach(() => { mockProjectsUsageDataHandler.mockResolvedValue({ - data: getProjectsUsageDataResponse(currentProjects, previousProjects), + data: getProjectsUsageDataResponse(projects), }); createComponent(); return waitForPromises(); @@ -152,7 +225,9 @@ describe('ProductAnalyticsGroupUsageChart', () => { describe('with one project', () => { beforeEach(() => { - mockProjectsUsageDataHandler.mockResolvedValue({ data: getProjectsUsageDataResponse() }); + mockProjectsUsageDataHandler.mockResolvedValue({ + data: getProjectsUsageDataResponse([getProjectWithYearsUsage()]), + }); createComponent(); return waitForPromises(); }); @@ -171,8 +246,18 @@ describe('ProductAnalyticsGroupUsageChart', () => { { name: 'Analytics events by month', data: [ - ['Dec 2022', 1234], - ['Jan 2023', 9876], + ['Feb 2022', 1], + ['Mar 2022', 1], + ['Apr 2022', 1], + ['May 2022', 1], + ['Jun 2022', 1], + ['Jul 2022', 1], + ['Aug 2022', 1], + ['Sep 2022', 1], + ['Oct 2022', 1], + ['Nov 2022', 1], + ['Dec 2022', 1], + ['Jan 2023', 1], ], }, ], @@ -183,18 +268,12 @@ describe('ProductAnalyticsGroupUsageChart', () => { describe('with many projects', () => { beforeEach(() => { mockProjectsUsageDataHandler.mockResolvedValue({ - data: getProjectsUsageDataResponse( - [ - getProjectUsage({ id: 1, name: 'onboarded1', numEvents: 1 }), - getProjectUsage({ id: 2, name: 'onboarded2', numEvents: 1 }), - getProjectUsage({ id: 3, name: 'onboarded3', numEvents: 1 }), - ], - [ - getProjectUsage({ id: 1, name: 'onboarded1', numEvents: 10 }), - getProjectUsage({ id: 2, name: 'onboarded2', numEvents: 20 }), - getProjectUsage({ id: 3, name: 'onboarded3', numEvents: 30 }), - ], - ), + data: getProjectsUsageDataResponse([ + getProjectWithYearsUsage({ + id: 1, + }), + getProjectWithYearsUsage({ id: 2 }), + ]), }); createComponent(); return waitForPromises(); @@ -206,8 +285,18 @@ describe('ProductAnalyticsGroupUsageChart', () => { { name: 'Analytics events by month', data: [ - ['Dec 2022', 60], - ['Jan 2023', 3], + ['Feb 2022', 2], + ['Mar 2022', 2], + ['Apr 2022', 2], + ['May 2022', 2], + ['Jun 2022', 2], + ['Jul 2022', 2], + ['Aug 2022', 2], + ['Sep 2022', 2], + ['Oct 2022', 2], + ['Nov 2022', 2], + ['Dec 2022', 2], + ['Jan 2023', 2], ], }, ], diff --git a/ee/spec/frontend/usage_quotas/product_analytics/components/product_analytics_usage_quota_app_spec.js b/ee/spec/frontend/usage_quotas/product_analytics/components/product_analytics_usage_quota_app_spec.js index 8da82329eb28..de5e0a09022e 100644 --- a/ee/spec/frontend/usage_quotas/product_analytics/components/product_analytics_usage_quota_app_spec.js +++ b/ee/spec/frontend/usage_quotas/product_analytics/components/product_analytics_usage_quota_app_spec.js @@ -2,7 +2,7 @@ import { nextTick } from 'vue'; import { shallowMount } from '@vue/test-utils'; import { GlEmptyState } from '@gitlab/ui'; import ProductAnalyticsUsageQuotaApp from 'ee/usage_quotas/product_analytics/components/product_analytics_usage_quota_app.vue'; -import ProductAnalyticsGroupUsageChart from 'ee/usage_quotas/product_analytics/components/product_analytics_group_usage_chart.vue'; +import ProductAnalyticsGroupUsage from 'ee/usage_quotas/product_analytics/components/product_analytics_group_usage.vue'; import ProductAnalyticsProjectsUsage from 'ee/usage_quotas/product_analytics/components/projects_usage/product_analytics_projects_usage.vue'; describe('ProductAnalyticsUsageQuotaApp', () => { @@ -10,8 +10,7 @@ describe('ProductAnalyticsUsageQuotaApp', () => { let wrapper; const findEmptyState = () => wrapper.findComponent(GlEmptyState); - const findProductAnalyticsGroupUsageChart = () => - wrapper.findComponent(ProductAnalyticsGroupUsageChart); + const findProductAnalyticsGroupUsage = () => wrapper.findComponent(ProductAnalyticsGroupUsage); const findProductAnalyticsProjectsUsage = () => wrapper.findComponent(ProductAnalyticsProjectsUsage); @@ -43,7 +42,7 @@ describe('ProductAnalyticsUsageQuotaApp', () => { beforeEach(() => createComponent({ productAnalyticsEnabled: true })); it('renders the monthly group usage chart', () => { - expect(findProductAnalyticsGroupUsageChart().exists()).toBe(true); + expect(findProductAnalyticsGroupUsage().exists()).toBe(true); }); it('renders the projects usage breakdown', () => { @@ -52,7 +51,7 @@ describe('ProductAnalyticsUsageQuotaApp', () => { describe('when there are no onboarded projects within the group', () => { beforeEach(() => { - findProductAnalyticsGroupUsageChart().vm.$emit('no-projects'); + findProductAnalyticsGroupUsage().vm.$emit('no-projects'); return nextTick(); }); diff --git a/ee/spec/frontend/usage_quotas/product_analytics/components/projects_usage/product_analytics_projects_usage_chart_spec.js b/ee/spec/frontend/usage_quotas/product_analytics/components/projects_usage/product_analytics_projects_usage_chart_spec.js index fe232bcf8e98..b9ec6d40f36c 100644 --- a/ee/spec/frontend/usage_quotas/product_analytics/components/projects_usage/product_analytics_projects_usage_chart_spec.js +++ b/ee/spec/frontend/usage_quotas/product_analytics/components/projects_usage/product_analytics_projects_usage_chart_spec.js @@ -2,11 +2,18 @@ import { GlSkeletonLoader } from '@gitlab/ui'; import { GlColumnChart } from '@gitlab/ui/dist/charts'; import ProductAnalyticsProjectsUsageChart from 'ee/usage_quotas/product_analytics/components/projects_usage/product_analytics_projects_usage_chart.vue'; import { mountExtended, shallowMountExtended } from 'helpers/vue_test_utils_helper'; +import { convertToGraphQLId } from '~/graphql_shared/utils'; +import { TYPENAME_PROJECT } from '~/graphql_shared/constants'; +import { getProjectUsage } from 'ee_jest/usage_quotas/product_analytics/graphql/mock_data'; +import { useFakeDate } from 'helpers/fake_date'; describe('ProductAnalyticsProjectsUsageChart', () => { /** @type {import('helpers/vue_test_utils_helper').ExtendedWrapper} */ let wrapper; + const mockNow = '2023-01-15T12:00:00Z'; + useFakeDate(mockNow); + const findLoadingState = () => wrapper.findComponent(GlSkeletonLoader); const findUsageChart = () => wrapper.findComponent(GlColumnChart); @@ -53,22 +60,20 @@ describe('ProductAnalyticsProjectsUsageChart', () => { }); describe('when there is project data', () => { - const projectsUsageData = [ - { - id: 1, - webUrl: '/test-project', - avatarUrl: '/test-project.jpg', - name: 'test-project', - currentEvents: 10, - previousEvents: 4, - }, - ]; - beforeEach(() => { createComponent( { isLoading: false, - projectsUsageData, + projectsUsageData: [ + getProjectUsage({ + id: convertToGraphQLId(TYPENAME_PROJECT, 1), + name: `test-project`, + usage: [ + { year: 2023, month: 1, count: 7 }, + { year: 2022, month: 12, count: 10 }, + ], + }), + ], }, mountExtended, ); @@ -82,12 +87,12 @@ describe('ProductAnalyticsProjectsUsageChart', () => { expect(findUsageChart().props()).toMatchObject({ bars: [ { - data: [['test-project', 4]], + data: [['test-project', 10]], name: 'Previous month', stack: 'previous', }, { - data: [['test-project', 10]], + data: [['test-project', 7]], name: 'Current month to date', stack: 'current', }, @@ -104,14 +109,16 @@ describe('ProductAnalyticsProjectsUsageChart', () => { createComponent( { isLoading: false, - projectsUsageData: Array.from({ length: 51 }).map((_, index) => ({ - id: index, - webUrl: `/test-project-${index}`, - avatarUrl: `/test-project-${index}.jpg`, - name: `test-project-${index}`, - currentEvents: 10, - previousEvents: 4, - })), + projectsUsageData: Array.from({ length: 51 }).map((_, index) => + getProjectUsage({ + id: convertToGraphQLId(TYPENAME_PROJECT, index), + name: `test-project-${index}`, + usage: [ + { year: 2023, month: 1, count: 7 }, + { year: 2022, month: 12, count: 10 }, + ], + }), + ), }, mountExtended, ); diff --git a/ee/spec/frontend/usage_quotas/product_analytics/components/projects_usage/product_analytics_projects_usage_spec.js b/ee/spec/frontend/usage_quotas/product_analytics/components/projects_usage/product_analytics_projects_usage_spec.js index 3a2a43d7e694..6fa6e159307a 100644 --- a/ee/spec/frontend/usage_quotas/product_analytics/components/projects_usage/product_analytics_projects_usage_spec.js +++ b/ee/spec/frontend/usage_quotas/product_analytics/components/projects_usage/product_analytics_projects_usage_spec.js @@ -6,16 +6,14 @@ import ProductAnalyticsProjectsUsage from 'ee/usage_quotas/product_analytics/com import ProductAnalyticsProjectsUsageChart from 'ee/usage_quotas/product_analytics/components/projects_usage/product_analytics_projects_usage_chart.vue'; import ProductAnalyticsProjectsUsageTable from 'ee/usage_quotas/product_analytics/components/projects_usage/product_analytics_projects_usage_table.vue'; import createMockApollo from 'helpers/mock_apollo_helper'; -import getGroupCurrentAndPrevProductAnalyticsUsage from 'ee/usage_quotas/product_analytics/graphql/queries/get_group_current_and_prev_product_analytics_usage.query.graphql'; +import getGroupCurrentAndPrevProductAnalyticsUsage from 'ee/usage_quotas/product_analytics/graphql/queries/get_group_product_analytics_usage.query.graphql'; import { shallowMountExtended } from 'helpers/vue_test_utils_helper'; import waitForPromises from 'helpers/wait_for_promises'; import { getProjectsUsageDataResponse } from 'ee_jest/usage_quotas/product_analytics/graphql/mock_data'; import { useFakeDate } from 'helpers/fake_date'; -import * as utils from 'ee/usage_quotas/product_analytics/graphql/utils'; Vue.use(VueApollo); -jest.mock('ee/usage_quotas/product_analytics/graphql/utils'); jest.mock('~/sentry/sentry_browser_wrapper'); describe('ProductAnalyticsProjectsUsage', () => { @@ -62,10 +60,16 @@ describe('ProductAnalyticsProjectsUsage', () => { expect(mockProjectsUsageDataHandler).toHaveBeenCalledWith({ namespacePath: 'some-group', - currentMonth: 1, - currentYear: 2023, - previousMonth: 12, - previousYear: 2022, + monthSelection: [ + { + month: 1, + year: 2023, + }, + { + month: 12, + year: 2022, + }, + ], }); }); @@ -121,17 +125,8 @@ describe('ProductAnalyticsProjectsUsage', () => { }); describe('and data has loaded', () => { - const projectsUsageData = [ - { - name: 'some onboarded project', - currentEvents: 9876, - previousEvents: 1234, - }, - ]; - beforeEach(() => { mockProjectsUsageDataHandler.mockResolvedValue({ data: getProjectsUsageDataResponse() }); - utils.mapProjectsUsageResponse.mockReturnValue(projectsUsageData); createComponent(); return waitForPromises(); }); @@ -143,14 +138,24 @@ describe('ProductAnalyticsProjectsUsage', () => { it('renders the chart', () => { expect(findProductAnalyticsProjectsUsageChart().props()).toMatchObject({ isLoading: false, - projectsUsageData, + projectsUsageData: [ + { + name: 'some onboarded project', + productAnalyticsEventsStored: [{ year: 2023, month: 11, count: 1234 }], + }, + ], }); }); it('renders the usage table', () => { expect(findProductAnalyticsProjectsUsageTable().props()).toMatchObject({ isLoading: false, - projectsUsageData, + projectsUsageData: [ + { + name: 'some onboarded project', + productAnalyticsEventsStored: [{ year: 2023, month: 11, count: 1234 }], + }, + ], }); }); }); diff --git a/ee/spec/frontend/usage_quotas/product_analytics/components/projects_usage/product_analytics_projects_usage_table_spec.js b/ee/spec/frontend/usage_quotas/product_analytics/components/projects_usage/product_analytics_projects_usage_table_spec.js index a283e93757b5..3139c206d3fe 100644 --- a/ee/spec/frontend/usage_quotas/product_analytics/components/projects_usage/product_analytics_projects_usage_table_spec.js +++ b/ee/spec/frontend/usage_quotas/product_analytics/components/projects_usage/product_analytics_projects_usage_table_spec.js @@ -2,11 +2,18 @@ import { GlSkeletonLoader, GlTableLite } from '@gitlab/ui'; import ProductAnalyticsProjectsUsageTable from 'ee/usage_quotas/product_analytics/components/projects_usage/product_analytics_projects_usage_table.vue'; import { mountExtended, shallowMountExtended } from 'helpers/vue_test_utils_helper'; import ProjectAvatar from '~/vue_shared/components/project_avatar.vue'; +import { getProjectUsage } from 'ee_jest/usage_quotas/product_analytics/graphql/mock_data'; +import { useFakeDate } from 'helpers/fake_date'; +import { convertToGraphQLId } from '~/graphql_shared/utils'; +import { TYPENAME_PROJECT } from '~/graphql_shared/constants'; describe('ProductAnalyticsProjectsUsageTable', () => { /** @type {import('helpers/vue_test_utils_helper').ExtendedWrapper} */ let wrapper; + const mockNow = '2023-01-15T12:00:00Z'; + useFakeDate(mockNow); + const findLoadingState = () => wrapper.findComponent(GlSkeletonLoader); const findUsageTableWrapper = () => wrapper.findByTestId('projects-usage-table'); const findUsageTable = () => wrapper.findComponent(GlTableLite); @@ -56,22 +63,20 @@ describe('ProductAnalyticsProjectsUsageTable', () => { }); describe('when there is project data', () => { - const projectsUsageData = [ - { - id: 1, - webUrl: '/test-project', - avatarUrl: '/test-project.jpg', - name: 'test-project', - currentEvents: 10, - previousEvents: 4, - }, - ]; - beforeEach(() => { createComponent( { isLoading: false, - projectsUsageData, + projectsUsageData: [ + getProjectUsage({ + id: convertToGraphQLId(TYPENAME_PROJECT, 1), + name: 'test-project', + usage: [ + { year: 2023, month: 1, count: 4 }, + { year: 2022, month: 12, count: 7 }, + ], + }), + ], }, mountExtended, ); @@ -92,7 +97,7 @@ describe('ProductAnalyticsProjectsUsageTable', () => { it('renders the project avatar', () => { expect(findProjectAvatar().props()).toMatchObject( expect.objectContaining({ - projectId: 1, + projectId: 'gid://gitlab/Project/1', projectAvatarUrl: '/test-project.jpg', projectName: 'test-project', alt: 'test-project', diff --git a/ee/spec/frontend/usage_quotas/product_analytics/components/utils_spec.js b/ee/spec/frontend/usage_quotas/product_analytics/components/utils_spec.js index d2eb172e45b8..e583ff4ce694 100644 --- a/ee/spec/frontend/usage_quotas/product_analytics/components/utils_spec.js +++ b/ee/spec/frontend/usage_quotas/product_analytics/components/utils_spec.js @@ -1,7 +1,33 @@ -import { projectsUsageDataValidator } from 'ee/usage_quotas/product_analytics/components/utils'; +import { + projectsUsageDataValidator, + findCurrentMonthUsage, + findPreviousMonthUsage, + mapMonthlyTotals, +} from 'ee/usage_quotas/product_analytics/components/utils'; +import { + getProjectUsage, + getProjectWithYearsUsage, +} from 'ee_jest/usage_quotas/product_analytics/graphql/mock_data'; +import { useFakeDate } from 'helpers/fake_date'; describe('Product analytics usage quota component utils', () => { describe('projectsUsageDataValidator', () => { + let validProject; + + beforeEach(() => { + validProject = getProjectUsage({ + id: 'gid://gitlab/Project/2', + name: 'another project', + usage: [ + { + year: 2023, + month: 11, + count: null, + }, + ], + }); + }); + it('returns true for empty array', () => { const result = projectsUsageDataValidator([]); @@ -9,18 +35,7 @@ describe('Product analytics usage quota component utils', () => { }); it('returns true when all items have all properties', () => { - const result = projectsUsageDataValidator([ - { - name: 'some project', - currentEvents: 1, - previousEvents: 1, - }, - { - name: 'another project', - currentEvents: 2, - previousEvents: 2, - }, - ]); + const result = projectsUsageDataValidator([validProject, getProjectUsage()]); expect(result).toBe(true); }); @@ -33,14 +48,10 @@ describe('Product analytics usage quota component utils', () => { it('returns false when one item is invalid', () => { const result = projectsUsageDataValidator([ + validProject, { - name: 'some project', - currentEvents: 1, - previousEvents: 1, - }, - { - currentEvents: 2, - previousEvents: 2, + ...validProject, + name: undefined, }, ]); @@ -49,16 +60,24 @@ describe('Product analytics usage quota component utils', () => { it.each([ { - currentEvents: 1, - previousEvents: 1, + ...validProject, + id: undefined, }, { - name: 'some project', - previousEvents: 1, + ...validProject, + name: undefined, }, { - name: 'some project', - currentEvents: 1, + ...validProject, + productAnalyticsEventsStored: undefined, + }, + { + ...validProject, + webUrl: undefined, + }, + { + ...validProject, + avatarUrl: undefined, }, ])('returns false when an item property is missing', (testCase) => { const result = projectsUsageDataValidator([testCase]); @@ -68,19 +87,24 @@ describe('Product analytics usage quota component utils', () => { it.each([ { - name: 12345, - currentEvents: 1, - previousEvents: 1, + ...validProject, + id: false, + }, + { + ...validProject, + name: false, }, { - name: 'some project', - currentEvents: 'invalid', - previousEvents: 1, + ...validProject, + productAnalyticsEventsStored: false, }, { - name: 'some project', - currentEvents: 1, - previousEvents: 'invalid', + ...validProject, + webUrl: false, + }, + { + ...validProject, + avatarUrl: false, }, ])('returns false when an item property is the wrong type', (testCase) => { const result = projectsUsageDataValidator([testCase]); @@ -88,4 +112,90 @@ describe('Product analytics usage quota component utils', () => { expect(result).toBe(false); }); }); + + describe('findCurrentMonthUsage', () => { + const mockNow = '2023-01-15T12:00:00Z'; + useFakeDate(mockNow); + + it('returns the expected usage', () => { + const result = findCurrentMonthUsage(getProjectWithYearsUsage()); + + expect(result).toMatchObject({ + count: 1, + month: 1, + year: 2023, + }); + }); + }); + + describe('findPreviousMonthUsage', () => { + const mockNow = '2023-01-15T12:00:00Z'; + useFakeDate(mockNow); + + it('returns the expected usage', () => { + const result = findPreviousMonthUsage(getProjectWithYearsUsage()); + + expect(result).toMatchObject({ + count: 1, + month: 12, + year: 2022, + }); + }); + }); + + describe('mapMonthlyTotals', () => { + it('returns an empty array for empty projects array', () => { + const result = mapMonthlyTotals([]); + expect(result).toEqual([]); + }); + + it('sums counts for the same month and year', () => { + const projects = [ + getProjectUsage({ usage: [{ year: 2023, month: 1, count: 10 }] }), + getProjectUsage({ usage: [{ year: 2023, month: 1, count: 5 }] }), + ]; + const result = mapMonthlyTotals(projects); + expect(result).toEqual([['Jan 2023', 15]]); + }); + + it('handles multiple months and years', () => { + const projects = [ + getProjectUsage({ usage: [{ year: 2023, month: 1, count: 10 }] }), + getProjectUsage({ usage: [{ year: 2022, month: 12, count: 5 }] }), + getProjectUsage({ usage: [{ year: 2023, month: 2, count: 8 }] }), + ]; + const result = mapMonthlyTotals(projects); + expect(result).toEqual([ + ['Dec 2022', 5], + ['Jan 2023', 10], + ['Feb 2023', 8], + ]); + }); + + it('returns sorted results', () => { + const projects = [ + getProjectUsage({ usage: [{ year: 2023, month: 2, count: 8 }] }), + getProjectUsage({ usage: [{ year: 2023, month: 1, count: 10 }] }), + getProjectUsage({ usage: [{ year: 2022, month: 12, count: 5 }] }), + ]; + const result = mapMonthlyTotals(projects); + expect(result).toEqual([ + ['Dec 2022', 5], + ['Jan 2023', 10], + ['Feb 2023', 8], + ]); + }); + + it('handles empty months', () => { + const projects = [ + getProjectUsage({ usage: [{ year: 2023, month: 1, count: 0 }] }), + getProjectUsage({ usage: [{ year: 2023, month: 2, count: 0 }] }), + ]; + const result = mapMonthlyTotals(projects); + expect(result).toEqual([ + ['Jan 2023', 0], + ['Feb 2023', 0], + ]); + }); + }); }); diff --git a/ee/spec/frontend/usage_quotas/product_analytics/graphql/mock_data.js b/ee/spec/frontend/usage_quotas/product_analytics/graphql/mock_data.js index cde4234b6092..525c6ca06960 100644 --- a/ee/spec/frontend/usage_quotas/product_analytics/graphql/mock_data.js +++ b/ee/spec/frontend/usage_quotas/product_analytics/graphql/mock_data.js @@ -1,48 +1,41 @@ import { convertToGraphQLId } from '~/graphql_shared/utils'; import { TYPENAME_GROUP, TYPENAME_PROJECT } from '~/graphql_shared/constants'; -export const getProjectUsage = ({ id, name, numEvents }) => ({ - id, - name, - productAnalyticsEventsStored: numEvents !== undefined ? numEvents : 2345, +export const getProjectUsage = ({ id, name, usage } = {}) => ({ + id: id || convertToGraphQLId(TYPENAME_PROJECT, 1), + name: name || 'some project', + productAnalyticsEventsStored: usage || [], webUrl: `/${name}`, avatarUrl: `/${name}.jpg`, __typename: 'Project', }); -export const getProjectsUsageDataResponse = (currentProjects, previousProjects) => ({ - previous: { +export const getProjectsUsageDataResponse = (projects) => ({ + group: { id: convertToGraphQLId(TYPENAME_GROUP, 1), projects: { - nodes: previousProjects || [ + nodes: projects || [ getProjectUsage({ id: convertToGraphQLId(TYPENAME_PROJECT, 1), name: 'some onboarded project', - numEvents: 1234, + usage: [ + { + year: 2023, + month: 11, + count: 1234, + }, + ], }), getProjectUsage({ id: convertToGraphQLId(TYPENAME_PROJECT, 2), name: 'not onboarded project', - numEvents: null, - }), - ], - __typename: 'ProjectConnection', - }, - __typename: 'Group', - }, - current: { - id: convertToGraphQLId(TYPENAME_GROUP, 1), - projects: { - nodes: currentProjects || [ - getProjectUsage({ - id: convertToGraphQLId(TYPENAME_PROJECT, 1), - name: 'some onboarded project', - numEvents: 9876, - }), - getProjectUsage({ - id: convertToGraphQLId(TYPENAME_PROJECT, 2), - name: 'not onboarded project', - numEvents: null, + usage: [ + { + year: 2023, + month: 11, + count: null, + }, + ], }), ], __typename: 'ProjectConnection', @@ -50,3 +43,71 @@ export const getProjectsUsageDataResponse = (currentProjects, previousProjects) __typename: 'Group', }, }); + +export const getProjectWithYearsUsage = ({ id, name } = {}) => + getProjectUsage({ + id, + name, + usage: [ + { + month: 1, + year: 2023, + count: 1, + }, + { + month: 12, + year: 2022, + count: 1, + }, + { + month: 11, + year: 2022, + count: 1, + }, + { + month: 10, + year: 2022, + count: 1, + }, + { + month: 9, + year: 2022, + count: 1, + }, + { + month: 8, + year: 2022, + count: 1, + }, + { + month: 7, + year: 2022, + count: 1, + }, + { + month: 6, + year: 2022, + count: 1, + }, + { + month: 5, + year: 2022, + count: 1, + }, + { + month: 4, + year: 2022, + count: 1, + }, + { + month: 3, + year: 2022, + count: 1, + }, + { + month: 2, + year: 2022, + count: 1, + }, + ], + }); diff --git a/ee/spec/frontend/usage_quotas/product_analytics/graphql/utils_spec.js b/ee/spec/frontend/usage_quotas/product_analytics/graphql/utils_spec.js deleted file mode 100644 index f2dd912f4820..000000000000 --- a/ee/spec/frontend/usage_quotas/product_analytics/graphql/utils_spec.js +++ /dev/null @@ -1,166 +0,0 @@ -import { - projectHasProductAnalyticsEnabled, - mapProjectsUsageResponse, -} from 'ee/usage_quotas/product_analytics/graphql/utils'; -import { getProjectsUsageDataResponse, getProjectUsage } from './mock_data'; - -describe('Product analytics usage quota graphql utils', () => { - describe('projectHasProductAnalyticsEnabled', () => { - it.each` - numEvents | expected - ${null} | ${false} - ${0} | ${true} - ${1} | ${true} - `('returns $expected when events stored count is $numEvents', ({ numEvents, expected }) => { - const result = projectHasProductAnalyticsEnabled( - getProjectUsage({ id: 1, name: 'some project', numEvents }), - ); - expect(result).toBe(expected); - }); - }); - - describe('mapProjectsUsageResponse', () => { - it('returns expected value when there are no projects', () => { - const response = getProjectsUsageDataResponse([], []); - - const mapped = mapProjectsUsageResponse(response); - - expect(mapped).toEqual([]); - }); - - it('returns expected value when there are no onboarded projects', () => { - const response = getProjectsUsageDataResponse( - [getProjectUsage({ id: 1, name: 'not onboarded', numEvents: null })], - [getProjectUsage({ id: 1, name: 'not onboarded', numEvents: null })], - ); - - const mapped = mapProjectsUsageResponse(response); - - expect(mapped).toEqual([]); - }); - - it('returns expected value when a project is onboarded in the current month', () => { - const response = getProjectsUsageDataResponse( - [getProjectUsage({ id: 1, name: 'just onboarded', numEvents: 1 })], - [getProjectUsage({ id: 1, name: 'just onboarded', numEvents: null })], - ); - - const mapped = mapProjectsUsageResponse(response); - - expect(mapped).toEqual([ - { - id: 1, - name: 'just onboarded', - webUrl: '/just onboarded', - avatarUrl: '/just onboarded.jpg', - currentEvents: 1, - previousEvents: 0, - }, - ]); - }); - - it('returns expected value when there is current but no previous projects', () => { - const response = getProjectsUsageDataResponse( - [getProjectUsage({ id: 1, name: 'onboarded', numEvents: 1234 })], - [], - ); - - const mapped = mapProjectsUsageResponse(response); - - expect(mapped).toEqual([ - { - id: 1, - name: 'onboarded', - webUrl: '/onboarded', - avatarUrl: '/onboarded.jpg', - currentEvents: 1234, - previousEvents: 0, - }, - ]); - }); - - it('returns expected value when there are previous but no current projects', () => { - const response = getProjectsUsageDataResponse( - [], - [getProjectUsage({ id: 1, name: 'onboarded', numEvents: 1234 })], - ); - - const mapped = mapProjectsUsageResponse(response); - - expect(mapped).toEqual([ - { - id: 1, - name: 'onboarded', - webUrl: '/onboarded', - avatarUrl: '/onboarded.jpg', - currentEvents: 0, - previousEvents: 1234, - }, - ]); - }); - - it('returns expected value when there are both current and previous projects', () => { - const response = getProjectsUsageDataResponse( - [getProjectUsage({ id: 1, name: 'onboarded', numEvents: 1234 })], - [getProjectUsage({ id: 1, name: 'onboarded', numEvents: 987 })], - ); - - const mapped = mapProjectsUsageResponse(response); - - expect(mapped).toEqual([ - { - id: 1, - name: 'onboarded', - webUrl: '/onboarded', - avatarUrl: '/onboarded.jpg', - currentEvents: 1234, - previousEvents: 987, - }, - ]); - }); - - it('returns expected value when there are mixed projects', () => { - const response = getProjectsUsageDataResponse( - [ - getProjectUsage({ id: 1, name: 'onboarded', numEvents: 1234 }), - getProjectUsage({ id: 2, name: 'current only project', numEvents: 55 }), - getProjectUsage({ id: 3, name: 'not onboarded', numEvents: null }), - ], - [ - getProjectUsage({ id: 1, name: 'onboarded', numEvents: 987 }), - getProjectUsage({ id: 4, name: 'previous only project', numEvents: 777 }), - getProjectUsage({ id: 3, name: 'not onboarded', numEvents: null }), - ], - ); - - const mapped = mapProjectsUsageResponse(response); - - expect(mapped).toEqual([ - { - id: 1, - name: 'onboarded', - webUrl: '/onboarded', - avatarUrl: '/onboarded.jpg', - currentEvents: 1234, - previousEvents: 987, - }, - { - id: 2, - name: 'current only project', - webUrl: '/current only project', - avatarUrl: '/current only project.jpg', - currentEvents: 55, - previousEvents: 0, - }, - { - id: 4, - name: 'previous only project', - webUrl: '/previous only project', - avatarUrl: '/previous only project.jpg', - currentEvents: 0, - previousEvents: 777, - }, - ]); - }); - }); -}); diff --git a/ee/spec/frontend/usage_quotas/product_analytics/utils_spec.js b/ee/spec/frontend/usage_quotas/product_analytics/utils_spec.js new file mode 100644 index 000000000000..c6d8766d0416 --- /dev/null +++ b/ee/spec/frontend/usage_quotas/product_analytics/utils_spec.js @@ -0,0 +1,18 @@ +import { projectHasProductAnalyticsEnabled } from 'ee/usage_quotas/product_analytics/utils'; +import { getProjectUsage } from './graphql/mock_data'; + +describe('Product analytics usage quota utils', () => { + describe('projectHasProductAnalyticsEnabled', () => { + it.each` + count | expected + ${null} | ${false} + ${0} | ${true} + ${1} | ${true} + `('returns $expected when events stored count is $count', ({ count, expected }) => { + const result = projectHasProductAnalyticsEnabled( + getProjectUsage({ id: 1, name: 'some project', usage: [{ year: 2023, month: 11, count }] }), + ); + expect(result).toBe(expected); + }); + }); +}); diff --git a/ee/spec/requests/api/graphql/project/product_analytics/events_stored_spec.rb b/ee/spec/requests/api/graphql/project/product_analytics/events_stored_spec.rb index 241daf3dbd08..605b886d0017 100644 --- a/ee/spec/requests/api/graphql/project/product_analytics/events_stored_spec.rb +++ b/ee/spec/requests/api/graphql/project/product_analytics/events_stored_spec.rb @@ -14,7 +14,14 @@ %( query { project(fullPath: "#{project.full_path}") { - productAnalyticsEventsStored + productAnalyticsEventsStored(monthSelection: [ + { year: 2023, month: 1 }, + { year: 2022, month: 12 } + ]) { + year + month + count + } } } ) @@ -29,10 +36,12 @@ end context 'when project does not have product analytics enabled' do - it "returns zero" do + it "returns zero for each months usage" do subject - expect(graphql_data.dig('project', 'productAnalyticsEventsStored')).to be_zero + graphql_data.dig('project', 'productAnalyticsEventsStored').each do |event| + expect(event['count']).to be_zero + end end end @@ -43,41 +52,21 @@ end end - it 'queries the ProjectUsageData interface' do - freeze_time do - expect_next_instance_of(Analytics::ProductAnalytics::ProjectUsageData) do |instance| - expect(instance) - .to receive(:events_stored_count).with(year: Time.current.year, month: Time.current.month).once - end - - subject - end - end - context 'when user is not a project member' do let_it_be(:user) { create(:user) } it { is_expected.to be_nil } end - context 'when setting a month and year' do - let(:query) do - %( - query { - project(fullPath: "#{project.full_path}") { - productAnalyticsEventsStored(year: 2021, month: 3) - } - } - ) - end + it 'queries the ProjectUsageData interface with the correct parameters' do + instance = Analytics::ProductAnalytics::ProjectUsageData.new(project_id: project.id) - it 'queries the ProjectUsageData interface with the correct parameters' do - expect_next_instance_of(Analytics::ProductAnalytics::ProjectUsageData) do |instance| - expect(instance).to receive(:events_stored_count).with(year: 2021, month: 3).once - end + allow(Analytics::ProductAnalytics::ProjectUsageData).to receive(:new).and_return(instance) - subject - end + expect(instance).to receive(:events_stored_count).with(year: 2023, month: 1).once + expect(instance).to receive(:events_stored_count).with(year: 2022, month: 12).once + + subject end end end -- GitLab