Commit 66a34007 authored by Scott Hampton's avatar Scott Hampton

Refactor chart and tests

Refactored the props for the area chart to be
more in line with what echarts expects.

Updated the specs to test a little more of the data in the chart.

Refactored the specs to use shallowMount.

Updated the documentation to point to the repo
analytics documentation.
parent ab9c3439
...@@ -854,19 +854,7 @@ With [GitLab Issue Analytics](issues_analytics/index.md), you can see a bar char ...@@ -854,19 +854,7 @@ With [GitLab Issue Analytics](issues_analytics/index.md), you can see a bar char
> - [Introduced](https://gitlab.com/gitlab-org/gitlab/-/issues/263478) in GitLab 13.6. > - [Introduced](https://gitlab.com/gitlab-org/gitlab/-/issues/263478) in GitLab 13.6.
> - [Feature flag removed](https://gitlab.com/gitlab-org/gitlab/-/issues/276003) in GitLab 13.7. > - [Feature flag removed](https://gitlab.com/gitlab-org/gitlab/-/issues/276003) in GitLab 13.7.
With [GitLab Repositories Analytics](repositories_analytics/index.md), you can download a CSV of the latest coverage data for all the projects in your group. With [GitLab Repositories Analytics](repositories_analytics/index.md), you can view overall activity of all projects with code coverage.
### Check code coverage for all projects
> [Introduced](https://gitlab.com/gitlab-org/gitlab/-/issues/263478) in [GitLab Premium](https://about.gitlab.com/pricing/) 13.7.
See the overall activity of all projects with code coverage with [GitLab Repositories Analytics](repositories_analytics/index.md).
It displays the current code coverage data available for your projects. In
[GitLab 13.9 and later](https://gitlab.com/gitlab-org/gitlab/-/issues/215140), it
also displays a graph of the average code coverage over the last 30 days:
![Group repositories analytics](img/group_code_coverage_analytics_v13_9.png)
## Dependency Proxy ## Dependency Proxy
......
...@@ -12,6 +12,25 @@ info: To determine the technical writer assigned to the Stage/Group associated w ...@@ -12,6 +12,25 @@ info: To determine the technical writer assigned to the Stage/Group associated w
WARNING: WARNING:
This feature might not be available to you. Check the **version history** note above for details. This feature might not be available to you. Check the **version history** note above for details.
![Group repositories analytics](../img/group_code_coverage_analytics_v13_9.png)
## Current group code coverage
> - [Introduced](https://gitlab.com/gitlab-org/gitlab/-/issues/263478) in [GitLab Premium](https://about.gitlab.com/pricing/) 13.7.
The **Analytics > Repositories** group page displays the overall test coverage of all your projects in your group.
In the **Overall activity** section, you can see:
- The number of projects with coverage reports.
- The average percentage of coverage across all your projects.
- The total number of pipeline jobs that produce coverage reports.
## Average group test coverage from the last 30 days
> - [Introduced](https://gitlab.com/gitlab-org/gitlab/-/issues/215140) in [GitLab Premium](https://about.gitlab.com/pricing/) 13.9.
The **Analytics > Repositories** group page displays the average test coverage of all your projects in your group in a graph for the last 30 days.
## Latest project test coverage list ## Latest project test coverage list
> - [Introduced](https://gitlab.com/gitlab-org/gitlab/-/issues/267624) in [GitLab Premium](https://about.gitlab.com/pricing/) 13.6. > - [Introduced](https://gitlab.com/gitlab-org/gitlab/-/issues/267624) in [GitLab Premium](https://about.gitlab.com/pricing/) 13.6.
......
...@@ -5,8 +5,11 @@ import { __, s__ } from '~/locale'; ...@@ -5,8 +5,11 @@ import { __, s__ } from '~/locale';
import MetricCard from '~/analytics/shared/components/metric_card.vue'; import MetricCard from '~/analytics/shared/components/metric_card.vue';
import { formatDate } from '~/lib/utils/datetime_utility'; import { formatDate } from '~/lib/utils/datetime_utility';
import ChartSkeletonLoader from '~/vue_shared/components/resizable_chart/skeleton_loader.vue'; import ChartSkeletonLoader from '~/vue_shared/components/resizable_chart/skeleton_loader.vue';
import { SUPPORTED_FORMATS, getFormatter } from '~/lib/utils/unit_format';
import getGroupTestCoverage from '../graphql/queries/get_group_test_coverage.query.graphql'; import getGroupTestCoverage from '../graphql/queries/get_group_test_coverage.query.graphql';
const formatPercent = getFormatter(SUPPORTED_FORMATS.percentHundred);
export default { export default {
name: 'TestCoverageSummary', name: 'TestCoverageSummary',
components: { components: {
...@@ -29,7 +32,7 @@ export default { ...@@ -29,7 +32,7 @@ export default {
return { return {
groupFullPath: this.groupFullPath, groupFullPath: this.groupFullPath,
startDate: new Date(Date.now() - THIRTY_DAYS), startDate: formatDate(new Date(Date.now() - THIRTY_DAYS), 'yyyy-mm-dd'),
}; };
}, },
result({ data }) { result({ data }) {
...@@ -42,11 +45,8 @@ export default { ...@@ -42,11 +45,8 @@ export default {
this.coverageCount = coverageCount; this.coverageCount = coverageCount;
this.groupCoverageChartData = [ this.groupCoverageChartData = [
{ {
name: this.$options.text.graphName, name: this.$options.i18n.graphName,
data: groupCoverage.map((coverage) => [ data: groupCoverage.map((coverage) => [coverage.date, coverage.averageCoverage]),
formatDate(coverage.date, 'mmm dd'),
coverage.averageCoverage,
]),
}, },
]; ];
}, },
...@@ -80,34 +80,44 @@ export default { ...@@ -80,34 +80,44 @@ export default {
{ {
key: 'projectCount', key: 'projectCount',
value: this.projectCount, value: this.projectCount,
label: this.$options.text.metrics.projectCountLabel, label: this.$options.i18n.metrics.projectCountLabel,
}, },
{ {
key: 'averageCoverage', key: 'averageCoverage',
value: this.averageCoverage, value: this.averageCoverage,
unit: '%', unit: '%',
label: this.$options.text.metrics.averageCoverageLabel, label: this.$options.i18n.metrics.averageCoverageLabel,
}, },
{ {
key: 'coverageCount', key: 'coverageCount',
value: this.coverageCount, value: this.coverageCount,
label: this.$options.text.metrics.coverageCountLabel, label: this.$options.i18n.metrics.coverageCountLabel,
}, },
]; ];
}, },
chartOptions() { chartOptions() {
return { return {
xAxis: { xAxis: {
name: this.$options.text.xAxisName, name: this.$options.i18n.xAxisName,
type: 'category', type: 'time',
axisLabel: {
formatter: (value) => formatDate(value, 'mmm dd'),
},
}, },
yAxis: { yAxis: {
name: this.$options.text.yAxisName, name: this.$options.i18n.yAxisName,
type: 'value', type: 'value',
min: 0, min: 0,
max: 100, max: 100,
axisLabel: { axisLabel: {
formatter: (value) => `${value}%`, /**
* We can't do `formatter: formatPercent` because
* formatter passes in a second argument of index, which
* formatPercent takes in as the number of decimal points
* we should include after. This formats 100 as 100.00000%
* instead of 100%.
*/
formatter: (value) => formatPercent(value),
}, },
}, },
}; };
...@@ -115,18 +125,18 @@ export default { ...@@ -115,18 +125,18 @@ export default {
}, },
methods: { methods: {
formatTooltipText(params) { formatTooltipText(params) {
this.tooltipTitle = params.value; this.tooltipTitle = formatDate(params.value, 'mmm dd');
this.coveragePercentage = params.seriesData?.[0]?.data?.[1]; this.coveragePercentage = formatPercent(params.seriesData?.[0]?.data?.[1], 2);
}, },
}, },
text: { i18n: {
graphCardHeader: s__('RepositoriesAnalytics|Average test coverage last 30 days'), graphCardHeader: s__('RepositoriesAnalytics|Average test coverage last 30 days'),
yAxisName: __('Coverage'), yAxisName: __('Coverage'),
xAxisName: __('Date'), xAxisName: __('Date'),
graphName: s__('RepositoriesAnalytics|Average coverage'), graphName: s__('RepositoriesAnalytics|Average coverage'),
graphTooltipMessage: __('Code Coverage: %{coveragePercentage}%{percentSymbol}'), graphTooltipMessage: __('Code Coverage: %{coveragePercentage}'),
metrics: { metrics: {
cardTitle: __('Overall Activity'), cardTitle: __('Overall activity'),
projectCountLabel: s__('RepositoriesAnalytics|Projects with Coverage'), projectCountLabel: s__('RepositoriesAnalytics|Projects with Coverage'),
averageCoverageLabel: s__('RepositoriesAnalytics|Average Coverage by Job'), averageCoverageLabel: s__('RepositoriesAnalytics|Average Coverage by Job'),
coverageCountLabel: s__('RepositoriesAnalytics|Jobs with Coverage'), coverageCountLabel: s__('RepositoriesAnalytics|Jobs with Coverage'),
...@@ -137,14 +147,14 @@ export default { ...@@ -137,14 +147,14 @@ export default {
<template> <template>
<div> <div>
<metric-card <metric-card
:title="$options.text.metrics.cardTitle" :title="$options.i18n.metrics.cardTitle"
:metrics="metrics" :metrics="metrics"
:is-loading="isLoading" :is-loading="isLoading"
/> />
<gl-card> <gl-card>
<template #header> <template #header>
<h5>{{ $options.text.graphCardHeader }}</h5> <h5>{{ $options.i18n.graphCardHeader }}</h5>
</template> </template>
<chart-skeleton-loader v-if="isLoading" data-testid="group-coverage-chart-loading" /> <chart-skeleton-loader v-if="isLoading" data-testid="group-coverage-chart-loading" />
...@@ -161,11 +171,10 @@ export default { ...@@ -161,11 +171,10 @@ export default {
{{ tooltipTitle }} {{ tooltipTitle }}
</template> </template>
<template #tooltip-content> <template #tooltip-content>
<gl-sprintf :message="$options.text.graphTooltipMessage"> <gl-sprintf :message="$options.i18n.graphTooltipMessage">
<template #coveragePercentage> <template #coveragePercentage>
{{ coveragePercentage }} {{ coveragePercentage }}
</template> </template>
<template #percentSymbol>%</template>
</gl-sprintf> </gl-sprintf>
</template> </template>
</gl-area-chart> </gl-area-chart>
......
// Jest Snapshot v1, https://goo.gl/fbAQLP
exports[`Test coverage table component when group code coverage is available renders area chart with correct data 1`] = `
Array [
Object {
"data": Array [
Array [
"2020-01-10",
77.9,
],
Array [
"2020-01-11",
78.7,
],
Array [
"2020-01-12",
79.6,
],
],
"name": "test",
},
]
`;
import { GlDeprecatedSkeletonLoading as GlSkeletonLoading } from '@gitlab/ui'; import { GlDeprecatedSkeletonLoading as GlSkeletonLoading } from '@gitlab/ui';
import { mount, createLocalVue } from '@vue/test-utils'; import { shallowMount, createLocalVue } from '@vue/test-utils';
import VueApollo from 'vue-apollo';
import TestCoverageSummary from 'ee/analytics/repository_analytics/components/test_coverage_summary.vue'; import TestCoverageSummary from 'ee/analytics/repository_analytics/components/test_coverage_summary.vue';
import getGroupTestCoverage from 'ee/analytics/repository_analytics/graphql/queries/get_group_test_coverage.query.graphql';
import { extendedWrapper } from 'helpers/vue_test_utils_helper'; import { extendedWrapper } from 'helpers/vue_test_utils_helper';
import createMockApollo from 'helpers/mock_apollo_helper'; import MetricCard from '~/analytics/shared/components/metric_card.vue';
import waitForPromises from 'helpers/wait_for_promises';
const localVue = createLocalVue(); const localVue = createLocalVue();
describe('Test coverage table component', () => { describe('Test coverage table component', () => {
let wrapper; let wrapper;
let fakeApollo;
const findProjectsWithTests = () => wrapper.find('.js-metric-card-item:nth-child(1) h3'); const findProjectsWithTests = () => wrapper.find('.js-metric-card-item:nth-child(1) h3');
const findAverageCoverage = () => wrapper.find('.js-metric-card-item:nth-child(2) h3'); const findAverageCoverage = () => wrapper.find('.js-metric-card-item:nth-child(2) h3');
...@@ -20,38 +16,34 @@ describe('Test coverage table component', () => { ...@@ -20,38 +16,34 @@ describe('Test coverage table component', () => {
const findChartLoadingState = () => wrapper.findByTestId('group-coverage-chart-loading'); const findChartLoadingState = () => wrapper.findByTestId('group-coverage-chart-loading');
const findLoadingState = () => wrapper.find(GlSkeletonLoading); const findLoadingState = () => wrapper.find(GlSkeletonLoading);
const createComponent = ({ data = {} } = {}, withApollo = false) => { const createComponent = ({ data = {} } = {}) => {
fakeApollo = createMockApollo([[getGroupTestCoverage, jest.fn().mockResolvedValue()]]); wrapper = extendedWrapper(
shallowMount(TestCoverageSummary, {
const props = { localVue,
localVue, data() {
data() { return {
return { projectCount: null,
projectCount: null, averageCoverage: null,
averageCoverage: null, coverageCount: null,
coverageCount: null, hasError: false,
hasError: false, isLoading: false,
isLoading: false, ...data,
...data, };
}; },
}, mocks: {
}; $apollo: {
if (withApollo) { queries: {
localVue.use(VueApollo); group: {
props.apolloProvider = fakeApollo; query: jest.fn().mockResolvedValue(),
} else { },
props.mocks = {
$apollo: {
queries: {
group: {
query: jest.fn().mockResolvedValue(),
}, },
}, },
}, },
}; stubs: {
} MetricCard,
},
wrapper = extendedWrapper(mount(TestCoverageSummary, props)); }),
);
}; };
afterEach(() => { afterEach(() => {
...@@ -97,16 +89,16 @@ describe('Test coverage table component', () => { ...@@ -97,16 +89,16 @@ describe('Test coverage table component', () => {
expect(findTotalCoverages().text()).toBe(coverageCount); expect(findTotalCoverages().text()).toBe(coverageCount);
}); });
it('renders area chart', () => { it('renders area chart with correct data', () => {
createComponent({ createComponent({
data: { data: {
groupCoverageChartData: [ groupCoverageChartData: [
{ {
name: 'test', name: 'test',
data: [ data: [
['Jan 10', 77.9], ['2020-01-10', 77.9],
['Jan 11', 78.7], ['2020-01-11', 78.7],
['Jan 12', 79.6], ['2020-01-12', 79.6],
], ],
}, },
], ],
...@@ -114,30 +106,29 @@ describe('Test coverage table component', () => { ...@@ -114,30 +106,29 @@ describe('Test coverage table component', () => {
}); });
expect(findGroupCoverageChart().exists()).toBe(true); expect(findGroupCoverageChart().exists()).toBe(true);
expect(findGroupCoverageChart().props('data')).toMatchSnapshot();
}); });
});
describe('when group has no coverage', () => { it('formats the area chart labels correctly', () => {
it('renders empty metrics', async () => {
createComponent({ createComponent({
withApollo: true, data: {
data: {}, groupCoverageChartData: [
queryData: { {
data: { name: 'test',
group: { data: [
codeCoverageActivities: { ['2020-01-10', 77.9],
nodes: [], ['2020-01-11', 78.7],
}, ['2020-01-12', 79.6],
],
}, },
}, ],
}, },
}); });
jest.runOnlyPendingTimers();
await waitForPromises();
expect(findProjectsWithTests().text()).toBe('-'); expect(findGroupCoverageChart().props('option').xAxis.axisLabel.formatter('2020-01-10')).toBe(
expect(findAverageCoverage().text()).toBe('-'); 'Jan 10',
expect(findTotalCoverages().text()).toBe('-'); );
expect(findGroupCoverageChart().props('option').yAxis.axisLabel.formatter(80)).toBe('80%');
}); });
}); });
}); });
...@@ -7196,6 +7196,9 @@ msgstr "" ...@@ -7196,6 +7196,9 @@ msgstr ""
msgid "Code" msgid "Code"
msgstr "" msgstr ""
msgid "Code Coverage: %{coveragePercentage}"
msgstr ""
msgid "Code Coverage: %{coveragePercentage}%{percentSymbol}" msgid "Code Coverage: %{coveragePercentage}%{percentSymbol}"
msgstr "" msgstr ""
...@@ -20916,7 +20919,7 @@ msgstr "" ...@@ -20916,7 +20919,7 @@ msgstr ""
msgid "Outdent" msgid "Outdent"
msgstr "" msgstr ""
msgid "Overall Activity" msgid "Overall activity"
msgstr "" msgstr ""
msgid "Overridden" msgid "Overridden"
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment