Commit 5f0f1aa6 authored by Scott Hampton's avatar Scott Hampton

Ensure we fetch each project only once

Because we were incrementally adding
projects to our query, each query was
different and thus we couldn't take
advantage of caching. By keeping track
of all projects we receive, we can ensure
we only fetch the projects we haven't
already fetched.

Also refactored:
- Loading state
- Spec finders
- Spec mount params
- Spec tests to use variables
parent 42c895d9
<script> <script>
import Vue from 'vue';
import { GlCard, GlEmptyState, GlSkeletonLoader, GlTable } from '@gitlab/ui'; import { GlCard, GlEmptyState, GlSkeletonLoader, GlTable } from '@gitlab/ui';
import { __, s__ } from '~/locale'; import { __, s__ } from '~/locale';
import TimeAgoTooltip from '~/vue_shared/components/time_ago_tooltip.vue'; import TimeAgoTooltip from '~/vue_shared/components/time_ago_tooltip.vue';
...@@ -15,23 +16,8 @@ export default { ...@@ -15,23 +16,8 @@ export default {
SelectProjectsDropdown, SelectProjectsDropdown,
TimeAgoTooltip, TimeAgoTooltip,
}, },
data() { apollo: {
return { coverageData: {
coverageData: [],
hasError: false,
allProjectsSelected: false,
selectedProjectIds: [],
isLoading: false,
};
},
computed: {
hasCoverageData() {
return this.coverageData.length;
},
},
methods: {
getCoverageData() {
this.$apollo.addSmartQuery('coverageData', {
query: getProjectsTestCoverage, query: getProjectsTestCoverage,
debounce: 500, debounce: 500,
variables() { variables() {
...@@ -39,8 +25,10 @@ export default { ...@@ -39,8 +25,10 @@ export default {
projectIds: this.selectedProjectIds, projectIds: this.selectedProjectIds,
}; };
}, },
update(data) { result({ data }) {
return data.projects.nodes; // Keep data from all queries so that we don't
// fetch the same data more than once
this.allCoverageData = [...this.allCoverageData, ...data.projects.nodes];
}, },
error() { error() {
this.handleError(); this.handleError();
...@@ -48,31 +36,61 @@ export default { ...@@ -48,31 +36,61 @@ export default {
watchLoading(isLoading) { watchLoading(isLoading) {
this.isLoading = isLoading; this.isLoading = isLoading;
}, },
}); skip() {
return this.skipQuery;
},
},
}, },
data() {
return {
allProjectsSelected: false,
allCoverageData: [],
hasError: false,
isLoading: false,
projectIds: {},
};
},
computed: {
hasCoverageData() {
return Boolean(this.selectedCoverageData.length);
},
skipQuery() {
// Skip if we haven't selected any projects yet
return !this.selectedProjectIds.length;
},
selectedProjectIds() {
// Get the IDs of the projects that we haven't requested yet
return Object.keys(this.projectIds).filter(
id => !this.allCoverageData.some(project => project.id === id),
);
},
selectedCoverageData() {
return this.allCoverageData.filter(({ id }) => this.projectIds[id]);
},
},
methods: {
handleError() { handleError() {
this.hasError = true; this.hasError = true;
}, },
selectAllProjects(allProjects) { selectAllProjects(allProjects) {
this.selectedProjectIds = allProjects.map(project => project.id); this.projectIds = Object.fromEntries(allProjects.map(({ id }) => [id, true]));
this.allProjectsSelected = true; this.allProjectsSelected = true;
this.getCoverageData();
}, },
selectProject({ id }) { toggleProject({ id }) {
if (this.allProjectsSelected) { if (this.allProjectsSelected) {
// Clear out all the selected projects // Reset all project selections to false
this.allProjectsSelected = false; this.allProjectsSelected = false;
this.selectedProjectIds = []; this.projectIds = Object.fromEntries(
Object.entries(this.projectIds).map(([key]) => [key, false]),
);
} }
const index = this.selectedProjectIds.indexOf(id); if (Object.prototype.hasOwnProperty.call(this.projectIds, id)) {
if (index < 0) { Vue.set(this.projectIds, id, !this.projectIds[id]);
this.selectedProjectIds.push(id); return;
} else {
this.selectedProjectIds.splice(index, 1);
} }
this.getCoverageData(); Vue.set(this.projectIds, id, true);
}, },
}, },
tableFields: [ tableFields: [
...@@ -99,6 +117,15 @@ export default { ...@@ -99,6 +117,15 @@ export default {
'RepositoriesAnalytics|Please select a project or multiple projects to display their most recent test coverage data.', 'RepositoriesAnalytics|Please select a project or multiple projects to display their most recent test coverage data.',
), ),
}, },
LOADING_STATE: {
rows: 4,
height: 10,
rx: 4,
groupXs: [0, 95, 180, 330],
widths: [90, 80, 145, 100],
totalWidth: 430,
totalHeight: 15,
},
}; };
</script> </script>
<template> <template>
...@@ -108,43 +135,36 @@ export default { ...@@ -108,43 +135,36 @@ export default {
class="gl-w-quarter" class="gl-w-quarter"
@projects-query-error="handleError" @projects-query-error="handleError"
@select-all-projects="selectAllProjects" @select-all-projects="selectAllProjects"
@select-project="selectProject" @select-project="toggleProject"
/> />
</template> </template>
<template v-if="isLoading">
<gl-skeleton-loader <gl-skeleton-loader
v-if="isLoading" v-for="index in $options.LOADING_STATE.rows"
:width="430" :key="index"
:height="55" :width="$options.LOADING_STATE.totalWidth"
:height="$options.LOADING_STATE.totalHeight"
data-testid="test-coverage-loading-state" data-testid="test-coverage-loading-state"
> >
<rect width="90" height="10" x="0" y="0" rx="4" /> <rect
<rect width="80" height="10" x="95" y="0" rx="4" /> v-for="(x, xIndex) in $options.LOADING_STATE.groupXs"
<rect width="145" height="10" x="180" y="0" rx="4" /> :key="`x-skeleton-${x}`"
<rect width="100" height="10" x="330" y="0" rx="4" /> :width="$options.LOADING_STATE.widths[xIndex]"
:height="$options.LOADING_STATE.height"
<rect width="90" height="10" x="0" y="15" rx="4" /> :x="x"
<rect width="80" height="10" x="95" y="15" rx="4" /> :y="0"
<rect width="145" height="10" x="180" y="15" rx="4" /> :rx="$options.LOADING_STATE.rx"
<rect width="100" height="10" x="330" y="15" rx="4" /> />
<rect width="90" height="10" x="0" y="30" rx="4" />
<rect width="80" height="10" x="95" y="30" rx="4" />
<rect width="145" height="10" x="180" y="30" rx="4" />
<rect width="100" height="10" x="330" y="30" rx="4" />
<rect width="90" height="10" x="0" y="45" rx="4" />
<rect width="80" height="10" x="95" y="45" rx="4" />
<rect width="145" height="10" x="180" y="45" rx="4" />
<rect width="100" height="10" x="330" y="45" rx="4" />
</gl-skeleton-loader> </gl-skeleton-loader>
</template>
<gl-table <gl-table
v-else-if="hasCoverageData" v-else-if="hasCoverageData"
data-testid="test-coverage-data-table" data-testid="test-coverage-data-table"
thead-class="thead-white" thead-class="thead-white"
:fields="$options.tableFields" :fields="$options.tableFields"
:items="coverageData" :items="selectedCoverageData"
> >
<template #head(project)="data"> <template #head(project)="data">
<div>{{ data.label }}</div> <div>{{ data.label }}</div>
......
...@@ -59,8 +59,10 @@ describe('Select projects dropdown component', () => { ...@@ -59,8 +59,10 @@ describe('Select projects dropdown component', () => {
}); });
describe('when selecting all project', () => { describe('when selecting all project', () => {
const initialData = { groupProjects: [{ id: 1, name: '1', isSelected: true }] };
beforeEach(() => { beforeEach(() => {
createComponent({ data: { groupProjects: [{ id: 1, name: '1', isSelected: true }] } }); createComponent({ data: initialData });
}); });
it('should reset all selected projects', () => { it('should reset all selected projects', () => {
...@@ -68,7 +70,7 @@ describe('Select projects dropdown component', () => { ...@@ -68,7 +70,7 @@ describe('Select projects dropdown component', () => {
return wrapper.vm.$nextTick().then(() => { return wrapper.vm.$nextTick().then(() => {
expect( expect(
findProjectById(1) findProjectById(initialData.groupProjects[0].id)
.find(GlIcon) .find(GlIcon)
.classes(), .classes(),
).toContain('gl-visibility-hidden'); ).toContain('gl-visibility-hidden');
...@@ -80,24 +82,30 @@ describe('Select projects dropdown component', () => { ...@@ -80,24 +82,30 @@ describe('Select projects dropdown component', () => {
selectAllProjects(); selectAllProjects();
expect(wrapper.vm.$emit).toHaveBeenCalledWith('select-all-projects', [ expect(wrapper.vm.$emit).toHaveBeenCalledWith('select-all-projects', [
{ id: 1, name: '1', isSelected: false }, { ...initialData.groupProjects[0], isSelected: false },
]); ]);
}); });
}); });
describe('when selecting a project', () => { describe('when selecting a project', () => {
const initialData = {
groupProjects: [{ id: 1, name: '1', isSelected: false }],
selectAllProjects: true,
};
beforeEach(() => { beforeEach(() => {
createComponent({ createComponent({
data: { groupProjects: [{ id: 1, name: '1', isSelected: false }], selectAllProjects: true }, data: initialData,
}); });
}); });
it('should check selected project', () => { it('should check selected project', () => {
selectProjectById(1); const project = initialData.groupProjects[0];
selectProjectById(project.id);
return wrapper.vm.$nextTick().then(() => { return wrapper.vm.$nextTick().then(() => {
expect( expect(
findProjectById(1) findProjectById(project.id)
.find(GlIcon) .find(GlIcon)
.classes(), .classes(),
).not.toContain('gl-visibility-hidden'); ).not.toContain('gl-visibility-hidden');
...@@ -105,7 +113,7 @@ describe('Select projects dropdown component', () => { ...@@ -105,7 +113,7 @@ describe('Select projects dropdown component', () => {
}); });
it('should uncheck select all projects', () => { it('should uncheck select all projects', () => {
selectProjectById(1); selectProjectById(initialData.groupProjects[0].id);
return wrapper.vm.$nextTick().then(() => { return wrapper.vm.$nextTick().then(() => {
expect( expect(
...@@ -117,12 +125,12 @@ describe('Select projects dropdown component', () => { ...@@ -117,12 +125,12 @@ describe('Select projects dropdown component', () => {
}); });
it('should emit select-project event', () => { it('should emit select-project event', () => {
const project = initialData.groupProjects[0];
jest.spyOn(wrapper.vm, '$emit'); jest.spyOn(wrapper.vm, '$emit');
selectProjectById(1); selectProjectById(project.id);
expect(wrapper.vm.$emit).toHaveBeenCalledWith('select-project', { expect(wrapper.vm.$emit).toHaveBeenCalledWith('select-project', {
id: 1, ...project,
name: '1',
isSelected: true, isSelected: true,
}); });
}); });
......
...@@ -8,16 +8,24 @@ describe('Test coverage table component', () => { ...@@ -8,16 +8,24 @@ describe('Test coverage table component', () => {
useFakeDate(); useFakeDate();
let wrapper; let wrapper;
const createComponent = (mountFn = shallowMount, data = {}) => { const findEmptyState = () => wrapper.find('[data-testid="test-coverage-table-empty-state"]');
const findLoadingState = () => wrapper.find('[data-testid="test-coverage-loading-state"');
const findTable = () => wrapper.find('[data-testid="test-coverage-data-table"');
const findProjectNameById = id => wrapper.find(`[data-testid="${id}-name"`);
const findProjectAverageById = id => wrapper.find(`[data-testid="${id}-average"`);
const findProjectCountById = id => wrapper.find(`[data-testid="${id}-count"`);
const findProjectDateById = id => wrapper.find(`[data-testid="${id}-date"`);
const createComponent = (data = {}, mountFn = shallowMount) => {
wrapper = mountFn(TestCoverageTable, { wrapper = mountFn(TestCoverageTable, {
localVue, localVue,
data() { data() {
return { return {
coverageData: [], allCoverageData: [],
hasError: false,
allProjectsSelected: false, allProjectsSelected: false,
selectedProjectIds: [], hasError: false,
isLoading: false, isLoading: false,
projectIds: {},
...data, ...data,
}; };
}, },
...@@ -41,18 +49,15 @@ describe('Test coverage table component', () => { ...@@ -41,18 +49,15 @@ describe('Test coverage table component', () => {
describe('when code coverage is empty', () => { describe('when code coverage is empty', () => {
it('renders empty state', () => { it('renders empty state', () => {
createComponent(); createComponent();
const emptyState = wrapper.find('[data-testid="test-coverage-table-empty-state"]'); expect(findEmptyState().exists()).toBe(true);
expect(emptyState.exists()).toBe(true);
}); });
}); });
describe('when query is loading', () => { describe('when query is loading', () => {
it('renders loading state', () => { it('renders loading state', () => {
createComponent(shallowMount, { isLoading: true }); createComponent({ isLoading: true });
const loadingState = wrapper.find('[data-testid="test-coverage-loading-state"');
expect(loadingState.exists()).toBe(true); expect(findLoadingState().exists()).toBe(true);
}); });
}); });
...@@ -65,8 +70,9 @@ describe('Test coverage table component', () => { ...@@ -65,8 +70,9 @@ describe('Test coverage table component', () => {
const yesterday = new Date(); const yesterday = new Date();
yesterday.setDate(yesterday.getDate() - 1); yesterday.setDate(yesterday.getDate() - 1);
createComponent(mount, { createComponent(
coverageData: [ {
allCoverageData: [
{ {
id, id,
name, name,
...@@ -77,18 +83,18 @@ describe('Test coverage table component', () => { ...@@ -77,18 +83,18 @@ describe('Test coverage table component', () => {
}, },
}, },
], ],
}); projectIds: {
const coverageTable = wrapper.find('[data-testid="test-coverage-data-table"'); [id]: true,
const expectedName = wrapper.find(`[data-testid="${id}-name"`); },
const expectedAverage = wrapper.find(`[data-testid="${id}-average"`); },
const expectedCount = wrapper.find(`[data-testid="${id}-count"`); mount,
const expectedDate = wrapper.find(`[data-testid="${id}-date"`); );
expect(coverageTable.exists()).toBe(true); expect(findTable().exists()).toBe(true);
expect(expectedName.text()).toBe(name); expect(findProjectNameById(id).text()).toBe(name);
expect(expectedAverage.text()).toBe(`${average}%`); expect(findProjectAverageById(id).text()).toBe(`${average}%`);
expect(expectedCount.text()).toBe(count); expect(findProjectCountById(id).text()).toBe(count);
expect(expectedDate.text()).toBe('1 day ago'); expect(findProjectDateById(id).text()).toBe('1 day ago');
}); });
}); });
}); });
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