blob: 1bef599cacfcde2845dad6f3b4a4384494df2867 [file] [log] [blame]
/**
* @license
* Copyright 2020 Google LLC
* SPDX-License-Identifier: Apache-2.0
*/
import {
AccountId,
AccountInfo,
ChangeInfo,
EmailAddress,
GroupId,
GroupInfo,
isAccount,
isDetailedLabelInfo,
isGroup,
ReviewerInput,
ServerInfo,
} from '../types/common';
import {AccountTag, ReviewerState} from '../constants/constants';
import {assertNever, hasOwnProperty} from './common-util';
import {AccountAddition} from '../elements/shared/gr-account-list/gr-account-list';
import {getDisplayName} from './display-name-util';
import {getApprovalInfo} from './label-util';
export const ACCOUNT_TEMPLATE_REGEX = '<GERRIT_ACCOUNT_(\\d+)>';
export function accountKey(account: AccountInfo): AccountId | EmailAddress {
if (account._account_id !== undefined) return account._account_id;
if (account.email) return account.email;
throw new Error('Account has neither _account_id nor email.');
}
export function mapReviewer(addition: AccountAddition): ReviewerInput {
if (addition.account) {
return {reviewer: accountKey(addition.account)};
}
if (addition.group) {
const reviewer = decodeURIComponent(addition.group.id) as GroupId;
const confirmed = addition.group.confirmed;
return {reviewer, confirmed};
}
throw new Error('Reviewer must be either an account or a group.');
}
export function isReviewerOrCC(
change: ChangeInfo,
reviewerAddition: AccountAddition
): boolean {
const reviewers = [
...(change.reviewers[ReviewerState.CC] ?? []),
...(change.reviewers[ReviewerState.REVIEWER] ?? []),
];
const reviewer = mapReviewer(reviewerAddition);
return reviewers.some(r => accountOrGroupKey(r) === reviewer.reviewer);
}
export function isServiceUser(account?: AccountInfo): boolean {
return !!account?.tags?.includes(AccountTag.SERVICE_USER);
}
export function isSelf(account?: AccountInfo, self?: AccountInfo): boolean {
return account?._account_id === self?._account_id;
}
export function removeServiceUsers(accounts?: AccountInfo[]): AccountInfo[] {
return accounts?.filter(a => !isServiceUser(a)) || [];
}
export function hasSameAvatar(account?: AccountInfo, other?: AccountInfo) {
return account?.avatars?.[0]?.url === other?.avatars?.[0]?.url;
}
export function accountOrGroupKey(entry: AccountInfo | GroupInfo) {
if (isAccount(entry)) return accountKey(entry);
if (isGroup(entry)) return entry.id;
assertNever(entry, 'entry must be account or group');
}
export function uniqueDefinedAvatar(
account: AccountInfo,
index: number,
accountArray: AccountInfo[]
) {
return (
index === accountArray.findIndex(other => hasSameAvatar(account, other))
);
}
/**
* Get account in pseudonymized form, that can be send to the backend.
*
* If account is not present, returns anonymous user name according to config.
*/
export function getAccountTemplate(account?: AccountInfo, config?: ServerInfo) {
return account?._account_id
? `<GERRIT_ACCOUNT_${account._account_id}>`
: getDisplayName(config);
}
/**
* Replace account templates with user display names in text, received from the backend.
*/
export function replaceTemplates(
text: string,
accountsInText?: AccountInfo[],
config?: ServerInfo
) {
return text.replace(
new RegExp(ACCOUNT_TEMPLATE_REGEX, 'g'),
(_accountIdTemplate, accountId) => {
const parsedAccountId = Number(accountId) as AccountId;
const accountInText = (accountsInText || []).find(
account => account._account_id === parsedAccountId
);
if (!accountInText) {
return `Gerrit Account ${parsedAccountId}`;
}
return getDisplayName(config, accountInText);
}
);
}
/**
* Returns max permitted score for reviewer.
*/
const getReviewerPermittedScore = (
change: ChangeInfo,
reviewer: AccountInfo,
label: string
) => {
// Note (issue 7874): sometimes the "all" list is not included in change
// detail responses, even when DETAILED_LABELS is included in options.
if (!change?.labels) {
return NaN;
}
const detailedLabel = change.labels[label];
if (!isDetailedLabelInfo(detailedLabel) || !detailedLabel.all) {
return NaN;
}
const approvalInfo = getApprovalInfo(detailedLabel, reviewer);
if (!approvalInfo) {
return NaN;
}
if (hasOwnProperty(approvalInfo, 'permitted_voting_range')) {
if (!approvalInfo.permitted_voting_range) return NaN;
return approvalInfo.permitted_voting_range.max;
} else if (hasOwnProperty(approvalInfo, 'value')) {
// If present, user can vote on the label.
return 0;
}
return NaN;
};
/**
* Explains which labels the user can vote on and which score they can
* give.
*/
export function computeVoteableText(change: ChangeInfo, reviewer: AccountInfo) {
if (!change || !change.labels) {
return '';
}
const maxScores = [];
for (const label of Object.keys(change.labels)) {
const maxScore = getReviewerPermittedScore(change, reviewer, label);
if (isNaN(maxScore) || maxScore < 0) {
continue;
}
const scoreLabel = maxScore > 0 ? `+${maxScore}` : `${maxScore}`;
maxScores.push(`${label}: ${scoreLabel}`);
}
return maxScores.join(', ');
}