|  | /** | 
|  | * @module modules/byblame-page-sk/byblameentry-sk | 
|  | * @description <h2><code>byblameentry-sk</code></h2> | 
|  | * | 
|  | * Displays a | 
|  | * [ByBlameEntry]{@link https://github.com/google/skia-buildbot/blob/0e14ae66aa226821e981bbd4c63dc8d07776997a/golden/go/web/web.go#L318}, | 
|  | * that is, a blame group of untriaged digests. | 
|  | */ | 
|  |  | 
|  | import { html } from 'lit/html.js'; | 
|  | import { define } from '../../../elements-sk/modules/define'; | 
|  | import { diffDate } from '../../../infra-sk/modules/human'; | 
|  | import { ElementSk } from '../../../infra-sk/modules/ElementSk'; | 
|  | import { baseRepoURL } from '../settings'; | 
|  | import { ByBlameEntry, Commit, TestRollup } from '../rpc_types'; | 
|  | import { detailHref } from '../common'; | 
|  |  | 
|  | const MAX_COMMITS = 10; | 
|  |  | 
|  | const commitHref = (hash: string) => { | 
|  | const repo = baseRepoURL(); | 
|  | if (!hash || !repo) { | 
|  | return ''; | 
|  | } | 
|  | const path = repo.indexOf('github.com') !== -1 ? 'commit' : '+show'; | 
|  | return `${repo}/${path}/${hash}`; | 
|  | }; | 
|  |  | 
|  | export class ByBlameEntrySk extends ElementSk { | 
|  | private static template = (el: ByBlameEntrySk) => html` | 
|  | <div class="blame"> | 
|  | <p> | 
|  | <a href=${el.blameHref()} class="triage" target="_blank" rel="noopener"> | 
|  | ${el.byBlameEntry!.nDigests === 1 | 
|  | ? '1 untriaged digest' | 
|  | : `${el.byBlameEntry!.nDigests} untriaged digests`} | 
|  | </a> | 
|  | </p> | 
|  |  | 
|  | ${ByBlameEntrySk.blameListTemplate(el.byBlameEntry?.commits)} | 
|  |  | 
|  | <h3>Tests affected</h3> | 
|  | <p class="num-tests-affected"> | 
|  | ${el.byBlameEntry!.nTests === 1 | 
|  | ? '1 test affected.' | 
|  | : `${el.byBlameEntry!.nTests} tests affected.`} | 
|  | </p> | 
|  |  | 
|  | ${ByBlameEntrySk.affectedTestsTemplate(el.byBlameEntry?.affectedTests)} | 
|  | </div> | 
|  | `; | 
|  |  | 
|  | private static blameListTemplate = (commits?: Commit[] | null) => { | 
|  | if (!commits || commits.length === 0) { | 
|  | return html`<p class="no-blamelist">No blamelist.</p>`; | 
|  | } | 
|  | const andNMore = commits.length - MAX_COMMITS; | 
|  |  | 
|  | commits = commits.slice(0, MAX_COMMITS); | 
|  |  | 
|  | return html` <h3>Blame</h3> | 
|  |  | 
|  | <ul class="blames"> | 
|  | ${commits.map( | 
|  | (commit: Commit) => | 
|  | html` <li> | 
|  | <a href=${commitHref(commit.hash)} target="_blank" rel="noopener"> | 
|  | ${commit.hash.slice(0, 7)} | 
|  | </a> | 
|  | <span class="commit-message"> ${commit.message} </span> | 
|  | <br /> | 
|  | <small> | 
|  | <span class="author">${commit.author}</span>, | 
|  | <span class="age"> | 
|  | ${diffDate(commit.commit_time * 1000)} | 
|  | </span> | 
|  | ago. | 
|  | </small> | 
|  | </li>` | 
|  | )} | 
|  | ${andNMore > 0 ? html`<li>And ${andNMore} other commit(s)</li>` : ''} | 
|  | </ul>`; | 
|  | }; | 
|  |  | 
|  | private static affectedTestsTemplate = ( | 
|  | affectedTests: TestRollup[] | undefined | null | 
|  | ) => { | 
|  | if (!affectedTests || affectedTests.length === 0) return ''; | 
|  | return html` | 
|  | <table class="affected-tests"> | 
|  | <thead> | 
|  | <tr> | 
|  | <th>Test</th> | 
|  | <th>Corpus</th> | 
|  | <th># Digests</th> | 
|  | <th>Example</th> | 
|  | </tr> | 
|  | </thead> | 
|  | <tbody> | 
|  | ${affectedTests.map( | 
|  | (test: TestRollup) => | 
|  | html` <tr> | 
|  | <td class="test">${test.grouping.name}</td> | 
|  | <td class="corpus">${test.grouping.source_type}</td> | 
|  | <td class="num-digests">${test.num}</td> | 
|  | <td> | 
|  | <a | 
|  | href=${detailHref(test.grouping, test.sample_digest)} | 
|  | class="example-link" | 
|  | target="_blank" | 
|  | rel="noopener"> | 
|  | ${test.sample_digest} | 
|  | </a> | 
|  | </td> | 
|  | </tr>` | 
|  | )} | 
|  | </tbody> | 
|  | </table> | 
|  | `; | 
|  | }; | 
|  |  | 
|  | private _byBlameEntry: ByBlameEntry | null = null; | 
|  |  | 
|  | private _corpus = ''; | 
|  |  | 
|  | constructor() { | 
|  | super(ByBlameEntrySk.template); | 
|  | } | 
|  |  | 
|  | connectedCallback(): void { | 
|  | super.connectedCallback(); | 
|  | this._render(); | 
|  | } | 
|  |  | 
|  | /** A ByBlameEntry object returned by the /json/v1/byblame RPC endpoint. */ | 
|  | get byBlameEntry(): ByBlameEntry | null { | 
|  | return this._byBlameEntry; | 
|  | } | 
|  |  | 
|  | set byBlameEntry(v: ByBlameEntry | null) { | 
|  | this._byBlameEntry = v; | 
|  | this._render(); | 
|  | } | 
|  |  | 
|  | /** The corpus corresponding to this blame group. */ | 
|  | get corpus(): string { | 
|  | return this._corpus; | 
|  | } | 
|  |  | 
|  | set corpus(v: string) { | 
|  | this._corpus = v; | 
|  | this._render(); | 
|  | } | 
|  |  | 
|  | private blameHref() { | 
|  | const blameID = this.byBlameEntry!.groupID; | 
|  |  | 
|  | return `/search?blame=${blameID}&corpus=${this.corpus}`; | 
|  | } | 
|  | } | 
|  |  | 
|  | define('byblameentry-sk', ByBlameEntrySk); |