/* * Copyright The OpenTelemetry Authors * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * https://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ import { otperformance as performance } from '../platform'; const NANOSECOND_DIGITS = 9; const SECOND_TO_NANOSECONDS = Math.pow(10, NANOSECOND_DIGITS); /** * Converts a number to HrTime, HrTime = [number, number]. * The first number is UNIX Epoch time in seconds since 00:00:00 UTC on 1 January 1970. * The second number represents the partial second elapsed since Unix Epoch time represented by first number in nanoseconds. * For example, 2021-01-01T12:30:10.150Z in UNIX Epoch time in milliseconds is represented as 1609504210150. * numberToHrtime calculates the first number by converting and truncating the Epoch time in milliseconds to seconds: * HrTime[0] = Math.trunc(1609504210150 / 1000) = 1609504210. * numberToHrtime calculates the second number by converting the digits after the decimal point of the subtraction, (1609504210150 / 1000) - HrTime[0], to nanoseconds: * HrTime[1] = Number((1609504210.150 - HrTime[0]).toFixed(9)) * SECOND_TO_NANOSECONDS = 150000000. * This is represented in HrTime format as [1609504210, 150000000]. * @param epochMillis */ function numberToHrtime(epochMillis) { const epochSeconds = epochMillis / 1000; // Decimals only. const seconds = Math.trunc(epochSeconds); // Round sub-nanosecond accuracy to nanosecond. const nanos = Number((epochSeconds - seconds).toFixed(NANOSECOND_DIGITS)) * SECOND_TO_NANOSECONDS; return [seconds, nanos]; } function getTimeOrigin() { let timeOrigin = performance.timeOrigin; if (typeof timeOrigin !== 'number') { const perf = performance; timeOrigin = perf.timing && perf.timing.fetchStart; } return timeOrigin; } /** * Returns an hrtime calculated via performance component. * @param performanceNow */ export function hrTime(performanceNow) { const timeOrigin = numberToHrtime(getTimeOrigin()); const now = numberToHrtime(typeof performanceNow === 'number' ? performanceNow : performance.now()); let seconds = timeOrigin[0] + now[0]; let nanos = timeOrigin[1] + now[1]; // Nanoseconds if (nanos > SECOND_TO_NANOSECONDS) { nanos -= SECOND_TO_NANOSECONDS; seconds += 1; } return [seconds, nanos]; } /** * * Converts a TimeInput to an HrTime, defaults to _hrtime(). * @param time */ export function timeInputToHrTime(time) { // process.hrtime if (isTimeInputHrTime(time)) { return time; } else if (typeof time === 'number') { // Must be a performance.now() if it's smaller than process start time. if (time < getTimeOrigin()) { return hrTime(time); } else { // epoch milliseconds or performance.timeOrigin return numberToHrtime(time); } } else if (time instanceof Date) { return numberToHrtime(time.getTime()); } else { throw TypeError('Invalid input type'); } } /** * Returns a duration of two hrTime. * @param startTime * @param endTime */ export function hrTimeDuration(startTime, endTime) { let seconds = endTime[0] - startTime[0]; let nanos = endTime[1] - startTime[1]; // overflow if (nanos < 0) { seconds -= 1; // negate nanos += SECOND_TO_NANOSECONDS; } return [seconds, nanos]; } /** * Convert hrTime to timestamp, for example "2019-05-14T17:00:00.000123456Z" * @param time */ export function hrTimeToTimeStamp(time) { const precision = NANOSECOND_DIGITS; const tmp = `${'0'.repeat(precision)}${time[1]}Z`; const nanoString = tmp.substr(tmp.length - precision - 1); const date = new Date(time[0] * 1000).toISOString(); return date.replace('000Z', nanoString); } /** * Convert hrTime to nanoseconds. * @param time */ export function hrTimeToNanoseconds(time) { return time[0] * SECOND_TO_NANOSECONDS + time[1]; } /** * Convert hrTime to milliseconds. * @param time */ export function hrTimeToMilliseconds(time) { return Math.round(time[0] * 1e3 + time[1] / 1e6); } /** * Convert hrTime to microseconds. * @param time */ export function hrTimeToMicroseconds(time) { return Math.round(time[0] * 1e6 + time[1] / 1e3); } /** * check if time is HrTime * @param value */ export function isTimeInputHrTime(value) { return (Array.isArray(value) && value.length === 2 && typeof value[0] === 'number' && typeof value[1] === 'number'); } /** * check if input value is a correct types.TimeInput * @param value */ export function isTimeInput(value) { return (isTimeInputHrTime(value) || typeof value === 'number' || value instanceof Date); } //# sourceMappingURL=time.js.map