2021-11-17 12:54:34 +00:00
|
|
|
import shim from './shim';
|
2021-11-28 16:58:39 +00:00
|
|
|
import time from './time';
|
2021-11-17 12:54:34 +00:00
|
|
|
const ntpClient_ = require('./vendor/ntp-client');
|
|
|
|
|
2023-05-10 12:50:48 +01:00
|
|
|
interface NtpServer {
|
|
|
|
domain: string;
|
|
|
|
port: number;
|
|
|
|
}
|
2021-11-17 12:54:34 +00:00
|
|
|
|
|
|
|
function ntpClient() {
|
|
|
|
ntpClient_.dgram = shim.dgram();
|
|
|
|
return ntpClient_;
|
|
|
|
}
|
|
|
|
|
2023-05-10 12:50:48 +01:00
|
|
|
const parseNtpServer = (ntpServer: string): NtpServer => {
|
|
|
|
const s = ntpServer.split(':');
|
|
|
|
if (s.length !== 2) throw new Error('NTP server URL must be in format `domain:port`');
|
|
|
|
return {
|
|
|
|
domain: s[0],
|
|
|
|
port: Number(s[1]),
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
export async function getNetworkTime(ntpServer: string): Promise<Date> {
|
2023-06-30 10:30:29 +01:00
|
|
|
// eslint-disable-next-line @typescript-eslint/ban-types -- Old code before rule was applied
|
2023-02-20 12:02:29 -03:00
|
|
|
return new Promise((resolve: Function, reject: Function) => {
|
2023-05-10 12:50:48 +01:00
|
|
|
const s = parseNtpServer(ntpServer);
|
2024-04-05 12:16:49 +01:00
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
2023-05-10 12:50:48 +01:00
|
|
|
ntpClient().getNetworkTime(s.domain, s.port, (error: any, date: Date) => {
|
2021-11-17 12:54:34 +00:00
|
|
|
if (error) {
|
|
|
|
reject(error);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
resolve(date);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2023-05-10 12:50:48 +01:00
|
|
|
export async function getDeviceTimeDrift(ntpServer: string): Promise<number> {
|
2021-11-28 16:58:39 +00:00
|
|
|
const maxTries = 3;
|
|
|
|
let tryCount = 0;
|
|
|
|
|
2021-11-17 12:54:34 +00:00
|
|
|
let ntpTime: Date = null;
|
2021-11-28 16:58:39 +00:00
|
|
|
|
|
|
|
while (true) {
|
|
|
|
tryCount++;
|
|
|
|
try {
|
2023-05-10 12:50:48 +01:00
|
|
|
ntpTime = await getNetworkTime(ntpServer);
|
2021-11-28 16:58:39 +00:00
|
|
|
break;
|
|
|
|
} catch (error) {
|
|
|
|
if (tryCount >= maxTries) {
|
2021-12-16 10:53:28 +01:00
|
|
|
const newError = typeof error === 'string' ? new Error(error) : error;
|
2023-05-10 12:50:48 +01:00
|
|
|
newError.message = `Cannot retrieve the network time from ${ntpServer}: ${newError.message}`;
|
2021-12-16 10:53:28 +01:00
|
|
|
throw newError;
|
2021-11-28 16:58:39 +00:00
|
|
|
} else {
|
|
|
|
await time.msleep(tryCount * 1000);
|
|
|
|
}
|
|
|
|
}
|
2021-11-17 12:54:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return ntpTime.getTime() - Date.now();
|
|
|
|
}
|
|
|
|
|
|
|
|
// export default async function(): Promise<Date> {
|
|
|
|
// if (shouldSyncTime()) {
|
|
|
|
// const release = await fetchingTimeMutex.acquire();
|
|
|
|
|
|
|
|
// try {
|
|
|
|
// if (shouldSyncTime()) {
|
|
|
|
// const date = await networkTime();
|
|
|
|
// nextSyncTime = Date.now() + 60 * 1000;
|
|
|
|
// timeOffset = date.getTime() - Date.now();
|
|
|
|
// }
|
|
|
|
// } catch (error) {
|
|
|
|
// logger.warn('Could not get NTP time - falling back to device time:', error);
|
|
|
|
// // Fallback to device time since
|
|
|
|
// // most of the time it's actually correct
|
|
|
|
// nextSyncTime = Date.now() + 20 * 1000;
|
|
|
|
// timeOffset = 0;
|
|
|
|
// } finally {
|
|
|
|
// release();
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
|
|
|
|
// return new Date(Date.now() + timeOffset);
|
|
|
|
// }
|