const hostRules = require('../../../lib/util/host-rules'); describe('platform/gitlab', () => { let gitlab; let get; beforeEach(() => { // clean up hostRules hostRules.clear(); // reset module jest.resetModules(); jest.mock('../../../lib/platform/gitlab/gl-got-wrapper'); gitlab = require('../../../lib/platform/gitlab'); get = require('../../../lib/platform/gitlab/gl-got-wrapper'); }); describe('getRepos', () => { function getRepos(...args) { // repo info get.mockImplementationOnce(() => ({ body: [ { path_with_namespace: 'a/b', }, { path_with_namespace: 'c/d', }, ], })); return gitlab.getRepos(...args); } it('should throw an error if no token is provided', async () => { await expect(gitlab.getRepos()).rejects.toThrow( Error('No token found for getRepos') ); }); it('should throw an error if it receives an error', async () => { get.mockImplementation(() => { throw new Error('getRepos error'); }); await expect(gitlab.getRepos('sometoken')).rejects.toThrow( Error('getRepos error') ); }); it('should return an array of repos', async () => { const repos = await getRepos('sometoken'); expect(get.mock.calls).toMatchSnapshot(); expect(repos).toMatchSnapshot(); }); it('should support a custom endpoint', async () => { const repos = await getRepos('sometoken', 'someendpoint'); expect(get.mock.calls).toMatchSnapshot(); expect(repos).toMatchSnapshot(); }); }); describe('getRepoStatus()', () => { it('exists', async () => { expect(await gitlab.getRepoStatus()).toEqual({}); }); }); describe('cleanRepo()', () => { it('exists', () => { gitlab.cleanRepo(); }); }); function initRepo(...args) { // projects/${config.repository get.mockImplementationOnce(() => ({ body: { default_branch: 'master', }, })); // user get.mockImplementationOnce(() => ({ body: { email: 'a@b.com', }, })); get.mockReturnValue({ body: [ { number: 1, source_branch: 'branch-a', title: 'branch a pr', state: 'opened', }, ], }); return gitlab.initRepo(...args); } describe('initRepo', () => { [ [undefined, 'mytoken', undefined], [undefined, 'mytoken', 'https://my.custom.endpoint/'], ['myenvtoken', 'myenvtoken', undefined], [undefined, 'mytoken', undefined, 'Renovate Bot '], ].forEach(([envToken, token, endpoint, gitAuthor], i) => { it(`should initialise the config for the repo - ${i}`, async () => { if (envToken !== undefined) { process.env.RENOVATE_TOKEN = envToken; } get.mockReturnValue({ body: [] }); const config = await initRepo({ repository: 'some/repo', token, endpoint, gitAuthor, }); expect(get.mock.calls).toMatchSnapshot(); expect(config).toMatchSnapshot(); }); }); it(`should escape all forward slashes in project names`, async () => { get.mockReturnValue({ body: [] }); await initRepo({ repository: 'some/repo/project', token: 'some-token' }); expect(get.mock.calls).toMatchSnapshot(); }); it('should throw an error if no token is provided', async () => { await expect( gitlab.initRepo({ repository: 'some/repo' }) ).rejects.toThrow( Error('No token found for GitLab repository some/repo') ); }); it('should throw an error if receiving an error', async () => { get.mockImplementation(() => { throw new Error('always error'); }); await expect( gitlab.initRepo({ repository: 'some/repo', token: 'sometoken' }) ).rejects.toThrow(Error('always error')); }); it('should throw an error if repository is archived', async () => { get.mockReturnValue({ body: { archived: true } }); await expect( gitlab.initRepo({ repository: 'some/repo', token: 'sometoken' }) ).rejects.toThrow(Error('archived')); }); it('should throw an error if repository is empty', async () => { get.mockReturnValue({ body: { default_branch: null } }); await expect( gitlab.initRepo({ repository: 'some/repo', token: 'sometoken' }) ).rejects.toThrow(Error('empty')); }); }); describe('getRepoForceRebase', () => { it('should return false', () => { expect(gitlab.getRepoForceRebase()).toBe(false); }); }); describe('setBaseBranch(branchName)', () => { it('sets the base branch', async () => { await gitlab.setBaseBranch('some-branch'); expect(get.mock.calls).toMatchSnapshot(); }); }); describe('getFileList', () => { it('returns empty array if error', async () => { get.mockImplementationOnce(() => { throw new Error('some error'); }); const files = await gitlab.getFileList(); expect(files).toEqual([]); }); it('warns if truncated result', async () => { get.mockImplementationOnce(() => ({ body: [], })); const files = await gitlab.getFileList(); expect(files).toHaveLength(0); }); it('caches the result', async () => { get.mockImplementationOnce(() => ({ body: [], })); let files = await gitlab.getFileList(); expect(files).toHaveLength(0); files = await gitlab.getFileList(); expect(files).toHaveLength(0); }); it('should return the files matching the fileName', async () => { get.mockImplementationOnce(() => ({ body: [ { type: 'blob', path: 'symlinks/package.json', mode: '120000' }, { type: 'blob', path: 'package.json' }, { type: 'blob', path: 'some-dir/package.json.some-thing-else', }, { type: 'blob', path: 'src/app/package.json' }, { type: 'blob', path: 'src/otherapp/package.json' }, ], })); const files = await gitlab.getFileList(); expect(files).toMatchSnapshot(); }); }); describe('branchExists(branchName)', () => { it('should return true if 200 OK', async () => { get.mockImplementationOnce(() => ({ statusCode: 200 })); const branchExists = await gitlab.branchExists('foo'); expect(branchExists).toBe(true); }); it('should return false if not 200 OK', async () => { get.mockImplementationOnce(() => ({ statusCode: 500 })); const branchExists = await gitlab.branchExists('foo'); expect(branchExists).toBe(false); }); it('should return false if 404 error received', async () => { get.mockImplementationOnce(() => Promise.reject({ statusCode: 404, }) ); const branchExists = await gitlab.branchExists('foo'); expect(branchExists).toBe(false); }); it('should return error if non-404 error thrown', async () => { get.mockImplementationOnce(() => Promise.reject({ statusCode: 500, }) ); await expect(gitlab.branchExists('foo')).rejects.toEqual({ statusCode: 500, }); }); }); describe('getAllRenovateBranches()', () => { it('should return all renovate branches', async () => { const search = 'renovate/'; get.mockImplementationOnce(path => { expect(path).toBe( 'projects/undefined/repository/branches?search=^renovate%2F' ); return { body: [ { name: 'renovate/a', }, { name: 'renovate/b', }, ], }; }); const res = await gitlab.getAllRenovateBranches(search); expect(res).toMatchSnapshot(); }); }); describe('isBranchStale()', () => { it('should return false if same SHA as master', async () => { // getBranchDetails - same as master get.mockImplementationOnce(() => ({ body: { commit: { parent_ids: ['1234'], }, }, })); // getBranchDetails - master get.mockImplementationOnce(() => ({ body: { commit: { id: '1234', }, }, })); expect(await gitlab.isBranchStale('thebranchname')).toBe(false); }); it('should return true if SHA different from master', async () => { // getBranchDetails - different from master get.mockImplementationOnce(() => ({ body: { commit: { parent_ids: ['12345678'], }, }, })); // getBranchDetails - master get.mockImplementationOnce(() => ({ body: { commit: { id: '1234', }, }, })); expect(await gitlab.isBranchStale('thebranchname')).toBe(true); }); }); describe('getBranchPr(branchName)', () => { it('should return null if branch does not exist', async () => { get.mockReturnValueOnce({ statusCode: 500 }); // branchExists const pr = await gitlab.getBranchPr('somebranch'); expect(pr).toBeNull(); }); it('should return null if no PR exists', async () => { get.mockReturnValueOnce({ statusCode: 200 }); // branchExists get.mockReturnValueOnce({ // branchExists body: [], }); const pr = await gitlab.getBranchPr('somebranch'); expect(pr).toBeNull(); }); it('should return the PR object', async () => { get.mockReturnValueOnce({ statusCode: 200 }); // branchExists get.mockReturnValueOnce({ body: [{ number: 91, source_branch: 'somebranch' }], }); get.mockReturnValueOnce({ body: { iid: 91, state: 'opened', additions: 1, deletions: 1, commits: 1, source_branch: 'some-branch', base: { sha: '1234', }, }, }); get.mockReturnValueOnce({ body: [] }); // get branch commit get.mockReturnValueOnce({ body: [{ status: 'success' }] }); // get commit statuses get.mockReturnValueOnce({ body: 'foo' }); const pr = await gitlab.getBranchPr('somebranch'); expect(pr).toMatchSnapshot(); }); }); describe('getBranchStatus(branchName, requiredStatusChecks)', () => { beforeEach(() => { get.mockReturnValueOnce({ body: { commit: { id: 1, }, }, }); }); it('returns success if requiredStatusChecks null', async () => { const res = await gitlab.getBranchStatus('somebranch', null); expect(res).toEqual('success'); }); it('return failed if unsupported requiredStatusChecks', async () => { const res = await gitlab.getBranchStatus('somebranch', ['foo']); expect(res).toEqual('failed'); }); it('returns pending if no results', async () => { get.mockReturnValueOnce({ body: [], }); const res = await gitlab.getBranchStatus('somebranch', []); expect(res).toEqual('pending'); }); it('returns success if all are success', async () => { get.mockReturnValueOnce({ body: [{ status: 'success' }, { status: 'success' }], }); const res = await gitlab.getBranchStatus('somebranch', []); expect(res).toEqual('success'); }); it('returns success if optional jobs fail', async () => { get.mockReturnValueOnce({ body: [ { status: 'success' }, { status: 'failed', allow_failure: true }, ], }); const res = await gitlab.getBranchStatus('somebranch', []); expect(res).toEqual('success'); }); it('returns failure if any mandatory jobs fails', async () => { get.mockReturnValueOnce({ body: [ { status: 'success' }, { status: 'failed', allow_failure: true }, { status: 'failed' }, ], }); const res = await gitlab.getBranchStatus('somebranch', []); expect(res).toEqual('failure'); }); it('returns custom statuses', async () => { get.mockReturnValueOnce({ body: [{ status: 'success' }, { status: 'foo' }], }); const res = await gitlab.getBranchStatus('somebranch', []); expect(res).toEqual('foo'); }); }); describe('getBranchStatusCheck', () => { beforeEach(() => { get.mockReturnValueOnce({ body: { commit: { id: 1, }, }, }); }); it('returns null if no results', async () => { get.mockReturnValueOnce({ body: [], }); const res = await gitlab.getBranchStatusCheck( 'somebranch', 'some-context' ); expect(res).toBeNull(); }); it('returns null if no matching results', async () => { get.mockReturnValueOnce({ body: [{ name: 'context-1', status: 'pending' }], }); const res = await gitlab.getBranchStatusCheck( 'somebranch', 'some-context' ); expect(res).toBeNull(); }); it('returns status if name found', async () => { get.mockReturnValueOnce({ body: [ { name: 'context-1', state: 'pending' }, { name: 'some-context', state: 'success' }, { name: 'context-3', state: 'failed' }, ], }); const res = await gitlab.getBranchStatusCheck( 'somebranch', 'some-context' ); expect(res).toEqual('success'); }); }); describe('setBranchStatus', () => { it('sets branch status', async () => { // getBranchCommit get.mockReturnValueOnce({ body: { commit: { id: 1, }, }, }); await gitlab.setBranchStatus( 'some-branch', 'some-context', 'some-description', 'some-state', 'some-url' ); expect(get.post).toHaveBeenCalledTimes(1); }); }); describe('mergeBranch(branchName)', () => { it('should perform a branch merge', async () => { await initRepo({ repository: 'some/repo', token: 'token', }); await gitlab.mergeBranch('thebranchname'); // deleteBranch get.delete.mockImplementationOnce(); expect(get.post.mock.calls).toMatchSnapshot(); expect(get.delete.mock.calls).toMatchSnapshot(); }); it('should throw if branch merge throws', async () => { await initRepo({ repository: 'some/repo', token: 'token', }); get.post.mockImplementationOnce(() => { throw new Error('branch-push failed'); }); await expect( gitlab.mergeBranch('thebranchname') ).rejects.toMatchSnapshot(); // deleteBranch get.delete.mockImplementationOnce(); expect(get.post.mock.calls).toMatchSnapshot(); expect(get.delete.mock.calls).toMatchSnapshot(); }); }); describe('deleteBranch(branchName)', () => { it('should send delete', async () => { get.delete = jest.fn(); await gitlab.deleteBranch('some-branch'); expect(get.delete).toHaveBeenCalledTimes(1); }); it('should close PR', async () => { get.delete = jest.fn(); get.mockReturnValueOnce({ body: [] }); // getBranchPr await gitlab.deleteBranch('some-branch', true); expect(get.delete).toHaveBeenCalledTimes(1); }); }); describe('getBranchLastCommitTime', () => { it('should return a Date', async () => { get.mockReturnValueOnce({ body: [ { id: 'ed899a2f4b50b4370feeea94676502b42383c746', short_id: 'ed899a2f4b5', title: 'Replace sanitize with escape once', author_name: 'Dmitriy Zaporozhets', author_email: 'dzaporozhets@sphereconsultinginc.com', authored_date: '2012-09-20T11:50:22+03:00', committer_name: 'Administrator', committer_email: 'admin@example.com', committed_date: '2012-09-20T11:50:22+03:00', created_at: '2012-09-20T11:50:22+03:00', message: 'Replace sanitize with escape once', parent_ids: ['6104942438c14ec7bd21c6cd5bd995272b3faff6'], }, { id: '6104942438c14ec7bd21c6cd5bd995272b3faff6', short_id: '6104942438c', title: 'Sanitize for network graph', author_name: 'randx', author_email: 'dmitriy.zaporozhets@gmail.com', committer_name: 'Dmitriy', committer_email: 'dmitriy.zaporozhets@gmail.com', created_at: '2012-09-20T09:06:12+03:00', message: 'Sanitize for network graph', parent_ids: ['ae1d9fb46aa2b07ee9836d49862ec4e2c46fbbba'], }, ], }); const res = await gitlab.getBranchLastCommitTime('some-branch'); expect(res).toMatchSnapshot(); }); it('handles error', async () => { get.mockReturnValueOnce({ body: [], }); const res = await gitlab.getBranchLastCommitTime('some-branch'); expect(res).toBeDefined(); }); }); describe('findIssue()', () => { it('returns null if no issue', async () => { get.mockReturnValueOnce({ body: [ { iid: 1, title: 'title-1', }, { iid: 2, title: 'title-2', }, ], }); const res = await gitlab.findIssue('title-3'); expect(res).toBeNull(); }); it('finds issue', async () => { get.mockReturnValueOnce({ body: [ { iid: 1, title: 'title-1', }, { iid: 2, title: 'title-2', }, ], }); get.mockReturnValueOnce({ body: { description: 'new-content' } }); const res = await gitlab.findIssue('title-2'); expect(res).not.toBeNull(); }); }); describe('ensureIssue()', () => { it('creates issue', async () => { get.mockImplementationOnce(() => ({ body: [ { iid: 1, title: 'title-1', }, { iid: 2, title: 'title-2', }, ], })); const res = await gitlab.ensureIssue('new-title', 'new-content'); expect(res).toEqual('created'); }); it('updates issue', async () => { get.mockReturnValueOnce({ body: [ { iid: 1, title: 'title-1', }, { iid: 2, title: 'title-2', }, ], }); get.mockReturnValueOnce({ body: { description: 'new-content' } }); const res = await gitlab.ensureIssue('title-2', 'newer-content'); expect(res).toEqual('updated'); }); it('skips update if unchanged', async () => { get.mockReturnValueOnce({ body: [ { number: 1, title: 'title-1', }, { number: 2, title: 'title-2', }, ], }); get.mockReturnValueOnce({ body: { description: 'newer-content' } }); const res = await gitlab.ensureIssue('title-2', 'newer-content'); expect(res).toBeNull(); }); }); describe('ensureIssueClosing()', () => { it('closes issue', async () => { get.mockImplementationOnce(() => ({ body: [ { number: 1, title: 'title-1', }, { number: 2, title: 'title-2', }, ], })); await gitlab.ensureIssueClosing('title-2'); }); }); describe('addAssignees(issueNo, assignees)', () => { it('should add the given assignees to the issue', async () => { get.mockReturnValueOnce({ body: [{ id: 123 }], }); await gitlab.addAssignees(42, ['someuser']); expect(get.put.mock.calls).toMatchSnapshot(); }); it('should warn if more than one assignee', async () => { get.mockReturnValueOnce({ body: [{ id: 123 }], }); await gitlab.addAssignees(42, ['someuser', 'someotheruser']); expect(get.put.mock.calls).toMatchSnapshot(); }); it('should swallow error', async () => { get.mockImplementationOnce({}); await gitlab.addAssignees(42, ['someuser', 'someotheruser']); expect(get.put).toHaveBeenCalledTimes(0); }); }); describe('addReviewers(issueNo, reviewers)', () => { it('should add the given reviewers to the PR', async () => { await gitlab.addReviewers(42, ['someuser', 'someotheruser']); }); }); describe('ensureComment', () => { it('add comment if not found', async () => { await initRepo({ repository: 'some/repo', token: 'token' }); get.mockReturnValueOnce({ body: [] }); await gitlab.ensureComment(42, 'some-subject', 'some\ncontent'); expect(get.post).toHaveBeenCalledTimes(1); expect(get.post.mock.calls).toMatchSnapshot(); }); it('add updates comment if necessary', async () => { await initRepo({ repository: 'some/repo', token: 'token' }); get.mockReturnValueOnce({ body: [{ id: 1234, body: '### some-subject\n\nblablabla' }], }); await gitlab.ensureComment(42, 'some-subject', 'some\ncontent'); expect(get.post).toHaveBeenCalledTimes(0); expect(get.put).toHaveBeenCalledTimes(1); expect(get.put.mock.calls).toMatchSnapshot(); }); it('skips comment', async () => { await initRepo({ repository: 'some/repo', token: 'token' }); get.mockReturnValueOnce({ body: [{ id: 1234, body: '### some-subject\n\nsome\ncontent' }], }); await gitlab.ensureComment(42, 'some-subject', 'some\ncontent'); expect(get.post).toHaveBeenCalledTimes(0); expect(get.put).toHaveBeenCalledTimes(0); }); it('handles comment with no description', async () => { await initRepo({ repository: 'some/repo', token: 'token' }); get.mockReturnValueOnce({ body: [{ id: 1234, body: '!merge' }] }); await gitlab.ensureComment(42, null, '!merge'); expect(get.post).toHaveBeenCalledTimes(0); expect(get.put).toHaveBeenCalledTimes(0); }); }); describe('ensureCommentRemoval', () => { it('deletes comment if found', async () => { await initRepo({ repository: 'some/repo', token: 'token' }); get.mockReturnValueOnce({ body: [{ id: 1234, body: '### some-subject\n\nblablabla' }], }); await gitlab.ensureCommentRemoval(42, 'some-subject'); expect(get.delete).toHaveBeenCalledTimes(1); }); }); describe('findPr(branchName, prTitle, state)', () => { it('returns true if no title and all state', async () => { get.mockReturnValueOnce({ body: [ { number: 1, source_branch: 'branch-a', title: 'branch a pr', state: 'opened', }, ], }); const res = await gitlab.findPr('branch-a', null); expect(res).toBeDefined(); }); it('returns true if not open', async () => { get.mockReturnValueOnce({ body: [ { number: 1, source_branch: 'branch-a', title: 'branch a pr', state: 'merged', }, ], }); const res = await gitlab.findPr('branch-a', null, '!open'); expect(res).toBeDefined(); }); it('caches pr list', async () => { get.mockReturnValueOnce({ body: [ { number: 1, source_branch: 'branch-a', title: 'branch a pr', state: 'opened', }, ], }); let res = await gitlab.findPr('branch-a', null); expect(res).toBeDefined(); res = await gitlab.findPr('branch-a', 'branch a pr'); expect(res).toBeDefined(); res = await gitlab.findPr('branch-a', 'branch a pr', 'open'); expect(res).toBeDefined(); res = await gitlab.findPr('branch-b'); expect(res).not.toBeDefined(); }); }); describe('createPr(branchName, title, body)', () => { it('returns the PR', async () => { get.post.mockReturnValueOnce({ body: { id: 1, iid: 12345, }, }); const pr = await gitlab.createPr( 'some-branch', 'some-title', 'the-body', null ); expect(pr).toMatchSnapshot(); expect(get.post.mock.calls).toMatchSnapshot(); }); it('uses default branch', async () => { get.post.mockReturnValueOnce({ body: { id: 1, iid: 12345, }, }); const pr = await gitlab.createPr( 'some-branch', 'some-title', 'the-body', [], true ); expect(pr).toMatchSnapshot(); expect(get.post.mock.calls).toMatchSnapshot(); }); }); describe('getPr(prNo)', () => { it('returns the PR', async () => { get.mockReturnValueOnce({ body: { id: 1, iid: 12345, description: 'a merge request', state: 'merged', merge_status: 'cannot_be_merged', diverged_commits_count: 5, source_branch: 'some-branch', }, }); get.mockReturnValueOnce({ body: { commit: {}, }, }); const pr = await gitlab.getPr(12345); expect(pr).toMatchSnapshot(); }); it('returns the PR with nonexisting branch', async () => { get.mockImplementationOnce(() => ({ body: { id: 1, iid: 12345, description: 'a merge request', state: 'open', merge_status: 'cannot_be_merged', diverged_commits_count: 2, source_branch: 'some-branch', }, })); get.mockImplementationOnce(() => Promise.reject({ statusCode: 404, }) ); const pr = await gitlab.getPr(12345); expect(pr).toMatchSnapshot(); }); }); describe('getPrFiles()', () => { it('should return empty if no mrNo is passed', async () => { const prFiles = await gitlab.getPrFiles(null); expect(prFiles).toEqual([]); }); it('returns files', async () => { get.mockReturnValueOnce({ body: { changes: [ { new_path: 'renovate.json' }, { new_path: 'not renovate.json' }, ], }, }); const prFiles = await gitlab.getPrFiles(123); expect(prFiles).toMatchSnapshot(); expect(prFiles).toHaveLength(2); }); }); describe('updatePr(prNo, title, body)', () => { jest.resetAllMocks(); it('updates the PR', async () => { await gitlab.updatePr(1, 'title', 'body'); expect(get.put.mock.calls.length).toEqual(1); }); }); describe('mergePr(pr)', () => { jest.resetAllMocks(); it('merges the PR', async () => { await gitlab.mergePr({ number: 1 }); expect(get.put.mock.calls.length).toEqual(1); }); }); const prBody = `https://github.com/foo/bar/issues/5 plus also [a link](https://github.com/foo/bar/issues/5 Pull Requests are the best, here are some PRs. ## Open These updates have all been created already. Click a checkbox below to force a retry/rebase of any. - [ ] [build(deps): update got packages (major)](../pull/2433) (\`gh-got\`, \`gl-got\`, \`got\`) `; describe('getPrBody(input)', () => { it('returns updated pr body', () => { expect(gitlab.getPrBody(prBody)).toMatchSnapshot(); }); }); describe('getFile(filePath, branchName)', () => { beforeEach(async () => { get.mockReturnValueOnce({ body: [{ type: 'blob', path: 'some-path' }] }); await gitlab.getFileList(); }); it('gets the file', async () => { get.mockReturnValueOnce({ body: { content: 'foo', }, }); const res = await gitlab.getFile('some-path'); expect(res).toMatchSnapshot(); }); it('short cuts 404', async () => { const res = await gitlab.getFile('some-missing-path'); expect(res).toBeNull(); }); it('returns null for 404', async () => { get.mockImplementationOnce(() => Promise.reject({ statusCode: 404 })); const res = await gitlab.getFile('some-path', 'some-branch'); expect(res).toBeNull(); }); it('throws error for non-404', async () => { get.mockImplementationOnce(() => Promise.reject({ statusCode: 403 })); await expect( gitlab.getFile('some-path', 'some-branch') ).rejects.toMatchSnapshot(); }); }); describe('commitFilesToBranch(branchName, files, message, parentBranch)', () => { it('creates file', async () => { get.mockImplementationOnce(() => Promise.reject({ statusCode: 404 })); // file exists get.mockImplementationOnce(() => Promise.reject({ statusCode: 404, }) ); // branch exists get.mockImplementationOnce(() => Promise.reject({ statusCode: 404, }) ); // branch exists const file = { name: 'some-new-file', contents: 'some new-contents', }; await gitlab.commitFilesToBranch( 'renovate/something', [file], 'Create something' ); expect(get.post.mock.calls).toMatchSnapshot(); expect(get.post).toHaveBeenCalledTimes(1); }); it('updates multiple files', async () => { // Two files exist get.mockReturnValueOnce({ body: { content: 'foo', }, }); get.mockReturnValueOnce({ body: { content: 'foo', }, }); // branch exists get.mockImplementationOnce(() => ({ statusCode: 200 })); get.mockImplementationOnce(() => Promise.reject({ statusCode: 404, }) ); // branch exists const files = [ { name: 'some-existing-file', contents: 'updated content', }, { name: 'some-other-existing-file', contents: 'other updated content', }, ]; await gitlab.commitFilesToBranch( 'renovate/something', files, 'Update something' ); expect(get.post.mock.calls).toMatchSnapshot(); expect(get.post).toHaveBeenCalledTimes(1); }); }); describe('getCommitMessages()', () => { it('returns commits messages', async () => { get.mockReturnValueOnce({ body: [ { title: 'foo', }, { title: 'bar', }, ], }); const res = await gitlab.getCommitMessages(); expect(res).toMatchSnapshot(); }); }); describe('getVulnerabilityAlerts()', () => { it('returns empty', async () => { const res = await gitlab.getVulnerabilityAlerts(); expect(res).toHaveLength(0); }); }); describe('deleteLabel(issueNo, label)', () => { it('should delete the label', async () => { get.mockReturnValueOnce({ body: { id: 1, iid: 12345, description: 'a merge request', state: 'merged', merge_status: 'cannot_be_merged', diverged_commits_count: 5, source_branch: 'some-branch', labels: ['foo', 'renovate', 'rebase'], }, }); get.mockReturnValueOnce({ body: { commit: {}, }, }); await gitlab.deleteLabel(42, 'rebase'); expect(get.put.mock.calls).toMatchSnapshot(); }); }); });