mirror of
https://github.com/Noratrieb/does-it-build.git
synced 2026-01-14 10:25:01 +01:00
159 lines
4.4 KiB
JavaScript
159 lines
4.4 KiB
JavaScript
class Table {
|
|
constructor(data, tableElemId, filterElemId, filterFailedElemId) {
|
|
this.data = data;
|
|
this.elem = document.getElementById(tableElemId);
|
|
|
|
document.getElementById(filterElemId).addEventListener("input", (e) => {
|
|
this.filter.search = e.target.value;
|
|
this.render();
|
|
});
|
|
document
|
|
.getElementById(filterFailedElemId)
|
|
.addEventListener("input", (e) => {
|
|
this.filter.filterFailed = e.target.checked;
|
|
this.render();
|
|
});
|
|
|
|
this.filter = {
|
|
search: "",
|
|
filterFailed: false,
|
|
};
|
|
}
|
|
|
|
update(data) {
|
|
this.data = data;
|
|
}
|
|
|
|
render() {
|
|
const allTargets = new Set();
|
|
const allNightlies = new Set();
|
|
|
|
// The infos grouped by target.
|
|
const targetInfos = new Map();
|
|
|
|
// Targets that have, at some point, errored
|
|
const targetsWithErrors = new Set();
|
|
|
|
// Whether a nightly is completely broken.
|
|
// These are still filtered out when filter failed is selected.
|
|
const isNightlyBroken = new Map();
|
|
|
|
// The first pass over the data, to find nightlies that are broken.
|
|
for (const info of this.data) {
|
|
if (!isNightlyBroken.has(info.nightly)) {
|
|
// Assume that a nightly is broken until proven otherwise.
|
|
isNightlyBroken.set(info.nightly, true);
|
|
}
|
|
if (info.status == "pass") {
|
|
// This nightly has built something, so it's clearly not broken :).
|
|
isNightlyBroken.set(info.nightly, false);
|
|
}
|
|
}
|
|
|
|
// Second pass over the data, group by nightly and prepare data for filter.
|
|
for (const info of this.data) {
|
|
allNightlies.add(info.nightly);
|
|
|
|
if (!info.target.includes(this.filter.search)) {
|
|
continue;
|
|
}
|
|
|
|
if (info.status === "error" && !isNightlyBroken.get(info.nightly)) {
|
|
targetsWithErrors.add(info.target);
|
|
}
|
|
|
|
allTargets.add(info.target);
|
|
if (!targetInfos.has(info.target)) {
|
|
targetInfos.set(info.target, new Map());
|
|
}
|
|
targetInfos.get(info.target).set(info.nightly, info);
|
|
}
|
|
|
|
const nightlies = Array.from(allNightlies);
|
|
nightlies.sort();
|
|
nightlies.reverse();
|
|
const targets = Array.from(allTargets);
|
|
targets.sort();
|
|
|
|
const header = document.createElement("tr");
|
|
const headerTarget = document.createElement("th");
|
|
headerTarget.innerText = "target";
|
|
header.appendChild(headerTarget);
|
|
nightlies.forEach((target) => {
|
|
const elem = document.createElement("th");
|
|
elem.classList.add("target-header");
|
|
elem.innerText = target;
|
|
header.appendChild(elem);
|
|
});
|
|
|
|
const rows = targets.flatMap((target) => {
|
|
if (this.filter.filterFailed && !targetsWithErrors.has(target)) {
|
|
return [];
|
|
}
|
|
|
|
const tr = document.createElement("tr");
|
|
|
|
const targetCol = document.createElement("td");
|
|
targetCol.innerText = target;
|
|
targetCol.classList.add("target-name-col");
|
|
tr.appendChild(targetCol);
|
|
|
|
const info = targetInfos.get(target) ?? new Map();
|
|
|
|
for (const nightly of nightlies) {
|
|
const td = document.createElement("td");
|
|
const targetInfo = info.get(nightly);
|
|
|
|
if (targetInfo) {
|
|
const a = document.createElement("a");
|
|
a.classList.add("build-info-a");
|
|
a.href = `build?nightly=${encodeURIComponent(
|
|
nightly
|
|
)}&target=${encodeURIComponent(target)}&mode=${encodeURIComponent(
|
|
targetInfo.mode
|
|
)}`;
|
|
a.innerText = targetInfo.status == "pass" ? "✅" : "❌";
|
|
td.appendChild(a);
|
|
td.classList.add("build-cell");
|
|
td.classList.add(targetInfo.status);
|
|
} else {
|
|
td.innerText = "";
|
|
td.classList.add("missing");
|
|
}
|
|
tr.appendChild(td);
|
|
}
|
|
|
|
return [tr];
|
|
});
|
|
this.elem.replaceChildren(header, ...rows);
|
|
}
|
|
}
|
|
|
|
const coreTable = new Table(
|
|
[],
|
|
"target-state",
|
|
"target-filter",
|
|
"target-filter-failed"
|
|
);
|
|
const miriTable = new Table(
|
|
[],
|
|
"target-state-miri",
|
|
"target-filter-miri",
|
|
"target-filter-failed-miri"
|
|
);
|
|
|
|
function fetchTargets() {
|
|
fetch("full-mega-monster")
|
|
.then((body) => body.json())
|
|
.then((body) => {
|
|
const core = body.filter((info) => info.mode === "core");
|
|
const miri = body.filter((info) => info.mode === "miri-std");
|
|
coreTable.update(core);
|
|
miriTable.update(miri);
|
|
coreTable.render();
|
|
miriTable.render();
|
|
});
|
|
}
|
|
|
|
// Initial fetch
|
|
fetchTargets();
|