mirror of
https://github.com/openclaw/openclaw.git
synced 2026-05-06 10:20:42 +00:00
fix(ui): render cron markdown summaries
This commit is contained in:
@@ -35,6 +35,7 @@ Docs: https://docs.openclaw.ai
|
||||
- Media-understanding/audio: migrate deprecated `{input}` placeholders in legacy `audio.transcription.command` configs to `{{MediaPath}}`, so custom audio transcribers no longer receive the literal placeholder after doctor repair. Fixes #72760. Thanks @krisfanue3-hash.
|
||||
- Ollama/onboarding: de-dupe suggested bare local models against installed `:latest` tags and skip redundant pulls, so setup shows the installed model once and no longer says it is downloading an already available model. Fixes #68952. Thanks @tleyden.
|
||||
- Compaction: skip oversized pre-compaction checkpoint snapshots and prune duplicate long user turns from compaction input and rotated successor transcripts, preventing retry storms from being preserved across checkpoint cycles. Fixes #72780. Thanks @SweetSophia.
|
||||
- Control UI/Cron: render cron job prompts and run summaries as sanitized markdown in the dashboard, with full-width block content, safer link clicks, and no duplicate error text when a failed run has no summary. Supersedes #48504. Thanks @garethdaine.
|
||||
- Control UI/Gateway: preserve WebChat client version labels across localhost, 127.0.0.1, and IPv6 loopback aliases on the same port, avoiding misleading `vcontrol-ui` connection logs while investigating duplicate-message reports. Refs #72753 and #72742. Thanks @LumenFromTheFuture and @allesgutefy.
|
||||
- Agents/reasoning: treat orphan closing reasoning tags with following answer text as a privacy boundary across delivery, history, streaming, and Control UI sanitizers so malformed local-model output cannot leak chain-of-thought text. Fixes #67092. Thanks @AnildoSilva.
|
||||
- Memory-core: run one-shot memory CLI commands through transient builtin and QMD managers so `memory index`, `memory status --index`, and `memory search` no longer start long-lived file watchers that can hit macOS `EMFILE` limits. Fixes #59101; carries forward #49851. Thanks @mbear469210-coder and @maoyuanxue.
|
||||
|
||||
@@ -1295,21 +1295,95 @@
|
||||
accent-color: var(--accent);
|
||||
}
|
||||
|
||||
.cron-run-entry {
|
||||
align-items: start;
|
||||
.list-item.cron-run-entry {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
padding: 0;
|
||||
overflow: hidden;
|
||||
}
|
||||
|
||||
.cron-run-entry__header {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
align-items: baseline;
|
||||
gap: 8px;
|
||||
padding: 12px;
|
||||
width: 100%;
|
||||
box-sizing: border-box;
|
||||
background: var(--secondary);
|
||||
border-radius: var(--radius-md) var(--radius-md) 0 0;
|
||||
border-bottom: 1px solid var(--border);
|
||||
}
|
||||
|
||||
@media (min-width: 1101px) {
|
||||
.cron-run-entry__header {
|
||||
display: grid;
|
||||
grid-template-columns: minmax(0, 1fr) minmax(200px, 260px);
|
||||
gap: 16px;
|
||||
align-items: start;
|
||||
}
|
||||
}
|
||||
|
||||
:root[data-theme-mode="light"] .cron-run-entry__header {
|
||||
background: var(--bg-muted);
|
||||
}
|
||||
|
||||
.cron-run-entry__meta {
|
||||
text-align: right;
|
||||
min-width: 220px;
|
||||
min-width: 0;
|
||||
}
|
||||
|
||||
.cron-run-entry__summary {
|
||||
white-space: pre-wrap;
|
||||
.cron-run-entry__body {
|
||||
padding: 12px;
|
||||
line-height: 1.45;
|
||||
min-width: 0;
|
||||
width: 100%;
|
||||
box-sizing: border-box;
|
||||
}
|
||||
|
||||
.cron-run-entry__body pre,
|
||||
.cron-run-entry__body blockquote,
|
||||
.cron-job-detail-value pre,
|
||||
.cron-job-detail-value blockquote {
|
||||
width: 100%;
|
||||
max-width: 100%;
|
||||
box-sizing: border-box;
|
||||
}
|
||||
|
||||
@media (max-width: 1100px) {
|
||||
.cron-job.list-item {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
padding: 0;
|
||||
overflow: hidden;
|
||||
max-width: 100%;
|
||||
}
|
||||
|
||||
.cron-job .cron-job-header {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 8px;
|
||||
border-radius: var(--radius-md) var(--radius-md) 0 0;
|
||||
width: 100%;
|
||||
}
|
||||
|
||||
.cron-job .list-meta {
|
||||
min-width: 0;
|
||||
}
|
||||
|
||||
.cron-job .cron-job-header .cron-job-state {
|
||||
display: flex;
|
||||
flex-wrap: wrap;
|
||||
align-items: baseline;
|
||||
gap: 4px 16px;
|
||||
}
|
||||
|
||||
.cron-job .cron-job-detail,
|
||||
.cron-job .cron-job-footer {
|
||||
width: 100%;
|
||||
box-sizing: border-box;
|
||||
}
|
||||
|
||||
.cron-summary-strip {
|
||||
flex-direction: column;
|
||||
}
|
||||
@@ -1373,6 +1447,27 @@
|
||||
min-width: 0;
|
||||
text-align: left;
|
||||
}
|
||||
|
||||
.list-item.cron-run-entry {
|
||||
padding: 0;
|
||||
}
|
||||
|
||||
.cron-run-entry__header {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 8px;
|
||||
padding: 10px;
|
||||
}
|
||||
|
||||
.cron-run-entry__body {
|
||||
padding: 10px;
|
||||
}
|
||||
|
||||
.cron-run-entry__body pre,
|
||||
.cron-run-entry__body table {
|
||||
overflow-x: auto;
|
||||
-webkit-overflow-scrolling: touch;
|
||||
}
|
||||
}
|
||||
|
||||
:root[data-theme-mode="light"] .field input,
|
||||
@@ -1852,17 +1947,40 @@
|
||||
.cron-job {
|
||||
grid-template-columns: minmax(0, 1fr) minmax(240px, 300px);
|
||||
grid-template-areas:
|
||||
"main meta"
|
||||
"header header"
|
||||
"payload payload"
|
||||
"footer footer";
|
||||
row-gap: 10px;
|
||||
row-gap: 0;
|
||||
padding: 0;
|
||||
overflow: hidden;
|
||||
}
|
||||
|
||||
.cron-job .list-main {
|
||||
grid-area: main;
|
||||
.cron-job .cron-job-header {
|
||||
grid-area: header;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
align-items: baseline;
|
||||
gap: 8px;
|
||||
padding: 12px;
|
||||
background: var(--secondary);
|
||||
border-radius: var(--radius-md) var(--radius-md) 0 0;
|
||||
border-bottom: 1px solid var(--border);
|
||||
}
|
||||
|
||||
@media (min-width: 1101px) {
|
||||
.cron-job .cron-job-header {
|
||||
display: grid;
|
||||
grid-template-columns: minmax(0, 1fr) minmax(240px, 300px);
|
||||
gap: 16px;
|
||||
align-items: start;
|
||||
}
|
||||
}
|
||||
|
||||
:root[data-theme-mode="light"] .cron-job .cron-job-header {
|
||||
background: var(--bg-muted);
|
||||
}
|
||||
|
||||
.cron-job .list-meta {
|
||||
grid-area: meta;
|
||||
min-width: 240px;
|
||||
gap: 8px;
|
||||
}
|
||||
@@ -1873,8 +1991,8 @@
|
||||
justify-content: space-between;
|
||||
align-items: center;
|
||||
gap: 12px;
|
||||
border-top: 1px solid var(--border);
|
||||
padding-top: 10px;
|
||||
border-top: none;
|
||||
padding: 4px 12px 12px;
|
||||
}
|
||||
|
||||
.cron-job-chips {
|
||||
@@ -1882,9 +2000,18 @@
|
||||
}
|
||||
|
||||
.cron-job-detail {
|
||||
grid-area: payload;
|
||||
display: grid;
|
||||
gap: 8px;
|
||||
margin-top: 2px;
|
||||
padding: 8px 12px 12px;
|
||||
overflow: hidden;
|
||||
}
|
||||
|
||||
.cron-job-detail-section {
|
||||
display: grid;
|
||||
gap: 3px;
|
||||
margin-top: 2px;
|
||||
min-width: 0;
|
||||
}
|
||||
|
||||
.cron-job-detail-label {
|
||||
@@ -1893,11 +2020,17 @@
|
||||
font-weight: 600;
|
||||
letter-spacing: 0.03em;
|
||||
text-transform: uppercase;
|
||||
padding: 4px 8px;
|
||||
background: var(--secondary);
|
||||
border-radius: var(--radius-sm);
|
||||
display: inline-block;
|
||||
width: fit-content;
|
||||
}
|
||||
|
||||
.cron-job-detail-value {
|
||||
font-size: 13px;
|
||||
line-height: 1.35;
|
||||
min-width: 0;
|
||||
overflow-wrap: anywhere;
|
||||
}
|
||||
|
||||
@@ -1988,8 +2121,8 @@
|
||||
.cron-job {
|
||||
grid-template-columns: 1fr;
|
||||
grid-template-areas:
|
||||
"main"
|
||||
"meta"
|
||||
"header"
|
||||
"payload"
|
||||
"footer";
|
||||
}
|
||||
|
||||
@@ -2891,6 +3024,101 @@ td.data-table-key-col {
|
||||
background: var(--secondary);
|
||||
}
|
||||
|
||||
/* Cron markdown overrides must come after .chat-text rules to win cascade. */
|
||||
.cron-run-entry__body.chat-text table,
|
||||
.cron-job-detail-value.chat-text table {
|
||||
display: table;
|
||||
width: 100%;
|
||||
}
|
||||
|
||||
.cron-run-entry__body.chat-text pre,
|
||||
.cron-run-entry__body.chat-text blockquote,
|
||||
.cron-job-detail-value.chat-text pre,
|
||||
.cron-job-detail-value.chat-text blockquote {
|
||||
width: 100%;
|
||||
max-width: 100%;
|
||||
box-sizing: border-box;
|
||||
}
|
||||
|
||||
.cron-run-entry__body.chat-text pre + p,
|
||||
.cron-run-entry__body.chat-text blockquote + p,
|
||||
.cron-run-entry__body.chat-text table + p,
|
||||
.cron-run-entry__body.chat-text p + pre,
|
||||
.cron-run-entry__body.chat-text p + blockquote,
|
||||
.cron-run-entry__body.chat-text p + table,
|
||||
.cron-job-detail-value.chat-text pre + p,
|
||||
.cron-job-detail-value.chat-text blockquote + p,
|
||||
.cron-job-detail-value.chat-text table + p,
|
||||
.cron-job-detail-value.chat-text p + pre,
|
||||
.cron-job-detail-value.chat-text p + blockquote,
|
||||
.cron-job-detail-value.chat-text p + table {
|
||||
margin-top: 0.75em;
|
||||
}
|
||||
|
||||
.cron-run-entry__body.chat-text pre,
|
||||
.cron-run-entry__body.chat-text blockquote,
|
||||
.cron-run-entry__body.chat-text table,
|
||||
.cron-job-detail-value.chat-text pre,
|
||||
.cron-job-detail-value.chat-text blockquote,
|
||||
.cron-job-detail-value.chat-text table {
|
||||
margin-bottom: 0.75em;
|
||||
}
|
||||
|
||||
.cron-run-entry__body.chat-text > :last-child,
|
||||
.cron-job-detail-value.chat-text > :last-child {
|
||||
margin-bottom: 0;
|
||||
}
|
||||
|
||||
@media (max-width: 600px) {
|
||||
.cron-job .cron-job-header {
|
||||
padding: 10px;
|
||||
gap: 6px;
|
||||
}
|
||||
|
||||
.cron-job .cron-job-header .cron-job-state {
|
||||
gap: 4px 12px;
|
||||
}
|
||||
|
||||
.cron-job .cron-job-detail {
|
||||
padding: 10px;
|
||||
}
|
||||
|
||||
.cron-job .cron-job-detail-value {
|
||||
font-size: 13px;
|
||||
overflow-x: auto;
|
||||
}
|
||||
|
||||
.cron-job .cron-job-detail-value table,
|
||||
.cron-run-entry__body table {
|
||||
display: block;
|
||||
overflow-x: auto;
|
||||
-webkit-overflow-scrolling: touch;
|
||||
}
|
||||
|
||||
.cron-job .cron-job-detail-value pre {
|
||||
overflow-x: auto;
|
||||
-webkit-overflow-scrolling: touch;
|
||||
}
|
||||
|
||||
.cron-job .cron-job-footer {
|
||||
padding: 8px 10px 10px;
|
||||
overflow-x: auto;
|
||||
-webkit-overflow-scrolling: touch;
|
||||
flex-wrap: nowrap;
|
||||
gap: 8px;
|
||||
}
|
||||
|
||||
.cron-job .cron-job-footer .chip-row,
|
||||
.cron-job .cron-job-footer .cron-job-actions {
|
||||
flex-wrap: nowrap;
|
||||
flex-shrink: 0;
|
||||
}
|
||||
|
||||
.cron-job .cron-job-footer .cron-job-actions {
|
||||
gap: 6px;
|
||||
}
|
||||
}
|
||||
|
||||
/* Tool cards */
|
||||
.chat-tool-card {
|
||||
margin-top: 8px;
|
||||
|
||||
@@ -232,7 +232,7 @@ describe("cron view", () => {
|
||||
expect(runHistoryCard).not.toBeUndefined();
|
||||
|
||||
const summaries = Array.from(
|
||||
runHistoryCard?.querySelectorAll(".list-item .list-sub") ?? [],
|
||||
runHistoryCard?.querySelectorAll(".cron-run-entry__body") ?? [],
|
||||
).map((el) => (el.textContent ?? "").trim());
|
||||
expect(summaries[0]).toBe("newer run");
|
||||
expect(summaries[1]).toBe("older run");
|
||||
@@ -299,6 +299,79 @@ describe("cron view", () => {
|
||||
expect(container.textContent).toContain("https://example.invalid/cron");
|
||||
});
|
||||
|
||||
it("renders cron job prompts and run summaries as sanitized markdown", () => {
|
||||
const container = document.createElement("div");
|
||||
const onLoadRuns = vi.fn();
|
||||
const job = {
|
||||
...createJob("job-md"),
|
||||
sessionTarget: "isolated" as const,
|
||||
payload: {
|
||||
kind: "agentTurn" as const,
|
||||
message: "## Plan\n\n- **Ship** [docs](https://example.com)\n\n<script>alert(1)</script>",
|
||||
},
|
||||
delivery: { mode: "announce" as const, channel: "telegram", to: "123" },
|
||||
};
|
||||
|
||||
render(
|
||||
renderCron(
|
||||
createProps({
|
||||
jobs: [job],
|
||||
runs: [
|
||||
{
|
||||
ts: 2,
|
||||
jobId: "job-md",
|
||||
status: "ok",
|
||||
summary: "Done with **markdown**\n\n| A | B |\n| - | - |\n| 1 | 2 |",
|
||||
},
|
||||
],
|
||||
onLoadRuns,
|
||||
}),
|
||||
),
|
||||
container,
|
||||
);
|
||||
|
||||
const prompt = container.querySelector(".cron-job-detail-value.chat-text");
|
||||
expect(prompt?.querySelector("strong")?.textContent).toBe("Ship");
|
||||
expect(prompt?.querySelector("a")?.getAttribute("href")).toBe("https://example.com");
|
||||
expect(prompt?.querySelector("script")).toBeNull();
|
||||
|
||||
const promptLink = prompt?.querySelector("a");
|
||||
promptLink?.dispatchEvent(new MouseEvent("click", { bubbles: true }));
|
||||
expect(onLoadRuns).not.toHaveBeenCalled();
|
||||
|
||||
const row = container.querySelector(".cron-job");
|
||||
row?.dispatchEvent(new MouseEvent("click", { bubbles: true }));
|
||||
expect(onLoadRuns).toHaveBeenCalledWith("job-md");
|
||||
|
||||
const runBody = container.querySelector(".cron-run-entry__body.chat-text");
|
||||
expect(runBody?.querySelector("strong")?.textContent).toBe("markdown");
|
||||
expect(runBody?.querySelector("table")).not.toBeNull();
|
||||
});
|
||||
|
||||
it("shows run errors in one place when no summary exists", () => {
|
||||
const container = document.createElement("div");
|
||||
render(
|
||||
renderCron(
|
||||
createProps({
|
||||
runs: [
|
||||
{
|
||||
ts: 2,
|
||||
jobId: "job-error",
|
||||
status: "error",
|
||||
error: "Failed with **markdown**",
|
||||
},
|
||||
],
|
||||
}),
|
||||
),
|
||||
container,
|
||||
);
|
||||
|
||||
expect(container.querySelector(".cron-run-entry__meta")?.textContent).not.toContain(
|
||||
"Failed with",
|
||||
);
|
||||
expect(container.querySelector(".cron-run-entry__body strong")?.textContent).toBe("markdown");
|
||||
});
|
||||
|
||||
it("wires the Edit action and shows save/cancel controls when editing", () => {
|
||||
const container = document.createElement("div");
|
||||
const onEdit = vi.fn();
|
||||
|
||||
@@ -1,5 +1,6 @@
|
||||
import { html, nothing } from "lit";
|
||||
import { ifDefined } from "lit/directives/if-defined.js";
|
||||
import { unsafeHTML } from "lit/directives/unsafe-html.js";
|
||||
import { t } from "../../i18n/index.ts";
|
||||
import type {
|
||||
CronFieldErrors,
|
||||
@@ -8,6 +9,7 @@ import type {
|
||||
CronJobsScheduleKindFilter,
|
||||
} from "../controllers/cron.ts";
|
||||
import { formatRelativeTimestamp, formatMs } from "../format.ts";
|
||||
import { toSanitizedMarkdownHtml } from "../markdown.ts";
|
||||
import { pathForTab } from "../navigation.ts";
|
||||
import { formatCronSchedule, formatNextRun } from "../presenter.ts";
|
||||
import type { ChannelUiMetaEntry, CronJob, CronRunLogEntry, CronStatus } from "../types.ts";
|
||||
@@ -1479,17 +1481,19 @@ function renderJob(job: CronJob, props: CronProps) {
|
||||
};
|
||||
return html`
|
||||
<div class=${itemClass} @click=${() => props.onLoadRuns(job.id)}>
|
||||
<div class="list-main">
|
||||
<div class="list-title">${job.name}</div>
|
||||
<div class="list-sub">${formatCronSchedule(job)}</div>
|
||||
${renderJobPayload(job)}
|
||||
${job.agentId
|
||||
? html`<div class="muted cron-job-agent">
|
||||
${t("cron.jobDetail.agent")}: ${job.agentId}
|
||||
</div>`
|
||||
: nothing}
|
||||
<div class="cron-job-header">
|
||||
<div class="list-main">
|
||||
<div class="list-title">${job.name}</div>
|
||||
<div class="list-sub">${formatCronSchedule(job)}</div>
|
||||
${job.agentId
|
||||
? html`<div class="muted cron-job-agent">
|
||||
${t("cron.jobDetail.agent")}: ${job.agentId}
|
||||
</div>`
|
||||
: nothing}
|
||||
</div>
|
||||
<div class="list-meta">${renderJobState(job)}</div>
|
||||
</div>
|
||||
<div class="list-meta">${renderJobState(job)}</div>
|
||||
${renderJobPayload(job)}
|
||||
<div class="cron-job-footer">
|
||||
<div class="chip-row cron-job-chips">
|
||||
<span class=${`chip ${job.enabled ? "chip-ok" : "chip-danger"}`}>
|
||||
@@ -1595,18 +1599,29 @@ function renderJobPayload(job: CronJob) {
|
||||
|
||||
return html`
|
||||
<div class="cron-job-detail">
|
||||
<span class="cron-job-detail-label">${t("cron.jobDetail.prompt")}</span>
|
||||
<span class="muted cron-job-detail-value">${job.payload.message}</span>
|
||||
<div class="cron-job-detail-section">
|
||||
<span class="cron-job-detail-label">${t("cron.jobDetail.prompt")}</span>
|
||||
<div class="muted cron-job-detail-value chat-text" @click=${stopPropagationForInteractive}>
|
||||
${unsafeHTML(toSanitizedMarkdownHtml(job.payload.message))}
|
||||
</div>
|
||||
</div>
|
||||
${delivery
|
||||
? html`<div class="cron-job-detail-section">
|
||||
<span class="cron-job-detail-label">${t("cron.jobDetail.delivery")}</span>
|
||||
<span class="muted cron-job-detail-value">${delivery.mode}${deliveryTarget}</span>
|
||||
</div>`
|
||||
: nothing}
|
||||
</div>
|
||||
${delivery
|
||||
? html`<div class="cron-job-detail">
|
||||
<span class="cron-job-detail-label">${t("cron.jobDetail.delivery")}</span>
|
||||
<span class="muted cron-job-detail-value">${delivery.mode}${deliveryTarget}</span>
|
||||
</div>`
|
||||
: nothing}
|
||||
`;
|
||||
}
|
||||
|
||||
function stopPropagationForInteractive(event: MouseEvent) {
|
||||
const target = event.target as HTMLElement | null;
|
||||
if (target?.closest("a,button,input,textarea,select,summary,[role='button'],[role='link']")) {
|
||||
event.stopPropagation();
|
||||
}
|
||||
}
|
||||
|
||||
function formatStateRelative(ms?: number) {
|
||||
if (typeof ms !== "number" || !Number.isFinite(ms)) {
|
||||
return t("common.na");
|
||||
@@ -1708,59 +1723,63 @@ function renderRun(
|
||||
: usage && typeof usage.input_tokens === "number" && typeof usage.output_tokens === "number"
|
||||
? `${usage.input_tokens} in / ${usage.output_tokens} out`
|
||||
: null;
|
||||
const bodySource = entry.summary ?? entry.error ?? t("cron.runEntry.noSummary");
|
||||
const showErrorInMeta = !!entry.error && entry.summary != null;
|
||||
return html`
|
||||
<div class="list-item cron-run-entry">
|
||||
<div class="list-main cron-run-entry__main">
|
||||
<div class="list-title cron-run-entry__title">
|
||||
${entry.jobName ?? entry.jobId}
|
||||
<span class="muted"> · ${status}</span>
|
||||
<div class="cron-run-entry__header">
|
||||
<div class="list-main cron-run-entry__main">
|
||||
<div class="list-title cron-run-entry__title">
|
||||
${entry.jobName ?? entry.jobId}
|
||||
<span class="muted"> · ${status}</span>
|
||||
</div>
|
||||
<div class="chip-row" style="margin-top: 4px;">
|
||||
<span class="chip">${delivery}</span>
|
||||
${entry.model ? html`<span class="chip">${entry.model}</span>` : nothing}
|
||||
${entry.provider ? html`<span class="chip">${entry.provider}</span>` : nothing}
|
||||
${usageSummary ? html`<span class="chip">${usageSummary}</span>` : nothing}
|
||||
</div>
|
||||
</div>
|
||||
<div class="list-sub cron-run-entry__summary">
|
||||
${entry.summary ?? entry.error ?? t("cron.runEntry.noSummary")}
|
||||
</div>
|
||||
<div class="chip-row" style="margin-top: 6px;">
|
||||
<span class="chip">${delivery}</span>
|
||||
${entry.model ? html`<span class="chip">${entry.model}</span>` : nothing}
|
||||
${entry.provider ? html`<span class="chip">${entry.provider}</span>` : nothing}
|
||||
${usageSummary ? html`<span class="chip">${usageSummary}</span>` : nothing}
|
||||
<div class="list-meta cron-run-entry__meta">
|
||||
<div>${formatMs(entry.ts)}</div>
|
||||
${typeof entry.runAtMs === "number"
|
||||
? html`<div class="muted">${t("cron.runEntry.runAt")} ${formatMs(entry.runAtMs)}</div>`
|
||||
: nothing}
|
||||
<div class="muted">${entry.durationMs ?? 0}ms</div>
|
||||
${typeof entry.nextRunAtMs === "number"
|
||||
? html`<div class="muted">${formatRunNextLabel(entry.nextRunAtMs)}</div>`
|
||||
: nothing}
|
||||
${chatUrl
|
||||
? html`<div>
|
||||
<a
|
||||
class="session-link"
|
||||
href=${chatUrl}
|
||||
@click=${(e: MouseEvent) => {
|
||||
if (
|
||||
e.defaultPrevented ||
|
||||
e.button !== 0 ||
|
||||
e.metaKey ||
|
||||
e.ctrlKey ||
|
||||
e.shiftKey ||
|
||||
e.altKey
|
||||
) {
|
||||
return;
|
||||
}
|
||||
if (onNavigateToChat && entry.sessionKey) {
|
||||
e.preventDefault();
|
||||
onNavigateToChat(entry.sessionKey);
|
||||
}
|
||||
}}
|
||||
>${t("cron.runEntry.openRunChat")}</a
|
||||
>
|
||||
</div>`
|
||||
: nothing}
|
||||
${showErrorInMeta ? html`<div class="muted">${entry.error}</div>` : nothing}
|
||||
${entry.deliveryError ? html`<div class="muted">${entry.deliveryError}</div>` : nothing}
|
||||
</div>
|
||||
</div>
|
||||
<div class="list-meta cron-run-entry__meta">
|
||||
<div>${formatMs(entry.ts)}</div>
|
||||
${typeof entry.runAtMs === "number"
|
||||
? html`<div class="muted">${t("cron.runEntry.runAt")} ${formatMs(entry.runAtMs)}</div>`
|
||||
: nothing}
|
||||
<div class="muted">${entry.durationMs ?? 0}ms</div>
|
||||
${typeof entry.nextRunAtMs === "number"
|
||||
? html`<div class="muted">${formatRunNextLabel(entry.nextRunAtMs)}</div>`
|
||||
: nothing}
|
||||
${chatUrl
|
||||
? html`<div>
|
||||
<a
|
||||
class="session-link"
|
||||
href=${chatUrl}
|
||||
@click=${(e: MouseEvent) => {
|
||||
if (
|
||||
e.defaultPrevented ||
|
||||
e.button !== 0 ||
|
||||
e.metaKey ||
|
||||
e.ctrlKey ||
|
||||
e.shiftKey ||
|
||||
e.altKey
|
||||
) {
|
||||
return;
|
||||
}
|
||||
if (onNavigateToChat && entry.sessionKey) {
|
||||
e.preventDefault();
|
||||
onNavigateToChat(entry.sessionKey);
|
||||
}
|
||||
}}
|
||||
>${t("cron.runEntry.openRunChat")}</a
|
||||
>
|
||||
</div>`
|
||||
: nothing}
|
||||
${entry.error ? html`<div class="muted">${entry.error}</div>` : nothing}
|
||||
${entry.deliveryError ? html`<div class="muted">${entry.deliveryError}</div>` : nothing}
|
||||
<div class="cron-run-entry__body chat-text">
|
||||
${unsafeHTML(toSanitizedMarkdownHtml(bodySource))}
|
||||
</div>
|
||||
</div>
|
||||
`;
|
||||
|
||||
Reference in New Issue
Block a user