blob: 34cb6eb0e032c505c978db0ccf6a8a461ea57121 [file] [log] [blame]
/**
* @license
* Copyright (C) 2015 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
import '../../../test/common-test-setup-karma';
import {
createAccountWithId,
createChange,
} from '../../../test/test-data-generators';
import {query, queryAndAssert, stubRestApi} from '../../../test/test-utils';
import {
AccountId,
BranchName,
ChangeInfo,
RepoName,
TopicName,
} from '../../../types/common';
import {GerritNav} from '../../core/gr-navigation/gr-navigation';
import {columnNames} from '../gr-change-list/gr-change-list';
import './gr-change-list-item';
import {GrChangeListItem, LabelCategory} from './gr-change-list-item';
const basicFixture = fixtureFromElement('gr-change-list-item');
suite('gr-change-list-item tests', () => {
const account = createAccountWithId();
const change: ChangeInfo = {
...createChange(),
internalHost: 'host',
project: 'a/test/repo' as RepoName,
topic: 'test-topic' as TopicName,
branch: 'test-branch' as BranchName,
};
let element: GrChangeListItem;
setup(() => {
stubRestApi('getLoggedIn').returns(Promise.resolve(false));
element = basicFixture.instantiate();
});
test('_computeLabelCategory', () => {
assert.equal(
element._computeLabelCategory({...change, labels: {}}, 'Verified'),
LabelCategory.NOT_APPLICABLE
);
assert.equal(
element._computeLabelCategory(
{...change, labels: {Verified: {approved: account, value: 1}}},
'Verified'
),
LabelCategory.APPROVED
);
assert.equal(
element._computeLabelCategory(
{...change, labels: {Verified: {rejected: account, value: -1}}},
'Verified'
),
LabelCategory.REJECTED
);
assert.equal(
element._computeLabelCategory(
{
...change,
labels: {'Code-Review': {approved: account, value: 1}},
unresolved_comment_count: 1,
},
'Code-Review'
),
LabelCategory.UNRESOLVED_COMMENTS
);
assert.equal(
element._computeLabelCategory(
{...change, labels: {'Code-Review': {value: 1}}},
'Code-Review'
),
LabelCategory.POSITIVE
);
assert.equal(
element._computeLabelCategory(
{...change, labels: {'Code-Review': {value: -1}}},
'Code-Review'
),
LabelCategory.NEGATIVE
);
assert.equal(
element._computeLabelCategory(
{...change, labels: {'Code-Review': {value: -1}}},
'Verified'
),
LabelCategory.NOT_APPLICABLE
);
});
test('_computeLabelClass', () => {
assert.equal(
element._computeLabelClass({...change, labels: {}}, 'Verified'),
'cell label u-gray-background'
);
assert.equal(
element._computeLabelClass(
{...change, labels: {Verified: {approved: account, value: 1}}},
'Verified'
),
'cell label u-green'
);
assert.equal(
element._computeLabelClass(
{...change, labels: {Verified: {rejected: account, value: -1}}},
'Verified'
),
'cell label u-red'
);
assert.equal(
element._computeLabelClass(
{...change, labels: {'Code-Review': {value: 1}}},
'Code-Review'
),
'cell label u-green u-monospace'
);
assert.equal(
element._computeLabelClass(
{...change, labels: {'Code-Review': {value: -1}}},
'Code-Review'
),
'cell label u-monospace u-red'
);
assert.equal(
element._computeLabelClass(
{...change, labels: {'Code-Review': {value: -1}}},
'Verified'
),
'cell label u-gray-background'
);
});
test('_computeLabelTitle', () => {
assert.equal(
element._computeLabelTitle({...change, labels: {}}, 'Verified'),
'Label not applicable'
);
assert.equal(
element._computeLabelTitle(
{...change, labels: {Verified: {approved: {name: 'Diffy'}}}},
'Verified'
),
'Verified by Diffy'
);
assert.equal(
element._computeLabelTitle(
{...change, labels: {Verified: {approved: {name: 'Diffy'}}}},
'Code-Review'
),
'Label not applicable'
);
assert.equal(
element._computeLabelTitle(
{...change, labels: {Verified: {rejected: {name: 'Diffy'}}}},
'Verified'
),
'Verified by Diffy'
);
assert.equal(
element._computeLabelTitle(
{
...change,
labels: {'Code-Review': {disliked: {name: 'Diffy'}, value: -1}},
},
'Code-Review'
),
'Code-Review by Diffy'
);
assert.equal(
element._computeLabelTitle(
{
...change,
labels: {'Code-Review': {recommended: {name: 'Diffy'}, value: 1}},
},
'Code-Review'
),
'Code-Review by Diffy'
);
assert.equal(
element._computeLabelTitle(
{
...change,
labels: {
'Code-Review': {
recommended: {name: 'Diffy'},
rejected: {name: 'Admin'},
},
},
},
'Code-Review'
),
'Code-Review by Admin'
);
assert.equal(
element._computeLabelTitle(
{
...change,
labels: {
'Code-Review': {
approved: {name: 'Diffy'},
rejected: {name: 'Admin'},
},
},
},
'Code-Review'
),
'Code-Review by Admin'
);
assert.equal(
element._computeLabelTitle(
{
...change,
labels: {
'Code-Review': {
recommended: {name: 'Diffy'},
disliked: {name: 'Admin'},
value: -1,
},
},
},
'Code-Review'
),
'Code-Review by Admin'
);
assert.equal(
element._computeLabelTitle(
{
...change,
labels: {
'Code-Review': {
approved: {name: 'Diffy'},
disliked: {name: 'Admin'},
value: -1,
},
},
},
'Code-Review'
),
'Code-Review by Diffy'
);
assert.equal(
element._computeLabelTitle(
{
...change,
labels: {'Code-Review': {approved: account, value: 1}},
unresolved_comment_count: 1,
},
'Code-Review'
),
'1 unresolved comment'
);
assert.equal(
element._computeLabelTitle(
{
...change,
labels: {'Code-Review': {approved: {name: 'Diffy'}, value: 1}},
unresolved_comment_count: 1,
},
'Code-Review'
),
'1 unresolved comment,\nCode-Review by Diffy'
);
assert.equal(
element._computeLabelTitle(
{
...change,
labels: {'Code-Review': {approved: account, value: 1}},
unresolved_comment_count: 2,
},
'Code-Review'
),
'2 unresolved comments'
);
});
test('_computeLabelIcon', () => {
assert.equal(
element._computeLabelIcon({...change, labels: {}}, 'missingLabel'),
''
);
assert.equal(
element._computeLabelIcon(
{...change, labels: {Verified: {approved: account, value: 1}}},
'Verified'
),
'gr-icons:check'
);
assert.equal(
element._computeLabelIcon(
{
...change,
labels: {'Code-Review': {approved: account, value: 1}},
unresolved_comment_count: 1,
},
'Code-Review'
),
'gr-icons:comment'
);
});
test('_computeLabelValue', () => {
assert.equal(
element._computeLabelValue({...change, labels: {}}, 'Verified'),
''
);
assert.equal(
element._computeLabelValue(
{...change, labels: {Verified: {approved: account, value: 1}}},
'Verified'
),
'✓'
);
assert.equal(
element._computeLabelValue(
{...change, labels: {Verified: {value: 1}}},
'Verified'
),
'+1'
);
assert.equal(
element._computeLabelValue(
{...change, labels: {Verified: {value: -1}}},
'Verified'
),
'-1'
);
assert.equal(
element._computeLabelValue(
{...change, labels: {Verified: {approved: account}}},
'Verified'
),
'✓'
);
assert.equal(
element._computeLabelValue(
{...change, labels: {Verified: {rejected: account}}},
'Verified'
),
'✕'
);
});
test('no hidden columns', async () => {
element.visibleChangeTableColumns = [
'Subject',
'Status',
'Owner',
'Assignee',
'Reviewers',
'Comments',
'Repo',
'Branch',
'Updated',
'Size',
];
await flush();
for (const column of columnNames) {
const elementClass = '.' + column.toLowerCase();
assert.isFalse(
queryAndAssert(element, elementClass).hasAttribute('hidden')
);
}
});
test('repo column hidden', async () => {
element.visibleChangeTableColumns = [
'Subject',
'Status',
'Owner',
'Assignee',
'Reviewers',
'Comments',
'Branch',
'Updated',
'Size',
];
await flush();
for (const column of columnNames) {
const elementClass = '.' + column.toLowerCase();
if (column === 'Repo') {
assert.isTrue(
queryAndAssert(element, elementClass).hasAttribute('hidden')
);
} else {
assert.isFalse(
queryAndAssert(element, elementClass).hasAttribute('hidden')
);
}
}
});
function checkComputeReviewers(
userId: number | undefined,
reviewerIds: number[],
reviewerNames: (string | undefined)[],
attSetIds: number[],
expected: number[]
) {
element.account = userId ? {_account_id: userId as AccountId} : null;
element.change = {
...change,
owner: {
_account_id: 99 as AccountId,
},
reviewers: {
REVIEWER: [],
},
attention_set: {},
};
for (let i = 0; i < reviewerIds.length; i++) {
element.change!.reviewers.REVIEWER!.push({
_account_id: reviewerIds[i] as AccountId,
name: reviewerNames[i],
});
}
attSetIds.forEach(id => (element.change!.attention_set![id] = {account}));
const actual = element
._computeReviewers(element.change)
.map(r => r._account_id);
assert.deepEqual(actual, expected as AccountId[]);
}
test('compute reviewers', () => {
checkComputeReviewers(undefined, [], [], [], []);
checkComputeReviewers(1, [], [], [], []);
checkComputeReviewers(1, [2], ['a'], [], [2]);
checkComputeReviewers(1, [2, 3], [undefined, 'a'], [], [2, 3]);
checkComputeReviewers(1, [2, 3], ['a', undefined], [], [3, 2]);
checkComputeReviewers(1, [99], ['owner'], [], []);
checkComputeReviewers(
1,
[2, 3, 4, 5],
['b', 'a', 'd', 'c'],
[3, 4],
[3, 4, 2, 5]
);
checkComputeReviewers(
1,
[2, 3, 1, 4, 5],
['b', 'a', 'x', 'd', 'c'],
[3, 4],
[1, 3, 4, 2, 5]
);
});
test('random column does not exist', async () => {
element.visibleChangeTableColumns = ['Bad'];
await flush();
const elementClass = '.bad';
assert.isNotOk(query(element, elementClass));
});
test('assignee only displayed if there is one', async () => {
element.change = change;
await flush();
assert.isNotOk(query(element, '.assignee gr-account-link'));
assert.equal(
queryAndAssert(element, '.assignee').textContent!.trim(),
'--'
);
element.change = {
...change,
assignee: {
name: 'test',
status: 'test',
},
};
await flush();
queryAndAssert(element, '.assignee gr-account-link');
});
test('TShirt sizing tooltip', () => {
assert.equal(
element._computeSizeTooltip({
...change,
insertions: NaN,
deletions: NaN,
}),
'Size unknown'
);
assert.equal(
element._computeSizeTooltip({...change, insertions: 0, deletions: 0}),
'Size unknown'
);
assert.equal(
element._computeSizeTooltip({...change, insertions: 1, deletions: 2}),
'added 1, removed 2 lines'
);
});
test('TShirt sizing', () => {
assert.equal(
element._computeChangeSize({
...change,
insertions: NaN,
deletions: NaN,
}),
null
);
assert.equal(
element._computeChangeSize({...change, insertions: 1, deletions: 1}),
'XS'
);
assert.equal(
element._computeChangeSize({...change, insertions: 9, deletions: 1}),
'S'
);
assert.equal(
element._computeChangeSize({...change, insertions: 10, deletions: 200}),
'M'
);
assert.equal(
element._computeChangeSize({...change, insertions: 99, deletions: 900}),
'L'
);
assert.equal(
element._computeChangeSize({...change, insertions: 99, deletions: 999}),
'XL'
);
});
test('change params passed to gr-navigation', async () => {
const navStub = sinon.stub(GerritNav);
element.change = change;
await flush();
assert.deepEqual(navStub.getUrlForChange.lastCall.args, [change]);
assert.deepEqual(navStub.getUrlForProjectChanges.lastCall.args, [
change.project,
true,
change.internalHost,
]);
assert.deepEqual(navStub.getUrlForBranch.lastCall.args, [
change.branch,
change.project,
undefined,
change.internalHost,
]);
assert.deepEqual(navStub.getUrlForTopic.lastCall.args, [
change.topic,
change.internalHost,
]);
});
test('_computeRepoDisplay', () => {
assert.equal(element._computeRepoDisplay(change), 'host/a/test/repo');
assert.equal(
element._computeTruncatedRepoDisplay(change),
'host/…/test/repo'
);
delete change.internalHost;
assert.equal(element._computeRepoDisplay(change), 'a/test/repo');
assert.equal(element._computeTruncatedRepoDisplay(change), '…/test/repo');
});
});