debian-mirror-gitlab/spec/frontend/packages/details/components/app_spec.js

386 lines
12 KiB
JavaScript
Raw Normal View History

2021-09-04 01:27:46 +05:30
import { GlEmptyState } from '@gitlab/ui';
2021-03-11 19:13:27 +05:30
import { mount, createLocalVue } from '@vue/test-utils';
2021-09-04 01:27:46 +05:30
import { nextTick } from 'vue';
2021-03-11 19:13:27 +05:30
import Vuex from 'vuex';
2021-10-27 15:23:28 +05:30
import { useMockLocationHelper } from 'helpers/mock_window_location_helper';
2020-10-24 23:57:45 +05:30
import stubChildren from 'helpers/stub_children';
import AdditionalMetadata from '~/packages/details/components/additional_metadata.vue';
2021-03-11 19:13:27 +05:30
import PackagesApp from '~/packages/details/components/app.vue';
import DependencyRow from '~/packages/details/components/dependency_row.vue';
2020-10-24 23:57:45 +05:30
import InstallationCommands from '~/packages/details/components/installation_commands.vue';
2021-02-22 17:27:13 +05:30
import PackageFiles from '~/packages/details/components/package_files.vue';
2021-03-11 19:13:27 +05:30
import PackageHistory from '~/packages/details/components/package_history.vue';
import PackageTitle from '~/packages/details/components/package_title.vue';
import * as getters from '~/packages/details/store/getters';
import PackageListRow from '~/packages/shared/components/package_list_row.vue';
import PackagesListLoader from '~/packages/shared/components/packages_list_loader.vue';
import { TrackingActions } from '~/packages/shared/constants';
import * as SharedUtils from '~/packages/shared/utils';
import Tracking from '~/tracking';
2020-10-24 23:57:45 +05:30
import {
composerPackage,
conanPackage,
mavenPackage,
mavenFiles,
npmPackage,
nugetPackage,
} from '../../mock_data';
const localVue = createLocalVue();
localVue.use(Vuex);
2021-10-27 15:23:28 +05:30
useMockLocationHelper();
2020-10-24 23:57:45 +05:30
describe('PackagesApp', () => {
let wrapper;
let store;
const fetchPackageVersions = jest.fn();
2020-11-24 15:15:51 +05:30
const deletePackage = jest.fn();
2021-09-04 01:27:46 +05:30
const deletePackageFile = jest.fn();
2020-11-24 15:15:51 +05:30
const defaultProjectName = 'bar';
2020-10-24 23:57:45 +05:30
function createComponent({
packageEntity = mavenPackage,
packageFiles = mavenFiles,
isLoading = false,
2020-11-24 15:15:51 +05:30
projectName = defaultProjectName,
2020-10-24 23:57:45 +05:30
} = {}) {
store = new Vuex.Store({
state: {
isLoading,
packageEntity,
packageFiles,
canDelete: true,
emptySvgPath: 'empty-illustration',
npmPath: 'foo',
npmHelpPath: 'foo',
2020-11-24 15:15:51 +05:30
projectName,
projectListUrl: 'project_url',
groupListUrl: 'group_url',
2020-10-24 23:57:45 +05:30
},
actions: {
2020-11-24 15:15:51 +05:30
deletePackage,
2020-10-24 23:57:45 +05:30
fetchPackageVersions,
2021-09-04 01:27:46 +05:30
deletePackageFile,
2020-10-24 23:57:45 +05:30
},
getters,
});
wrapper = mount(PackagesApp, {
localVue,
store,
stubs: {
...stubChildren(PackagesApp),
2020-11-24 15:15:51 +05:30
PackageTitle: false,
TitleArea: false,
2020-10-24 23:57:45 +05:30
GlButton: false,
GlModal: false,
GlTab: false,
GlTabs: false,
GlTable: false,
},
});
}
const packageTitle = () => wrapper.find(PackageTitle);
const emptyState = () => wrapper.find(GlEmptyState);
const deleteButton = () => wrapper.find('.js-delete-button');
2021-09-04 01:27:46 +05:30
const findDeleteModal = () => wrapper.find({ ref: 'deleteModal' });
const findDeleteFileModal = () => wrapper.find({ ref: 'deleteFileModal' });
2020-10-24 23:57:45 +05:30
const versionsTab = () => wrapper.find('.js-versions-tab > a');
const packagesLoader = () => wrapper.find(PackagesListLoader);
const packagesVersionRows = () => wrapper.findAll(PackageListRow);
const noVersionsMessage = () => wrapper.find('[data-testid="no-versions-message"]');
const dependenciesTab = () => wrapper.find('.js-dependencies-tab > a');
const dependenciesCountBadge = () => wrapper.find('[data-testid="dependencies-badge"]');
const noDependenciesMessage = () => wrapper.find('[data-testid="no-dependencies-message"]');
const dependencyRows = () => wrapper.findAll(DependencyRow);
const findPackageHistory = () => wrapper.find(PackageHistory);
const findAdditionalMetadata = () => wrapper.find(AdditionalMetadata);
const findInstallationCommands = () => wrapper.find(InstallationCommands);
2021-02-22 17:27:13 +05:30
const findPackageFiles = () => wrapper.find(PackageFiles);
2020-10-24 23:57:45 +05:30
afterEach(() => {
wrapper.destroy();
});
2021-09-04 01:27:46 +05:30
it('renders the app and displays the package title', async () => {
2020-10-24 23:57:45 +05:30
createComponent();
2021-09-04 01:27:46 +05:30
await nextTick();
expect(packageTitle().exists()).toBe(true);
2020-10-24 23:57:45 +05:30
});
it('renders an empty state component when no an invalid package is passed as a prop', () => {
createComponent({
packageEntity: {},
});
2021-09-04 01:27:46 +05:30
expect(emptyState().exists()).toBe(true);
2020-10-24 23:57:45 +05:30
});
it('package history has the right props', () => {
createComponent();
expect(findPackageHistory().exists()).toBe(true);
expect(findPackageHistory().props('packageEntity')).toEqual(wrapper.vm.packageEntity);
expect(findPackageHistory().props('projectName')).toEqual(wrapper.vm.projectName);
});
it('additional metadata has the right props', () => {
createComponent();
expect(findAdditionalMetadata().exists()).toBe(true);
expect(findAdditionalMetadata().props('packageEntity')).toEqual(wrapper.vm.packageEntity);
});
it('installation commands has the right props', () => {
createComponent();
expect(findInstallationCommands().exists()).toBe(true);
expect(findInstallationCommands().props('packageEntity')).toEqual(wrapper.vm.packageEntity);
});
it('hides the files table if package type is COMPOSER', () => {
createComponent({ packageEntity: composerPackage });
2021-02-22 17:27:13 +05:30
expect(findPackageFiles().exists()).toBe(false);
2020-10-24 23:57:45 +05:30
});
describe('deleting packages', () => {
beforeEach(() => {
createComponent();
deleteButton().trigger('click');
});
it('shows the delete confirmation modal when delete is clicked', () => {
2021-09-04 01:27:46 +05:30
expect(findDeleteModal().exists()).toBe(true);
});
});
describe('deleting package files', () => {
it('shows the delete confirmation modal when delete is clicked', () => {
createComponent();
findPackageFiles().vm.$emit('delete-file', mavenFiles[0]);
expect(findDeleteFileModal().exists()).toBe(true);
2020-10-24 23:57:45 +05:30
});
});
describe('versions', () => {
describe('api call', () => {
beforeEach(() => {
createComponent();
});
it('makes api request on first click of tab', () => {
versionsTab().trigger('click');
expect(fetchPackageVersions).toHaveBeenCalled();
});
});
it('displays the loader when state is loading', () => {
createComponent({ isLoading: true });
expect(packagesLoader().exists()).toBe(true);
});
it('displays the correct version count when the package has versions', () => {
createComponent({ packageEntity: npmPackage });
expect(packagesVersionRows()).toHaveLength(npmPackage.versions.length);
});
it('displays the no versions message when there are none', () => {
createComponent();
expect(noVersionsMessage().exists()).toBe(true);
});
});
describe('dependency links', () => {
it('does not show the dependency links for a non nuget package', () => {
createComponent();
expect(dependenciesTab().exists()).toBe(false);
});
it('shows the dependencies tab with 0 count when a nuget package with no dependencies', () => {
createComponent({
packageEntity: {
...nugetPackage,
dependency_links: [],
},
});
return wrapper.vm.$nextTick(() => {
const dependenciesBadge = dependenciesCountBadge();
expect(dependenciesTab().exists()).toBe(true);
expect(dependenciesBadge.exists()).toBe(true);
expect(dependenciesBadge.text()).toBe('0');
expect(noDependenciesMessage().exists()).toBe(true);
});
});
it('renders the correct number of dependency rows for a nuget package', () => {
createComponent({ packageEntity: nugetPackage });
return wrapper.vm.$nextTick(() => {
const dependenciesBadge = dependenciesCountBadge();
expect(dependenciesTab().exists()).toBe(true);
expect(dependenciesBadge.exists()).toBe(true);
expect(dependenciesBadge.text()).toBe(nugetPackage.dependency_links.length.toString());
expect(dependencyRows()).toHaveLength(nugetPackage.dependency_links.length);
});
});
});
2020-11-24 15:15:51 +05:30
describe('tracking and delete', () => {
2021-09-04 01:27:46 +05:30
describe('delete package', () => {
2020-11-24 15:15:51 +05:30
const originalReferrer = document.referrer;
const setReferrer = (value = defaultProjectName) => {
Object.defineProperty(document, 'referrer', {
value,
configurable: true,
});
};
afterEach(() => {
Object.defineProperty(document, 'referrer', {
value: originalReferrer,
configurable: true,
});
});
2020-10-24 23:57:45 +05:30
2021-09-04 01:27:46 +05:30
it('calls the proper vuex action', () => {
2020-11-24 15:15:51 +05:30
createComponent({ packageEntity: npmPackage });
2021-09-04 01:27:46 +05:30
findDeleteModal().vm.$emit('primary');
2020-11-24 15:15:51 +05:30
expect(deletePackage).toHaveBeenCalled();
});
2020-10-24 23:57:45 +05:30
2020-11-24 15:15:51 +05:30
it('when referrer contains project name calls window.replace with project url', async () => {
setReferrer();
deletePackage.mockResolvedValue();
createComponent({ packageEntity: npmPackage });
2021-09-04 01:27:46 +05:30
findDeleteModal().vm.$emit('primary');
2020-11-24 15:15:51 +05:30
await deletePackage();
expect(window.location.replace).toHaveBeenCalledWith(
'project_url?showSuccessDeleteAlert=true',
);
});
it('when referrer does not contain project name calls window.replace with group url', async () => {
setReferrer('baz');
deletePackage.mockResolvedValue();
createComponent({ packageEntity: npmPackage });
2021-09-04 01:27:46 +05:30
findDeleteModal().vm.$emit('primary');
2020-11-24 15:15:51 +05:30
await deletePackage();
expect(window.location.replace).toHaveBeenCalledWith(
'group_url?showSuccessDeleteAlert=true',
);
});
2020-10-24 23:57:45 +05:30
});
2021-09-04 01:27:46 +05:30
describe('delete file', () => {
it('calls the proper vuex action', () => {
createComponent({ packageEntity: npmPackage });
findPackageFiles().vm.$emit('delete-file', mavenFiles[0]);
findDeleteFileModal().vm.$emit('primary');
expect(deletePackageFile).toHaveBeenCalled();
});
});
2020-11-24 15:15:51 +05:30
describe('tracking', () => {
let eventSpy;
let utilSpy;
const category = 'foo';
beforeEach(() => {
eventSpy = jest.spyOn(Tracking, 'event');
utilSpy = jest.spyOn(SharedUtils, 'packageTypeToTrackCategory').mockReturnValue(category);
});
it('tracking category calls packageTypeToTrackCategory', () => {
createComponent({ packageEntity: conanPackage });
expect(wrapper.vm.tracking.category).toBe(category);
expect(utilSpy).toHaveBeenCalledWith('conan');
});
2021-09-04 01:27:46 +05:30
it(`delete button on delete modal call event with ${TrackingActions.DELETE_PACKAGE}`, () => {
2020-11-24 15:15:51 +05:30
createComponent({ packageEntity: npmPackage });
2021-09-04 01:27:46 +05:30
findDeleteModal().vm.$emit('primary');
2020-10-24 23:57:45 +05:30
expect(eventSpy).toHaveBeenCalledWith(
category,
TrackingActions.DELETE_PACKAGE,
expect.any(Object),
);
});
2021-09-04 01:27:46 +05:30
it(`canceling a package deletion tracks ${TrackingActions.CANCEL_DELETE_PACKAGE}`, () => {
createComponent({ packageEntity: npmPackage });
findDeleteModal().vm.$emit('canceled');
expect(eventSpy).toHaveBeenCalledWith(
category,
TrackingActions.CANCEL_DELETE_PACKAGE,
expect.any(Object),
);
});
it(`request a file deletion tracks ${TrackingActions.REQUEST_DELETE_PACKAGE_FILE}`, () => {
createComponent({ packageEntity: npmPackage });
findPackageFiles().vm.$emit('delete-file', mavenFiles[0]);
expect(eventSpy).toHaveBeenCalledWith(
category,
TrackingActions.REQUEST_DELETE_PACKAGE_FILE,
expect.any(Object),
);
});
it(`confirming a file deletion tracks ${TrackingActions.DELETE_PACKAGE_FILE}`, () => {
createComponent({ packageEntity: npmPackage });
findPackageFiles().vm.$emit('delete-file', npmPackage);
findDeleteFileModal().vm.$emit('primary');
expect(eventSpy).toHaveBeenCalledWith(
category,
TrackingActions.REQUEST_DELETE_PACKAGE_FILE,
expect.any(Object),
);
});
it(`canceling a file deletion tracks ${TrackingActions.CANCEL_DELETE_PACKAGE_FILE}`, () => {
createComponent({ packageEntity: npmPackage });
findPackageFiles().vm.$emit('delete-file', npmPackage);
findDeleteFileModal().vm.$emit('canceled');
expect(eventSpy).toHaveBeenCalledWith(
category,
TrackingActions.CANCEL_DELETE_PACKAGE_FILE,
expect.any(Object),
);
});
2020-11-24 15:15:51 +05:30
it(`file download link call event with ${TrackingActions.PULL_PACKAGE}`, () => {
createComponent({ packageEntity: conanPackage });
2020-10-24 23:57:45 +05:30
2021-02-22 17:27:13 +05:30
findPackageFiles().vm.$emit('download-file');
2020-11-24 15:15:51 +05:30
expect(eventSpy).toHaveBeenCalledWith(
category,
TrackingActions.PULL_PACKAGE,
expect.any(Object),
);
});
2020-10-24 23:57:45 +05:30
});
});
});