mirror of
https://gitlab.com/gitlab-org/gitlab-foss.git
synced 2025-08-13 13:31:19 +00:00
373 lines
13 KiB
JavaScript
373 lines
13 KiB
JavaScript
import {
|
|
GlAlert,
|
|
GlFormInputGroup,
|
|
GlFormGroup,
|
|
GlModal,
|
|
GlSprintf,
|
|
GlSkeletonLoader,
|
|
GlDisclosureDropdown,
|
|
GlDisclosureDropdownItem,
|
|
} from '@gitlab/ui';
|
|
import Vue, { nextTick } from 'vue';
|
|
import VueApollo from 'vue-apollo';
|
|
import MockAdapter from 'axios-mock-adapter';
|
|
import createMockApollo from 'helpers/mock_apollo_helper';
|
|
import { createMockDirective, getBinding } from 'helpers/vue_mock_directive';
|
|
import { shallowMountExtended } from 'helpers/vue_test_utils_helper';
|
|
import waitForPromises from 'helpers/wait_for_promises';
|
|
import { GRAPHQL_PAGE_SIZE } from '~/packages_and_registries/dependency_proxy/constants';
|
|
import axios from '~/lib/utils/axios_utils';
|
|
import { HTTP_STATUS_ACCEPTED } from '~/lib/utils/http_status';
|
|
import setWindowLocation from 'helpers/set_window_location_helper';
|
|
import { TEST_HOST } from 'helpers/test_constants';
|
|
import DependencyProxyApp from '~/packages_and_registries/dependency_proxy/app.vue';
|
|
import TitleArea from '~/vue_shared/components/registry/title_area.vue';
|
|
import ClipboardButton from '~/vue_shared/components/clipboard_button.vue';
|
|
import ManifestsList from '~/packages_and_registries/dependency_proxy/components/manifests_list.vue';
|
|
import createRouter from '~/packages_and_registries/dependency_proxy/router';
|
|
import getDependencyProxyDetailsQuery from '~/packages_and_registries/dependency_proxy/graphql/queries/get_dependency_proxy_details.query.graphql';
|
|
|
|
import { proxyDetailsQuery, proxyData, pagination, proxyManifests } from './mock_data';
|
|
|
|
const dummyApiVersion = 'v3000';
|
|
const dummyGrouptId = 1;
|
|
const dummyUrlRoot = '/gitlab';
|
|
const expectedUrl = `${dummyUrlRoot}/api/${dummyApiVersion}/groups/${dummyGrouptId}/dependency_proxy/cache`;
|
|
|
|
Vue.use(VueApollo);
|
|
|
|
describe('DependencyProxyApp', () => {
|
|
let wrapper;
|
|
let router;
|
|
let apolloProvider;
|
|
let resolver;
|
|
let mock;
|
|
|
|
const provideDefaults = {
|
|
groupPath: 'gitlab-org',
|
|
groupId: dummyGrouptId,
|
|
noManifestsIllustration: 'noManifestsIllustration',
|
|
canClearCache: true,
|
|
settingsPath: 'path',
|
|
};
|
|
|
|
function createComponent({ provide = provideDefaults } = {}) {
|
|
const requestHandlers = [[getDependencyProxyDetailsQuery, resolver]];
|
|
|
|
apolloProvider = createMockApollo(requestHandlers);
|
|
router = createRouter('/');
|
|
|
|
wrapper = shallowMountExtended(DependencyProxyApp, {
|
|
apolloProvider,
|
|
provide,
|
|
router,
|
|
stubs: {
|
|
GlAlert,
|
|
GlFormGroup,
|
|
GlModal,
|
|
GlSprintf,
|
|
TitleArea,
|
|
},
|
|
directives: {
|
|
GlTooltip: createMockDirective('gl-tooltip'),
|
|
},
|
|
});
|
|
}
|
|
|
|
const findClipBoardButton = () => wrapper.findComponent(ClipboardButton);
|
|
const findFormGroup = () => wrapper.findComponent(GlFormGroup);
|
|
const findFormInputGroup = () => wrapper.findComponent(GlFormInputGroup);
|
|
const findProxyCountText = () => wrapper.findByTestId('proxy-count');
|
|
const findManifestList = () => wrapper.findComponent(ManifestsList);
|
|
const findLoader = () => wrapper.findComponent(GlSkeletonLoader);
|
|
const findClearCacheDropdownList = () => wrapper.findComponent(GlDisclosureDropdown);
|
|
const findClearCacheModal = () => wrapper.findComponent(GlModal);
|
|
const findClearCacheAlert = () => wrapper.findComponent(GlAlert);
|
|
const findSettingsLink = () => wrapper.findByTestId('settings-link');
|
|
|
|
beforeEach(() => {
|
|
resolver = jest.fn().mockResolvedValue(proxyDetailsQuery());
|
|
|
|
window.gon = {
|
|
api_version: dummyApiVersion,
|
|
relative_url_root: dummyUrlRoot,
|
|
};
|
|
|
|
mock = new MockAdapter(axios);
|
|
mock.onDelete(expectedUrl).reply(HTTP_STATUS_ACCEPTED, {});
|
|
setWindowLocation(TEST_HOST);
|
|
});
|
|
|
|
afterEach(() => {
|
|
mock.restore();
|
|
});
|
|
|
|
describe('when the dependency proxy is available', () => {
|
|
describe('when is loading', () => {
|
|
beforeEach(() => {
|
|
createComponent();
|
|
});
|
|
|
|
it('renders loading component & sets loading prop', () => {
|
|
expect(findLoader().exists()).toBe(true);
|
|
expect(findManifestList().props('loading')).toBe(true);
|
|
});
|
|
|
|
it('does not render a form group with label', () => {
|
|
expect(findFormGroup().exists()).toBe(false);
|
|
});
|
|
});
|
|
|
|
describe('when the app is loaded', () => {
|
|
describe('when the dependency proxy is enabled', () => {
|
|
beforeEach(() => {
|
|
createComponent();
|
|
return waitForPromises();
|
|
});
|
|
|
|
it('resolver is called with right arguments', () => {
|
|
expect(resolver).toHaveBeenCalledWith({
|
|
first: GRAPHQL_PAGE_SIZE,
|
|
fullPath: provideDefaults.groupPath,
|
|
});
|
|
});
|
|
|
|
it('renders a form group with a label', () => {
|
|
expect(findFormGroup().attributes('label')).toBe(
|
|
DependencyProxyApp.i18n.proxyImagePrefix,
|
|
);
|
|
expect(findFormGroup().attributes('labelfor')).toBe('proxy-url');
|
|
});
|
|
|
|
it('renders a form input group', () => {
|
|
expect(findFormInputGroup().exists()).toBe(true);
|
|
expect(findFormInputGroup().attributes('id')).toBe('proxy-url');
|
|
expect(findFormInputGroup().props('value')).toBe(proxyData().dependencyProxyImagePrefix);
|
|
expect(findFormInputGroup().attributes('readonly')).toBeDefined();
|
|
expect(findFormInputGroup().props('selectOnClick')).toBe(true);
|
|
});
|
|
|
|
it('form input group has a clipboard button', () => {
|
|
expect(findClipBoardButton().exists()).toBe(true);
|
|
expect(findClipBoardButton().props()).toMatchObject({
|
|
text: proxyData().dependencyProxyImagePrefix,
|
|
title: DependencyProxyApp.i18n.copyImagePrefixText,
|
|
});
|
|
});
|
|
|
|
it('form group has a description with proxy count', () => {
|
|
expect(findProxyCountText().text()).toBe('Contains 2 blobs of images (1024 Bytes)');
|
|
});
|
|
|
|
describe('link to settings', () => {
|
|
it('is rendered', () => {
|
|
expect(findSettingsLink().exists()).toBe(true);
|
|
});
|
|
|
|
it('has the right icon', () => {
|
|
expect(findSettingsLink().props('icon')).toBe('settings');
|
|
});
|
|
|
|
it('has the right attributes', () => {
|
|
expect(findSettingsLink().attributes()).toMatchObject({
|
|
'aria-label': DependencyProxyApp.i18n.settingsText,
|
|
href: 'path',
|
|
});
|
|
});
|
|
|
|
it('sets tooltip with right label', () => {
|
|
const tooltip = getBinding(findSettingsLink().element, 'gl-tooltip');
|
|
|
|
expect(tooltip.value).toBe(DependencyProxyApp.i18n.settingsText);
|
|
});
|
|
});
|
|
|
|
describe('manifest lists', () => {
|
|
describe('when there are no manifests', () => {
|
|
beforeEach(() => {
|
|
resolver = jest.fn().mockResolvedValue(
|
|
proxyDetailsQuery({
|
|
extend: { dependencyProxyManifests: { nodes: [], pageInfo: pagination() } },
|
|
}),
|
|
);
|
|
createComponent();
|
|
|
|
return waitForPromises();
|
|
});
|
|
|
|
it('renders the list', () => {
|
|
expect(findManifestList().exists()).toBe(true);
|
|
});
|
|
});
|
|
|
|
describe('when there are manifests', () => {
|
|
it('shows list', () => {
|
|
expect(findManifestList().props()).toMatchObject({
|
|
dependencyProxyImagePrefix: proxyData().dependencyProxyImagePrefix,
|
|
manifests: proxyManifests(),
|
|
pagination: pagination(),
|
|
});
|
|
});
|
|
|
|
describe('prev-page event on list', () => {
|
|
beforeEach(() => {
|
|
findManifestList().vm.$emit('prev-page');
|
|
});
|
|
|
|
describe('while loading', () => {
|
|
it('does not render loading component & sets loading prop', () => {
|
|
expect(findLoader().exists()).toBe(false);
|
|
expect(findManifestList().props('loading')).toBe(true);
|
|
});
|
|
|
|
it('renders form group with label', () => {
|
|
expect(findFormGroup().exists()).toBe(true);
|
|
});
|
|
});
|
|
|
|
it('list fetches the previous page', async () => {
|
|
await waitForPromises();
|
|
|
|
expect(resolver).toHaveBeenCalledWith({
|
|
before: pagination().startCursor,
|
|
first: null,
|
|
fullPath: provideDefaults.groupPath,
|
|
last: GRAPHQL_PAGE_SIZE,
|
|
});
|
|
expect(window.location.search).toBe(`?before=${pagination().startCursor}`);
|
|
});
|
|
});
|
|
|
|
describe('next-page event on list', () => {
|
|
beforeEach(() => {
|
|
findManifestList().vm.$emit('next-page');
|
|
});
|
|
|
|
describe('while loading', () => {
|
|
it('does not render loading component & sets loading prop', () => {
|
|
expect(findLoader().exists()).toBe(false);
|
|
expect(findManifestList().props('loading')).toBe(true);
|
|
});
|
|
|
|
it('renders form group with label', () => {
|
|
expect(findFormGroup().exists()).toBe(true);
|
|
});
|
|
});
|
|
|
|
it('fetches the next page', async () => {
|
|
await waitForPromises();
|
|
|
|
expect(resolver).toHaveBeenCalledWith({
|
|
after: pagination().endCursor,
|
|
first: GRAPHQL_PAGE_SIZE,
|
|
fullPath: provideDefaults.groupPath,
|
|
});
|
|
expect(window.location.search).toBe(`?after=${pagination().endCursor}`);
|
|
});
|
|
});
|
|
|
|
describe('triggering page event on list', () => {
|
|
it('renders form group with label', async () => {
|
|
findManifestList().vm.$emit('next-page');
|
|
await nextTick();
|
|
|
|
expect(findFormGroup().attributes('label')).toEqual(
|
|
expect.stringMatching(DependencyProxyApp.i18n.proxyImagePrefix),
|
|
);
|
|
});
|
|
});
|
|
|
|
it('shows the clear cache dropdown list', () => {
|
|
expect(findClearCacheDropdownList().exists()).toBe(true);
|
|
|
|
const clearCacheDropdownItem =
|
|
findClearCacheDropdownList().findComponent(GlDisclosureDropdownItem);
|
|
|
|
expect(clearCacheDropdownItem.text()).toBe('Clear cache');
|
|
});
|
|
|
|
it('shows the clear cache confirmation modal', () => {
|
|
const modal = findClearCacheModal();
|
|
|
|
expect(modal.find('.modal-title').text()).toContain('Clear 2 images from cache?');
|
|
expect(modal.props('actionPrimary').text).toBe('Clear cache');
|
|
});
|
|
|
|
it('submits the clear cache request', async () => {
|
|
findClearCacheModal().vm.$emit('primary', { preventDefault: jest.fn() });
|
|
|
|
await waitForPromises();
|
|
|
|
expect(findClearCacheAlert().exists()).toBe(true);
|
|
expect(findClearCacheAlert().text()).toBe(
|
|
'All items in the cache are scheduled for removal.',
|
|
);
|
|
});
|
|
|
|
describe('when user has no permission to clear cache', () => {
|
|
beforeEach(() => {
|
|
createComponent({
|
|
provide: {
|
|
...provideDefaults,
|
|
canClearCache: false,
|
|
},
|
|
});
|
|
});
|
|
|
|
it('does not show the clear cache dropdown list', () => {
|
|
expect(findClearCacheDropdownList().exists()).toBe(false);
|
|
});
|
|
|
|
it('does not show link to settings', () => {
|
|
expect(findSettingsLink().exists()).toBe(false);
|
|
});
|
|
});
|
|
});
|
|
});
|
|
});
|
|
|
|
describe('pagination params', () => {
|
|
it('after is set from the url params', async () => {
|
|
setWindowLocation('?after=1234');
|
|
createComponent();
|
|
await waitForPromises();
|
|
|
|
expect(resolver).toHaveBeenCalledWith({
|
|
first: GRAPHQL_PAGE_SIZE,
|
|
after: '1234',
|
|
fullPath: provideDefaults.groupPath,
|
|
});
|
|
});
|
|
|
|
it('before is set from the url params', async () => {
|
|
setWindowLocation('?before=1234');
|
|
createComponent();
|
|
await waitForPromises();
|
|
|
|
expect(resolver).toHaveBeenCalledWith({
|
|
first: null,
|
|
last: GRAPHQL_PAGE_SIZE,
|
|
before: '1234',
|
|
fullPath: provideDefaults.groupPath,
|
|
});
|
|
});
|
|
|
|
describe('when url params are changed', () => {
|
|
it('after is set from the url params', async () => {
|
|
createComponent();
|
|
await waitForPromises();
|
|
router.push('?after=1234');
|
|
await waitForPromises();
|
|
|
|
expect(resolver).toHaveBeenCalledWith({
|
|
first: GRAPHQL_PAGE_SIZE,
|
|
after: '1234',
|
|
fullPath: provideDefaults.groupPath,
|
|
});
|
|
});
|
|
});
|
|
});
|
|
});
|
|
});
|
|
});
|