Migration to Lit/Typescript of Code Owners
Also show shortened directory names in code-owners
Google-Bug-Id: b/237256547,b/241905566
Release-Notes: skip
Change-Id: I4d4b3bac9247f34e8710888275a71f85cfb638e4
diff --git a/web/owner-status-column.ts b/web/owner-status-column.ts
new file mode 100644
index 0000000..4b6f44d
--- /dev/null
+++ b/web/owner-status-column.ts
@@ -0,0 +1,259 @@
+/**
+ * @license
+ * Copyright (C) 2020 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 {
+ BasePatchSetNum,
+ RevisionPatchSetNum,
+} from '@gerritcodereview/typescript-api/rest-api';
+import {css, html, LitElement, nothing, PropertyValues} from 'lit';
+import {customElement, property} from 'lit/decorators';
+import {OwnerStatus} from './code-owners-api';
+import {FileStatus} from './code-owners-model';
+import {CodeOwnersModelMixin} from './code-owners-model-mixin';
+
+// TODO: Extend the API for plugins.
+export interface PatchRange {
+ patchNum: RevisionPatchSetNum;
+ basePatchNum: BasePatchSetNum;
+}
+
+const MAGIC_FILES = ['/COMMIT_MSG', '/MERGE_LIST', '/PATCHSET_LEVEL'];
+const STATUS_CODE = {
+ PENDING: 'pending',
+ PENDING_OLD_PATH: 'pending-old-path',
+ MISSING: 'missing',
+ MISSING_OLD_PATH: 'missing-old-path',
+ APPROVED: 'approved',
+ ERROR: 'error',
+ ERROR_OLD_PATH: 'error-old-path',
+};
+
+const STATUS_PRIORITY_ORDER = [
+ STATUS_CODE.ERROR,
+ STATUS_CODE.ERROR_OLD_PATH,
+ STATUS_CODE.MISSING,
+ STATUS_CODE.PENDING,
+ STATUS_CODE.MISSING_OLD_PATH,
+ STATUS_CODE.PENDING_OLD_PATH,
+ STATUS_CODE.APPROVED,
+];
+
+const STATUS_ICON = {
+ [STATUS_CODE.PENDING]: 'schedule',
+ [STATUS_CODE.MISSING]: 'close',
+ [STATUS_CODE.PENDING_OLD_PATH]: 'schedule',
+ [STATUS_CODE.MISSING_OLD_PATH]: ':close',
+ [STATUS_CODE.APPROVED]: 'check',
+ [STATUS_CODE.ERROR]: 'info',
+};
+
+const STATUS_TOOLTIP = {
+ [STATUS_CODE.PENDING]: 'Pending code owner approval',
+ [STATUS_CODE.MISSING]: 'Missing code owner approval',
+ [STATUS_CODE.PENDING_OLD_PATH]:
+ 'Pending code owner approval on pre-renamed file',
+ [STATUS_CODE.MISSING_OLD_PATH]:
+ 'Missing code owner approval on pre-renamed file',
+ [STATUS_CODE.APPROVED]: 'Approved by code owner',
+ [STATUS_CODE.ERROR]: 'Failed to fetch code owner status',
+ [STATUS_CODE.ERROR_OLD_PATH]: 'Failed to fetch code owner status',
+};
+
+const base = CodeOwnersModelMixin(LitElement);
+
+class BaseEl extends base {
+ @property({type: Object})
+ patchRange?: PatchRange;
+
+ computeHidden() {
+ const newerPatchsetUploaded = this.status?.newerPatchsetUploaded;
+ if (
+ this.change === undefined ||
+ this.patchRange === undefined ||
+ newerPatchsetUploaded === undefined
+ ) {
+ return true;
+ }
+ // if code-owners is not a submit requirement, don't show status column
+ if (
+ this.change.requirements &&
+ !this.change.requirements.find(r => r.type === 'code-owners')
+ ) {
+ return true;
+ }
+
+ if (newerPatchsetUploaded) return true;
+
+ const latestPatchset =
+ this.change.revisions![this.change.current_revision!];
+ // Note: in some special cases, patchNum is undefined on latest patchset
+ // like after publishing the edit, still show for them
+ // TODO: this should be fixed in Gerrit
+ if (this.patchRange?.patchNum === undefined) return false;
+ // only show if its latest patchset
+ if (`${this.patchRange.patchNum}` !== `${latestPatchset._number}`)
+ return true;
+ return false;
+ }
+}
+
+export const OWNERS_STATUS_COLUMN_HEADER = 'owner-status-column-header';
+/**
+ * Column header element for owner status.
+ */
+@customElement(OWNERS_STATUS_COLUMN_HEADER)
+export class OwnerStatusColumnHeader extends BaseEl {
+ static override get styles() {
+ return [
+ css`
+ :host() {
+ display: block;
+ padding-right: var(--spacing-m);
+ width: 3em;
+ }
+ `,
+ ];
+ }
+
+ override render() {
+ if (this.computeHidden()) return nothing;
+ return html`<div></div>`;
+ }
+}
+
+export const OWNER_STATUS_COLUMN_CONTENT = 'owner-status-column-content';
+/**
+ * Row content element for owner status.
+ */
+@customElement(OWNER_STATUS_COLUMN_CONTENT)
+export class OwnerStatusColumnContent extends BaseEl {
+ @property({type: String})
+ path?: string;
+
+ @property({type: String})
+ oldPath?: string;
+
+ @property({type: Array})
+ cleanlyMergedPaths?: Array<string>;
+
+ @property({type: Array})
+ cleanlyMergedOldPaths?: Array<string>;
+
+ @property({type: String, reflect: true, attribute: 'owner-status'})
+ ownerStatus?: string;
+
+ static override get styles() {
+ return [
+ css`
+ :host {
+ display: block;
+ padding-right: var(--spacing-m);
+ width: 3em;
+ text-align: center;
+ }
+ iron-icon {
+ padding: var(--spacing-xs) 0px;
+ }
+ :host([owner-status='approved']) iron-icon {
+ color: var(--positive-green-text-color);
+ }
+ :host([owner-status='pending']) iron-icon {
+ color: #ffa62f;
+ }
+ :host([owner-status='missing']) iron-icon,
+ :host([owner-status='error']) iron-icon {
+ color: var(--negative-red-text-color);
+ }
+ `,
+ ];
+ }
+
+ protected override willUpdate(changedProperties: PropertyValues): void {
+ super.willUpdate(changedProperties);
+ this.computeStatus();
+ }
+
+ override render() {
+ if (this.computeHidden() || this.status === undefined) return nothing;
+
+ const statusInfo = this.computeTooltip();
+ const statusIcon = this.computeIcon();
+ return html`
+ <gr-tooltip-content title=${statusInfo} has-tooltip>
+ <gr-icon icon=${statusIcon}></gr-icon>
+ </gr-tooltip-content>
+ `;
+ }
+
+ override loadPropertiesAfterModelChanged() {
+ super.loadPropertiesAfterModelChanged();
+ this.modelLoader?.loadStatus();
+ }
+
+ private computeStatus() {
+ if (
+ this.status === undefined ||
+ (this.cleanlyMergedPaths === undefined &&
+ (this.path === undefined || this.oldPath === undefined))
+ ) {
+ return;
+ }
+
+ const codeOwnerStatusMap = this.status.codeOwnerStatusMap;
+ const paths =
+ this.path === undefined ? this.cleanlyMergedPaths : [this.path];
+ const oldPaths =
+ this.oldPath === undefined ? this.cleanlyMergedOldPaths : [this.oldPath];
+
+ const statuses = (paths ?? [])
+ .filter(path => !MAGIC_FILES.includes(path))
+ .map(path => this.extractStatus(codeOwnerStatusMap.get(path)!, false));
+ // oldPath may contain null, so filter that as well.
+ const oldStatuses = (oldPaths ?? [])
+ .filter(path => !MAGIC_FILES.includes(path) && !!path)
+ .map(path => this.extractStatus(codeOwnerStatusMap.get(path)!, true));
+ const allStatuses = statuses.concat(oldStatuses);
+ if (allStatuses.length === 0) {
+ return;
+ }
+ this.ownerStatus = allStatuses.reduce((a, b) =>
+ STATUS_PRIORITY_ORDER.indexOf(a) < STATUS_PRIORITY_ORDER.indexOf(b)
+ ? a
+ : b
+ );
+ }
+
+ private computeIcon() {
+ return STATUS_ICON[this.ownerStatus!];
+ }
+
+ private computeTooltip() {
+ return STATUS_TOOLTIP[this.ownerStatus!];
+ }
+
+ private extractStatus(statusItem: FileStatus, oldPath: boolean) {
+ if (statusItem === undefined) {
+ return oldPath ? STATUS_CODE.ERROR_OLD_PATH : STATUS_CODE.ERROR;
+ } else if (statusItem.status === OwnerStatus.INSUFFICIENT_REVIEWERS) {
+ return oldPath ? STATUS_CODE.MISSING_OLD_PATH : STATUS_CODE.MISSING;
+ } else if (statusItem.status === OwnerStatus.PENDING) {
+ return oldPath ? STATUS_CODE.PENDING_OLD_PATH : STATUS_CODE.PENDING;
+ } else {
+ return STATUS_CODE.APPROVED;
+ }
+ }
+}