Commit e1bf0f6a authored by Kushal Pandya's avatar Kushal Pandya

Merge branch '229266-mlunoe-analytics-filter-module-list-values-follow-up' into 'master'

Follow-up on "Feat(Analytics filter module): lists + values"

See merge request gitlab-org/gitlab!40975
parents bc886f7f 3c6a8d53
import { isEmpty } from 'lodash';
import { queryToObject } from '~/lib/utils/url_utility';
/** /**
* Strips enclosing quotations from a string if it has one. * Strips enclosing quotations from a string if it has one.
* *
...@@ -29,3 +32,133 @@ export const uniqueTokens = tokens => { ...@@ -29,3 +32,133 @@ export const uniqueTokens = tokens => {
return uniques; return uniques;
}, []); }, []);
}; };
/**
* Creates a token from a type and a filter. Example returned object
* { type: 'myType', value: { data: 'myData', operator: '= '} }
* @param {String} type the name of the filter
* @param {Object}
* @param {Object.value} filter value to be returned as token data
* @param {Object.operator} filter operator to be retuned as token operator
* @return {Object}
* @return {Object.type} token type
* @return {Object.value} token value
*/
function createToken(type, filter) {
return { type, value: { data: filter.value, operator: filter.operator } };
}
/**
* This function takes a filter object and translates it into a token array
* @param {Object} filters
* @param {Object.myFilterName} a single filter value or an array of filters
* @return {Array} tokens an array of tokens created from filter values
*/
export function prepareTokens(filters = {}) {
return Object.keys(filters).reduce((memo, key) => {
const value = filters[key];
if (!value) {
return memo;
}
if (Array.isArray(value)) {
return [...memo, ...value.map(filterValue => createToken(key, filterValue))];
}
return [...memo, createToken(key, value)];
}, []);
}
export function processFilters(filters) {
return filters.reduce((acc, token) => {
const { type, value } = token;
const { operator } = value;
const tokenValue = value.data;
if (!acc[type]) {
acc[type] = [];
}
acc[type].push({ value: tokenValue, operator });
return acc;
}, {});
}
/**
* This function takes a filter object and maps it into a query object. Example filter:
* { myFilterName: { value: 'foo', operator: '=' } }
* gets translated into:
* { myFilterName: 'foo', 'not[myFilterName]': null }
* @param {Object} filters
* @param {Object.myFilterName} a single filter value or an array of filters
* @return {Object} query object with both filter name and not-name with values
*/
export function filterToQueryObject(filters = {}) {
return Object.keys(filters).reduce((memo, key) => {
const filter = filters[key];
let selected;
let unselected;
if (Array.isArray(filter)) {
selected = filter.filter(item => item.operator === '=').map(item => item.value);
unselected = filter.filter(item => item.operator === '!=').map(item => item.value);
} else {
selected = filter?.operator === '=' ? filter.value : null;
unselected = filter?.operator === '!=' ? filter.value : null;
}
if (isEmpty(selected)) {
selected = null;
}
if (isEmpty(unselected)) {
unselected = null;
}
return { ...memo, [key]: selected, [`not[${key}]`]: unselected };
}, {});
}
/**
* Extracts filter name from url name, e.g. `not[my_filter]` => `my_filter`
* and returns the operator with it depending on the filter name
* @param {String} filterName from url
* @return {Object}
* @return {Object.filterName} extracted filtern ame
* @return {Object.operator} `=` or `!=`
*/
function extractNameAndOperator(filterName) {
// eslint-disable-next-line @gitlab/require-i18n-strings
if (filterName.startsWith('not[') && filterName.endsWith(']')) {
return { filterName: filterName.slice(4, -1), operator: '!=' };
}
return { filterName, operator: '=' };
}
/**
* This function takes a URL query string and maps it into a filter object. Example query string:
* '?myFilterName=foo'
* gets translated into:
* { myFilterName: { value: 'foo', operator: '=' } }
* @param {String} query URL quert string, e.g. from `window.location.search`
* @return {Object} filter object with filter names and their values
*/
export function urlQueryToFilter(query = '') {
const filters = queryToObject(query, { gatherArrays: true });
return Object.keys(filters).reduce((memo, key) => {
const value = filters[key];
if (!value) {
return memo;
}
const { filterName, operator } = extractNameAndOperator(key);
let previousValues = [];
if (Array.isArray(memo[filterName])) {
previousValues = memo[filterName];
}
if (Array.isArray(value)) {
const newAdditions = value.filter(Boolean).map(item => ({ value: item, operator }));
return { ...memo, [filterName]: [...previousValues, ...newAdditions] };
}
return { ...memo, [filterName]: { value, operator } };
}, {});
}
---
title: Fixed an issue where not all URL query parameters would apply to the filter
bar on initial load in the Value Stream Analytics page
merge_request: 40975
author:
type: fixed
...@@ -4,7 +4,7 @@ import { __ } from '~/locale'; ...@@ -4,7 +4,7 @@ import { __ } from '~/locale';
import MilestoneToken from '../../shared/components/tokens/milestone_token.vue'; import MilestoneToken from '../../shared/components/tokens/milestone_token.vue';
import LabelToken from '../../shared/components/tokens/label_token.vue'; import LabelToken from '../../shared/components/tokens/label_token.vue';
import FilteredSearchBar from '~/vue_shared/components/filtered_search_bar/filtered_search_bar_root.vue'; import FilteredSearchBar from '~/vue_shared/components/filtered_search_bar/filtered_search_bar_root.vue';
import { processFilters } from '../../shared/utils'; import { processFilters } from '~/vue_shared/components/filtered_search_bar/filtered_search_utils';
export default { export default {
components: { components: {
......
...@@ -10,7 +10,11 @@ import { ...@@ -10,7 +10,11 @@ import {
DEFAULT_LABEL_NONE, DEFAULT_LABEL_NONE,
DEFAULT_LABEL_ANY, DEFAULT_LABEL_ANY,
} from '~/vue_shared/components/filtered_search_bar/constants'; } from '~/vue_shared/components/filtered_search_bar/constants';
import { prepareTokens, processFilters } from '../../shared/utils'; import {
prepareTokens,
processFilters,
filterToQueryObject,
} from '~/vue_shared/components/filtered_search_bar/filtered_search_utils';
export default { export default {
name: 'FilterBar', name: 'FilterBar',
...@@ -75,6 +79,7 @@ export default { ...@@ -75,6 +79,7 @@ export default {
title: __('Assignees'), title: __('Assignees'),
type: 'assignees', type: 'assignees',
token: AuthorToken, token: AuthorToken,
defaultAuthors: [],
initialAuthors: this.assigneesData, initialAuthors: this.assigneesData,
unique: false, unique: false,
operators: [{ value: '=', description: 'is', default: 'true' }], operators: [{ value: '=', description: 'is', default: 'true' }],
...@@ -83,17 +88,12 @@ export default { ...@@ -83,17 +88,12 @@ export default {
]; ];
}, },
query() { query() {
const selectedLabelList = this.selectedLabelList?.length ? this.selectedLabelList : null; return filterToQueryObject({
const selectedAssigneeList = this.selectedAssigneeList?.length
? this.selectedAssigneeList
: null;
return {
milestone_title: this.selectedMilestone, milestone_title: this.selectedMilestone,
author_username: this.selectedAuthor, author_username: this.selectedAuthor,
label_name: selectedLabelList, label_name: this.selectedLabelList,
assignee_username: selectedAssigneeList, assignee_username: this.selectedAssigneeList,
}; });
}, },
}, },
methods: { methods: {
...@@ -105,22 +105,21 @@ export default { ...@@ -105,22 +105,21 @@ export default {
'fetchAssignees', 'fetchAssignees',
]), ]),
initialFilterValue() { initialFilterValue() {
const { return prepareTokens({
selectedMilestone: milestone = null, milestone: this.selectedMilestone,
selectedAuthor: author = null, author: this.selectedAuthor,
selectedAssigneeList: assignees = [], assignees: this.selectedAssigneeList,
selectedLabelList: labels = [], labels: this.selectedLabelList,
} = this; });
return prepareTokens({ milestone, author, assignees, labels });
}, },
handleFilter(filters) { handleFilter(filters) {
const { labels, milestone, author, assignees } = processFilters(filters); const { labels, milestone, author, assignees } = processFilters(filters);
this.setFilters({ this.setFilters({
selectedAuthor: author ? author[0].value : null, selectedAuthor: author ? author[0] : null,
selectedMilestone: milestone ? milestone[0].value : null, selectedMilestone: milestone ? milestone[0] : null,
selectedAssigneeList: assignees ? assignees.map(a => a.value) : [], selectedAssigneeList: assignees || [],
selectedLabelList: labels ? labels.map(l => l.value) : [], selectedLabelList: labels || [],
}); });
}, },
}, },
......
...@@ -3,6 +3,7 @@ import { GlToast } from '@gitlab/ui'; ...@@ -3,6 +3,7 @@ import { GlToast } from '@gitlab/ui';
import CycleAnalytics from './components/base.vue'; import CycleAnalytics from './components/base.vue';
import createStore from './store'; import createStore from './store';
import { buildCycleAnalyticsInitialData } from '../shared/utils'; import { buildCycleAnalyticsInitialData } from '../shared/utils';
import { urlQueryToFilter } from '~/vue_shared/components/filtered_search_bar/filtered_search_utils';
import { parseBoolean } from '~/lib/utils/common_utils'; import { parseBoolean } from '~/lib/utils/common_utils';
Vue.use(GlToast); Vue.use(GlToast);
...@@ -19,8 +20,19 @@ export default () => { ...@@ -19,8 +20,19 @@ export default () => {
analyticsSimilaritySearch: hasAnalyticsSimilaritySearch = false, analyticsSimilaritySearch: hasAnalyticsSimilaritySearch = false,
} = gon?.features; } = gon?.features;
const {
author_username = null,
milestone_title = null,
assignee_username = [],
label_name = [],
} = urlQueryToFilter(window.location.search);
store.dispatch('initializeCycleAnalytics', { store.dispatch('initializeCycleAnalytics', {
...initialData, ...initialData,
selectedAuthor: author_username,
selectedMilestone: milestone_title,
selectedAssigneeList: assignee_username,
selectedLabelList: label_name,
featureFlags: { featureFlags: {
hasDurationChart, hasDurationChart,
hasPathNavigation, hasPathNavigation,
......
...@@ -283,8 +283,8 @@ export const initializeCycleAnalytics = ({ dispatch, commit }, initialData = {}) ...@@ -283,8 +283,8 @@ export const initializeCycleAnalytics = ({ dispatch, commit }, initialData = {})
labelsPath, labelsPath,
selectedAuthor, selectedAuthor,
selectedMilestone, selectedMilestone,
selectedAssignees, selectedAssigneeList,
selectedLabels, selectedLabelList,
} = initialData; } = initialData;
commit(types.SET_FEATURE_FLAGS, featureFlags); commit(types.SET_FEATURE_FLAGS, featureFlags);
...@@ -294,8 +294,8 @@ export const initializeCycleAnalytics = ({ dispatch, commit }, initialData = {}) ...@@ -294,8 +294,8 @@ export const initializeCycleAnalytics = ({ dispatch, commit }, initialData = {})
dispatch('filters/initialize', { dispatch('filters/initialize', {
selectedAuthor, selectedAuthor,
selectedMilestone, selectedMilestone,
selectedAssignees, selectedAssigneeList,
selectedLabels, selectedLabelList,
}), }),
dispatch('durationChart/setLoading', true), dispatch('durationChart/setLoading', true),
dispatch('typeOfWork/setLoading', true), dispatch('typeOfWork/setLoading', true),
......
import dateFormat from 'dateformat'; import dateFormat from 'dateformat';
import { dateFormats } from './constants';
import { convertObjectPropsToCamelCase } from '~/lib/utils/common_utils'; import { convertObjectPropsToCamelCase } from '~/lib/utils/common_utils';
import { dateFormats } from './constants';
export const toYmd = date => dateFormat(date, dateFormats.isoDate); export const toYmd = date => dateFormat(date, dateFormats.isoDate);
...@@ -79,10 +79,6 @@ export const buildCycleAnalyticsInitialData = ({ ...@@ -79,10 +79,6 @@ export const buildCycleAnalyticsInitialData = ({
groupFullPath = null, groupFullPath = null,
groupParentId = null, groupParentId = null,
groupAvatarUrl = null, groupAvatarUrl = null,
author = null,
milestone = null,
labels = null,
assignees = null,
labelsPath = '', labelsPath = '',
milestonesPath = '', milestonesPath = '',
} = {}) => ({ } = {}) => ({
...@@ -102,10 +98,6 @@ export const buildCycleAnalyticsInitialData = ({ ...@@ -102,10 +98,6 @@ export const buildCycleAnalyticsInitialData = ({
selectedProjects: projects selectedProjects: projects
? buildProjectsFromJSON(projects).map(convertObjectPropsToCamelCase) ? buildProjectsFromJSON(projects).map(convertObjectPropsToCamelCase)
: [], : [],
selectedAuthor: author,
selectedMilestone: milestone,
selectedLabels: labels ? JSON.parse(labels) : [],
selectedAssignees: assignees ? JSON.parse(assignees) : [],
labelsPath, labelsPath,
milestonesPath, milestonesPath,
}); });
...@@ -114,28 +106,3 @@ export const filterBySearchTerm = (data = [], searchTerm = '', filterByKey = 'na ...@@ -114,28 +106,3 @@ export const filterBySearchTerm = (data = [], searchTerm = '', filterByKey = 'na
if (!searchTerm?.length) return data; if (!searchTerm?.length) return data;
return data.filter(item => item[filterByKey].toLowerCase().includes(searchTerm.toLowerCase())); return data.filter(item => item[filterByKey].toLowerCase().includes(searchTerm.toLowerCase()));
}; };
export const prepareTokens = (tokens = {}) => {
const { milestone = null, author = null, assignees = [], labels = [] } = tokens;
const authorToken = author ? [{ type: 'author', value: { data: author } }] : [];
const milestoneToken = milestone ? [{ type: 'milestone', value: { data: milestone } }] : [];
const assigneeTokens = assignees?.map(data => ({ type: 'assignees', value: { data } })) || [];
const labelTokens = labels?.map(data => ({ type: 'labels', value: { data } })) || [];
return [...authorToken, ...milestoneToken, ...assigneeTokens, ...labelTokens];
};
export function processFilters(filters) {
return filters.reduce((acc, token) => {
const { type, value } = token;
const { operator } = value;
const tokenValue = value.data;
if (!acc[type]) {
acc[type] = [];
}
acc[type].push({ value: tokenValue, operator });
return acc;
}, {});
}
import { createLocalVue, shallowMount } from '@vue/test-utils'; import { createLocalVue, shallowMount } from '@vue/test-utils';
import Vuex from 'vuex'; import Vuex from 'vuex';
import * as utils from 'ee/analytics/shared/utils';
import FilterBar from 'ee/analytics/code_review_analytics/components/filter_bar.vue'; import FilterBar from 'ee/analytics/code_review_analytics/components/filter_bar.vue';
import createFiltersState from 'ee/analytics/shared/store/modules/filters/state'; import createFiltersState from 'ee/analytics/shared/store/modules/filters/state';
import FilteredSearchBar from '~/vue_shared/components/filtered_search_bar/filtered_search_bar_root.vue'; import FilteredSearchBar from '~/vue_shared/components/filtered_search_bar/filtered_search_bar_root.vue';
import * as utils from '~/vue_shared/components/filtered_search_bar/filtered_search_utils';
import { mockMilestones, mockLabels } from '../mock_data'; import { mockMilestones, mockLabels } from '../mock_data';
const localVue = createLocalVue(); const localVue = createLocalVue();
......
...@@ -2,12 +2,12 @@ import { createLocalVue, shallowMount } from '@vue/test-utils'; ...@@ -2,12 +2,12 @@ import { createLocalVue, shallowMount } from '@vue/test-utils';
import Vuex from 'vuex'; import Vuex from 'vuex';
import axios from 'axios'; import axios from 'axios';
import MockAdapter from 'axios-mock-adapter'; import MockAdapter from 'axios-mock-adapter';
import * as utils from 'ee/analytics/shared/utils';
import storeConfig from 'ee/analytics/cycle_analytics/store'; import storeConfig from 'ee/analytics/cycle_analytics/store';
import FilterBar from 'ee/analytics/cycle_analytics/components/filter_bar.vue'; import FilterBar from 'ee/analytics/cycle_analytics/components/filter_bar.vue';
import initialFiltersState from 'ee/analytics/shared/store/modules/filters/state'; import initialFiltersState from 'ee/analytics/shared/store/modules/filters/state';
import FilteredSearchBar from '~/vue_shared/components/filtered_search_bar/filtered_search_bar_root.vue'; import FilteredSearchBar from '~/vue_shared/components/filtered_search_bar/filtered_search_bar_root.vue';
import UrlSync from '~/vue_shared/components/url_sync.vue'; import UrlSync from '~/vue_shared/components/url_sync.vue';
import * as utils from '~/vue_shared/components/filtered_search_bar/filtered_search_utils';
import { filterMilestones, filterLabels } from '../../shared/store/modules/filters/mock_data'; import { filterMilestones, filterLabels } from '../../shared/store/modules/filters/mock_data';
import * as commonUtils from '~/lib/utils/common_utils'; import * as commonUtils from '~/lib/utils/common_utils';
import * as urlUtils from '~/lib/utils/url_utility'; import * as urlUtils from '~/lib/utils/url_utility';
...@@ -29,9 +29,13 @@ const initialFilterBarState = { ...@@ -29,9 +29,13 @@ const initialFilterBarState = {
const defaultParams = { const defaultParams = {
milestone_title: null, milestone_title: null,
'not[milestone_title]': null,
author_username: null, author_username: null,
'not[author_username]': null,
assignee_username: null, assignee_username: null,
'not[assignee_username]': null,
label_name: null, label_name: null,
'not[label_name]': null,
}; };
async function shouldMergeUrlParams(wrapper, result) { async function shouldMergeUrlParams(wrapper, result) {
...@@ -167,8 +171,8 @@ describe('Filter bar', () => { ...@@ -167,8 +171,8 @@ describe('Filter bar', () => {
expect(setFiltersMock).toHaveBeenCalledWith( expect(setFiltersMock).toHaveBeenCalledWith(
expect.anything(), expect.anything(),
{ {
selectedLabelList: [selectedLabelList[0].title], selectedLabelList: [{ value: selectedLabelList[0].title, operator: '=' }],
selectedMilestone: selectedMilestone[0].title, selectedMilestone: { value: selectedMilestone[0].title, operator: '=' },
selectedAssigneeList: [], selectedAssigneeList: [],
selectedAuthor: null, selectedAuthor: null,
}, },
...@@ -177,13 +181,22 @@ describe('Filter bar', () => { ...@@ -177,13 +181,22 @@ describe('Filter bar', () => {
}); });
}); });
describe.each` describe.each([
stateKey | payload | paramKey ['selectedMilestone', 'milestone_title', { value: '12.0', operator: '=' }, '12.0'],
${'selectedMilestone'} | ${'12.0'} | ${'milestone_title'} ['selectedAuthor', 'author_username', { value: 'rootUser', operator: '=' }, 'rootUser'],
${'selectedAuthor'} | ${'rootUser'} | ${'author_username'} [
${'selectedLabelList'} | ${['Afternix', 'Brouceforge']} | ${'label_name'} 'selectedLabelList',
${'selectedAssigneeList'} | ${['rootUser', 'secondaryUser']} | ${'assignee_username'} 'label_name',
`('with a $stateKey updates the $paramKey url parameter', ({ stateKey, payload, paramKey }) => { [{ value: 'Afternix', operator: '=' }, { value: 'Brouceforge', operator: '=' }],
['Afternix', 'Brouceforge'],
],
[
'selectedAssigneeList',
'assignee_username',
[{ value: 'rootUser', operator: '=' }, { value: 'secondaryUser', operator: '=' }],
['rootUser', 'secondaryUser'],
],
])('with a %s updates the %s url parameter', (stateKey, paramKey, payload, result) => {
beforeEach(() => { beforeEach(() => {
commonUtils.historyPushState = jest.fn(); commonUtils.historyPushState = jest.fn();
urlUtils.mergeUrlParams = jest.fn(); urlUtils.mergeUrlParams = jest.fn();
...@@ -199,7 +212,7 @@ describe('Filter bar', () => { ...@@ -199,7 +212,7 @@ describe('Filter bar', () => {
it(`sets the ${paramKey} url parameter`, () => { it(`sets the ${paramKey} url parameter`, () => {
return shouldMergeUrlParams(wrapper, { return shouldMergeUrlParams(wrapper, {
...defaultParams, ...defaultParams,
[paramKey]: payload, [paramKey]: result,
}); });
}); });
}); });
......
...@@ -3,8 +3,6 @@ import { ...@@ -3,8 +3,6 @@ import {
buildProjectFromDataset, buildProjectFromDataset,
buildCycleAnalyticsInitialData, buildCycleAnalyticsInitialData,
filterBySearchTerm, filterBySearchTerm,
prepareTokens,
processFilters,
} from 'ee/analytics/shared/utils'; } from 'ee/analytics/shared/utils';
const groupDataset = { const groupDataset = {
...@@ -78,17 +76,13 @@ describe('buildProjectFromDataset', () => { ...@@ -78,17 +76,13 @@ describe('buildProjectFromDataset', () => {
describe('buildCycleAnalyticsInitialData', () => { describe('buildCycleAnalyticsInitialData', () => {
it.each` it.each`
field | value field | value
${'group'} | ${null} ${'group'} | ${null}
${'createdBefore'} | ${null} ${'createdBefore'} | ${null}
${'createdAfter'} | ${null} ${'createdAfter'} | ${null}
${'selectedProjects'} | ${[]} ${'selectedProjects'} | ${[]}
${'selectedAuthor'} | ${null} ${'labelsPath'} | ${''}
${'selectedMilestone'} | ${null} ${'milestonesPath'} | ${''}
${'selectedLabels'} | ${[]}
${'selectedAssignees'} | ${[]}
${'labelsPath'} | ${''}
${'milestonesPath'} | ${''}
`('will set a default value for "$field" if is not present', ({ field, value }) => { `('will set a default value for "$field" if is not present', ({ field, value }) => {
expect(buildCycleAnalyticsInitialData()).toMatchObject({ expect(buildCycleAnalyticsInitialData()).toMatchObject({
[field]: value, [field]: value,
...@@ -140,62 +134,6 @@ describe('buildCycleAnalyticsInitialData', () => { ...@@ -140,62 +134,6 @@ describe('buildCycleAnalyticsInitialData', () => {
}); });
}); });
describe('selectedAssignees', () => {
it('will be set given an array of assignees', () => {
const selectedAssignees = ['krillin', 'chiao-tzu'];
expect(
buildCycleAnalyticsInitialData({ assignees: JSON.stringify(selectedAssignees) }),
).toMatchObject({
selectedAssignees,
});
});
it.each`
field | value
${'selectedAssignees'} | ${null}
${'selectedAssignees'} | ${[]}
${'selectedAssignees'} | ${''}
`('will be an empty array if given a value of `$value`', ({ value, field }) => {
expect(buildCycleAnalyticsInitialData({ projects: value })).toMatchObject({
[field]: [],
});
});
});
describe('selectedLabels', () => {
it('will be set given an array of labels', () => {
const selectedLabels = ['krillin', 'chiao-tzu'];
expect(
buildCycleAnalyticsInitialData({ labels: JSON.stringify(selectedLabels) }),
).toMatchObject({ selectedLabels });
});
it.each`
field | value
${'selectedLabels'} | ${null}
${'selectedLabels'} | ${[]}
${'selectedLabels'} | ${''}
`('will be an empty array if given a value of `$value`', ({ value, field }) => {
expect(buildCycleAnalyticsInitialData({ projects: value })).toMatchObject({
[field]: [],
});
});
});
describe.each`
field | key | value
${'milestone'} | ${'selectedMilestone'} | ${'cell-saga'}
${'author'} | ${'selectedAuthor'} | ${'cell'}
`('$field', ({ field, value, key }) => {
it(`will set ${key} field with the given value`, () => {
expect(buildCycleAnalyticsInitialData({ [field]: value })).toMatchObject({ [key]: value });
});
it(`will set ${key} to null if omitted`, () => {
expect(buildCycleAnalyticsInitialData()).toMatchObject({ [key]: null });
});
});
describe.each` describe.each`
field | value field | value
${'createdBefore'} | ${'2019-12-31'} ${'createdBefore'} | ${'2019-12-31'}
...@@ -235,50 +173,3 @@ describe('buildCycleAnalyticsInitialData', () => { ...@@ -235,50 +173,3 @@ describe('buildCycleAnalyticsInitialData', () => {
}); });
}); });
}); });
describe('prepareTokens', () => {
describe('with empty data', () => {
it('returns an empty array', () => {
expect(prepareTokens()).toEqual([]);
expect(prepareTokens({})).toEqual([]);
expect(prepareTokens({ milestone: null, author: null, assignees: [], labels: [] })).toEqual(
[],
);
});
});
it.each`
token | value | result
${'milestone'} | ${'v1.0'} | ${[{ type: 'milestone', value: { data: 'v1.0' } }]}
${'author'} | ${'mr.popo'} | ${[{ type: 'author', value: { data: 'mr.popo' } }]}
${'labels'} | ${['z-fighters']} | ${[{ type: 'labels', value: { data: 'z-fighters' } }]}
${'assignees'} | ${['krillin', 'piccolo']} | ${[{ type: 'assignees', value: { data: 'krillin' } }, { type: 'assignees', value: { data: 'piccolo' } }]}
`('with $token=$value sets the $token key', ({ token, value, result }) => {
const res = prepareTokens({ [token]: value });
expect(res).toEqual(result);
});
});
describe('processFilters', () => {
it('processes multiple filter values', () => {
const result = processFilters([
{ type: 'milestone', value: { data: 'my-milestone', operator: '=' } },
{ type: 'labels', value: { data: 'my-label', operator: '=' } },
]);
expect(result).toStrictEqual({
labels: [{ value: 'my-label', operator: '=' }],
milestone: [{ value: 'my-milestone', operator: '=' }],
});
});
it('does not remove wrapping double quotes from the data', () => {
const result = processFilters([
{ type: 'milestone', value: { data: '"milestone with spaces"', operator: '=' } },
]);
expect(result).toStrictEqual({
milestone: [{ value: '"milestone with spaces"', operator: '=' }],
});
});
});
import * as filteredSearchUtils from '~/vue_shared/components/filtered_search_bar/filtered_search_utils'; import {
stripQuotes,
uniqueTokens,
prepareTokens,
processFilters,
filterToQueryObject,
urlQueryToFilter,
} from '~/vue_shared/components/filtered_search_bar/filtered_search_utils';
import { import {
tokenValueAuthor, tokenValueAuthor,
...@@ -20,7 +27,7 @@ describe('Filtered Search Utils', () => { ...@@ -20,7 +27,7 @@ describe('Filtered Search Utils', () => {
`( `(
'returns string $outputValue when called with string $inputValue', 'returns string $outputValue when called with string $inputValue',
({ inputValue, outputValue }) => { ({ inputValue, outputValue }) => {
expect(filteredSearchUtils.stripQuotes(inputValue)).toBe(outputValue); expect(stripQuotes(inputValue)).toBe(outputValue);
}, },
); );
}); });
...@@ -28,7 +35,7 @@ describe('Filtered Search Utils', () => { ...@@ -28,7 +35,7 @@ describe('Filtered Search Utils', () => {
describe('uniqueTokens', () => { describe('uniqueTokens', () => {
it('returns tokens array with duplicates removed', () => { it('returns tokens array with duplicates removed', () => {
expect( expect(
filteredSearchUtils.uniqueTokens([ uniqueTokens([
tokenValueAuthor, tokenValueAuthor,
tokenValueLabel, tokenValueLabel,
tokenValueMilestone, tokenValueMilestone,
...@@ -40,13 +47,172 @@ describe('Filtered Search Utils', () => { ...@@ -40,13 +47,172 @@ describe('Filtered Search Utils', () => {
it('returns tokens array as it is if it does not have duplicates', () => { it('returns tokens array as it is if it does not have duplicates', () => {
expect( expect(
filteredSearchUtils.uniqueTokens([ uniqueTokens([tokenValueAuthor, tokenValueLabel, tokenValueMilestone, tokenValuePlain]),
tokenValueAuthor,
tokenValueLabel,
tokenValueMilestone,
tokenValuePlain,
]),
).toHaveLength(4); ).toHaveLength(4);
}); });
}); });
}); });
describe('prepareTokens', () => {
describe('with empty data', () => {
it('returns an empty array', () => {
expect(prepareTokens()).toEqual([]);
expect(prepareTokens({})).toEqual([]);
expect(prepareTokens({ milestone: null, author: null, assignees: [], labels: [] })).toEqual(
[],
);
});
});
it.each([
[
'milestone',
{ value: 'v1.0', operator: '=' },
[{ type: 'milestone', value: { data: 'v1.0', operator: '=' } }],
],
[
'author',
{ value: 'mr.popo', operator: '!=' },
[{ type: 'author', value: { data: 'mr.popo', operator: '!=' } }],
],
[
'labels',
[{ value: 'z-fighters', operator: '=' }],
[{ type: 'labels', value: { data: 'z-fighters', operator: '=' } }],
],
[
'assignees',
[{ value: 'krillin', operator: '=' }, { value: 'piccolo', operator: '!=' }],
[
{ type: 'assignees', value: { data: 'krillin', operator: '=' } },
{ type: 'assignees', value: { data: 'piccolo', operator: '!=' } },
],
],
[
'foo',
[{ value: 'bar', operator: '!=' }, { value: 'baz', operator: '!=' }],
[
{ type: 'foo', value: { data: 'bar', operator: '!=' } },
{ type: 'foo', value: { data: 'baz', operator: '!=' } },
],
],
])('gathers %s=%j into result=%j', (token, value, result) => {
const res = prepareTokens({ [token]: value });
expect(res).toEqual(result);
});
});
describe('processFilters', () => {
it('processes multiple filter values', () => {
const result = processFilters([
{ type: 'foo', value: { data: 'foo', operator: '=' } },
{ type: 'bar', value: { data: 'bar1', operator: '=' } },
{ type: 'bar', value: { data: 'bar2', operator: '!=' } },
]);
expect(result).toStrictEqual({
foo: [{ value: 'foo', operator: '=' }],
bar: [{ value: 'bar1', operator: '=' }, { value: 'bar2', operator: '!=' }],
});
});
it('does not remove wrapping double quotes from the data', () => {
const result = processFilters([
{ type: 'foo', value: { data: '"value with spaces"', operator: '=' } },
]);
expect(result).toStrictEqual({
foo: [{ value: '"value with spaces"', operator: '=' }],
});
});
});
describe('filterToQueryObject', () => {
describe('with empty data', () => {
it('returns an empty object', () => {
expect(filterToQueryObject()).toEqual({});
expect(filterToQueryObject({})).toEqual({});
expect(filterToQueryObject({ author_username: null, label_name: [] })).toEqual({
author_username: null,
label_name: null,
'not[author_username]': null,
'not[label_name]': null,
});
});
});
it.each([
[
'author_username',
{ value: 'v1.0', operator: '=' },
{ author_username: 'v1.0', 'not[author_username]': null },
],
[
'author_username',
{ value: 'v1.0', operator: '!=' },
{ author_username: null, 'not[author_username]': 'v1.0' },
],
[
'label_name',
[{ value: 'z-fighters', operator: '=' }],
{ label_name: ['z-fighters'], 'not[label_name]': null },
],
[
'label_name',
[{ value: 'z-fighters', operator: '!=' }],
{ label_name: null, 'not[label_name]': ['z-fighters'] },
],
[
'foo',
[{ value: 'bar', operator: '=' }, { value: 'baz', operator: '=' }],
{ foo: ['bar', 'baz'], 'not[foo]': null },
],
[
'foo',
[{ value: 'bar', operator: '!=' }, { value: 'baz', operator: '!=' }],
{ foo: null, 'not[foo]': ['bar', 'baz'] },
],
[
'foo',
[{ value: 'bar', operator: '!=' }, { value: 'baz', operator: '=' }],
{ foo: ['baz'], 'not[foo]': ['bar'] },
],
])('gathers filter values %s=%j into query object=%j', (token, value, result) => {
const res = filterToQueryObject({ [token]: value });
expect(res).toEqual(result);
});
});
describe('urlQueryToFilter', () => {
describe('with empty data', () => {
it('returns an empty object', () => {
expect(urlQueryToFilter()).toEqual({});
expect(urlQueryToFilter('')).toEqual({});
expect(urlQueryToFilter('author_username=&milestone_title=&')).toEqual({});
});
});
it.each([
['author_username=v1.0', { author_username: { value: 'v1.0', operator: '=' } }],
['not[author_username]=v1.0', { author_username: { value: 'v1.0', operator: '!=' } }],
['foo=bar&foo=baz', { foo: { value: 'baz', operator: '=' } }],
['foo=bar&foo[]=baz', { foo: [{ value: 'baz', operator: '=' }] }],
['not[foo]=bar&foo=baz', { foo: { value: 'baz', operator: '=' } }],
[
'foo[]=bar&foo[]=baz&not[foo]=',
{ foo: [{ value: 'bar', operator: '=' }, { value: 'baz', operator: '=' }] },
],
[
'foo[]=&not[foo][]=bar&not[foo][]=baz',
{ foo: [{ value: 'bar', operator: '!=' }, { value: 'baz', operator: '!=' }] },
],
[
'foo[]=baz&not[foo][]=bar',
{ foo: [{ value: 'baz', operator: '=' }, { value: 'bar', operator: '!=' }] },
],
['not[foo][]=bar', { foo: [{ value: 'bar', operator: '!=' }] }],
])('gathers filter values %s into query object=%j', (query, result) => {
const res = urlQueryToFilter(query);
expect(res).toEqual(result);
});
});
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