Commit 816c58c5 authored by Ezekiel Kigbo's avatar Ezekiel Kigbo Committed by Martin Wortschack

Minor review feedback

Update gitlab.pot file

Generate values in the time_summary fixture

Adds the prepare_cycle_time_data method
to generate cycle time values
parent 23fbced2
...@@ -18,6 +18,7 @@ import TimeMetricsCard from './time_metrics_card.vue'; ...@@ -18,6 +18,7 @@ import TimeMetricsCard from './time_metrics_card.vue';
import StageTableNav from './stage_table_nav.vue'; import StageTableNav from './stage_table_nav.vue';
import CustomStageForm from './custom_stage_form.vue'; import CustomStageForm from './custom_stage_form.vue';
import PathNavigation from './path_navigation.vue'; import PathNavigation from './path_navigation.vue';
import MetricCard from '../../shared/components/metric_card.vue';
export default { export default {
name: 'CycleAnalytics', name: 'CycleAnalytics',
...@@ -35,6 +36,7 @@ export default { ...@@ -35,6 +36,7 @@ export default {
CustomStageForm, CustomStageForm,
StageTableNav, StageTableNav,
PathNavigation, PathNavigation,
MetricCard,
}, },
mixins: [glFeatureFlagsMixin(), UrlSyncMixin], mixins: [glFeatureFlagsMixin(), UrlSyncMixin],
props: { props: {
...@@ -267,12 +269,15 @@ export default { ...@@ -267,12 +269,15 @@ export default {
" "
/> />
<div v-else-if="!errorCode"> <div v-else-if="!errorCode">
<div class="js-recent-activity mt-3 gl-display-flex"> <div class="js-recent-activity gl-mt-3 gl-display-flex">
<div class="gl-flex-fill-1 gl-pr-2"> <div class="gl-flex-fill-1 gl-pr-2">
<time-metrics-card <time-metrics-card
#default="{ metrics, loading }"
:group-path="currentGroupPath" :group-path="currentGroupPath"
:additional-params="cycleAnalyticsRequestParams" :additional-params="cycleAnalyticsRequestParams"
/> >
<metric-card :title="__('Time')" :metrics="metrics" :is-loading="loading" />
</time-metrics-card>
</div> </div>
<div class="gl-flex-fill-1 gl-pl-2"> <div class="gl-flex-fill-1 gl-pl-2">
<recent-activity-card <recent-activity-card
......
...@@ -19,7 +19,7 @@ export default { ...@@ -19,7 +19,7 @@ export default {
additionalParams: { additionalParams: {
type: Object, type: Object,
required: false, required: false,
default: null, default: () => ({}),
}, },
}, },
data() { data() {
...@@ -40,13 +40,10 @@ export default { ...@@ -40,13 +40,10 @@ export default {
fetchData() { fetchData() {
removeFlash(); removeFlash();
this.loading = true; this.loading = true;
return Api.cycleAnalyticsTimeSummaryData( return Api.cycleAnalyticsTimeSummaryData(this.groupPath, this.additionalParams)
this.groupPath,
this.additionalParams ? this.additionalParams : {},
)
.then(({ data }) => { .then(({ data }) => {
this.data = data.map(({ title: label, value }) => ({ this.data = data.map(({ title: label, ...rest }) => ({
value: value || '-', ...rest,
label, label,
key: slugify(label), key: slugify(label),
})); }));
......
...@@ -26,12 +26,11 @@ export default { ...@@ -26,12 +26,11 @@ export default {
valueText(metric) { valueText(metric) {
const { value = null, unit = null } = metric; const { value = null, unit = null } = metric;
if (!value) return '-'; if (!value) return '-';
return unit ? `${value} ${unit}` : value; return unit && value ? `${value} ${unit}` : value;
}, },
}, },
}; };
</script> </script>
<template> <template>
<gl-card> <gl-card>
<template #header> <template #header>
......
...@@ -289,14 +289,14 @@ describe 'Group Value Stream Analytics', :js do ...@@ -289,14 +289,14 @@ describe 'Group Value Stream Analytics', :js do
it 'displays the lead time' do it 'displays the lead time' do
lead_time = page.all(card_metric_selector).first lead_time = page.all(card_metric_selector).first
expect(lead_time).to have_content(n_('Lead time', 'Lead time', 10)) expect(lead_time).to have_content(_('Lead Time'))
expect(lead_time).to have_content('10') expect(lead_time).to have_content('-')
end end
it 'displays the cycle time' do it 'displays the cycle time' do
cycle_time = page.all(card_metric_selector)[1] cycle_time = page.all(card_metric_selector)[1]
expect(cycle_time).to have_content(n_('Cycle time', 'Cycle time', 0)) expect(cycle_time).to have_content(_('Cycle Time'))
expect(cycle_time).to have_content('-') expect(cycle_time).to have_content('-')
end end
end end
......
...@@ -27,7 +27,7 @@ exports[`RecentActivityCard matches the snapshot 1`] = ` ...@@ -27,7 +27,7 @@ exports[`RecentActivityCard matches the snapshot 1`] = `
<h3 <h3
class="my-2" class="my-2"
> >
3 4
</h3> </h3>
<p <p
......
// Jest Snapshot v1, https://goo.gl/fbAQLP
exports[`TimeMetricsCard matches the snapshot 1`] = `
<div
class="card"
>
<!---->
<div
class="card-header"
>
<strong>
Time
</strong>
</div>
<div
class="card-body"
>
<!---->
<!---->
<div
class="d-flex"
>
<div
class="js-metric-card-item flex-grow text-center"
>
<h3
class="my-2"
>
-
</h3>
<p
class="text-secondary gl-font-sm mb-2"
>
Lead Time
</p>
</div>
<div
class="js-metric-card-item flex-grow text-center"
>
<h3
class="my-2"
>
-
</h3>
<p
class="text-secondary gl-font-sm mb-2"
>
Cycle Time
</p>
</div>
</div>
</div>
<!---->
<!---->
</div>
`;
...@@ -8,6 +8,7 @@ import MockAdapter from 'axios-mock-adapter'; ...@@ -8,6 +8,7 @@ import MockAdapter from 'axios-mock-adapter';
import GroupsDropdownFilter from 'ee/analytics/shared/components/groups_dropdown_filter.vue'; import GroupsDropdownFilter from 'ee/analytics/shared/components/groups_dropdown_filter.vue';
import ProjectsDropdownFilter from 'ee/analytics/shared/components/projects_dropdown_filter.vue'; import ProjectsDropdownFilter from 'ee/analytics/shared/components/projects_dropdown_filter.vue';
import RecentActivityCard from 'ee/analytics/cycle_analytics/components/recent_activity_card.vue'; import RecentActivityCard from 'ee/analytics/cycle_analytics/components/recent_activity_card.vue';
import TimeMetricsCard from 'ee/analytics/cycle_analytics/components/time_metrics_card.vue';
import PathNavigation from 'ee/analytics/cycle_analytics/components/path_navigation.vue'; import PathNavigation from 'ee/analytics/cycle_analytics/components/path_navigation.vue';
import StageTable from 'ee/analytics/cycle_analytics/components/stage_table.vue'; import StageTable from 'ee/analytics/cycle_analytics/components/stage_table.vue';
import 'bootstrap'; import 'bootstrap';
...@@ -123,6 +124,10 @@ describe('Cycle Analytics component', () => { ...@@ -123,6 +124,10 @@ describe('Cycle Analytics component', () => {
expect(wrapper.find(RecentActivityCard).exists()).toBe(flag); expect(wrapper.find(RecentActivityCard).exists()).toBe(flag);
}; };
const displaysTimeMetricsCard = flag => {
expect(wrapper.find(TimeMetricsCard).exists()).toBe(flag);
};
const displaysStageTable = flag => { const displaysStageTable = flag => {
expect(wrapper.find(StageTable).exists()).toBe(flag); expect(wrapper.find(StageTable).exists()).toBe(flag);
}; };
...@@ -179,10 +184,14 @@ describe('Cycle Analytics component', () => { ...@@ -179,10 +184,14 @@ describe('Cycle Analytics component', () => {
displaysDateRangePicker(false); displaysDateRangePicker(false);
}); });
it('does not display the recent activity table', () => { it('does not display the recent activity card', () => {
displaysRecentActivityCard(false); displaysRecentActivityCard(false);
}); });
it('does not display the time metrics card', () => {
displaysTimeMetricsCard(false);
});
it('does not display the stage table', () => { it('does not display the stage table', () => {
displaysStageTable(false); displaysStageTable(false);
}); });
...@@ -243,10 +252,14 @@ describe('Cycle Analytics component', () => { ...@@ -243,10 +252,14 @@ describe('Cycle Analytics component', () => {
displaysDateRangePicker(true); displaysDateRangePicker(true);
}); });
it('displays the recent activity table', () => { it('displays the recent activity card', () => {
displaysRecentActivityCard(true); displaysRecentActivityCard(true);
}); });
it('displays the time metrics card', () => {
displaysTimeMetricsCard(true);
});
it('displays the stage table', () => { it('displays the stage table', () => {
displaysStageTable(true); displaysStageTable(true);
}); });
...@@ -352,10 +365,14 @@ describe('Cycle Analytics component', () => { ...@@ -352,10 +365,14 @@ describe('Cycle Analytics component', () => {
displaysDateRangePicker(false); displaysDateRangePicker(false);
}); });
it('does not display the recent activity table', () => { it('does not display the recent activity card', () => {
displaysRecentActivityCard(false); displaysRecentActivityCard(false);
}); });
it('does not display the time metrics card', () => {
displaysTimeMetricsCard(false);
});
it('does not display the stage table', () => { it('does not display the stage table', () => {
displaysStageTable(false); displaysStageTable(false);
}); });
......
import { mount } from '@vue/test-utils'; import { shallowMount } from '@vue/test-utils';
import createFlash from '~/flash'; import createFlash from '~/flash';
import TimeMetricsCard from 'ee/analytics/cycle_analytics/components/time_metrics_card.vue'; import TimeMetricsCard from 'ee/analytics/cycle_analytics/components/time_metrics_card.vue';
import { group, timeMetricsData } from '../mock_data'; import { group, timeMetricsData } from '../mock_data';
...@@ -10,17 +10,18 @@ describe('TimeMetricsCard', () => { ...@@ -10,17 +10,18 @@ describe('TimeMetricsCard', () => {
const { full_path: groupPath } = group; const { full_path: groupPath } = group;
let wrapper; let wrapper;
const createComponent = (additionalParams = {}) => { const createComponent = ({ additionalParams = {} } = {}) => {
return mount(TimeMetricsCard, { return shallowMount(TimeMetricsCard, {
propsData: { propsData: {
groupPath, groupPath,
additionalParams, additionalParams,
}, },
slots: {
default: 'mockMetricCard',
},
}); });
}; };
const findMetricCards = () => wrapper.findAll('.js-metric-card-item');
beforeEach(() => { beforeEach(() => {
jest.spyOn(Api, 'cycleAnalyticsTimeSummaryData').mockResolvedValue({ data: timeMetricsData }); jest.spyOn(Api, 'cycleAnalyticsTimeSummaryData').mockResolvedValue({ data: timeMetricsData });
...@@ -31,27 +32,10 @@ describe('TimeMetricsCard', () => { ...@@ -31,27 +32,10 @@ describe('TimeMetricsCard', () => {
wrapper.destroy(); wrapper.destroy();
wrapper = null; wrapper = null;
}); });
it('matches the snapshot', () => {
expect(wrapper.element).toMatchSnapshot();
});
it('fetches the time metrics data', () => { it('fetches the time metrics data', () => {
expect(Api.cycleAnalyticsTimeSummaryData).toHaveBeenCalledWith(groupPath, {}); expect(Api.cycleAnalyticsTimeSummaryData).toHaveBeenCalledWith(groupPath, {});
}); });
describe('with data', () => {
it.each`
metric | value | index
${'Lead Time'} | ${'-'} | ${0}
${'Cycle Time'} | ${'-'} | ${1}
`('Renders the $metric', ({ metric, value, index }) => {
const card = findMetricCards().at(index);
expect(card.html()).toContain(metric);
expect(card.html()).toContain(value);
});
});
describe('with a failing request', () => { describe('with a failing request', () => {
beforeEach(() => { beforeEach(() => {
jest.spyOn(Api, 'cycleAnalyticsTimeSummaryData').mockRejectedValue(); jest.spyOn(Api, 'cycleAnalyticsTimeSummaryData').mockRejectedValue();
...@@ -69,9 +53,11 @@ describe('TimeMetricsCard', () => { ...@@ -69,9 +53,11 @@ describe('TimeMetricsCard', () => {
describe('with additional params', () => { describe('with additional params', () => {
beforeEach(() => { beforeEach(() => {
wrapper = createComponent({ wrapper = createComponent({
additionalParams: {
'project_ids[]': [1], 'project_ids[]': [1],
created_after: '2020-01-01', created_after: '2020-01-01',
created_before: '2020-02-01', created_before: '2020-02-01',
},
}); });
}); });
......
...@@ -7,9 +7,7 @@ describe 'Analytics (JavaScript fixtures)', :sidekiq_inline do ...@@ -7,9 +7,7 @@ describe 'Analytics (JavaScript fixtures)', :sidekiq_inline do
let(:group) { create(:group)} let(:group) { create(:group)}
let(:project) { create(:project, :repository, namespace: group) } let(:project) { create(:project, :repository, namespace: group) }
let(:user) { create(:user, :admin) } let(:user) { create(:user, :admin) }
# let(:issue) { create(:issue, project: project, created_at: 4.days.ago) }
let(:milestone) { create(:milestone, project: project) } let(:milestone) { create(:milestone, project: project) }
# let(:mr) { create_merge_request_closing_issue(user, project, issue, commit_message: "References #{issue.to_reference}") }
let(:issue) { create(:issue, project: project, created_at: 4.days.ago) } let(:issue) { create(:issue, project: project, created_at: 4.days.ago) }
let(:issue_1) { create(:issue, project: project, created_at: 5.days.ago) } let(:issue_1) { create(:issue, project: project, created_at: 5.days.ago) }
...@@ -189,11 +187,22 @@ describe 'Analytics (JavaScript fixtures)', :sidekiq_inline do ...@@ -189,11 +187,22 @@ describe 'Analytics (JavaScript fixtures)', :sidekiq_inline do
let(:params) { { created_after: 3.months.ago, created_before: Time.now, group_id: group.full_path } } let(:params) { { created_after: 3.months.ago, created_before: Time.now, group_id: group.full_path } }
def prepare_cycle_time_data
issue.update!(created_at: 5.days.ago)
issue.metrics.update!(first_mentioned_in_commit_at: 4.days.ago)
issue.update!(closed_at: 3.days.ago)
issue_1.update!(created_at: 8.days.ago)
issue_1.metrics.update!(first_mentioned_in_commit_at: 6.days.ago)
issue_1.update!(closed_at: 1.day.ago)
end
before do before do
stub_feature_flags(Gitlab::Analytics::CYCLE_ANALYTICS_FEATURE_FLAG => true) stub_feature_flags(Gitlab::Analytics::CYCLE_ANALYTICS_FEATURE_FLAG => true)
stub_licensed_features(cycle_analytics_for_groups: true) stub_licensed_features(cycle_analytics_for_groups: true)
prepare_cycle_analytics_data prepare_cycle_analytics_data
prepare_cycle_time_data
sign_in(user) sign_in(user)
end end
...@@ -203,24 +212,9 @@ describe 'Analytics (JavaScript fixtures)', :sidekiq_inline do ...@@ -203,24 +212,9 @@ describe 'Analytics (JavaScript fixtures)', :sidekiq_inline do
expect(response).to be_successful expect(response).to be_successful
end end
end
describe Analytics::CycleAnalytics::TimeSummaryController, type: :controller do
render_views
let(:params) { { created_after: 3.months.ago, created_before: Time.now, group_id: group.full_path } }
before do
stub_feature_flags(Gitlab::Analytics::CYCLE_ANALYTICS_FEATURE_FLAG => true)
stub_licensed_features(cycle_analytics_for_groups: true)
prepare_cycle_analytics_data
sign_in(user)
end
it 'analytics/value_stream_analytics/time_summary.json' do it 'analytics/value_stream_analytics/time_summary.json' do
get(:show, params: params, format: :json) get(:time_summary, params: params, format: :json)
expect(response).to be_successful expect(response).to be_successful
end end
......
...@@ -21680,6 +21680,9 @@ msgstr "" ...@@ -21680,6 +21680,9 @@ msgstr ""
msgid "There was an error while fetching value stream analytics recent activity data." msgid "There was an error while fetching value stream analytics recent activity data."
msgstr "" msgstr ""
msgid "There was an error while fetching value stream analytics time summary data."
msgstr ""
msgid "There was an error with the reCAPTCHA. Please solve the reCAPTCHA again." msgid "There was an error with the reCAPTCHA. Please solve the reCAPTCHA again."
msgstr "" msgstr ""
......
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