Commit bf3ddac6 authored by Mark Florian's avatar Mark Florian

Set window location in tests more consistently

Jest/JSDOM provides a more robust and simple way of setting the mocked
`window.location`, which preserves all the expected behaviours and
properties of `window.location`.

By using it, we can get rid of hacks which attempt to recreate
behaviours of `window.location` that we lost by mutating it ourselves
(for instance, adding a `toString` method).
parent 427d13a4
...@@ -423,6 +423,40 @@ it('does something', () => { ...@@ -423,6 +423,40 @@ it('does something', () => {
}); });
``` ```
### Mocking the current location in Jest
If your tests require `window.location.href` to take a particular value, use
the `setWindowLocation` helper:
```javascript
import setWindowLocation from 'helpers/set_window_location';
it('passes', () => {
setWindowLocation('https://gitlab.test/foo?bar=true');
expect(window.location).toMatchObject({
hostname: 'gitlab.test',
pathname: '/foo',
search: '?bar=true',
});
});
```
If your tests need to assert that certain `window.location` methods were
called, use the `useMockLocationHelper` helper:
```javascript
import { useMockLocationHelper } from 'helpers/mock_window_location_helper';
useMockLocationHelper();
it('passes', () => {
window.location.reload();
expect(window.location.reload).toHaveBeenCalled();
});
```
### Waiting in tests ### Waiting in tests
Sometimes a test needs to wait for something to happen in the application before it continues. Sometimes a test needs to wait for something to happen in the application before it continues.
......
...@@ -2,6 +2,7 @@ import { GlPagination, GlTable } from '@gitlab/ui'; ...@@ -2,6 +2,7 @@ import { GlPagination, GlTable } from '@gitlab/ui';
import { mount } from '@vue/test-utils'; import { mount } from '@vue/test-utils';
import AuditEventsTable from 'ee/audit_events/components/audit_events_table.vue'; import AuditEventsTable from 'ee/audit_events/components/audit_events_table.vue';
import setWindowLocation from 'helpers/set_window_location_helper';
import createEvents from '../mock_data'; import createEvents from '../mock_data';
const EVENTS = createEvents(); const EVENTS = createEvents();
...@@ -24,8 +25,7 @@ describe('AuditEventsTable component', () => { ...@@ -24,8 +25,7 @@ describe('AuditEventsTable component', () => {
}; };
beforeEach(() => { beforeEach(() => {
delete window.location; setWindowLocation('https://localhost');
window.location = new URL('https://localhost');
wrapper = createComponent(); wrapper = createComponent();
}); });
...@@ -60,21 +60,21 @@ describe('AuditEventsTable component', () => { ...@@ -60,21 +60,21 @@ describe('AuditEventsTable component', () => {
}); });
it('should get the page number from the URL', () => { it('should get the page number from the URL', () => {
window.location.search = '?page=2'; setWindowLocation('?page=2');
wrapper = createComponent(); wrapper = createComponent();
expect(wrapper.find(GlPagination).props().value).toBe(2); expect(wrapper.find(GlPagination).props().value).toBe(2);
}); });
it('should not have a prevPage if the page is 1', () => { it('should not have a prevPage if the page is 1', () => {
window.location.search = '?page=1'; setWindowLocation('?page=1');
wrapper = createComponent(); wrapper = createComponent();
expect(wrapper.find(GlPagination).props().prevPage).toBe(null); expect(wrapper.find(GlPagination).props().prevPage).toBe(null);
}); });
it('should set the prevPage to 1 if the page is 2', () => { it('should set the prevPage to 1 if the page is 2', () => {
window.location.search = '?page=2'; setWindowLocation('?page=2');
wrapper = createComponent(); wrapper = createComponent();
expect(wrapper.find(GlPagination).props().prevPage).toBe(1); expect(wrapper.find(GlPagination).props().prevPage).toBe(1);
...@@ -88,7 +88,7 @@ describe('AuditEventsTable component', () => { ...@@ -88,7 +88,7 @@ describe('AuditEventsTable component', () => {
}); });
it('should set the nextPage to 2 if the page is 1', () => { it('should set the nextPage to 2 if the page is 1', () => {
window.location.search = '?page=1'; setWindowLocation('?page=1');
wrapper = createComponent(); wrapper = createComponent();
expect(wrapper.find(GlPagination).props().nextPage).toBe(2); expect(wrapper.find(GlPagination).props().nextPage).toBe(2);
......
...@@ -8,6 +8,7 @@ import actions, { gqlClient } from 'ee/boards/stores/actions'; ...@@ -8,6 +8,7 @@ import actions, { gqlClient } from 'ee/boards/stores/actions';
import boardsStoreEE from 'ee/boards/stores/boards_store_ee'; import boardsStoreEE from 'ee/boards/stores/boards_store_ee';
import * as types from 'ee/boards/stores/mutation_types'; import * as types from 'ee/boards/stores/mutation_types';
import mutations from 'ee/boards/stores/mutations'; import mutations from 'ee/boards/stores/mutations';
import setWindowLocation from 'helpers/set_window_location_helper';
import { TEST_HOST } from 'helpers/test_constants'; import { TEST_HOST } from 'helpers/test_constants';
import testAction from 'helpers/vuex_action_helper'; import testAction from 'helpers/vuex_action_helper';
import { mockMoveIssueParams, mockMoveData, mockMoveState } from 'jest/boards/mock_data'; import { mockMoveIssueParams, mockMoveData, mockMoveState } from 'jest/boards/mock_data';
...@@ -32,6 +33,7 @@ Vue.use(Vuex); ...@@ -32,6 +33,7 @@ Vue.use(Vuex);
let mock; let mock;
beforeEach(() => { beforeEach(() => {
setWindowLocation(TEST_HOST);
mock = new MockAdapter(axios); mock = new MockAdapter(axios);
window.gon = { features: {} }; window.gon = { features: {} };
jest.spyOn(commonUtils, 'historyPushState'); jest.spyOn(commonUtils, 'historyPushState');
...@@ -615,9 +617,7 @@ describe('fetchIssuesForEpic', () => { ...@@ -615,9 +617,7 @@ describe('fetchIssuesForEpic', () => {
describe('toggleEpicSwimlanes', () => { describe('toggleEpicSwimlanes', () => {
it('should commit mutation TOGGLE_EPICS_SWIMLANES', () => { it('should commit mutation TOGGLE_EPICS_SWIMLANES', () => {
const startURl = `${TEST_HOST}/groups/gitlab-org/-/boards/1?group_by=epic`; const startURl = `${TEST_HOST}/groups/gitlab-org/-/boards/1?group_by=epic`;
global.jsdom.reconfigure({ setWindowLocation(startURl);
url: startURl,
});
const state = { const state = {
isShowingEpicsSwimlanes: false, isShowingEpicsSwimlanes: false,
...@@ -643,9 +643,7 @@ describe('toggleEpicSwimlanes', () => { ...@@ -643,9 +643,7 @@ describe('toggleEpicSwimlanes', () => {
}); });
it('should dispatch fetchEpicsSwimlanes and fetchLists actions when isShowingEpicsSwimlanes is true', () => { it('should dispatch fetchEpicsSwimlanes and fetchLists actions when isShowingEpicsSwimlanes is true', () => {
global.jsdom.reconfigure({ setWindowLocation(`${TEST_HOST}/groups/gitlab-org/-/boards/1`);
url: `${TEST_HOST}/groups/gitlab-org/-/boards/1`,
});
jest.spyOn(gqlClient, 'query').mockResolvedValue({}); jest.spyOn(gqlClient, 'query').mockResolvedValue({});
......
import { GlPagination } from '@gitlab/ui'; import { GlPagination } from '@gitlab/ui';
import { shallowMount } from '@vue/test-utils'; import { shallowMount } from '@vue/test-utils';
import Pagination from 'ee/compliance_dashboard/components/shared/pagination.vue'; import Pagination from 'ee/compliance_dashboard/components/shared/pagination.vue';
import setWindowLocation from 'helpers/set_window_location_helper';
describe('Pagination component', () => { describe('Pagination component', () => {
let wrapper; let wrapper;
const origin = 'https://localhost';
const findGlPagination = () => wrapper.find(GlPagination); const findGlPagination = () => wrapper.find(GlPagination);
const getLink = (query) => wrapper.find(query).element.getAttribute('href'); const getLink = (query) => wrapper.find(query).element.getAttribute('href');
...@@ -23,8 +24,7 @@ describe('Pagination component', () => { ...@@ -23,8 +24,7 @@ describe('Pagination component', () => {
}; };
beforeEach(() => { beforeEach(() => {
delete window.location; setWindowLocation(origin);
window.location = new URL('https://localhost');
}); });
afterEach(() => { afterEach(() => {
...@@ -33,7 +33,7 @@ describe('Pagination component', () => { ...@@ -33,7 +33,7 @@ describe('Pagination component', () => {
describe('when initialized', () => { describe('when initialized', () => {
beforeEach(() => { beforeEach(() => {
window.location.search = '?page=2'; setWindowLocation('?page=2');
wrapper = createComponent(); wrapper = createComponent();
}); });
...@@ -42,17 +42,17 @@ describe('Pagination component', () => { ...@@ -42,17 +42,17 @@ describe('Pagination component', () => {
}); });
it('should create a link to the previous page', () => { it('should create a link to the previous page', () => {
expect(findPrevPageLink()).toEqual('https://localhost/?page=1'); expect(findPrevPageLink()).toBe(`${origin}/?page=1`);
}); });
it('should create a link to the next page', () => { it('should create a link to the next page', () => {
expect(findNextPageLink()).toEqual('https://localhost/?page=3'); expect(findNextPageLink()).toBe(`${origin}/?page=3`);
}); });
}); });
describe('when on last page', () => { describe('when on last page', () => {
beforeEach(() => { beforeEach(() => {
window.location.search = '?page=2'; setWindowLocation('?page=2');
wrapper = createComponent(true); wrapper = createComponent(true);
}); });
...@@ -63,7 +63,7 @@ describe('Pagination component', () => { ...@@ -63,7 +63,7 @@ describe('Pagination component', () => {
describe('when there is only one page', () => { describe('when there is only one page', () => {
beforeEach(() => { beforeEach(() => {
window.location.search = '?page=1'; setWindowLocation('?page=1');
wrapper = createComponent(true); wrapper = createComponent(true);
}); });
......
...@@ -11,15 +11,15 @@ import dastScannerProfilesQuery from 'ee/security_configuration/dast_profiles/gr ...@@ -11,15 +11,15 @@ import dastScannerProfilesQuery from 'ee/security_configuration/dast_profiles/gr
import dastSiteProfilesQuery from 'ee/security_configuration/dast_profiles/graphql/dast_site_profiles.query.graphql'; import dastSiteProfilesQuery from 'ee/security_configuration/dast_profiles/graphql/dast_site_profiles.query.graphql';
import { useLocalStorageSpy } from 'helpers/local_storage_helper'; import { useLocalStorageSpy } from 'helpers/local_storage_helper';
import createApolloProvider from 'helpers/mock_apollo_helper'; import createApolloProvider from 'helpers/mock_apollo_helper';
import setWindowLocation from 'helpers/set_window_location_helper';
import { stubComponent } from 'helpers/stub_component'; import { stubComponent } from 'helpers/stub_component';
import waitForPromises from 'helpers/wait_for_promises'; import waitForPromises from 'helpers/wait_for_promises';
import { redirectTo, setUrlParams } from '~/lib/utils/url_utility'; import { redirectTo } from '~/lib/utils/url_utility';
import RefSelector from '~/ref/components/ref_selector.vue'; import RefSelector from '~/ref/components/ref_selector.vue';
import LocalStorageSync from '~/vue_shared/components/local_storage_sync.vue'; import LocalStorageSync from '~/vue_shared/components/local_storage_sync.vue';
import * as responses from '../mocks/apollo_mocks'; import * as responses from '../mocks/apollo_mocks';
import { scannerProfiles, siteProfiles } from '../mocks/mock_data'; import { scannerProfiles, siteProfiles } from '../mocks/mock_data';
const URL_HOST = 'https://localhost/';
const helpPagePath = '/application_security/dast/index#on-demand-scans'; const helpPagePath = '/application_security/dast/index#on-demand-scans';
const projectPath = 'group/project'; const projectPath = 'group/project';
const defaultBranch = 'main'; const defaultBranch = 'main';
...@@ -52,7 +52,6 @@ useLocalStorageSpy(); ...@@ -52,7 +52,6 @@ useLocalStorageSpy();
jest.mock('~/lib/utils/url_utility', () => ({ jest.mock('~/lib/utils/url_utility', () => ({
isAbsolute: jest.requireActual('~/lib/utils/url_utility').isAbsolute, isAbsolute: jest.requireActual('~/lib/utils/url_utility').isAbsolute,
queryToObject: jest.requireActual('~/lib/utils/url_utility').queryToObject, queryToObject: jest.requireActual('~/lib/utils/url_utility').queryToObject,
setUrlParams: jest.requireActual('~/lib/utils/url_utility').setUrlParams,
redirectTo: jest.fn(), redirectTo: jest.fn(),
})); }));
...@@ -578,27 +577,21 @@ describe('OnDemandScansForm', () => { ...@@ -578,27 +577,21 @@ describe('OnDemandScansForm', () => {
const [scannerProfile] = scannerProfiles; const [scannerProfile] = scannerProfiles;
it('scanner profile', () => { it('scanner profile', () => {
global.jsdom.reconfigure({ setWindowLocation('?scanner_profile_id=1');
url: setUrlParams({ scanner_profile_id: 1 }, URL_HOST),
});
createShallowComponent(); createShallowComponent();
expect(wrapper.find(ScannerProfileSelector).attributes('value')).toBe(scannerProfile.id); expect(wrapper.find(ScannerProfileSelector).attributes('value')).toBe(scannerProfile.id);
}); });
it('site profile', () => { it('site profile', () => {
global.jsdom.reconfigure({ setWindowLocation('?site_profile_id=1');
url: setUrlParams({ site_profile_id: 1 }, URL_HOST),
});
createShallowComponent(); createShallowComponent();
expect(wrapper.find(SiteProfileSelector).attributes('value')).toBe(siteProfile.id); expect(wrapper.find(SiteProfileSelector).attributes('value')).toBe(siteProfile.id);
}); });
it('both scanner & site profile', () => { it('both scanner & site profile', () => {
global.jsdom.reconfigure({ setWindowLocation('?site_profile_id=1&scanner_profile_id=1');
url: setUrlParams({ site_profile_id: 1, scanner_profile_id: 1 }, URL_HOST),
});
createShallowComponent(); createShallowComponent();
expect(wrapper.find(SiteProfileSelector).attributes('value')).toBe(siteProfile.id); expect(wrapper.find(SiteProfileSelector).attributes('value')).toBe(siteProfile.id);
...@@ -614,10 +607,6 @@ describe('OnDemandScansForm', () => { ...@@ -614,10 +607,6 @@ describe('OnDemandScansForm', () => {
}), }),
); );
global.jsdom.reconfigure({
url: setUrlParams({ site_profile_id: 1, scanner_profile_id: 1 }, URL_HOST),
});
createShallowComponent(); createShallowComponent();
await wrapper.vm.$nextTick(); await wrapper.vm.$nextTick();
......
...@@ -8,9 +8,9 @@ import SiteProfileSelector from 'ee/on_demand_scans/components/profile_selector/ ...@@ -8,9 +8,9 @@ import SiteProfileSelector from 'ee/on_demand_scans/components/profile_selector/
import dastScannerProfilesQuery from 'ee/security_configuration/dast_profiles/graphql/dast_scanner_profiles.query.graphql'; import dastScannerProfilesQuery from 'ee/security_configuration/dast_profiles/graphql/dast_scanner_profiles.query.graphql';
import dastSiteProfilesQuery from 'ee/security_configuration/dast_profiles/graphql/dast_site_profiles.query.graphql'; import dastSiteProfilesQuery from 'ee/security_configuration/dast_profiles/graphql/dast_site_profiles.query.graphql';
import createApolloProvider from 'helpers/mock_apollo_helper'; import createApolloProvider from 'helpers/mock_apollo_helper';
import setWindowLocation from 'helpers/set_window_location_helper';
import { shallowMountExtended } from 'helpers/vue_test_utils_helper'; import { shallowMountExtended } from 'helpers/vue_test_utils_helper';
import waitForPromises from 'helpers/wait_for_promises'; import waitForPromises from 'helpers/wait_for_promises';
import { setUrlParams } from '~/lib/utils/url_utility';
import * as responses from '../../mocks/apollo_mocks'; import * as responses from '../../mocks/apollo_mocks';
import { scannerProfiles, siteProfiles } from '../../mocks/mock_data'; import { scannerProfiles, siteProfiles } from '../../mocks/mock_data';
...@@ -21,6 +21,10 @@ const fullPath = '/project/path'; ...@@ -21,6 +21,10 @@ const fullPath = '/project/path';
const [passiveScannerProfile, activeScannerProfile] = scannerProfiles; const [passiveScannerProfile, activeScannerProfile] = scannerProfiles;
const [nonValidatedSiteProfile, validatedSiteProfile] = siteProfiles; const [nonValidatedSiteProfile, validatedSiteProfile] = siteProfiles;
beforeEach(() => {
setWindowLocation(URL_HOST);
});
describe('EE - DAST Profiles Selector', () => { describe('EE - DAST Profiles Selector', () => {
let wrapper; let wrapper;
let localVue; let localVue;
...@@ -179,27 +183,21 @@ describe('EE - DAST Profiles Selector', () => { ...@@ -179,27 +183,21 @@ describe('EE - DAST Profiles Selector', () => {
const [scannerProfile] = scannerProfiles; const [scannerProfile] = scannerProfiles;
it('scanner profile', () => { it('scanner profile', () => {
global.jsdom.reconfigure({ setWindowLocation(`?scanner_profile_id=1`);
url: setUrlParams({ scanner_profile_id: 1 }, URL_HOST),
});
createComponent(); createComponent();
expect(findScannerProfilesSelector().attributes('value')).toBe(scannerProfile.id); expect(findScannerProfilesSelector().attributes('value')).toBe(scannerProfile.id);
}); });
it('site profile', () => { it('site profile', () => {
global.jsdom.reconfigure({ setWindowLocation(`?site_profile_id=1`);
url: setUrlParams({ site_profile_id: 1 }, URL_HOST),
});
createComponent(); createComponent();
expect(findSiteProfilesSelector().attributes('value')).toBe(siteProfile.id); expect(findSiteProfilesSelector().attributes('value')).toBe(siteProfile.id);
}); });
it('both scanner & site profile', () => { it('both scanner & site profile', () => {
global.jsdom.reconfigure({ setWindowLocation(`?site_profile_id=1&scanner_profile_id=1`);
url: setUrlParams({ site_profile_id: 1, scanner_profile_id: 1 }, URL_HOST),
});
createComponent(); createComponent();
expect(wrapper.find(SiteProfileSelector).attributes('value')).toBe(siteProfile.id); expect(wrapper.find(SiteProfileSelector).attributes('value')).toBe(siteProfile.id);
......
...@@ -10,6 +10,7 @@ import { addScheduleModalId } from 'ee/oncall_schedules/components/oncall_schedu ...@@ -10,6 +10,7 @@ import { addScheduleModalId } from 'ee/oncall_schedules/components/oncall_schedu
import updateOncallScheduleMutation from 'ee/oncall_schedules/graphql/mutations/update_oncall_schedule.mutation.graphql'; import updateOncallScheduleMutation from 'ee/oncall_schedules/graphql/mutations/update_oncall_schedule.mutation.graphql';
import getOncallSchedulesWithRotationsQuery from 'ee/oncall_schedules/graphql/queries/get_oncall_schedules.query.graphql'; import getOncallSchedulesWithRotationsQuery from 'ee/oncall_schedules/graphql/queries/get_oncall_schedules.query.graphql';
import createMockApollo from 'helpers/mock_apollo_helper'; import createMockApollo from 'helpers/mock_apollo_helper';
import { useMockLocationHelper } from 'helpers/mock_window_location_helper';
import waitForPromises from 'helpers/wait_for_promises'; import waitForPromises from 'helpers/wait_for_promises';
import { import {
getOncallSchedulesQueryResponse, getOncallSchedulesQueryResponse,
...@@ -250,19 +251,7 @@ describe('AddScheduleModal', () => { ...@@ -250,19 +251,7 @@ describe('AddScheduleModal', () => {
}); });
describe('when the schedule timezone is updated', () => { describe('when the schedule timezone is updated', () => {
const { location } = window; useMockLocationHelper();
beforeEach(() => {
delete window.location;
window.location = {
reload: jest.fn(),
hash: location.hash,
};
});
afterEach(() => {
window.location = location;
});
it('it should not reload the page if the timezone has not changed', async () => { it('it should not reload the page if the timezone has not changed', async () => {
mutate.mockResolvedValueOnce({}); mutate.mockResolvedValueOnce({});
......
...@@ -5,12 +5,17 @@ import { merge } from 'lodash'; ...@@ -5,12 +5,17 @@ import { merge } from 'lodash';
import DastFailedSiteValidations from 'ee/security_configuration/dast_profiles/components/dast_failed_site_validations.vue'; import DastFailedSiteValidations from 'ee/security_configuration/dast_profiles/components/dast_failed_site_validations.vue';
import DastProfiles from 'ee/security_configuration/dast_profiles/components/dast_profiles.vue'; import DastProfiles from 'ee/security_configuration/dast_profiles/components/dast_profiles.vue';
import setWindowLocation from 'helpers/set_window_location_helper'; import setWindowLocation from 'helpers/set_window_location_helper';
import { TEST_HOST } from 'helpers/test_constants';
const TEST_NEW_DAST_SAVED_SCAN_PATH = '/-/on_demand_scans/new'; const TEST_NEW_DAST_SAVED_SCAN_PATH = '/-/on_demand_scans/new';
const TEST_NEW_DAST_SCANNER_PROFILE_PATH = '/-/on_demand_scans/scanner_profiles/new'; const TEST_NEW_DAST_SCANNER_PROFILE_PATH = '/-/on_demand_scans/scanner_profiles/new';
const TEST_NEW_DAST_SITE_PROFILE_PATH = '/-/on_demand_scans/site_profiles/new'; const TEST_NEW_DAST_SITE_PROFILE_PATH = '/-/on_demand_scans/site_profiles/new';
const TEST_PROJECT_FULL_PATH = '/namespace/project'; const TEST_PROJECT_FULL_PATH = '/namespace/project';
beforeEach(() => {
setWindowLocation(TEST_HOST);
});
describe('EE - DastProfiles', () => { describe('EE - DastProfiles', () => {
let wrapper; let wrapper;
...@@ -115,8 +120,6 @@ describe('EE - DastProfiles', () => { ...@@ -115,8 +120,6 @@ describe('EE - DastProfiles', () => {
}); });
describe('tabs', () => { describe('tabs', () => {
const originalLocation = window.location;
describe('without location hash set', () => { describe('without location hash set', () => {
beforeEach(() => { beforeEach(() => {
createFullComponent(); createFullComponent();
...@@ -148,19 +151,15 @@ describe('EE - DastProfiles', () => { ...@@ -148,19 +151,15 @@ describe('EE - DastProfiles', () => {
describe.each` describe.each`
tabName | index | givenLocationHash tabName | index | givenLocationHash
${'Saved Scans'} | ${0} | ${'saved-scans'} ${'Saved Scans'} | ${0} | ${'#saved-scans'}
${'Site Profiles'} | ${1} | ${'site-profiles'} ${'Site Profiles'} | ${1} | ${'#site-profiles'}
${'Scanner Profiles'} | ${2} | ${'scanner-profiles'} ${'Scanner Profiles'} | ${2} | ${'#scanner-profiles'}
`('with location hash set to "$givenLocationHash"', ({ tabName, index, givenLocationHash }) => { `('with location hash set to "$givenLocationHash"', ({ tabName, index, givenLocationHash }) => {
beforeEach(() => { beforeEach(() => {
setWindowLocation(`http://foo.com/index#${givenLocationHash}`); setWindowLocation(givenLocationHash);
createFullComponent(); createFullComponent();
}); });
afterEach(() => {
window.location = originalLocation;
});
it(`has "${tabName}" selected`, () => { it(`has "${tabName}" selected`, () => {
const tab = getTab({ const tab = getTab({
tabName, tabName,
...@@ -171,7 +170,8 @@ describe('EE - DastProfiles', () => { ...@@ -171,7 +170,8 @@ describe('EE - DastProfiles', () => {
}); });
it('updates the browsers URL to contain the selected tab', () => { it('updates the browsers URL to contain the selected tab', () => {
window.location.hash = ''; setWindowLocation('#');
expect(window.location.hash).toBe('');
getTabsComponent().vm.$emit('input', index); getTabsComponent().vm.$emit('input', index);
......
import initSecurityDashboard from 'ee/security_dashboard/security_dashboard_init'; import initSecurityDashboard from 'ee/security_dashboard/security_dashboard_init';
import { DASHBOARD_TYPES } from 'ee/security_dashboard/store/constants'; import { DASHBOARD_TYPES } from 'ee/security_dashboard/store/constants';
import setWindowLocation from 'helpers/set_window_location_helper';
import { TEST_HOST } from 'helpers/test_constants'; import { TEST_HOST } from 'helpers/test_constants';
const EMPTY_DIV = document.createElement('div'); const EMPTY_DIV = document.createElement('div');
...@@ -19,9 +20,7 @@ describe('Security Dashboard', () => { ...@@ -19,9 +20,7 @@ describe('Security Dashboard', () => {
root = document.createElement('div'); root = document.createElement('div');
document.body.appendChild(root); document.body.appendChild(root);
global.jsdom.reconfigure({ setWindowLocation(`${TEST_HOST}/-/security/dashboard`);
url: `${TEST_HOST}/-/security/dashboard`,
});
}); });
afterEach(() => { afterEach(() => {
......
import { DASHBOARD_TYPES } from 'ee/security_dashboard/store/constants'; import { DASHBOARD_TYPES } from 'ee/security_dashboard/store/constants';
import initVulnerabilityReport from 'ee/security_dashboard/vulnerability_report_init'; import initVulnerabilityReport from 'ee/security_dashboard/vulnerability_report_init';
import setWindowLocation from 'helpers/set_window_location_helper';
import { TEST_HOST } from 'helpers/test_constants'; import { TEST_HOST } from 'helpers/test_constants';
const EMPTY_DIV = document.createElement('div'); const EMPTY_DIV = document.createElement('div');
...@@ -25,9 +26,7 @@ describe('Vulnerability Report', () => { ...@@ -25,9 +26,7 @@ describe('Vulnerability Report', () => {
root = document.createElement('div'); root = document.createElement('div');
document.body.appendChild(root); document.body.appendChild(root);
global.jsdom.reconfigure({ setWindowLocation(`${TEST_HOST}/-/security/vulnerabilities`);
url: `${TEST_HOST}/-/security/vulnerabilities`,
});
}); });
afterEach(() => { afterEach(() => {
......
/** /**
* setWindowLocation allows for setting `window.location` * setWindowLocation allows for setting `window.location` within Jest.
* (doing so directly is causing an error in jsdom)
* *
* Example usage: * The jsdom environment at the time of writing does not support changing the
* assert(window.location.hash === undefined); * current location (see
* setWindowLocation('http://example.com#foo') * https://github.com/jsdom/jsdom/blob/16.4.0/lib/jsdom/living/window/navigation.js#L76),
* assert(window.location.hash === '#foo'); * hence this helper.
* *
* More information: * This helper mutates the current `window.location` very similarly to how
* https://github.com/facebook/jest/issues/890 * a direct assignment to `window.location.href` would in a browser (but
* without the navigation/reload behaviour). For instance:
* *
* @param url * - Set the full href by passing an absolute URL, e.g.:
*
* setWindowLocation('https://gdk.test');
* // window.location.href is now 'https://gdk.test'
*
* - Set the path, search and/or hash components by passing a relative URL:
*
* setWindowLocation('/foo/bar');
* // window.location.href is now 'http://test.host/foo/bar'
*
* setWindowLocation('?foo=bar');
* // window.location.href is now 'http://test.host/?foo=bar'
*
* setWindowLocation('#foo');
* // window.location.href is now 'http://test.host/#foo'
*
* setWindowLocation('/a/b/foo.html?bar=1#qux');
* // window.location.href is now 'http://test.host/a/b/foo.html?bar=1#qux
*
* Both approaches also automatically update the rest of the properties on
* `window.locaton`. For instance:
*
* setWindowLocation('http://test.host/a/b/foo.html?bar=1#qux');
* // window.location.origin is now 'http://test.host'
* // window.location.pathname is now '/a/b/foo.html'
* // window.location.search is now '?bar=1'
* // window.location.searchParams is now { bar: 1 }
* // window.location.hash is now '#qux'
*
* @param {string} url A string representing an absolute or relative URL.
* @returns {undefined}
*/ */
export default function setWindowLocation(url) { export default function setWindowLocation(url) {
const parsedUrl = new URL(url); if (typeof url !== 'string') {
throw new TypeError(`Expected string; got ${url} (${typeof url})`);
}
const newLocationValue = [ const newUrl = new URL(url, window.location.href);
'hash',
'host',
'hostname',
'href',
'origin',
'pathname',
'port',
'protocol',
'search',
].reduce(
(location, prop) => ({
...location,
[prop]: parsedUrl[prop],
}),
{},
);
Object.defineProperty(window, 'location', { global.jsdom.reconfigure({ url: newUrl.href });
value: newLocationValue,
writable: true,
});
} }
import setWindowLocation from './set_window_location_helper'; import setWindowLocation from './set_window_location_helper';
describe('setWindowLocation', () => { describe('helpers/set_window_location_helper', () => {
const originalLocation = window.location; const originalLocation = window.location.href;
afterEach(() => { beforeEach(() => {
window.location = originalLocation; setWindowLocation(originalLocation);
}); });
describe('setWindowLocation', () => {
describe('given a complete URL', () => {
it.each` it.each`
url | property | value url | property | value
${'https://gitlab.com#foo'} | ${'hash'} | ${'#foo'} ${'https://gitlab.com#foo'} | ${'hash'} | ${'#foo'}
...@@ -16,25 +18,89 @@ describe('setWindowLocation', () => { ...@@ -16,25 +18,89 @@ describe('setWindowLocation', () => {
${'http://gitlab.com'} | ${'origin'} | ${'http://gitlab.com'} ${'http://gitlab.com'} | ${'origin'} | ${'http://gitlab.com'}
${'http://gitlab.com/foo/bar/baz'} | ${'pathname'} | ${'/foo/bar/baz'} ${'http://gitlab.com/foo/bar/baz'} | ${'pathname'} | ${'/foo/bar/baz'}
${'https://gitlab.com'} | ${'protocol'} | ${'https:'} ${'https://gitlab.com'} | ${'protocol'} | ${'https:'}
${'http://gitlab.com#foo'} | ${'protocol'} | ${'http:'} ${'ftp://gitlab.com#foo'} | ${'protocol'} | ${'ftp:'}
${'http://gitlab.com:8080'} | ${'port'} | ${'8080'} ${'http://gitlab.com:8080'} | ${'port'} | ${'8080'}
${'http://gitlab.com?foo=bar&bar=foo'} | ${'search'} | ${'?foo=bar&bar=foo'} ${'http://gitlab.com?foo=bar&bar=foo'} | ${'search'} | ${'?foo=bar&bar=foo'}
`( `(
'sets "window.location.$property" to be "$value" when called with: "$url"', 'sets "window.location.$property" to be "$value" when called with: "$url"',
({ url, property, value }) => { ({ url, property, value }) => {
expect(window.location).toBe(originalLocation); expect(window.location.href).toBe(originalLocation);
setWindowLocation(url); setWindowLocation(url);
expect(window.location[property]).toBe(value); expect(window.location[property]).toBe(value);
}, },
); );
});
describe('given a partial URL', () => {
it.each`
partialURL | href
${'//foo.test:3000/'} | ${'http://foo.test:3000/'}
${'/foo/bar'} | ${`${originalLocation}foo/bar`}
${'foo/bar'} | ${`${originalLocation}foo/bar`}
${'?foo=bar'} | ${`${originalLocation}?foo=bar`}
${'#a-thing'} | ${`${originalLocation}#a-thing`}
`('$partialURL sets location.href to $href', ({ partialURL, href }) => {
expect(window.location.href).toBe(originalLocation);
setWindowLocation(partialURL);
expect(window.location.href).toBe(href);
});
});
it.each([null, 1, undefined, false, '', 'gitlab.com'])( describe('relative path', () => {
describe.each`
initialHref | path | newHref
${'https://gdk.test/foo/bar'} | ${'/qux'} | ${'https://gdk.test/qux'}
${'https://gdk.test/foo/bar/'} | ${'/qux'} | ${'https://gdk.test/qux'}
${'https://gdk.test/foo/bar'} | ${'qux'} | ${'https://gdk.test/foo/qux'}
${'https://gdk.test/foo/bar/'} | ${'qux'} | ${'https://gdk.test/foo/bar/qux'}
${'https://gdk.test/foo/bar'} | ${'../qux'} | ${'https://gdk.test/qux'}
${'https://gdk.test/foo/bar/'} | ${'../qux'} | ${'https://gdk.test/foo/qux'}
`('when location is $initialHref', ({ initialHref, path, newHref }) => {
beforeEach(() => {
setWindowLocation(initialHref);
});
it(`${path} sets window.location.href to ${newHref}`, () => {
expect(window.location.href).toBe(initialHref);
setWindowLocation(path);
expect(window.location.href).toBe(newHref);
});
});
});
it.each([null, 1, undefined, false, 'https://', 'https:', { foo: 1 }, []])(
'throws an error when called with an invalid url: "%s"', 'throws an error when called with an invalid url: "%s"',
(invalidUrl) => { (invalidUrl) => {
expect(() => setWindowLocation(invalidUrl)).toThrow(/Invalid URL/); expect(() => setWindowLocation(invalidUrl)).toThrow();
expect(window.location).toBe(originalLocation); expect(window.location.href).toBe(originalLocation);
},
);
describe('affects links', () => {
it.each`
url | hrefAttr | expectedHref
${'http://gitlab.com/'} | ${'foo'} | ${'http://gitlab.com/foo'}
${'http://gitlab.com/bar/'} | ${'foo'} | ${'http://gitlab.com/bar/foo'}
${'http://gitlab.com/bar/'} | ${'/foo'} | ${'http://gitlab.com/foo'}
${'http://gdk.test:3000/?foo=bar'} | ${'?qux=1'} | ${'http://gdk.test:3000/?qux=1'}
${'https://gdk.test:3000/?foo=bar'} | ${'//other.test'} | ${'https://other.test/'}
`(
'given $url, <a href="$hrefAttr"> points to $expectedHref',
({ url, hrefAttr, expectedHref }) => {
setWindowLocation(url);
const link = document.createElement('a');
link.setAttribute('href', hrefAttr);
expect(link.href).toBe(expectedHref);
}, },
); );
});
});
}); });
import { mount, createLocalVue } from '@vue/test-utils'; import { mount, createLocalVue } from '@vue/test-utils';
import Vuex from 'vuex'; import Vuex from 'vuex';
import setWindowLocation from 'helpers/set_window_location_helper';
import { TEST_HOST } from 'helpers/test_constants'; import { TEST_HOST } from 'helpers/test_constants';
import { trimText } from 'helpers/text_helper'; import { trimText } from 'helpers/text_helper';
import CompareVersionsComponent from '~/diffs/components/compare_versions.vue'; import CompareVersionsComponent from '~/diffs/components/compare_versions.vue';
...@@ -13,6 +14,10 @@ localVue.use(Vuex); ...@@ -13,6 +14,10 @@ localVue.use(Vuex);
const NEXT_COMMIT_URL = `${TEST_HOST}/?commit_id=next`; const NEXT_COMMIT_URL = `${TEST_HOST}/?commit_id=next`;
const PREV_COMMIT_URL = `${TEST_HOST}/?commit_id=prev`; const PREV_COMMIT_URL = `${TEST_HOST}/?commit_id=prev`;
beforeEach(() => {
setWindowLocation(TEST_HOST);
});
describe('CompareVersions', () => { describe('CompareVersions', () => {
let wrapper; let wrapper;
let store; let store;
...@@ -215,15 +220,7 @@ describe('CompareVersions', () => { ...@@ -215,15 +220,7 @@ describe('CompareVersions', () => {
describe('prev commit', () => { describe('prev commit', () => {
beforeAll(() => { beforeAll(() => {
global.jsdom.reconfigure({ setWindowLocation(`${TEST_HOST}?commit_id=${mrCommit.id}`);
url: `${TEST_HOST}?commit_id=${mrCommit.id}`,
});
});
afterAll(() => {
global.jsdom.reconfigure({
url: TEST_HOST,
});
}); });
beforeEach(() => { beforeEach(() => {
...@@ -258,15 +255,7 @@ describe('CompareVersions', () => { ...@@ -258,15 +255,7 @@ describe('CompareVersions', () => {
describe('next commit', () => { describe('next commit', () => {
beforeAll(() => { beforeAll(() => {
global.jsdom.reconfigure({ setWindowLocation(`${TEST_HOST}?commit_id=${mrCommit.id}`);
url: `${TEST_HOST}?commit_id=${mrCommit.id}`,
});
});
afterAll(() => {
global.jsdom.reconfigure({
url: TEST_HOST,
});
}); });
beforeEach(() => { beforeEach(() => {
......
import setWindowLocation from 'helpers/set_window_location_helper';
import { import {
DIFF_COMPARE_BASE_VERSION_INDEX, DIFF_COMPARE_BASE_VERSION_INDEX,
DIFF_COMPARE_HEAD_VERSION_INDEX, DIFF_COMPARE_HEAD_VERSION_INDEX,
...@@ -47,15 +48,12 @@ describe('Compare diff version dropdowns', () => { ...@@ -47,15 +48,12 @@ describe('Compare diff version dropdowns', () => {
let expectedFirstVersion; let expectedFirstVersion;
let expectedBaseVersion; let expectedBaseVersion;
let expectedHeadVersion; let expectedHeadVersion;
const originalLocation = window.location; const originalLocation = window.location.href;
const setupTest = (includeDiffHeadParam) => { const setupTest = (includeDiffHeadParam) => {
const diffHeadParam = includeDiffHeadParam ? '?diff_head=true' : ''; const diffHeadParam = includeDiffHeadParam ? '?diff_head=true' : '';
Object.defineProperty(window, 'location', { setWindowLocation(diffHeadParam);
writable: true,
value: { search: diffHeadParam },
});
expectedFirstVersion = { expectedFirstVersion = {
...diffsMockData[1], ...diffsMockData[1],
...@@ -91,7 +89,7 @@ describe('Compare diff version dropdowns', () => { ...@@ -91,7 +89,7 @@ describe('Compare diff version dropdowns', () => {
}; };
afterEach(() => { afterEach(() => {
window.location = originalLocation; setWindowLocation(originalLocation);
}); });
it('base version selected', () => { it('base version selected', () => {
......
...@@ -6,6 +6,7 @@ import { ...@@ -6,6 +6,7 @@ import {
import { shallowMount } from '@vue/test-utils'; import { shallowMount } from '@vue/test-utils';
import axios from 'axios'; import axios from 'axios';
import MockAdapter from 'axios-mock-adapter'; import MockAdapter from 'axios-mock-adapter';
import setWindowLocation from 'helpers/set_window_location_helper';
import { TEST_HOST } from 'helpers/test_constants'; import { TEST_HOST } from 'helpers/test_constants';
import waitForPromises from 'helpers/wait_for_promises'; import waitForPromises from 'helpers/wait_for_promises';
import createFlash from '~/flash'; import createFlash from '~/flash';
...@@ -27,11 +28,6 @@ const TEST_ENDPOINT = '/issues'; ...@@ -27,11 +28,6 @@ const TEST_ENDPOINT = '/issues';
const TEST_CREATE_ISSUES_PATH = '/createIssue'; const TEST_CREATE_ISSUES_PATH = '/createIssue';
const TEST_SVG_PATH = '/emptySvg'; const TEST_SVG_PATH = '/emptySvg';
const setUrl = (query) => {
window.location.href = `${TEST_LOCATION}${query}`;
window.location.search = query;
};
const MOCK_ISSUES = Array(PAGE_SIZE_MANUAL) const MOCK_ISSUES = Array(PAGE_SIZE_MANUAL)
.fill(0) .fill(0)
.map((_, i) => ({ .map((_, i) => ({
...@@ -40,7 +36,6 @@ const MOCK_ISSUES = Array(PAGE_SIZE_MANUAL) ...@@ -40,7 +36,6 @@ const MOCK_ISSUES = Array(PAGE_SIZE_MANUAL)
})); }));
describe('Issuables list component', () => { describe('Issuables list component', () => {
let oldLocation;
let mockAxios; let mockAxios;
let wrapper; let wrapper;
let apiSpy; let apiSpy;
...@@ -75,19 +70,13 @@ describe('Issuables list component', () => { ...@@ -75,19 +70,13 @@ describe('Issuables list component', () => {
beforeEach(() => { beforeEach(() => {
mockAxios = new MockAdapter(axios); mockAxios = new MockAdapter(axios);
oldLocation = window.location; setWindowLocation(TEST_LOCATION);
Object.defineProperty(window, 'location', {
writable: true,
value: { href: '', search: '' },
});
window.location.href = TEST_LOCATION;
}); });
afterEach(() => { afterEach(() => {
wrapper.destroy(); wrapper.destroy();
wrapper = null; wrapper = null;
mockAxios.restore(); mockAxios.restore();
window.location = oldLocation;
}); });
describe('with failed issues response', () => { describe('with failed issues response', () => {
...@@ -314,7 +303,7 @@ describe('Issuables list component', () => { ...@@ -314,7 +303,7 @@ describe('Issuables list component', () => {
'?assignee_username=root&author_username=root&confidential=yes&label_name%5B%5D=Aquapod&label_name%5B%5D=Astro&milestone_title=v3.0&my_reaction_emoji=airplane&scope=all&sort=priority&state=opened&weight=0&not[label_name][]=Afterpod&not[milestone_title][]=13'; '?assignee_username=root&author_username=root&confidential=yes&label_name%5B%5D=Aquapod&label_name%5B%5D=Astro&milestone_title=v3.0&my_reaction_emoji=airplane&scope=all&sort=priority&state=opened&weight=0&not[label_name][]=Afterpod&not[milestone_title][]=13';
beforeEach(() => { beforeEach(() => {
setUrl(query); setWindowLocation(query);
setupApiMock(() => [200, MOCK_ISSUES.slice(0)]); setupApiMock(() => [200, MOCK_ISSUES.slice(0)]);
factory({ sortKey: 'milestone_due_desc' }); factory({ sortKey: 'milestone_due_desc' });
...@@ -358,7 +347,7 @@ describe('Issuables list component', () => { ...@@ -358,7 +347,7 @@ describe('Issuables list component', () => {
'?assignee_username=root&author_username=root&confidential=yes&label_name%5B%5D=Aquapod&label_name%5B%5D=Astro&milestone_title=v3.0&my_reaction_emoji=airplane&scope=all&sort=priority&state=opened&weight=0&page=3'; '?assignee_username=root&author_username=root&confidential=yes&label_name%5B%5D=Aquapod&label_name%5B%5D=Astro&milestone_title=v3.0&my_reaction_emoji=airplane&scope=all&sort=priority&state=opened&weight=0&page=3';
beforeEach(() => { beforeEach(() => {
setUrl(query); setWindowLocation(query);
setupApiMock(() => [200, MOCK_ISSUES.slice(0)]); setupApiMock(() => [200, MOCK_ISSUES.slice(0)]);
factory({ sortKey: 'milestone_due_desc' }); factory({ sortKey: 'milestone_due_desc' });
...@@ -387,7 +376,7 @@ describe('Issuables list component', () => { ...@@ -387,7 +376,7 @@ describe('Issuables list component', () => {
describe('with hash in window.location', () => { describe('with hash in window.location', () => {
beforeEach(() => { beforeEach(() => {
window.location.href = `${TEST_LOCATION}#stuff`; setWindowLocation(`${TEST_LOCATION}#stuff`);
setupApiMock(() => [200, MOCK_ISSUES.slice(0)]); setupApiMock(() => [200, MOCK_ISSUES.slice(0)]);
factory(); factory();
return waitForPromises(); return waitForPromises();
...@@ -422,7 +411,7 @@ describe('Issuables list component', () => { ...@@ -422,7 +411,7 @@ describe('Issuables list component', () => {
describe('with query in window location', () => { describe('with query in window location', () => {
beforeEach(() => { beforeEach(() => {
window.location.search = '?weight=Any'; setWindowLocation('?weight=Any');
factory(); factory();
...@@ -436,7 +425,7 @@ describe('Issuables list component', () => { ...@@ -436,7 +425,7 @@ describe('Issuables list component', () => {
describe('with closed state', () => { describe('with closed state', () => {
beforeEach(() => { beforeEach(() => {
window.location.search = '?state=closed'; setWindowLocation('?state=closed');
factory(); factory();
...@@ -450,7 +439,7 @@ describe('Issuables list component', () => { ...@@ -450,7 +439,7 @@ describe('Issuables list component', () => {
describe('with all state', () => { describe('with all state', () => {
beforeEach(() => { beforeEach(() => {
window.location.search = '?state=all'; setWindowLocation('?state=all');
factory(); factory();
...@@ -565,7 +554,7 @@ describe('Issuables list component', () => { ...@@ -565,7 +554,7 @@ describe('Issuables list component', () => {
}); });
it('sets value according to query', () => { it('sets value according to query', () => {
setUrl(query); setWindowLocation(query);
factory({ type: 'jira' }); factory({ type: 'jira' });
...@@ -583,7 +572,7 @@ describe('Issuables list component', () => { ...@@ -583,7 +572,7 @@ describe('Issuables list component', () => {
it('sets value according to query', () => { it('sets value according to query', () => {
const query = '?search=free+text'; const query = '?search=free+text';
setUrl(query); setWindowLocation(query);
factory({ type: 'jira' }); factory({ type: 'jira' });
......
...@@ -7,6 +7,7 @@ import VueApollo from 'vue-apollo'; ...@@ -7,6 +7,7 @@ import VueApollo from 'vue-apollo';
import getIssuesQuery from 'ee_else_ce/issues_list/queries/get_issues.query.graphql'; import getIssuesQuery from 'ee_else_ce/issues_list/queries/get_issues.query.graphql';
import getIssuesCountQuery from 'ee_else_ce/issues_list/queries/get_issues_count.query.graphql'; import getIssuesCountQuery from 'ee_else_ce/issues_list/queries/get_issues_count.query.graphql';
import createMockApollo from 'helpers/mock_apollo_helper'; import createMockApollo from 'helpers/mock_apollo_helper';
import setWindowLocation from 'helpers/set_window_location_helper';
import { TEST_HOST } from 'helpers/test_constants'; import { TEST_HOST } from 'helpers/test_constants';
import waitForPromises from 'helpers/wait_for_promises'; import waitForPromises from 'helpers/wait_for_promises';
import { import {
...@@ -42,7 +43,6 @@ import eventHub from '~/issues_list/eventhub'; ...@@ -42,7 +43,6 @@ import eventHub from '~/issues_list/eventhub';
import { getSortOptions } from '~/issues_list/utils'; import { getSortOptions } from '~/issues_list/utils';
import axios from '~/lib/utils/axios_utils'; import axios from '~/lib/utils/axios_utils';
import { scrollUp } from '~/lib/utils/scroll_utils'; import { scrollUp } from '~/lib/utils/scroll_utils';
import { setUrlParams } from '~/lib/utils/url_utility';
jest.mock('~/flash'); jest.mock('~/flash');
jest.mock('~/lib/utils/scroll_utils', () => ({ jest.mock('~/lib/utils/scroll_utils', () => ({
...@@ -115,11 +115,11 @@ describe('IssuesListApp component', () => { ...@@ -115,11 +115,11 @@ describe('IssuesListApp component', () => {
}; };
beforeEach(() => { beforeEach(() => {
setWindowLocation(TEST_HOST);
axiosMock = new AxiosMockAdapter(axios); axiosMock = new AxiosMockAdapter(axios);
}); });
afterEach(() => { afterEach(() => {
global.jsdom.reconfigure({ url: TEST_HOST });
axiosMock.reset(); axiosMock.reset();
wrapper.destroy(); wrapper.destroy();
}); });
...@@ -186,7 +186,7 @@ describe('IssuesListApp component', () => { ...@@ -186,7 +186,7 @@ describe('IssuesListApp component', () => {
const search = '?search=refactor&sort=created_date&state=opened'; const search = '?search=refactor&sort=created_date&state=opened';
beforeEach(() => { beforeEach(() => {
global.jsdom.reconfigure({ url: `${TEST_HOST}${search}` }); setWindowLocation(search);
wrapper = mountComponent({ wrapper = mountComponent({
provide: { ...defaultProvide, isSignedIn: true }, provide: { ...defaultProvide, isSignedIn: true },
...@@ -258,7 +258,7 @@ describe('IssuesListApp component', () => { ...@@ -258,7 +258,7 @@ describe('IssuesListApp component', () => {
describe('initial url params', () => { describe('initial url params', () => {
describe('due_date', () => { describe('due_date', () => {
it('is set from the url params', () => { it('is set from the url params', () => {
global.jsdom.reconfigure({ url: `${TEST_HOST}?${PARAM_DUE_DATE}=${DUE_DATE_OVERDUE}` }); setWindowLocation(`?${PARAM_DUE_DATE}=${DUE_DATE_OVERDUE}`);
wrapper = mountComponent(); wrapper = mountComponent();
...@@ -268,7 +268,7 @@ describe('IssuesListApp component', () => { ...@@ -268,7 +268,7 @@ describe('IssuesListApp component', () => {
describe('search', () => { describe('search', () => {
it('is set from the url params', () => { it('is set from the url params', () => {
global.jsdom.reconfigure({ url: `${TEST_HOST}${locationSearch}` }); setWindowLocation(locationSearch);
wrapper = mountComponent(); wrapper = mountComponent();
...@@ -278,9 +278,7 @@ describe('IssuesListApp component', () => { ...@@ -278,9 +278,7 @@ describe('IssuesListApp component', () => {
describe('sort', () => { describe('sort', () => {
it.each(Object.keys(urlSortParams))('is set as %s from the url params', (sortKey) => { it.each(Object.keys(urlSortParams))('is set as %s from the url params', (sortKey) => {
global.jsdom.reconfigure({ setWindowLocation(`?sort=${urlSortParams[sortKey]}`);
url: setUrlParams({ sort: urlSortParams[sortKey] }, TEST_HOST),
});
wrapper = mountComponent(); wrapper = mountComponent();
...@@ -297,7 +295,7 @@ describe('IssuesListApp component', () => { ...@@ -297,7 +295,7 @@ describe('IssuesListApp component', () => {
it('is set from the url params', () => { it('is set from the url params', () => {
const initialState = IssuableStates.All; const initialState = IssuableStates.All;
global.jsdom.reconfigure({ url: setUrlParams({ state: initialState }, TEST_HOST) }); setWindowLocation(`?state=${initialState}`);
wrapper = mountComponent(); wrapper = mountComponent();
...@@ -307,7 +305,7 @@ describe('IssuesListApp component', () => { ...@@ -307,7 +305,7 @@ describe('IssuesListApp component', () => {
describe('filter tokens', () => { describe('filter tokens', () => {
it('is set from the url params', () => { it('is set from the url params', () => {
global.jsdom.reconfigure({ url: `${TEST_HOST}${locationSearch}` }); setWindowLocation(locationSearch);
wrapper = mountComponent(); wrapper = mountComponent();
...@@ -347,7 +345,7 @@ describe('IssuesListApp component', () => { ...@@ -347,7 +345,7 @@ describe('IssuesListApp component', () => {
describe('when there are issues', () => { describe('when there are issues', () => {
describe('when search returns no results', () => { describe('when search returns no results', () => {
beforeEach(() => { beforeEach(() => {
global.jsdom.reconfigure({ url: `${TEST_HOST}?search=no+results` }); setWindowLocation(`?search=no+results`);
wrapper = mountComponent({ provide: { hasProjectIssues: true }, mountFn: mount }); wrapper = mountComponent({ provide: { hasProjectIssues: true }, mountFn: mount });
}); });
...@@ -377,9 +375,7 @@ describe('IssuesListApp component', () => { ...@@ -377,9 +375,7 @@ describe('IssuesListApp component', () => {
describe('when "Closed" tab has no issues', () => { describe('when "Closed" tab has no issues', () => {
beforeEach(() => { beforeEach(() => {
global.jsdom.reconfigure({ setWindowLocation(`?state=${IssuableStates.Closed}`);
url: setUrlParams({ state: IssuableStates.Closed }, TEST_HOST),
});
wrapper = mountComponent({ provide: { hasProjectIssues: true }, mountFn: mount }); wrapper = mountComponent({ provide: { hasProjectIssues: true }, mountFn: mount });
}); });
......
import setWindowLocation from 'helpers/set_window_location_helper';
import { TEST_HOST } from 'helpers/test_constants'; import { TEST_HOST } from 'helpers/test_constants';
import * as urlUtils from '~/lib/utils/url_utility'; import * as urlUtils from '~/lib/utils/url_utility';
...@@ -16,24 +17,11 @@ const shas = { ...@@ -16,24 +17,11 @@ const shas = {
], ],
}; };
const setWindowLocation = (value) => { beforeEach(() => {
Object.defineProperty(window, 'location', { setWindowLocation(TEST_HOST);
writable: true, });
value,
});
};
describe('URL utility', () => { describe('URL utility', () => {
let originalLocation;
beforeAll(() => {
originalLocation = window.location;
});
afterAll(() => {
window.location = originalLocation;
});
describe('webIDEUrl', () => { describe('webIDEUrl', () => {
afterEach(() => { afterEach(() => {
gon.relative_url_root = ''; gon.relative_url_root = '';
...@@ -68,14 +56,7 @@ describe('URL utility', () => { ...@@ -68,14 +56,7 @@ describe('URL utility', () => {
describe('getParameterValues', () => { describe('getParameterValues', () => {
beforeEach(() => { beforeEach(() => {
setWindowLocation({ setWindowLocation('https://gitlab.com?test=passing&multiple=1&multiple=2');
href: 'https://gitlab.com?test=passing&multiple=1&multiple=2',
// make our fake location act like real window.location.toString
// URL() (used in getParameterValues) does this if passed an object
toString() {
return this.href;
},
});
}); });
it('returns empty array for no params', () => { it('returns empty array for no params', () => {
...@@ -330,9 +311,7 @@ describe('URL utility', () => { ...@@ -330,9 +311,7 @@ describe('URL utility', () => {
describe('doesHashExistInUrl', () => { describe('doesHashExistInUrl', () => {
beforeEach(() => { beforeEach(() => {
setWindowLocation({ setWindowLocation('#note_1');
hash: 'https://gitlab.com/gitlab-org/gitlab-test/issues/1#note_1',
});
}); });
it('should return true when the given string exists in the URL hash', () => { it('should return true when the given string exists in the URL hash', () => {
...@@ -442,10 +421,7 @@ describe('URL utility', () => { ...@@ -442,10 +421,7 @@ describe('URL utility', () => {
describe('getBaseURL', () => { describe('getBaseURL', () => {
beforeEach(() => { beforeEach(() => {
setWindowLocation({ setWindowLocation('https://gitlab.com');
protocol: 'https:',
host: 'gitlab.com',
});
}); });
it('returns correct base URL', () => { it('returns correct base URL', () => {
...@@ -637,10 +613,7 @@ describe('URL utility', () => { ...@@ -637,10 +613,7 @@ describe('URL utility', () => {
${'http:'} | ${'ws:'} ${'http:'} | ${'ws:'}
${'https:'} | ${'wss:'} ${'https:'} | ${'wss:'}
`('returns "$expectation" with "$protocol" protocol', ({ protocol, expectation }) => { `('returns "$expectation" with "$protocol" protocol', ({ protocol, expectation }) => {
setWindowLocation({ setWindowLocation(`${protocol}//example.com`);
protocol,
host: 'example.com',
});
expect(urlUtils.getWebSocketProtocol()).toEqual(expectation); expect(urlUtils.getWebSocketProtocol()).toEqual(expectation);
}); });
...@@ -648,10 +621,7 @@ describe('URL utility', () => { ...@@ -648,10 +621,7 @@ describe('URL utility', () => {
describe('getWebSocketUrl', () => { describe('getWebSocketUrl', () => {
it('joins location host to path', () => { it('joins location host to path', () => {
setWindowLocation({ setWindowLocation('http://example.com');
protocol: 'http:',
host: 'example.com',
});
const path = '/lorem/ipsum?a=bc'; const path = '/lorem/ipsum?a=bc';
...@@ -724,32 +694,32 @@ describe('URL utility', () => { ...@@ -724,32 +694,32 @@ describe('URL utility', () => {
const { getParameterByName } = urlUtils; const { getParameterByName } = urlUtils;
it('should return valid parameter', () => { it('should return valid parameter', () => {
setWindowLocation({ search: '?scope=all&p=2' }); setWindowLocation('?scope=all&p=2');
expect(getParameterByName('p')).toEqual('2'); expect(getParameterByName('p')).toEqual('2');
expect(getParameterByName('scope')).toBe('all'); expect(getParameterByName('scope')).toBe('all');
}); });
it('should return invalid parameter', () => { it('should return invalid parameter', () => {
setWindowLocation({ search: '?scope=all&p=2' }); setWindowLocation('?scope=all&p=2');
expect(getParameterByName('fakeParameter')).toBe(null); expect(getParameterByName('fakeParameter')).toBe(null);
}); });
it('should return a parameter with spaces', () => { it('should return a parameter with spaces', () => {
setWindowLocation({ search: '?search=my terms' }); setWindowLocation('?search=my terms');
expect(getParameterByName('search')).toBe('my terms'); expect(getParameterByName('search')).toBe('my terms');
}); });
it('should return a parameter with encoded spaces', () => { it('should return a parameter with encoded spaces', () => {
setWindowLocation({ search: '?search=my%20terms' }); setWindowLocation('?search=my%20terms');
expect(getParameterByName('search')).toBe('my terms'); expect(getParameterByName('search')).toBe('my terms');
}); });
it('should return a parameter with plus signs as spaces', () => { it('should return a parameter with plus signs as spaces', () => {
setWindowLocation({ search: '?search=my+terms' }); setWindowLocation('?search=my+terms');
expect(getParameterByName('search')).toBe('my terms'); expect(getParameterByName('search')).toBe('my terms');
}); });
...@@ -842,18 +812,20 @@ describe('URL utility', () => { ...@@ -842,18 +812,20 @@ describe('URL utility', () => {
}); });
describe('urlIsDifferent', () => { describe('urlIsDifferent', () => {
const current = 'http://current.test/';
beforeEach(() => { beforeEach(() => {
setWindowLocation('current'); setWindowLocation(current);
}); });
it('should compare against the window location if no compare value is provided', () => { it('should compare against the window location if no compare value is provided', () => {
expect(urlUtils.urlIsDifferent('different')).toBeTruthy(); expect(urlUtils.urlIsDifferent('different')).toBeTruthy();
expect(urlUtils.urlIsDifferent('current')).toBeFalsy(); expect(urlUtils.urlIsDifferent(current)).toBeFalsy();
}); });
it('should use the provided compare value', () => { it('should use the provided compare value', () => {
expect(urlUtils.urlIsDifferent('different', 'current')).toBeTruthy(); expect(urlUtils.urlIsDifferent('different', current)).toBeTruthy();
expect(urlUtils.urlIsDifferent('current', 'current')).toBeFalsy(); expect(urlUtils.urlIsDifferent(current, current)).toBeFalsy();
}); });
}); });
...@@ -944,9 +916,8 @@ describe('URL utility', () => { ...@@ -944,9 +916,8 @@ describe('URL utility', () => {
it.each([[httpProtocol], [httpsProtocol]])( it.each([[httpProtocol], [httpsProtocol]])(
'when no url passed, returns correct protocol for %i from window location', 'when no url passed, returns correct protocol for %i from window location',
(protocol) => { (protocol) => {
setWindowLocation({ setWindowLocation(`${protocol}//test.host`);
protocol,
});
expect(urlUtils.getHTTPProtocol()).toBe(protocol.slice(0, -1)); expect(urlUtils.getHTTPProtocol()).toBe(protocol.slice(0, -1));
}, },
); );
...@@ -979,10 +950,8 @@ describe('URL utility', () => { ...@@ -979,10 +950,8 @@ describe('URL utility', () => {
describe('getURLOrigin', () => { describe('getURLOrigin', () => {
it('when no url passed, returns correct origin from window location', () => { it('when no url passed, returns correct origin from window location', () => {
const origin = 'https://foo.bar'; setWindowLocation('https://user:pass@origin.test:1234/foo/bar?foo=1#bar');
expect(urlUtils.getURLOrigin()).toBe('https://origin.test:1234');
setWindowLocation({ origin });
expect(urlUtils.getURLOrigin()).toBe(origin);
}); });
it.each` it.each`
...@@ -1032,10 +1001,6 @@ describe('URL utility', () => { ...@@ -1032,10 +1001,6 @@ describe('URL utility', () => {
// eslint-disable-next-line no-script-url // eslint-disable-next-line no-script-url
const javascriptUrl = 'javascript:alert(1)'; const javascriptUrl = 'javascript:alert(1)';
beforeEach(() => {
setWindowLocation({ origin: TEST_HOST });
});
it.each` it.each`
url | expected url | expected
${TEST_HOST} | ${true} ${TEST_HOST} | ${true}
......
import { GlAlert } from '@gitlab/ui'; import { GlAlert } from '@gitlab/ui';
import { shallowMount } from '@vue/test-utils'; import { shallowMount } from '@vue/test-utils';
import setWindowLocation from 'helpers/set_window_location_helper';
import { TEST_HOST } from 'helpers/test_constants'; import { TEST_HOST } from 'helpers/test_constants';
import CodeSnippetAlert from '~/pipeline_editor/components/code_snippet_alert/code_snippet_alert.vue'; import CodeSnippetAlert from '~/pipeline_editor/components/code_snippet_alert/code_snippet_alert.vue';
import { CODE_SNIPPET_SOURCES } from '~/pipeline_editor/components/code_snippet_alert/constants'; import { CODE_SNIPPET_SOURCES } from '~/pipeline_editor/components/code_snippet_alert/constants';
...@@ -12,6 +13,10 @@ import { ...@@ -12,6 +13,10 @@ import {
LOAD_FAILURE_UNKNOWN, LOAD_FAILURE_UNKNOWN,
} from '~/pipeline_editor/constants'; } from '~/pipeline_editor/constants';
beforeEach(() => {
setWindowLocation(TEST_HOST);
});
describe('Pipeline Editor messages', () => { describe('Pipeline Editor messages', () => {
let wrapper; let wrapper;
...@@ -95,9 +100,7 @@ describe('Pipeline Editor messages', () => { ...@@ -95,9 +100,7 @@ describe('Pipeline Editor messages', () => {
describe('code snippet alert', () => { describe('code snippet alert', () => {
const setCodeSnippetUrlParam = (value) => { const setCodeSnippetUrlParam = (value) => {
global.jsdom.reconfigure({ setWindowLocation(`${TEST_HOST}/?code_snippet_copied_from=${value}`);
url: `${TEST_HOST}/?code_snippet_copied_from=${value}`,
});
}; };
it('does not show by default', () => { it('does not show by default', () => {
......
...@@ -4,6 +4,8 @@ import { mount } from '@vue/test-utils'; ...@@ -4,6 +4,8 @@ import { mount } from '@vue/test-utils';
import MockAdapter from 'axios-mock-adapter'; import MockAdapter from 'axios-mock-adapter';
import { chunk } from 'lodash'; import { chunk } from 'lodash';
import { nextTick } from 'vue'; import { nextTick } from 'vue';
import setWindowLocation from 'helpers/set_window_location_helper';
import { TEST_HOST } from 'helpers/test_constants';
import { extendedWrapper } from 'helpers/vue_test_utils_helper'; import { extendedWrapper } from 'helpers/vue_test_utils_helper';
import waitForPromises from 'helpers/wait_for_promises'; import waitForPromises from 'helpers/wait_for_promises';
import Api from '~/api'; import Api from '~/api';
...@@ -40,7 +42,6 @@ const mockPipelineWithStages = mockPipelinesResponse.pipelines.find( ...@@ -40,7 +42,6 @@ const mockPipelineWithStages = mockPipelinesResponse.pipelines.find(
describe('Pipelines', () => { describe('Pipelines', () => {
let wrapper; let wrapper;
let mock; let mock;
let origWindowLocation;
const paths = { const paths = {
emptyStateSvgPath: '/assets/illustrations/pipelines_empty.svg', emptyStateSvgPath: '/assets/illustrations/pipelines_empty.svg',
...@@ -98,17 +99,8 @@ describe('Pipelines', () => { ...@@ -98,17 +99,8 @@ describe('Pipelines', () => {
); );
}; };
beforeAll(() => { beforeEach(() => {
origWindowLocation = window.location; setWindowLocation(TEST_HOST);
delete window.location;
window.location = {
search: '',
protocol: 'https:',
};
});
afterAll(() => {
window.location = origWindowLocation;
}); });
beforeEach(() => { beforeEach(() => {
......
...@@ -4,6 +4,7 @@ import MockAdapter from 'axios-mock-adapter'; ...@@ -4,6 +4,7 @@ import MockAdapter from 'axios-mock-adapter';
import { merge } from 'lodash'; import { merge } from 'lodash';
import Vuex from 'vuex'; import Vuex from 'vuex';
import { getJSONFixture } from 'helpers/fixtures'; import { getJSONFixture } from 'helpers/fixtures';
import setWindowLocation from 'helpers/set_window_location_helper';
import { TEST_HOST } from 'helpers/test_constants'; import { TEST_HOST } from 'helpers/test_constants';
import * as commonUtils from '~/lib/utils/common_utils'; import * as commonUtils from '~/lib/utils/common_utils';
import ReleaseEditNewApp from '~/releases/components/app_edit_new.vue'; import ReleaseEditNewApp from '~/releases/components/app_edit_new.vue';
...@@ -77,7 +78,7 @@ describe('Release edit/new component', () => { ...@@ -77,7 +78,7 @@ describe('Release edit/new component', () => {
}; };
beforeEach(() => { beforeEach(() => {
global.jsdom.reconfigure({ url: TEST_HOST }); setWindowLocation(TEST_HOST);
mock = new MockAdapter(axios); mock = new MockAdapter(axios);
gon.api_version = 'v4'; gon.api_version = 'v4';
...@@ -164,9 +165,7 @@ describe('Release edit/new component', () => { ...@@ -164,9 +165,7 @@ describe('Release edit/new component', () => {
`when the URL contains a "${BACK_URL_PARAM}=$backUrl" parameter`, `when the URL contains a "${BACK_URL_PARAM}=$backUrl" parameter`,
({ backUrl, expectedHref }) => { ({ backUrl, expectedHref }) => {
beforeEach(async () => { beforeEach(async () => {
global.jsdom.reconfigure({ setWindowLocation(`${TEST_HOST}?${BACK_URL_PARAM}=${encodeURIComponent(backUrl)}`);
url: `${TEST_HOST}?${BACK_URL_PARAM}=${encodeURIComponent(backUrl)}`,
});
await factory(); await factory();
}); });
......
...@@ -2,6 +2,7 @@ import { GlLink } from '@gitlab/ui'; ...@@ -2,6 +2,7 @@ import { GlLink } from '@gitlab/ui';
import { shallowMount } from '@vue/test-utils'; import { shallowMount } from '@vue/test-utils';
import { merge } from 'lodash'; import { merge } from 'lodash';
import { getJSONFixture } from 'helpers/fixtures'; import { getJSONFixture } from 'helpers/fixtures';
import setWindowLocation from 'helpers/set_window_location_helper';
import { convertObjectPropsToCamelCase } from '~/lib/utils/common_utils'; import { convertObjectPropsToCamelCase } from '~/lib/utils/common_utils';
import ReleaseBlockHeader from '~/releases/components/release_block_header.vue'; import ReleaseBlockHeader from '~/releases/components/release_block_header.vue';
import { BACK_URL_PARAM } from '~/releases/constants'; import { BACK_URL_PARAM } from '~/releases/constants';
...@@ -60,12 +61,7 @@ describe('Release block header', () => { ...@@ -60,12 +61,7 @@ describe('Release block header', () => {
const currentUrl = 'https://example.gitlab.com/path'; const currentUrl = 'https://example.gitlab.com/path';
beforeEach(() => { beforeEach(() => {
Object.defineProperty(window, 'location', { setWindowLocation(currentUrl);
writable: true,
value: {
href: currentUrl,
},
});
factory(); factory();
}); });
......
import { createLocalVue, mount, shallowMount } from '@vue/test-utils'; import { createLocalVue, mount, shallowMount } from '@vue/test-utils';
import VueApollo from 'vue-apollo'; import VueApollo from 'vue-apollo';
import createMockApollo from 'helpers/mock_apollo_helper'; import createMockApollo from 'helpers/mock_apollo_helper';
import { TEST_HOST } from 'helpers/test_constants'; import setWindowLocation from 'helpers/set_window_location_helper';
import waitForPromises from 'helpers/wait_for_promises'; import waitForPromises from 'helpers/wait_for_promises';
import createFlash from '~/flash'; import createFlash from '~/flash';
import { updateHistory } from '~/lib/utils/url_utility'; import { updateHistory } from '~/lib/utils/url_utility';
...@@ -43,7 +43,6 @@ localVue.use(VueApollo); ...@@ -43,7 +43,6 @@ localVue.use(VueApollo);
describe('AdminRunnersApp', () => { describe('AdminRunnersApp', () => {
let wrapper; let wrapper;
let mockRunnersQuery; let mockRunnersQuery;
let originalLocation;
const findRunnerTypeHelp = () => wrapper.findComponent(RunnerTypeHelp); const findRunnerTypeHelp = () => wrapper.findComponent(RunnerTypeHelp);
const findRunnerManualSetupHelp = () => wrapper.findComponent(RunnerManualSetupHelp); const findRunnerManualSetupHelp = () => wrapper.findComponent(RunnerManualSetupHelp);
...@@ -65,22 +64,8 @@ describe('AdminRunnersApp', () => { ...@@ -65,22 +64,8 @@ describe('AdminRunnersApp', () => {
}); });
}; };
const setQuery = (query) => {
window.location.href = `${TEST_HOST}/admin/runners?${query}`;
window.location.search = query;
};
beforeAll(() => {
originalLocation = window.location;
Object.defineProperty(window, 'location', { writable: true, value: { href: '', search: '' } });
});
afterAll(() => {
window.location = originalLocation;
});
beforeEach(async () => { beforeEach(async () => {
setQuery(''); setWindowLocation('/admin/runners');
mockRunnersQuery = jest.fn().mockResolvedValue(runnersData); mockRunnersQuery = jest.fn().mockResolvedValue(runnersData);
createComponentWithApollo(); createComponentWithApollo();
...@@ -116,7 +101,7 @@ describe('AdminRunnersApp', () => { ...@@ -116,7 +101,7 @@ describe('AdminRunnersApp', () => {
describe('when a filter is preselected', () => { describe('when a filter is preselected', () => {
beforeEach(async () => { beforeEach(async () => {
setQuery(`?status[]=${STATUS_ACTIVE}&runner_type[]=${INSTANCE_TYPE}&tag[]=tag1`); setWindowLocation(`?status[]=${STATUS_ACTIVE}&runner_type[]=${INSTANCE_TYPE}&tag[]=tag1`);
createComponentWithApollo(); createComponentWithApollo();
await waitForPromises(); await waitForPromises();
......
import setHighlightClass from 'ee_else_ce/search/highlight_blob_search_result'; import setHighlightClass from 'ee_else_ce/search/highlight_blob_search_result';
import setWindowLocation from 'helpers/set_window_location_helper';
import { initSearchApp } from '~/search'; import { initSearchApp } from '~/search';
import createStore from '~/search/store'; import createStore from '~/search/store';
...@@ -8,25 +9,6 @@ jest.mock('~/search/sidebar'); ...@@ -8,25 +9,6 @@ jest.mock('~/search/sidebar');
jest.mock('ee_else_ce/search/highlight_blob_search_result'); jest.mock('ee_else_ce/search/highlight_blob_search_result');
describe('initSearchApp', () => { describe('initSearchApp', () => {
let defaultLocation;
const setUrl = (query) => {
window.location.href = `https://localhost:3000/search${query}`;
window.location.search = query;
};
beforeEach(() => {
defaultLocation = window.location;
Object.defineProperty(window, 'location', {
writable: true,
value: { href: '', search: '' },
});
});
afterEach(() => {
window.location = defaultLocation;
});
describe.each` describe.each`
search | decodedSearch search | decodedSearch
${'test'} | ${'test'} ${'test'} | ${'test'}
...@@ -38,7 +20,7 @@ describe('initSearchApp', () => { ...@@ -38,7 +20,7 @@ describe('initSearchApp', () => {
${'test+%2520+this+%2520+stuff'} | ${'test %20 this %20 stuff'} ${'test+%2520+this+%2520+stuff'} | ${'test %20 this %20 stuff'}
`('parameter decoding', ({ search, decodedSearch }) => { `('parameter decoding', ({ search, decodedSearch }) => {
beforeEach(() => { beforeEach(() => {
setUrl(`?search=${search}`); setWindowLocation(`/search?search=${search}`);
initSearchApp(); initSearchApp();
}); });
......
import { waitFor } from '@testing-library/dom'; import { waitFor } from '@testing-library/dom';
import setWindowLocation from 'helpers/set_window_location_helper';
import { TEST_HOST } from 'helpers/test_constants'; import { TEST_HOST } from 'helpers/test_constants';
import initDiffsApp from '~/diffs'; import initDiffsApp from '~/diffs';
import { createStore } from '~/mr_notes/stores'; import { createStore } from '~/mr_notes/stores';
...@@ -111,9 +112,7 @@ describe('diffs third party interoperability', () => { ...@@ -111,9 +112,7 @@ describe('diffs third party interoperability', () => {
${'parallel view right side'} | ${'parallel'} | ${'.diff-tr.line_holder'} | ${'.diff-td.line_content.right-side'} | ${EXPECT_PARALLEL_RIGHT_SIDE} ${'parallel view right side'} | ${'parallel'} | ${'.diff-tr.line_holder'} | ${'.diff-td.line_content.right-side'} | ${EXPECT_PARALLEL_RIGHT_SIDE}
`('$desc', ({ view, rowSelector, codeSelector, expectation }) => { `('$desc', ({ view, rowSelector, codeSelector, expectation }) => {
beforeEach(async () => { beforeEach(async () => {
global.jsdom.reconfigure({ setWindowLocation(`${TEST_HOST}/${TEST_BASE_URL}/diffs?view=${view}`);
url: `${TEST_HOST}/${TEST_BASE_URL}/diffs?view=${view}`,
});
vm = startDiffsApp(); vm = startDiffsApp();
......
/* global monaco */ /* global monaco */
import setWindowLocation from 'helpers/set_window_location_helper';
import { TEST_HOST } from 'helpers/test_constants'; import { TEST_HOST } from 'helpers/test_constants';
import { initIde } from '~/ide'; import { initIde } from '~/ide';
import extendStore from '~/ide/stores/extend'; import extendStore from '~/ide/stores/extend';
...@@ -9,9 +10,7 @@ export default (container, { isRepoEmpty = false, path = '', mrId = '' } = {}) = ...@@ -9,9 +10,7 @@ export default (container, { isRepoEmpty = false, path = '', mrId = '' } = {}) =
const projectName = isRepoEmpty ? 'lorem-ipsum-empty' : 'lorem-ipsum'; const projectName = isRepoEmpty ? 'lorem-ipsum-empty' : 'lorem-ipsum';
const pathSuffix = mrId ? `merge_requests/${mrId}` : `tree/master/-/${path}`; const pathSuffix = mrId ? `merge_requests/${mrId}` : `tree/master/-/${path}`;
global.jsdom.reconfigure({ setWindowLocation(`${TEST_HOST}/-/ide/project/gitlab-test/${projectName}/${pathSuffix}`);
url: `${TEST_HOST}/-/ide/project/gitlab-test/${projectName}/${pathSuffix}`,
});
const el = document.createElement('div'); const el = document.createElement('div');
Object.assign(el.dataset, IDE_DATASET); Object.assign(el.dataset, IDE_DATASET);
......
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