From 492caf4cc7a4e4f259d0bb59c67f6f76a1f4661d Mon Sep 17 00:00:00 2001 From: Jeremy Clements <79224539+jeclrsg@users.noreply.github.com> Date: Thu, 17 Oct 2024 16:31:15 -0400 Subject: [PATCH] HPCC-32820 ECL Watch v9 logs view timestamp formatting support conditionally formatting the timestamp returned by the logging engine (loki returns nanoseconds rather than unixtime or a Date) Signed-off-by: Jeremy Clements <79224539+jeclrsg@users.noreply.github.com> --- esp/src/src-react/components/Logs.tsx | 14 ++++++++++++-- esp/src/src/Utility.ts | 23 +++++++++++++++++++++++ 2 files changed, 35 insertions(+), 2 deletions(-) diff --git a/esp/src/src-react/components/Logs.tsx b/esp/src/src-react/components/Logs.tsx index 40e65d0fc1a..3fcb095bdc2 100644 --- a/esp/src/src-react/components/Logs.tsx +++ b/esp/src/src-react/components/Logs.tsx @@ -3,7 +3,7 @@ import { CommandBar, ContextualMenuItemType, ICommandBarItemProps } from "@fluen import { GetLogsExRequest, LogaccessService, LogType, TargetAudience, WsLogaccess } from "@hpcc-js/comms"; import { Level, scopedLogger } from "@hpcc-js/util"; import nlsHPCC from "src/nlsHPCC"; -import { logColor, removeAllExcept, wuidToDate, wuidToTime } from "src/Utility"; +import { formatDateString, logColor, removeAllExcept, timestampToDate, wuidToDate, wuidToTime } from "src/Utility"; import { useLogAccessInfo } from "../hooks/platform"; import { HolyGrail } from "../layouts/HolyGrail"; import { pushParams } from "../util/history"; @@ -132,7 +132,17 @@ export const Logs: React.FunctionComponent = ({ } }); const retVal = { - timestamp: { label: nlsHPCC.TimeStamp, width: 140, sortable: false, }, + timestamp: { + label: nlsHPCC.TimeStamp, width: 140, sortable: false, + formatter: ts => { + if (ts) { + if (ts.indexOf(":") < 0) { + return timestampToDate(ts).toISOString(); + } + return formatDateString(ts); + } + }, + }, message: { label: nlsHPCC.Message, width: 600, sortable: false, }, components: { label: nlsHPCC.ContainerName, width: 150, sortable: false }, audience: { label: nlsHPCC.Audience, width: 60, sortable: false, }, diff --git a/esp/src/src/Utility.ts b/esp/src/src/Utility.ts index 32c0a912f59..d17b93257cb 100644 --- a/esp/src/src/Utility.ts +++ b/esp/src/src/Utility.ts @@ -1242,6 +1242,29 @@ export function format(labelTpl, obj) { .join("\n") ; } + +const TEN_TRILLION = 10000000000000; +export function nanosToMillis(timestamp: number): number { + if (timestamp > TEN_TRILLION) { + return Math.round(timestamp / 1000000); + } else { + return timestamp; + } +} + +export function timestampToDate(timestamp: number): Date { + const millis = nanosToMillis(timestamp); + return new Date(millis); +} + +export function formatDateString(dateStr: string): string { + const matches = dateStr.match(/([0-9]{4}(?:-[0-9]{1,2})+)([T\s])((?:[0-9]{1,2}:)+[0-9]{1,2}\.[0-9]{1,3})(Z*)/); + if (matches) { + return `${matches[1]}T${matches[3]}${matches[4] ? matches[4] : "Z"}`; + } + return dateStr; +} + const theme = getTheme(); const { semanticColors } = theme;