blob: 45138a9910b35de1e71e81b6dffbe2b96950563b [file] [log] [blame]
/**
* @license
* Copyright 2016 Google LLC
* SPDX-License-Identifier: Apache-2.0
*/
import '../../../test/common-test-setup-karma';
import './gr-change-list-view';
import {GrChangeListView} from './gr-change-list-view';
import {page} from '../../../utils/page-wrapper-utils';
import {GerritNav} from '../../core/gr-navigation/gr-navigation';
import {
mockPromise,
query,
stubRestApi,
queryAndAssert,
stubFlags,
} from '../../../test/test-utils';
import {createChange} from '../../../test/test-data-generators';
import {
ChangeInfo,
EmailAddress,
NumericChangeId,
RepoName,
} from '../../../api/rest-api';
import {fixture, html, waitUntil, assert} from '@open-wc/testing';
import {GerritView} from '../../../services/router/router-model';
const CHANGE_ID = 'IcA3dAB3edAB9f60B8dcdA6ef71A75980e4B7127';
const COMMIT_HASH = '12345678';
suite('gr-change-list-view tests', () => {
let element: GrChangeListView;
setup(async () => {
stubRestApi('getLoggedIn').returns(Promise.resolve(false));
stubRestApi('getChanges').returns(Promise.resolve([]));
stubRestApi('getAccountDetails').returns(Promise.resolve(undefined));
stubRestApi('getAccountStatus').returns(Promise.resolve(undefined));
element = await fixture(html`<gr-change-list-view></gr-change-list-view>`);
});
teardown(async () => {
await element.updateComplete;
});
test('render', async () => {
element.changes = Array(25)
.fill(0)
.map(_ => createChange());
element.changesPerPage = 10;
element.loading = false;
await element.updateComplete;
assert.shadowDom.equal(
element,
/* HTML */ `
<div class="loading" hidden="">Loading...</div>
<div>
<gr-change-list> </gr-change-list>
<nav>
Page
<a href="" id="prevArrow">
<gr-icon icon="chevron_left" aria-label="Older"></gr-icon>
</a>
</nav>
</div>
`
);
});
suite('bulk actions', () => {
let getChangesStub: sinon.SinonStub;
setup(async () => {
stubFlags('isEnabled').returns(true);
getChangesStub = sinon.stub(element, 'getChanges');
getChangesStub.returns(Promise.resolve([createChange()]));
element.loading = false;
element.reload();
await waitUntil(() => element.loading === false);
element.requestUpdate();
await element.updateComplete;
});
test('checkboxes remain checked after soft reload', async () => {
let checkbox = queryAndAssert<HTMLInputElement>(
query(
query(query(element, 'gr-change-list'), 'gr-change-list-section'),
'gr-change-list-item'
),
'.selection > label > input'
);
checkbox.click();
await waitUntil(() => checkbox.checked);
getChangesStub.restore();
getChangesStub.returns(Promise.resolve([[createChange()]]));
element.reload();
await element.updateComplete;
checkbox = queryAndAssert<HTMLInputElement>(
query(
query(query(element, 'gr-change-list'), 'gr-change-list-section'),
'gr-change-list-item'
),
'.selection > label > input'
);
assert.isTrue(checkbox.checked);
});
});
test('computePage', () => {
element.offset = 0;
element.changesPerPage = 25;
assert.equal(element.computePage(), 1);
element.offset = 50;
element.changesPerPage = 25;
assert.equal(element.computePage(), 3);
});
test('limitFor', () => {
const defaultLimit = 25;
const limitFor = (q: string) => element.limitFor(q, defaultLimit);
assert.equal(limitFor(''), defaultLimit);
assert.equal(limitFor('limit:10'), 10);
assert.equal(limitFor('xlimit:10'), defaultLimit);
assert.equal(limitFor('x(limit:10'), 10);
});
test('computeNavLink', () => {
const getUrlStub = sinon
.stub(GerritNav, 'getUrlForSearchQuery')
.returns('');
element.query = 'status:open';
element.offset = 0;
element.changesPerPage = 5;
let direction = 1;
element.computeNavLink(direction);
assert.equal(getUrlStub.lastCall.args[1], 5);
direction = -1;
element.computeNavLink(direction);
assert.equal(getUrlStub.lastCall.args[1], 0);
element.offset = 5;
direction = 1;
element.computeNavLink(direction);
assert.equal(getUrlStub.lastCall.args[1], 10);
});
test('prevArrow', async () => {
element.changes = Array(25)
.fill(0)
.map(_ => createChange());
element.offset = 0;
element.loading = false;
await element.updateComplete;
assert.isNotOk(query(element, '#prevArrow'));
element.offset = 5;
await element.updateComplete;
assert.isOk(query(element, '#prevArrow'));
});
test('nextArrow', async () => {
element.changes = Array(25)
.fill(0)
.map(_ => ({...createChange(), _more_changes: true} as ChangeInfo));
element.loading = false;
await element.updateComplete;
assert.isOk(query(element, '#nextArrow'));
element.changes = Array(25)
.fill(0)
.map(_ => createChange());
await element.updateComplete;
assert.isNotOk(query(element, '#nextArrow'));
});
test('handleNextPage', async () => {
const showStub = sinon.stub(page, 'show');
element.changes = Array(25)
.fill(0)
.map(_ => createChange());
element.changesPerPage = 10;
element.loading = false;
await element.updateComplete;
element.handleNextPage();
assert.isFalse(showStub.called);
element.changes = Array(25)
.fill(0)
.map(_ => ({...createChange(), _more_changes: true} as ChangeInfo));
element.loading = false;
await element.updateComplete;
element.handleNextPage();
assert.isTrue(showStub.called);
});
test('handlePreviousPage', async () => {
const showStub = sinon.stub(page, 'show');
element.offset = 0;
element.changes = Array(25)
.fill(0)
.map(_ => createChange());
element.changesPerPage = 10;
element.loading = false;
await element.updateComplete;
element.handlePreviousPage();
assert.isFalse(showStub.called);
element.offset = 25;
await element.updateComplete;
element.handlePreviousPage();
assert.isTrue(showStub.called);
});
test('userId query', async () => {
assert.isNull(element.userId);
element.query = 'owner: foo@bar';
element.changes = [
{...createChange(), owner: {email: 'foo@bar' as EmailAddress}},
];
await element.updateComplete;
assert.equal(element.userId, 'foo@bar' as EmailAddress);
element.query = 'foo bar baz';
element.changes = [
{...createChange(), owner: {email: 'foo@bar' as EmailAddress}},
];
await element.updateComplete;
assert.isNull(element.userId);
});
test('userId query without email', async () => {
assert.isNull(element.userId);
element.query = 'owner: foo@bar';
element.changes = [{...createChange(), owner: {}}];
await element.updateComplete;
assert.isNull(element.userId);
});
test('repo query', async () => {
assert.isNull(element.repo);
element.query = 'project: test-repo';
element.changes = [
{
...createChange(),
owner: {email: 'foo@bar' as EmailAddress},
project: 'test-repo' as RepoName,
},
];
await element.updateComplete;
assert.equal(element.repo, 'test-repo' as RepoName);
element.query = 'foo bar baz';
element.changes = [
{...createChange(), owner: {email: 'foo@bar' as EmailAddress}},
];
await element.updateComplete;
assert.isNull(element.repo);
});
test('repo query with open status', async () => {
assert.isNull(element.repo);
element.query = 'project:test-repo status:open';
element.changes = [
{
...createChange(),
owner: {email: 'foo@bar' as EmailAddress},
project: 'test-repo' as RepoName,
},
];
await element.updateComplete;
assert.equal(element.repo, 'test-repo' as RepoName);
element.query = 'foo bar baz';
element.changes = [
{...createChange(), owner: {email: 'foo@bar' as EmailAddress}},
];
await element.updateComplete;
assert.isNull(element.repo);
});
suite('query based navigation', () => {
setup(() => {});
teardown(async () => {
await element.updateComplete;
sinon.restore();
});
test('Searching for a change ID redirects to change', async () => {
const change = {...createChange(), _number: 1 as NumericChangeId};
sinon.stub(element, 'getChanges').returns(Promise.resolve([change]));
const promise = mockPromise();
sinon.stub(GerritNav, 'navigateToChange').callsFake((url, opt) => {
assert.equal(url, change);
// eslint-disable-next-line @typescript-eslint/no-unnecessary-type-assertion
assert.isTrue(opt!.redirect);
promise.resolve();
});
element.params = {
view: GerritView.SEARCH,
query: CHANGE_ID,
offset: '',
};
await promise;
});
test('Searching for a change num redirects to change', async () => {
const change = {...createChange(), _number: 1 as NumericChangeId};
sinon.stub(element, 'getChanges').returns(Promise.resolve([change]));
const promise = mockPromise();
sinon.stub(GerritNav, 'navigateToChange').callsFake((url, opt) => {
assert.equal(url, change);
// eslint-disable-next-line @typescript-eslint/no-unnecessary-type-assertion
assert.isTrue(opt!.redirect);
promise.resolve();
});
element.params = {view: GerritView.SEARCH, query: '1', offset: ''};
await promise;
});
test('Commit hash redirects to change', async () => {
const change = {...createChange(), _number: 1 as NumericChangeId};
sinon.stub(element, 'getChanges').returns(Promise.resolve([change]));
const promise = mockPromise();
sinon.stub(GerritNav, 'navigateToChange').callsFake((url, opt) => {
assert.equal(url, change);
// eslint-disable-next-line @typescript-eslint/no-unnecessary-type-assertion
assert.isTrue(opt!.redirect);
promise.resolve();
});
element.params = {
view: GerritView.SEARCH,
query: COMMIT_HASH,
offset: '',
};
await promise;
});
test('Searching for an invalid change ID searches', async () => {
sinon.stub(element, 'getChanges').returns(Promise.resolve([]));
const stub = sinon.stub(GerritNav, 'navigateToChange');
element.params = {
view: GerritView.SEARCH,
query: CHANGE_ID,
offset: '',
};
await element.updateComplete;
assert.isFalse(stub.called);
});
test('Change ID with multiple search results searches', async () => {
sinon.stub(element, 'getChanges').returns(Promise.resolve(undefined));
const stub = sinon.stub(GerritNav, 'navigateToChange');
element.params = {
view: GerritView.SEARCH,
query: CHANGE_ID,
offset: '',
};
await element.updateComplete;
assert.isFalse(stub.called);
});
});
});