mirror of
https://github.com/Ellpeck/ObsidianSimpleTimeTracker.git
synced 2024-11-16 07:23:12 +01:00
392 lines
48 KiB
Text
392 lines
48 KiB
Text
|
/*
|
||
|
THIS IS A GENERATED/BUNDLED FILE BY ESBUILD
|
||
|
if you want to view the source, please visit the github repository of this plugin
|
||
|
*/
|
||
|
var __create = Object.create;
|
||
|
var __defProp = Object.defineProperty;
|
||
|
var __defProps = Object.defineProperties;
|
||
|
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
|
||
|
var __getOwnPropDescs = Object.getOwnPropertyDescriptors;
|
||
|
var __getOwnPropNames = Object.getOwnPropertyNames;
|
||
|
var __getOwnPropSymbols = Object.getOwnPropertySymbols;
|
||
|
var __getProtoOf = Object.getPrototypeOf;
|
||
|
var __hasOwnProp = Object.prototype.hasOwnProperty;
|
||
|
var __propIsEnum = Object.prototype.propertyIsEnumerable;
|
||
|
var __defNormalProp = (obj, key, value) => key in obj ? __defProp(obj, key, { enumerable: true, configurable: true, writable: true, value }) : obj[key] = value;
|
||
|
var __spreadValues = (a, b) => {
|
||
|
for (var prop in b || (b = {}))
|
||
|
if (__hasOwnProp.call(b, prop))
|
||
|
__defNormalProp(a, prop, b[prop]);
|
||
|
if (__getOwnPropSymbols)
|
||
|
for (var prop of __getOwnPropSymbols(b)) {
|
||
|
if (__propIsEnum.call(b, prop))
|
||
|
__defNormalProp(a, prop, b[prop]);
|
||
|
}
|
||
|
return a;
|
||
|
};
|
||
|
var __spreadProps = (a, b) => __defProps(a, __getOwnPropDescs(b));
|
||
|
var __markAsModule = (target) => __defProp(target, "__esModule", { value: true });
|
||
|
var __export = (target, all) => {
|
||
|
__markAsModule(target);
|
||
|
for (var name in all)
|
||
|
__defProp(target, name, { get: all[name], enumerable: true });
|
||
|
};
|
||
|
var __reExport = (target, module2, desc) => {
|
||
|
if (module2 && typeof module2 === "object" || typeof module2 === "function") {
|
||
|
for (let key of __getOwnPropNames(module2))
|
||
|
if (!__hasOwnProp.call(target, key) && key !== "default")
|
||
|
__defProp(target, key, { get: () => module2[key], enumerable: !(desc = __getOwnPropDesc(module2, key)) || desc.enumerable });
|
||
|
}
|
||
|
return target;
|
||
|
};
|
||
|
var __toModule = (module2) => {
|
||
|
return __reExport(__markAsModule(__defProp(module2 != null ? __create(__getProtoOf(module2)) : {}, "default", module2 && module2.__esModule && "default" in module2 ? { get: () => module2.default, enumerable: true } : { value: module2, enumerable: true })), module2);
|
||
|
};
|
||
|
var __async = (__this, __arguments, generator) => {
|
||
|
return new Promise((resolve, reject) => {
|
||
|
var fulfilled = (value) => {
|
||
|
try {
|
||
|
step(generator.next(value));
|
||
|
} catch (e) {
|
||
|
reject(e);
|
||
|
}
|
||
|
};
|
||
|
var rejected = (value) => {
|
||
|
try {
|
||
|
step(generator.throw(value));
|
||
|
} catch (e) {
|
||
|
reject(e);
|
||
|
}
|
||
|
};
|
||
|
var step = (x) => x.done ? resolve(x.value) : Promise.resolve(x.value).then(fulfilled, rejected);
|
||
|
step((generator = generator.apply(__this, __arguments)).next());
|
||
|
});
|
||
|
};
|
||
|
|
||
|
// src/main.ts
|
||
|
__export(exports, {
|
||
|
default: () => SimpleTimeTrackerPlugin
|
||
|
});
|
||
|
var import_obsidian3 = __toModule(require("obsidian"));
|
||
|
|
||
|
// src/settings.ts
|
||
|
var defaultSettings = {
|
||
|
timestampFormat: "YY-MM-DD hh:mm:ss",
|
||
|
csvDelimiter: ","
|
||
|
};
|
||
|
|
||
|
// src/settings-tab.ts
|
||
|
var import_obsidian = __toModule(require("obsidian"));
|
||
|
var SimpleTimeTrackerSettingsTab = class extends import_obsidian.PluginSettingTab {
|
||
|
constructor(app, plugin) {
|
||
|
super(app, plugin);
|
||
|
this.plugin = plugin;
|
||
|
}
|
||
|
display() {
|
||
|
this.containerEl.empty();
|
||
|
this.containerEl.createEl("h2", { text: "Super Simple Time Tracker Settings" });
|
||
|
new import_obsidian.Setting(this.containerEl).setName("Timestamp Display Format").setDesc(createFragment((f) => {
|
||
|
f.createSpan({ text: "The way that timestamps in time tracker tables should be displayed. Uses " });
|
||
|
f.createEl("a", { text: "moment.js", href: "https://momentjs.com/docs/#/parsing/string-format/" });
|
||
|
f.createSpan({ text: " syntax." });
|
||
|
})).addText((t) => {
|
||
|
t.setValue(String(this.plugin.settings.timestampFormat));
|
||
|
t.onChange((v) => __async(this, null, function* () {
|
||
|
this.plugin.settings.timestampFormat = v.length ? v : defaultSettings.timestampFormat;
|
||
|
yield this.plugin.saveSettings();
|
||
|
}));
|
||
|
});
|
||
|
new import_obsidian.Setting(this.containerEl).setName("CSV Delimiter").setDesc("The delimiter character that should be used when copying a tracker table as CSV. For example, some languages use a semicolon instead of a comma.").addText((t) => {
|
||
|
t.setValue(String(this.plugin.settings.csvDelimiter));
|
||
|
t.onChange((v) => __async(this, null, function* () {
|
||
|
this.plugin.settings.csvDelimiter = v.length ? v : defaultSettings.csvDelimiter;
|
||
|
yield this.plugin.saveSettings();
|
||
|
}));
|
||
|
});
|
||
|
this.containerEl.createEl("hr");
|
||
|
this.containerEl.createEl("p", { text: "If you like this plugin and want to support its development, you can do so through my website by clicking this fancy image!" });
|
||
|
this.containerEl.createEl("a", { href: "https://ellpeck.de/support" }).createEl("img", {
|
||
|
attr: { src: "https://ellpeck.de/res/generalsupport.png" },
|
||
|
cls: "simple-time-tracker-support"
|
||
|
});
|
||
|
}
|
||
|
};
|
||
|
|
||
|
// src/tracker.ts
|
||
|
var import_obsidian2 = __toModule(require("obsidian"));
|
||
|
function saveTracker(tracker, app, fileName, section) {
|
||
|
return __async(this, null, function* () {
|
||
|
let file = app.vault.getAbstractFileByPath(fileName);
|
||
|
if (!file)
|
||
|
return;
|
||
|
let content = yield app.vault.read(file);
|
||
|
let lines = content.split("\n");
|
||
|
let prev = lines.filter((_, i) => i <= section.lineStart).join("\n");
|
||
|
let next = lines.filter((_, i) => i >= section.lineEnd).join("\n");
|
||
|
content = `${prev}
|
||
|
${JSON.stringify(tracker)}
|
||
|
${next}`;
|
||
|
yield app.vault.modify(file, content);
|
||
|
});
|
||
|
}
|
||
|
function loadTracker(json) {
|
||
|
if (json) {
|
||
|
try {
|
||
|
return JSON.parse(json);
|
||
|
} catch (e) {
|
||
|
console.log(`Failed to parse Tracker from ${json}`);
|
||
|
}
|
||
|
}
|
||
|
return { entries: [] };
|
||
|
}
|
||
|
function displayTracker(tracker, element, file, getSectionInfo, settings) {
|
||
|
let running = isRunning(tracker);
|
||
|
let btn = new import_obsidian2.ButtonComponent(element).setClass("clickable-icon").setIcon(`lucide-${running ? "stop" : "play"}-circle`).setTooltip(running ? "End" : "Start").onClick(() => __async(this, null, function* () {
|
||
|
if (running) {
|
||
|
endRunningEntry(tracker);
|
||
|
} else {
|
||
|
startNewEntry(tracker, newSegmentNameBox.getValue());
|
||
|
}
|
||
|
yield saveTracker(tracker, this.app, file, getSectionInfo());
|
||
|
}));
|
||
|
btn.buttonEl.addClass("simple-time-tracker-btn");
|
||
|
let newSegmentNameBox = new import_obsidian2.TextComponent(element).setPlaceholder("Segment name").setDisabled(running);
|
||
|
newSegmentNameBox.inputEl.addClass("simple-time-tracker-txt");
|
||
|
let timer = element.createDiv({ cls: "simple-time-tracker-timers" });
|
||
|
let currentDiv = timer.createEl("div", { cls: "simple-time-tracker-timer" });
|
||
|
let current = currentDiv.createEl("span", { cls: "simple-time-tracker-timer-time" });
|
||
|
currentDiv.createEl("span", { text: "Current" });
|
||
|
let totalDiv = timer.createEl("div", { cls: "simple-time-tracker-timer" });
|
||
|
let total = totalDiv.createEl("span", { cls: "simple-time-tracker-timer-time", text: "0s" });
|
||
|
totalDiv.createEl("span", { text: "Total" });
|
||
|
if (tracker.entries.length > 0) {
|
||
|
let table = element.createEl("table", { cls: "simple-time-tracker-table" });
|
||
|
table.createEl("tr").append(createEl("th", { text: "Segment" }), createEl("th", { text: "Start time" }), createEl("th", { text: "End time" }), createEl("th", { text: "Duration" }), createEl("th"));
|
||
|
for (let entry of tracker.entries)
|
||
|
addEditableTableRow(tracker, entry, table, newSegmentNameBox, running, file, getSectionInfo, settings, 0);
|
||
|
let buttons = element.createEl("div", { cls: "simple-time-tracker-bottom" });
|
||
|
new import_obsidian2.ButtonComponent(buttons).setButtonText("Copy as table").onClick(() => navigator.clipboard.writeText(createMarkdownTable(tracker, settings)));
|
||
|
new import_obsidian2.ButtonComponent(buttons).setButtonText("Copy as CSV").onClick(() => navigator.clipboard.writeText(createCsv(tracker, settings)));
|
||
|
}
|
||
|
setCountdownValues(tracker, current, total, currentDiv);
|
||
|
let intervalId = window.setInterval(() => {
|
||
|
if (!element.isConnected) {
|
||
|
window.clearInterval(intervalId);
|
||
|
return;
|
||
|
}
|
||
|
setCountdownValues(tracker, current, total, currentDiv);
|
||
|
}, 1e3);
|
||
|
}
|
||
|
function startSubEntry(entry, name) {
|
||
|
if (!entry.subEntries) {
|
||
|
entry.subEntries = [__spreadProps(__spreadValues({}, entry), { name: `Part 1` })];
|
||
|
entry.startTime = null;
|
||
|
entry.endTime = null;
|
||
|
}
|
||
|
if (!name)
|
||
|
name = `Part ${entry.subEntries.length + 1}`;
|
||
|
entry.subEntries.push({ name, startTime: (0, import_obsidian2.moment)().unix(), endTime: null, subEntries: null });
|
||
|
}
|
||
|
function startNewEntry(tracker, name) {
|
||
|
if (!name)
|
||
|
name = `Segment ${tracker.entries.length + 1}`;
|
||
|
let entry = { name, startTime: (0, import_obsidian2.moment)().unix(), endTime: null, subEntries: null };
|
||
|
tracker.entries.push(entry);
|
||
|
}
|
||
|
function endRunningEntry(tracker) {
|
||
|
let entry = getRunningEntry(tracker.entries);
|
||
|
entry.endTime = (0, import_obsidian2.moment)().unix();
|
||
|
}
|
||
|
function removeEntry(entries, toRemove) {
|
||
|
if (entries.contains(toRemove)) {
|
||
|
entries.remove(toRemove);
|
||
|
return true;
|
||
|
} else {
|
||
|
for (let entry of entries) {
|
||
|
if (entry.subEntries && removeEntry(entry.subEntries, toRemove)) {
|
||
|
if (entry.subEntries.length == 1) {
|
||
|
let single = entry.subEntries[0];
|
||
|
entry.startTime = single.startTime;
|
||
|
entry.endTime = single.endTime;
|
||
|
entry.subEntries = null;
|
||
|
}
|
||
|
return true;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
return false;
|
||
|
}
|
||
|
function isRunning(tracker) {
|
||
|
return !!getRunningEntry(tracker.entries);
|
||
|
}
|
||
|
function getRunningEntry(entries) {
|
||
|
for (let entry of entries) {
|
||
|
if (entry.subEntries) {
|
||
|
let running = getRunningEntry(entry.subEntries);
|
||
|
if (running)
|
||
|
return running;
|
||
|
} else {
|
||
|
if (!entry.endTime)
|
||
|
return entry;
|
||
|
}
|
||
|
}
|
||
|
return null;
|
||
|
}
|
||
|
function getDuration(entry) {
|
||
|
if (entry.subEntries) {
|
||
|
return getTotalDuration(entry.subEntries);
|
||
|
} else {
|
||
|
let endTime = entry.endTime ? import_obsidian2.moment.unix(entry.endTime) : (0, import_obsidian2.moment)();
|
||
|
return endTime.diff(import_obsidian2.moment.unix(entry.startTime));
|
||
|
}
|
||
|
}
|
||
|
function getTotalDuration(entries) {
|
||
|
let ret = 0;
|
||
|
for (let entry of entries)
|
||
|
ret += getDuration(entry);
|
||
|
return ret;
|
||
|
}
|
||
|
function setCountdownValues(tracker, current, total, currentDiv) {
|
||
|
let running = getRunningEntry(tracker.entries);
|
||
|
if (running && !running.endTime) {
|
||
|
current.setText(formatDuration(getDuration(running)));
|
||
|
currentDiv.hidden = false;
|
||
|
} else {
|
||
|
currentDiv.hidden = true;
|
||
|
}
|
||
|
total.setText(formatDuration(getTotalDuration(tracker.entries)));
|
||
|
}
|
||
|
function formatTimestamp(timestamp, settings) {
|
||
|
return import_obsidian2.moment.unix(timestamp).format(settings.timestampFormat);
|
||
|
}
|
||
|
function formatDuration(totalTime) {
|
||
|
let duration = import_obsidian2.moment.duration(totalTime);
|
||
|
let ret = "";
|
||
|
if (duration.years() > 0)
|
||
|
ret += duration.years() + "y ";
|
||
|
if (duration.months() > 0)
|
||
|
ret += duration.months() + "m ";
|
||
|
if (duration.days() > 0)
|
||
|
ret += duration.days() + "d ";
|
||
|
if (duration.hours() > 0)
|
||
|
ret += duration.hours() + "h ";
|
||
|
if (duration.minutes() > 0)
|
||
|
ret += duration.minutes() + "m ";
|
||
|
ret += duration.seconds() + "s";
|
||
|
return ret;
|
||
|
}
|
||
|
function createMarkdownTable(tracker, settings) {
|
||
|
let table = [["Segment", "Start time", "End time", "Duration"]];
|
||
|
for (let entry of tracker.entries)
|
||
|
table.push(...createTableSection(entry, settings));
|
||
|
table.push(["**Total**", "", "", `**${formatDuration(getTotalDuration(tracker.entries))}**`]);
|
||
|
let ret = "";
|
||
|
let widths = Array.from(Array(4).keys()).map((i) => Math.max(...table.map((a) => a[i].length)));
|
||
|
for (let r = 0; r < table.length; r++) {
|
||
|
if (r == 1)
|
||
|
ret += Array.from(Array(4).keys()).map((i) => "-".repeat(widths[i])).join(" | ") + "\n";
|
||
|
let row = [];
|
||
|
for (let i = 0; i < 4; i++)
|
||
|
row.push(table[r][i].padEnd(widths[i], " "));
|
||
|
ret += row.join(" | ") + "\n";
|
||
|
}
|
||
|
return ret;
|
||
|
}
|
||
|
function createCsv(tracker, settings) {
|
||
|
let ret = "";
|
||
|
for (let entry of tracker.entries) {
|
||
|
for (let row of createTableSection(entry, settings))
|
||
|
ret += row.join(settings.csvDelimiter) + "\n";
|
||
|
}
|
||
|
return ret;
|
||
|
}
|
||
|
function createTableSection(entry, settings) {
|
||
|
let ret = [[
|
||
|
entry.name,
|
||
|
entry.startTime ? formatTimestamp(entry.startTime, settings) : "",
|
||
|
entry.endTime ? formatTimestamp(entry.endTime, settings) : "",
|
||
|
entry.endTime || entry.subEntries ? formatDuration(getDuration(entry)) : ""
|
||
|
]];
|
||
|
if (entry.subEntries) {
|
||
|
for (let sub of entry.subEntries)
|
||
|
ret.push(...createTableSection(sub, settings));
|
||
|
}
|
||
|
return ret;
|
||
|
}
|
||
|
function addEditableTableRow(tracker, entry, table, newSegmentNameBox, running, file, getSectionInfo, settings, indent) {
|
||
|
let row = table.createEl("tr");
|
||
|
let name = row.createEl("td");
|
||
|
let namePar = name.createEl("span", { text: entry.name });
|
||
|
namePar.style.marginLeft = `${indent}em`;
|
||
|
let nameBox = new import_obsidian2.TextComponent(name).setValue(entry.name);
|
||
|
nameBox.inputEl.hidden = true;
|
||
|
row.createEl("td", { text: entry.startTime ? formatTimestamp(entry.startTime, settings) : "" });
|
||
|
row.createEl("td", { text: entry.endTime ? formatTimestamp(entry.endTime, settings) : "" });
|
||
|
row.createEl("td", { text: entry.endTime || entry.subEntries ? formatDuration(getDuration(entry)) : "" });
|
||
|
let entryButtons = row.createEl("td");
|
||
|
if (!running) {
|
||
|
new import_obsidian2.ButtonComponent(entryButtons).setClass("clickable-icon").setIcon(`lucide-play`).setTooltip("Continue").onClick(() => __async(this, null, function* () {
|
||
|
startSubEntry(entry, newSegmentNameBox.getValue());
|
||
|
yield saveTracker(tracker, this.app, file, getSectionInfo());
|
||
|
}));
|
||
|
}
|
||
|
let editButton = new import_obsidian2.ButtonComponent(entryButtons).setClass("clickable-icon").setTooltip("Edit").setIcon("lucide-pencil").onClick(() => __async(this, null, function* () {
|
||
|
if (namePar.hidden) {
|
||
|
namePar.hidden = false;
|
||
|
nameBox.inputEl.hidden = true;
|
||
|
editButton.setIcon("lucide-pencil");
|
||
|
if (nameBox.getValue()) {
|
||
|
entry.name = nameBox.getValue();
|
||
|
namePar.setText(entry.name);
|
||
|
yield saveTracker(tracker, this.app, file, getSectionInfo());
|
||
|
}
|
||
|
} else {
|
||
|
namePar.hidden = true;
|
||
|
nameBox.inputEl.hidden = false;
|
||
|
nameBox.setValue(entry.name);
|
||
|
editButton.setIcon("lucide-check");
|
||
|
}
|
||
|
}));
|
||
|
new import_obsidian2.ButtonComponent(entryButtons).setClass("clickable-icon").setTooltip("Remove").setIcon("lucide-trash").onClick(() => __async(this, null, function* () {
|
||
|
removeEntry(tracker.entries, entry);
|
||
|
yield saveTracker(tracker, this.app, file, getSectionInfo());
|
||
|
}));
|
||
|
if (entry.subEntries) {
|
||
|
for (let sub of entry.subEntries)
|
||
|
addEditableTableRow(tracker, sub, table, newSegmentNameBox, running, file, getSectionInfo, settings, indent + 1);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// src/main.ts
|
||
|
var SimpleTimeTrackerPlugin = class extends import_obsidian3.Plugin {
|
||
|
onload() {
|
||
|
return __async(this, null, function* () {
|
||
|
yield this.loadSettings();
|
||
|
this.addSettingTab(new SimpleTimeTrackerSettingsTab(this.app, this));
|
||
|
this.registerMarkdownCodeBlockProcessor("simple-time-tracker", (s, e, i) => {
|
||
|
let tracker = loadTracker(s);
|
||
|
e.empty();
|
||
|
displayTracker(tracker, e, i.sourcePath, () => i.getSectionInfo(e), this.settings);
|
||
|
});
|
||
|
this.addCommand({
|
||
|
id: `insert`,
|
||
|
name: `Insert Time Tracker`,
|
||
|
editorCallback: (e, _) => {
|
||
|
e.replaceSelection("```simple-time-tracker\n```\n");
|
||
|
}
|
||
|
});
|
||
|
});
|
||
|
}
|
||
|
loadSettings() {
|
||
|
return __async(this, null, function* () {
|
||
|
this.settings = Object.assign({}, defaultSettings, yield this.loadData());
|
||
|
});
|
||
|
}
|
||
|
saveSettings() {
|
||
|
return __async(this, null, function* () {
|
||
|
yield this.saveData(this.settings);
|
||
|
});
|
||
|
}
|
||
|
};
|
||
|
//# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsic3JjL21haW4udHMiLCAic3JjL3NldHRpbmdzLnRzIiwgInNyYy9zZXR0aW5ncy10YWIudHMiLCAic3JjL3RyYWNrZXIudHMiXSwKICAic291cmNlc0NvbnRlbnQiOiBbImltcG9ydCB7IFBsdWdpbiB9IGZyb20gXCJvYnNpZGlhblwiO1xyXG5pbXBvcnQgeyBkZWZhdWx0U2V0dGluZ3MsIFNpbXBsZVRpbWVUcmFja2VyU2V0dGluZ3MgfSBmcm9tIFwiLi9zZXR0aW5nc1wiO1xyXG5pbXBvcnQgeyBTaW1wbGVUaW1lVHJhY2tlclNldHRpbmdzVGFiIH0gZnJvbSBcIi4vc2V0dGluZ3MtdGFiXCI7XHJcbmltcG9ydCB7IGRpc3BsYXlUcmFja2VyLCBsb2FkVHJhY2tlciwgVHJhY2tlciB9IGZyb20gXCIuL3RyYWNrZXJcIjtcclxuXHJcbmV4cG9ydCBkZWZhdWx0IGNsYXNzIFNpbXBsZVRpbWVUcmFja2VyUGx1Z2luIGV4dGVuZHMgUGx1Z2luIHtcclxuXHJcbiAgICBzZXR0aW5nczogU2ltcGxlVGltZVRyYWNrZXJTZXR0aW5ncztcclxuXHJcbiAgICBhc3luYyBvbmxvYWQoKTogUHJvbWlzZTx2b2lkPiB7XHJcbiAgICAgICAgYXdhaXQgdGhpcy5sb2FkU2V0dGluZ3MoKTtcclxuXHJcbiAgICAgICAgdGhpcy5hZGRTZXR0aW5nVGFiKG5ldyBTaW1wbGVUaW1lVHJhY2tlclNldHRpbmdzVGFiKHRoaXMuYXBwLCB0aGlzKSk7XHJcblxyXG4gICAgICAgIHRoaXMucmVnaXN0ZXJNYXJrZG93bkNvZGVCbG9ja1Byb2Nlc3NvcihcInNpbXBsZS10aW1lLXRyYWNrZXJcIiwgKHMsIGUsIGkpID0+IHtcclxuICAgICAgICAgICAgbGV0IHRyYWNrZXI6IFRyYWNrZXIgPSBsb2FkVHJhY2tlcihzKTtcclxuICAgICAgICAgICAgZS5lbXB0eSgpO1xyXG4gICAgICAgICAgICBkaXNwbGF5VHJhY2tlcih0cmFja2VyLCBlLCBpLnNvdXJjZVBhdGgsICgpID0+IGkuZ2V0U2VjdGlvbkluZm8oZSksIHRoaXMuc2V0dGluZ3MpO1xyXG4gICAgICAgIH0pO1xyXG5cclxuICAgICAgICB0aGlzLmFkZENvbW1hbmQoe1xyXG4gICAgICAgICAgICBpZDogYGluc2VydGAsXHJcbiAgICAgICAgICAgIG5hbWU6IGBJbnNlcnQgVGltZSBUcmFja2VyYCxcclxuICAgICAgICAgICAgZWRpdG9yQ2FsbGJhY2s6IChlLCBfKSA9PiB7XHJcbiAgICAgICAgICAgICAgICBlLnJlcGxhY2VTZWxlY3Rpb24oXCJgYGBzaW1wbGUtdGltZS10cmFja2VyXFxuYGBgXFxuXCIpO1xyXG4gICAgICAgICAgICB9XHJcbiAgICAgICAgfSk7XHJcbiAgICB9XHJcblxyXG4gICAgYXN5bmMgbG9hZFNldHRpbmdzKCk6IFByb21pc2U8dm9pZD4ge1xyXG4gICAgICAgIHRoaXMuc2V0dGluZ3MgPSBPYmplY3QuYXNzaWduKHt9LCBkZWZhdWx0U2V0dGluZ3MsIGF3YWl0IHRoaXMubG9hZERhdGEoKSk7XHJcbiAgICB9XHJcblxyXG4gICAgYXN5bmMgc2F2ZVNldHRpbmdzKCk6IFByb21pc2U8dm9pZD4ge1xyXG4gICAgICAgIGF3YWl0IHRoaXMuc2F2ZURhdGEodGhpcy5zZXR0aW5ncyk7XHJcbiAgICB9XHJcbn1cclxuIiwgImV4cG9ydCBjb25zdCBkZWZhdWx0U2V0dGluZ3M6IFNpbXBsZVRpbWVUcmFja2VyU2V0dGluZ3MgPSB7XHJcbiAgICB0aW1lc3RhbXBGb3JtYXQ6IFwiWVktTU0tREQgaGg6bW06c3NcIixcclxuICAgIGNzdkRlbGltaXRlcjogXCIsXCJcclxufTtcclxuXHJcbmV4cG9ydCBpbnRlcmZhY2UgU2ltcGxlVGltZVRyYWNrZXJTZXR0aW5ncyB7XHJcblxyXG4gICAgdGltZXN0YW1wRm9ybWF0OiBzdHJpbmc7XHJcbiAgICBjc3ZEZWxpbWl0ZXI6IHN0cmluZztcclxuXHJcbn1cclxuIiwgImltcG9ydCB7IEFwcCwgUGx1Z2luU2V0dGluZ1RhYiwgU2V0dGluZyB9IGZyb20gXCJvYnNpZGlhblwiO1xyXG5pbXBvcnQgU2ltcGxlVGltZVRyYWNrZXJQbHVnaW4gZnJvbSBcIi4vbWFpblwiO1xyXG5pbXBvcnQgeyBkZWZhdWx0U2V0dGluZ3MgfSBmcm9tIFwiLi9zZXR0aW5nc1wiO1xyXG5cclxuZXhwb3J0IGNsYXNzIFNpbXBsZVRpbWVUcmFja2VyU2V0dGluZ3NUYWIgZXh0ZW5kcyBQbHVnaW5TZXR0aW5nVGFiIHtcclxuXHJcbiAgICBwbHVnaW46IFNpbXBsZVRpbWVUcmFja2VyUGx1Z2luO1xyXG5cclxuICAgIGNvbnN0cnVjdG9yKGFwcDogQXBwLCBwbHVnaW46IFNpbXBsZVRpbWVUcmFja2VyUGx1Z2luKSB7XHJcbiAgICAgICAgc3VwZXIoYXBwLCBwbHVnaW4pO1xyXG4gICAgICAgIHRoaXMucGx1Z2luID0gcGx1Z2luO1xyXG4gICAgfVxyXG5cclxuICAgIGRpc3BsYXkoKTogdm9pZCB7XHJcbiAgICAgICAgdGhpcy5jb250YWluZXJFbC5lbXB0eSgpO1xyXG4gICAgICAgIHRoaXMuY29udGFpbmVyRWwuY3JlYXRlRWwoXCJoMlwiLCB7dGV4dDogXCJTdXBlciBTaW1wbGUgVGltZSBUcmFja2VyIFNldHRpbmdzXCJ9KTtcclxuXHJcbiAgICAgICAgbmV3IFNldHRpbmcodGhpcy5jb250YWluZXJFbClcclxuICAgICAgICAgICAgLnNldE5hbWUoXCJUaW1lc3RhbXAgRGlzcGxheSBGb3JtYXRcIilcclxuICAgICAgICAgICAgLnNldERlc2MoY3JlYXRlRnJhZ21lbnQoZiA9PiB7XHJcbiAgICAgICAgICAgICAgICBmLmNyZWF0ZVNwYW4oe3RleHQ6IFwiVGhlIHdheSB0aGF0IHRpbWVzdGFtcHMgaW4gdGltZSB0cmFja2VyIHRhYmxlcyBzaG91bGQgYmUgZGlzcGxheWVkLiBVc2VzIFwifSk7XHJcbiAgICAgICAgICAgICAgICBmLmNyZWF0ZUVsKFwiYVwiLCB7dGV4dDogXCJtb21lbnQuanNcIiwgaHJlZjogXCJodHRwczovL21vbWVudGpzLmNvbS9kb2NzLyMvcGFyc2luZy9zdHJpbmctZm9ybWF0L1wifSk7XHJcbiAgICAgICAgICAgICAgICBmLmNyZWF0ZVNwYW4oe3RleHQ6IFwiIHN5bnRheC5cIn0pO1xyXG4gICAgICAgICAgICB9KSlcclxuICAgICAgICAgICAgLmFkZFRleHQodCA9PiB7XHJcbiAgICAgICAgICAgICAgICB0LnNldFZhbHVlKFN0cmluZyh0aGlzLnBsdWdpbi5zZXR0aW5ncy50aW1lc3RhbXBGb3JtYXQpKTtcclxuICAgICAgICAgICAgICAgIHQub25DaGFuZ2UoYXN5bmMgdiA9PiB7XHJcbiAgICAgICAgICAgICAgICAgICAgdGhpcy5wbHVnaW4uc2V0dGluZ3MudGltZXN0YW1wRm9ybWF0ID0gdi5sZW5ndGggPyB2IDogZGVmY
|