render refs in log viewer

This commit is contained in:
Bruno Windels 2021-02-23 22:11:01 +01:00
parent 851c469727
commit 113b47540a
2 changed files with 98 additions and 30 deletions

View file

@ -110,7 +110,7 @@
margin: 0; margin: 0;
} }
.timeline div.item { .timeline .item {
--hue: 100deg; --hue: 100deg;
--brightness: 80%; --brightness: 80%;
background-color: hsl(var(--hue), 60%, var(--brightness)); background-color: hsl(var(--hue), 60%, var(--brightness));
@ -121,15 +121,16 @@
margin: 1px; margin: 1px;
flex: 1; flex: 1;
min-width: 0; min-width: 0;
cursor: pointer; color: inherit;
text-decoration: none;
} }
.timeline div.item:not(.has-children) { .timeline .item:not(.has-children) {
margin-left: calc(24px + 4px + 1px); margin-left: calc(24px + 4px + 1px);
} }
.timeline div.item .caption { .timeline .item .caption {
white-space: nowrap; white-space: nowrap;
text-overflow: ellipsis; text-overflow: ellipsis;
overflow: hidden; overflow: hidden;
@ -157,15 +158,15 @@
color: white; color: white;
} }
.timeline div.item.type-network { .timeline .item.type-network {
--hue: 30deg; --hue: 30deg;
} }
.timeline div.item.type-navigation { .timeline .item.type-navigation {
--hue: 200deg; --hue: 200deg;
} }
.timeline div.item.selected { .timeline .item.selected {
background-color: Highlight; background-color: Highlight;
border-color: Highlight; border-color: Highlight;
color: HighlightText; color: HighlightText;

View file

@ -21,36 +21,54 @@ const main = document.querySelector("main");
let selectedItemNode; let selectedItemNode;
let rootItem; let rootItem;
let itemByRef;
const logLevels = [undefined, "All", "Debug", "Detail", "Info", "Warn", "Error", "Fatal", "Off"]; const logLevels = [undefined, "All", "Debug", "Detail", "Info", "Warn", "Error", "Fatal", "Off"];
main.addEventListener("click", event => { main.addEventListener("click", event => {
if (selectedItemNode) {
selectedItemNode.classList.remove("selected");
selectedItemNode = null;
}
if (event.target.classList.contains("toggleExpanded")) { if (event.target.classList.contains("toggleExpanded")) {
const li = event.target.parentElement.parentElement; const li = event.target.parentElement.parentElement;
li.classList.toggle("expanded"); li.classList.toggle("expanded");
} else { } else {
// allow clicking any links other than .item in the timeline, like refs
if (event.target.tagName === "A" && !event.target.classList.contains("item")) {
return;
}
const itemNode = event.target.closest(".item"); const itemNode = event.target.closest(".item");
if (itemNode) { if (itemNode) {
// we don't want scroll to jump when clicking
// so prevent default behaviour, and select and push to history manually
event.preventDefault();
selectNode(itemNode);
history.pushState(null, null, `#${itemNode.id}`);
}
}
});
window.addEventListener("hashchange", () => {
const id = window.location.hash.substr(1);
const itemNode = document.getElementById(id);
if (itemNode && itemNode.closest("main")) {
selectNode(itemNode);
itemNode.scrollIntoView({behavior: "smooth", block: "nearest"});
}
});
function selectNode(itemNode) {
if (selectedItemNode) {
selectedItemNode.classList.remove("selected");
}
selectedItemNode = itemNode; selectedItemNode = itemNode;
selectedItemNode.classList.add("selected"); selectedItemNode.classList.add("selected");
const path = selectedItemNode.dataset.path;
let item = rootItem; let item = rootItem;
let parent; let parent;
if (path.length) { const indices = selectedItemNode.id.split("/").map(i => parseInt(i, 10));
const indices = path.split("/").map(i => parseInt(i, 10));
for(const i of indices) { for(const i of indices) {
parent = item; parent = item;
item = itemChildren(item)[i]; item = itemChildren(item)[i];
} }
showItemDetails(item, parent, selectedItemNode);
} }
showItemDetails(item, parent, itemNode);
}
}
});
function stringifyItemValue(value) { function stringifyItemValue(value) {
if (typeof value === "object" && value !== null) { if (typeof value === "object" && value !== null) {
@ -75,9 +93,20 @@ function showItemDetails(item, parent, itemNode) {
t.p([t.strong("Forced finish: "), (itemForcedFinish(item) || false) + ""]), t.p([t.strong("Forced finish: "), (itemForcedFinish(item) || false) + ""]),
t.p(t.strong("Values:")), t.p(t.strong("Values:")),
t.ul({class: "values"}, Object.entries(itemValues(item)).map(([key, value]) => { t.ul({class: "values"}, Object.entries(itemValues(item)).map(([key, value]) => {
let valueNode;
if (key === "ref") {
const refItem = itemByRef.get(value);
if (refItem) {
valueNode = t.a({href: `#${refItem.id}`}, itemCaption(refItem));
} else {
valueNode = `unknown ref ${value}`;
}
} else {
valueNode = stringifyItemValue(value);
}
return t.li([ return t.li([
t.span({className: "key"}, normalizeValueKey(key)), t.span({className: "key"}, normalizeValueKey(key)),
t.span({className: "value"}, stringifyItemValue(value)) t.span({className: "value"}, valueNode)
]); ]);
})), })),
t.p(expandButton) t.p(expandButton)
@ -117,6 +146,9 @@ async function loadFile() {
const logs = JSON.parse(json); const logs = JSON.parse(json);
logs.items.sort((a, b) => itemStart(a) - itemStart(b)); logs.items.sort((a, b) => itemStart(a) - itemStart(b));
rootItem = {c: logs.items}; rootItem = {c: logs.items};
itemByRef = new Map();
preprocessRecursively(rootItem, itemByRef, []);
const fragment = logs.items.reduce((fragment, item, i, items) => { const fragment = logs.items.reduce((fragment, item, i, items) => {
const prevItem = i === 0 ? null : items[i - 1]; const prevItem = i === 0 ? null : items[i - 1];
fragment.appendChild(t.section([ fragment.appendChild(t.section([
@ -128,6 +160,21 @@ async function loadFile() {
main.replaceChildren(fragment); main.replaceChildren(fragment);
} }
function preprocessRecursively(item, refsMap, path) {
if (itemRefSource(item)) {
refsMap.set(itemRefSource(item), item);
}
if (itemChildren(item)) {
for (let i = 0; i < itemChildren(item).length; i += 1) {
// do it in advance for a child as we don't want to do it for the rootItem
const child = itemChildren(item)[i];
const childPath = path.concat(i);
child.id = childPath.join("/");
preprocessRecursively(child, refsMap, childPath);
}
}
}
function formatTime(ms) { function formatTime(ms) {
if (ms < 1000) { if (ms < 1000) {
return `${ms}ms`; return `${ms}ms`;
@ -152,6 +199,8 @@ function itemLabel(item) { return item.v?.l; }
function itemType(item) { return item.v?.t; } function itemType(item) { return item.v?.t; }
function itemError(item) { return item.e; } function itemError(item) { return item.e; }
function itemForcedFinish(item) { return item.f; } function itemForcedFinish(item) { return item.f; }
function itemRef(item) { return item.v?.ref; }
function itemRefSource(item) { return item.v?.refId; }
function itemShortErrorMessage(item) { function itemShortErrorMessage(item) {
if (itemError(item)) { if (itemError(item)) {
const e = itemError(item); const e = itemError(item);
@ -168,6 +217,13 @@ function itemCaption(item) {
return `${itemLabel(item)} (${itemValues(item).status})`; return `${itemLabel(item)} (${itemValues(item).status})`;
} else if (itemLabel(item) && itemError(item)) { } else if (itemLabel(item) && itemError(item)) {
return `${itemLabel(item)} (${itemShortErrorMessage(item)})`; return `${itemLabel(item)} (${itemShortErrorMessage(item)})`;
} else if (itemRef(item)) {
const refItem = itemByRef.get(itemRef(item));
if (refItem) {
return `ref "${itemCaption(refItem)}"`
} else {
return `unknown ref ${itemRef(item)}`
}
} else { } else {
return itemLabel(item) || itemType(item); return itemLabel(item) || itemType(item);
} }
@ -181,7 +237,7 @@ function normalizeValueKey(key) {
} }
// returns the node and the total range (recursively) occupied by the node // returns the node and the total range (recursively) occupied by the node
function itemToNode(item, path) { function itemToNode(item) {
const hasChildren = !!itemChildren(item)?.length; const hasChildren = !!itemChildren(item)?.length;
const className = { const className = {
item: true, item: true,
@ -191,18 +247,29 @@ function itemToNode(item, path) {
[`level-${itemLevel(item)}`]: true, [`level-${itemLevel(item)}`]: true,
}; };
const id = item.id;
let captionNode;
if (itemRef(item)) {
const refItem = itemByRef.get(itemRef(item));
if (refItem) {
captionNode = ["ref ", t.a({href: `#${refItem.id}`}, itemCaption(refItem))];
}
}
if (!captionNode) {
captionNode = itemCaption(item);
}
const li = t.li([ const li = t.li([
t.div([ t.div([
hasChildren ? t.button({className: "toggleExpanded"}) : "", hasChildren ? t.button({className: "toggleExpanded"}) : "",
t.div({className, "data-path": path.join("/")}, [ t.a({className, id, href: `#${id}`}, [
t.span({class: "caption"}, itemCaption(item)), t.span({class: "caption"}, captionNode),
t.span({class: "duration"}, `(${itemDuration(item)}ms)`), t.span({class: "duration"}, `(${itemDuration(item)}ms)`),
]) ])
]) ])
]); ]);
if (itemChildren(item) && itemChildren(item).length) { if (itemChildren(item) && itemChildren(item).length) {
li.appendChild(t.ol(itemChildren(item).map((item, i) => { li.appendChild(t.ol(itemChildren(item).map(item => {
return itemToNode(item, path.concat(i)); return itemToNode(item);
}))); })));
} }
return li; return li;