Commit 52e79b25 authored by pburdette's avatar pburdette

Move api calls to tokens

Move api calls out of filtered
search and into tokens. This
is needed to display url params
values when the component is mounted.
parent 8570e64d
...@@ -3,9 +3,6 @@ import { GlFilteredSearch } from '@gitlab/ui'; ...@@ -3,9 +3,6 @@ import { GlFilteredSearch } from '@gitlab/ui';
import { __, s__ } from '~/locale'; import { __, s__ } from '~/locale';
import PipelineTriggerAuthorToken from './tokens/pipeline_trigger_author_token.vue'; import PipelineTriggerAuthorToken from './tokens/pipeline_trigger_author_token.vue';
import PipelineBranchNameToken from './tokens/pipeline_branch_name_token.vue'; import PipelineBranchNameToken from './tokens/pipeline_branch_name_token.vue';
import Api from '~/api';
import createFlash from '~/flash';
import { FETCH_AUTHOR_ERROR_MESSAGE, FETCH_BRANCH_ERROR_MESSAGE } from '../constants';
export default { export default {
components: { components: {
...@@ -25,12 +22,6 @@ export default { ...@@ -25,12 +22,6 @@ export default {
required: true, required: true,
}, },
}, },
data() {
return {
projectUsers: null,
projectBranches: null,
};
},
computed: { computed: {
tokens() { tokens() {
return [ return [
...@@ -41,7 +32,6 @@ export default { ...@@ -41,7 +32,6 @@ export default {
unique: true, unique: true,
token: PipelineTriggerAuthorToken, token: PipelineTriggerAuthorToken,
operators: [{ value: '=', description: __('is'), default: 'true' }], operators: [{ value: '=', description: __('is'), default: 'true' }],
triggerAuthors: this.projectUsers,
projectId: this.projectId, projectId: this.projectId,
}, },
{ {
...@@ -51,7 +41,6 @@ export default { ...@@ -51,7 +41,6 @@ export default {
unique: true, unique: true,
token: PipelineBranchNameToken, token: PipelineBranchNameToken,
operators: [{ value: '=', description: __('is'), default: 'true' }], operators: [{ value: '=', description: __('is'), default: 'true' }],
branches: this.projectBranches,
projectId: this.projectId, projectId: this.projectId,
}, },
]; ];
...@@ -83,25 +72,6 @@ export default { ...@@ -83,25 +72,6 @@ export default {
return valueArray; return valueArray;
}, },
}, },
created() {
Api.projectUsers(this.projectId)
.then(users => {
this.projectUsers = users;
})
.catch(err => {
createFlash(FETCH_AUTHOR_ERROR_MESSAGE);
throw err;
});
Api.branches(this.projectId)
.then(({ data }) => {
this.projectBranches = data.map(branch => branch.name);
})
.catch(err => {
createFlash(FETCH_BRANCH_ERROR_MESSAGE);
throw err;
});
},
methods: { methods: {
onSubmit(filters) { onSubmit(filters) {
this.$emit('filterPipelines', filters); this.$emit('filterPipelines', filters);
......
...@@ -23,15 +23,18 @@ export default { ...@@ -23,15 +23,18 @@ export default {
}, },
data() { data() {
return { return {
branches: this.config.branches, branches: null,
loading: true, loading: true,
}; };
}, },
created() {
this.fetchBranches();
},
methods: { methods: {
fetchBranchBySearchTerm(searchTerm) { fetchBranches(searchterm) {
Api.branches(this.config.projectId, searchTerm) Api.branches(this.config.projectId, searchterm)
.then(res => { .then(({ data }) => {
this.branches = res.data.map(branch => branch.name); this.branches = data.map(branch => branch.name);
this.loading = false; this.loading = false;
}) })
.catch(err => { .catch(err => {
...@@ -41,7 +44,7 @@ export default { ...@@ -41,7 +44,7 @@ export default {
}); });
}, },
searchBranches: debounce(function debounceSearch({ data }) { searchBranches: debounce(function debounceSearch({ data }) {
this.fetchBranchBySearchTerm(data); this.fetchBranches(data);
}, FILTER_PIPELINES_SEARCH_DELAY), }, FILTER_PIPELINES_SEARCH_DELAY),
}, },
}; };
...@@ -55,7 +58,7 @@ export default { ...@@ -55,7 +58,7 @@ export default {
@input="searchBranches" @input="searchBranches"
> >
<template #suggestions> <template #suggestions>
<gl-loading-icon v-if="loading && !value" /> <gl-loading-icon v-if="loading" />
<template v-else> <template v-else>
<gl-filtered-search-suggestion <gl-filtered-search-suggestion
v-for="(branch, index) in branches" v-for="(branch, index) in branches"
......
...@@ -36,7 +36,7 @@ export default { ...@@ -36,7 +36,7 @@ export default {
}, },
data() { data() {
return { return {
users: this.config.triggerAuthors, users: [],
loading: true, loading: true,
}; };
}, },
...@@ -45,16 +45,19 @@ export default { ...@@ -45,16 +45,19 @@ export default {
return this.value.data.toLowerCase(); return this.value.data.toLowerCase();
}, },
activeUser() { activeUser() {
return this.config.triggerAuthors.find(user => { return this.users.find(user => {
return user.username.toLowerCase() === this.currentValue; return user.username.toLowerCase() === this.currentValue;
}); });
}, },
}, },
created() {
this.fetchProjectUsers();
},
methods: { methods: {
fetchAuthorBySearchTerm(searchTerm) { fetchProjectUsers(searchTerm) {
Api.projectUsers(this.config.projectId, searchTerm) Api.projectUsers(this.config.projectId, searchTerm)
.then(res => { .then(users => {
this.users = res; this.users = users;
this.loading = false; this.loading = false;
}) })
.catch(err => { .catch(err => {
...@@ -64,7 +67,7 @@ export default { ...@@ -64,7 +67,7 @@ export default {
}); });
}, },
searchAuthors: debounce(function debounceSearch({ data }) { searchAuthors: debounce(function debounceSearch({ data }) {
this.fetchAuthorBySearchTerm(data); this.fetchProjectUsers(data);
}, FILTER_PIPELINES_SEARCH_DELAY), }, FILTER_PIPELINES_SEARCH_DELAY),
}, },
}; };
...@@ -72,7 +75,6 @@ export default { ...@@ -72,7 +75,6 @@ export default {
<template> <template>
<gl-filtered-search-token <gl-filtered-search-token
v-if="config.triggerAuthors"
:config="config" :config="config"
v-bind="{ ...$props, ...$attrs }" v-bind="{ ...$props, ...$attrs }"
v-on="$listeners" v-on="$listeners"
...@@ -94,7 +96,7 @@ export default { ...@@ -94,7 +96,7 @@ export default {
}}</gl-filtered-search-suggestion> }}</gl-filtered-search-suggestion>
<gl-dropdown-divider /> <gl-dropdown-divider />
<gl-loading-icon v-if="loading && !value" /> <gl-loading-icon v-if="loading" />
<template v-else> <template v-else>
<gl-filtered-search-suggestion <gl-filtered-search-suggestion
v-for="user in users" v-for="user in users"
......
...@@ -3,13 +3,7 @@ import { mount } from '@vue/test-utils'; ...@@ -3,13 +3,7 @@ import { mount } from '@vue/test-utils';
import MockAdapter from 'axios-mock-adapter'; import MockAdapter from 'axios-mock-adapter';
import axios from '~/lib/utils/axios_utils'; import axios from '~/lib/utils/axios_utils';
import PipelinesFilteredSearch from '~/pipelines/components/pipelines_filtered_search.vue'; import PipelinesFilteredSearch from '~/pipelines/components/pipelines_filtered_search.vue';
import { import { users, mockSearch, pipelineWithStages, branches } from '../mock_data';
users,
mockSearch,
pipelineWithStages,
branches,
mockBranchesAfterMap,
} from '../mock_data';
import { GlFilteredSearch } from '@gitlab/ui'; import { GlFilteredSearch } from '@gitlab/ui';
describe('Pipelines filtered search', () => { describe('Pipelines filtered search', () => {
...@@ -22,11 +16,12 @@ describe('Pipelines filtered search', () => { ...@@ -22,11 +16,12 @@ describe('Pipelines filtered search', () => {
.props('availableTokens') .props('availableTokens')
.find(token => token.type === type); .find(token => token.type === type);
const createComponent = () => { const createComponent = (params = {}) => {
wrapper = mount(PipelinesFilteredSearch, { wrapper = mount(PipelinesFilteredSearch, {
propsData: { propsData: {
pipelines: [pipelineWithStages], pipelines: [pipelineWithStages],
projectId: '21', projectId: '21',
params,
}, },
attachToDocument: true, attachToDocument: true,
}); });
...@@ -60,7 +55,6 @@ describe('Pipelines filtered search', () => { ...@@ -60,7 +55,6 @@ describe('Pipelines filtered search', () => {
icon: 'user', icon: 'user',
title: 'Trigger author', title: 'Trigger author',
unique: true, unique: true,
triggerAuthors: users,
projectId: '21', projectId: '21',
operators: [expect.objectContaining({ value: '=' })], operators: [expect.objectContaining({ value: '=' })],
}); });
...@@ -70,28 +64,51 @@ describe('Pipelines filtered search', () => { ...@@ -70,28 +64,51 @@ describe('Pipelines filtered search', () => {
icon: 'branch', icon: 'branch',
title: 'Branch name', title: 'Branch name',
unique: true, unique: true,
branches: mockBranchesAfterMap,
projectId: '21', projectId: '21',
operators: [expect.objectContaining({ value: '=' })], operators: [expect.objectContaining({ value: '=' })],
}); });
}); });
it('fetches and sets project users', () => { it('emits filterPipelines on submit with correct filter', () => {
expect(Api.projectUsers).toHaveBeenCalled(); findFilteredSearch().vm.$emit('submit', mockSearch);
expect(wrapper.vm.projectUsers).toEqual(users); expect(wrapper.emitted('filterPipelines')).toBeTruthy();
expect(wrapper.emitted('filterPipelines')[0]).toEqual([mockSearch]);
}); });
it('fetches and sets branches', () => { describe('Url query params', () => {
expect(Api.branches).toHaveBeenCalled(); const params = {
page: '1',
ref: 'master',
scope: 'all',
username: 'deja.green',
};
expect(wrapper.vm.projectBranches).toEqual(mockBranchesAfterMap); beforeEach(() => {
createComponent(params);
}); });
it('emits filterPipelines on submit with correct filter', () => { it('sets default value if url query params', () => {
findFilteredSearch().vm.$emit('submit', mockSearch); const expectedValueProp = [
{
type: 'username',
value: {
data: params.username,
operator: '=',
},
},
{
type: 'ref',
value: {
data: params.ref,
operator: '=',
},
},
{ type: 'filtered-search-term', value: { data: '' } },
];
expect(wrapper.emitted('filterPipelines')).toBeTruthy(); expect(findFilteredSearch().props('value')).toEqual(expectedValueProp);
expect(wrapper.emitted('filterPipelines')[0]).toEqual([mockSearch]); expect(findFilteredSearch().props('value')).toHaveLength(expectedValueProp.length);
});
}); });
}); });
...@@ -56,6 +56,7 @@ describe('Pipelines', () => { ...@@ -56,6 +56,7 @@ describe('Pipelines', () => {
propsData: { propsData: {
store: new Store(), store: new Store(),
projectId: '21', projectId: '21',
params: {},
...props, ...props,
}, },
methods: { methods: {
......
import Api from '~/api';
import MockAdapter from 'axios-mock-adapter';
import axios from '~/lib/utils/axios_utils';
import { GlFilteredSearchToken, GlFilteredSearchSuggestion, GlLoadingIcon } from '@gitlab/ui'; import { GlFilteredSearchToken, GlFilteredSearchSuggestion, GlLoadingIcon } from '@gitlab/ui';
import { shallowMount } from '@vue/test-utils'; import { shallowMount } from '@vue/test-utils';
import PipelineBranchNameToken from '~/pipelines/components/tokens/pipeline_branch_name_token.vue'; import PipelineBranchNameToken from '~/pipelines/components/tokens/pipeline_branch_name_token.vue';
import { branches } from '../mock_data'; import { branches, mockBranchesAfterMap } from '../mock_data';
describe('Pipeline Branch Name Token', () => { describe('Pipeline Branch Name Token', () => {
let wrapper; let wrapper;
let mock;
const findFilteredSearchToken = () => wrapper.find(GlFilteredSearchToken); const findFilteredSearchToken = () => wrapper.find(GlFilteredSearchToken);
const findAllFilteredSearchSuggestions = () => wrapper.findAll(GlFilteredSearchSuggestion); const findAllFilteredSearchSuggestions = () => wrapper.findAll(GlFilteredSearchSuggestion);
...@@ -46,10 +50,15 @@ describe('Pipeline Branch Name Token', () => { ...@@ -46,10 +50,15 @@ describe('Pipeline Branch Name Token', () => {
}; };
beforeEach(() => { beforeEach(() => {
mock = new MockAdapter(axios);
jest.spyOn(Api, 'branches').mockResolvedValue({ data: branches });
createComponent(); createComponent();
}); });
afterEach(() => { afterEach(() => {
mock.restore();
wrapper.destroy(); wrapper.destroy();
wrapper = null; wrapper = null;
}); });
...@@ -58,6 +67,12 @@ describe('Pipeline Branch Name Token', () => { ...@@ -58,6 +67,12 @@ describe('Pipeline Branch Name Token', () => {
expect(findFilteredSearchToken().props('config')).toEqual(defaultProps.config); expect(findFilteredSearchToken().props('config')).toEqual(defaultProps.config);
}); });
it('fetches and sets project branches', () => {
expect(Api.branches).toHaveBeenCalled();
expect(wrapper.vm.branches).toEqual(mockBranchesAfterMap);
});
describe('displays loading icon correctly', () => { describe('displays loading icon correctly', () => {
it('shows loading icon', () => { it('shows loading icon', () => {
createComponent({ stubs }, { loading: true }); createComponent({ stubs }, { loading: true });
......
import Api from '~/api';
import MockAdapter from 'axios-mock-adapter';
import axios from '~/lib/utils/axios_utils';
import { GlFilteredSearchToken, GlFilteredSearchSuggestion, GlLoadingIcon } from '@gitlab/ui'; import { GlFilteredSearchToken, GlFilteredSearchSuggestion, GlLoadingIcon } from '@gitlab/ui';
import { shallowMount } from '@vue/test-utils'; import { shallowMount } from '@vue/test-utils';
import PipelineTriggerAuthorToken from '~/pipelines/components/tokens/pipeline_trigger_author_token.vue'; import PipelineTriggerAuthorToken from '~/pipelines/components/tokens/pipeline_trigger_author_token.vue';
...@@ -5,6 +8,7 @@ import { users } from '../mock_data'; ...@@ -5,6 +8,7 @@ import { users } from '../mock_data';
describe('Pipeline Trigger Author Token', () => { describe('Pipeline Trigger Author Token', () => {
let wrapper; let wrapper;
let mock;
const findFilteredSearchToken = () => wrapper.find(GlFilteredSearchToken); const findFilteredSearchToken = () => wrapper.find(GlFilteredSearchToken);
const findAllFilteredSearchSuggestions = () => wrapper.findAll(GlFilteredSearchSuggestion); const findAllFilteredSearchSuggestions = () => wrapper.findAll(GlFilteredSearchSuggestion);
...@@ -45,10 +49,15 @@ describe('Pipeline Trigger Author Token', () => { ...@@ -45,10 +49,15 @@ describe('Pipeline Trigger Author Token', () => {
}; };
beforeEach(() => { beforeEach(() => {
mock = new MockAdapter(axios);
jest.spyOn(Api, 'projectUsers').mockResolvedValue(users);
createComponent(); createComponent();
}); });
afterEach(() => { afterEach(() => {
mock.restore();
wrapper.destroy(); wrapper.destroy();
wrapper = null; wrapper = null;
}); });
...@@ -57,6 +66,12 @@ describe('Pipeline Trigger Author Token', () => { ...@@ -57,6 +66,12 @@ describe('Pipeline Trigger Author Token', () => {
expect(findFilteredSearchToken().props('config')).toEqual(defaultProps.config); expect(findFilteredSearchToken().props('config')).toEqual(defaultProps.config);
}); });
it('fetches and sets project users', () => {
expect(Api.projectUsers).toHaveBeenCalled();
expect(wrapper.vm.users).toEqual(users);
});
describe('displays loading icon correctly', () => { describe('displays loading icon correctly', () => {
it('shows loading icon', () => { it('shows loading icon', () => {
createComponent({ stubs }, { loading: true }); createComponent({ stubs }, { loading: true });
......
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