From 449eb8536bb106a58883fc1f78dd19c5796c6d89 Mon Sep 17 00:00:00 2001 From: vincentbavitz <58160433+vincentbavitz@users.noreply.github.com> Date: Wed, 4 Dec 2019 16:07:20 +1100 Subject: [PATCH] System clock (#677) * Show toast when client clock is out of sync * Fixed syntactical misarrangements * Simplify getServerTime method * Futher simplify getServerTime method * Update js/modules/loki_app_dot_net_api.js Co-Authored-By: Mikunj Varsani * Update js/modules/loki_app_dot_net_api.js Co-Authored-By: Mikunj Varsani * Update loki_app_dot_net_api.js Use cached timestamp value * Update preload.js Update some values to function scope * Updated syntax * Semifinal changes * Improved efficiency and eliminated need for restart on clock update * Remove await and immediately invoked function --- _locales/en/messages.json | 6 +++++ js/background.js | 5 ++++- js/modules/loki_app_dot_net_api.js | 36 ++++++++++++++++++++++++++++++ js/views/conversation_view.js | 10 +++++++++ 4 files changed, 56 insertions(+), 1 deletion(-) diff --git a/_locales/en/messages.json b/_locales/en/messages.json index 91f32bdd7..38ae0a326 100644 --- a/_locales/en/messages.json +++ b/_locales/en/messages.json @@ -1429,6 +1429,12 @@ "description": "Warning notification that this version of the app has expired" }, + "clockOutOfSync": { + "message": + "Your clock is out of sync. Please update your clock and try again.", + "description": + "Notifcation that user's clock is out of sync with Loki's servers." + }, "upgrade": { "message": "Upgrade", "description": diff --git a/js/background.js b/js/background.js index 13ef99a9b..b191176ea 100644 --- a/js/background.js +++ b/js/background.js @@ -11,7 +11,7 @@ libloki, libsignal, StringView, - BlockedNumberController + BlockedNumberController, */ // eslint-disable-next-line func-names @@ -125,6 +125,9 @@ 'loki/loki_icon_128.png', ]); + // Set server-client time difference + window.LokiPublicChatAPI.setClockParams(); + // We add this to window here because the default Node context is erased at the end // of preload.js processing window.setImmediate = window.nodeSetImmediate; diff --git a/js/modules/loki_app_dot_net_api.js b/js/modules/loki_app_dot_net_api.js index 5a3710d65..cc6f756e0 100644 --- a/js/modules/loki_app_dot_net_api.js +++ b/js/modules/loki_app_dot_net_api.js @@ -63,6 +63,42 @@ class LokiAppDotNetAPI extends EventEmitter { return thisServer; } + static async getServerTime() { + const url = `${window.getDefaultFileServer()}/loki/v1/time`; + let timestamp = NaN; + + try { + const res = await nodeFetch(url); + if (res.ok) { + timestamp = await res.text(); + } + } catch (e) { + return timestamp; + } + + return Number(timestamp); + } + + static async getTimeDifferential() { + // Get time differential between server and client in seconds + const serverTime = await this.getServerTime(); + const clientTime = Math.ceil(Date.now() / 1000); + + if (Number.isNaN(serverTime)) { + return 0; + } + return serverTime - clientTime; + } + + static async setClockParams() { + // Set server-client time difference + const maxTimeDifferential = 30; + const timeDifferential = await this.getTimeDifferential(); + + window.clientClockSynced = Math.abs(timeDifferential) < maxTimeDifferential; + return window.clientClockSynced; + } + // channel getter/factory async findOrCreateChannel(serverUrl, channelId, conversationId) { const server = await this.findOrCreateServer(serverUrl); diff --git a/js/views/conversation_view.js b/js/views/conversation_view.js index 60af7147c..b65fddcba 100644 --- a/js/views/conversation_view.js +++ b/js/views/conversation_view.js @@ -28,6 +28,11 @@ return { toastMessage: i18n('expiredWarning') }; }, }); + Whisper.ClockOutOfSyncToast = Whisper.ToastView.extend({ + render_attributes() { + return { toastMessage: i18n('clockOutOfSync') }; + }, + }); Whisper.BlockedToast = Whisper.ToastView.extend({ render_attributes() { return { toastMessage: i18n('unblockToSend') }; @@ -1956,6 +1961,11 @@ if (extension.expired()) { toast = new Whisper.ExpiredToast(); } + if (!window.clientClockSynced) { + // Check to see if user has updated their clock to current time + const clockSynced = await window.LokiPublicChatAPI.setClockParams(); + toast = clockSynced ? toast : new Whisper.ClockOutOfSyncToast(); + } if (this.model.isPrivate() && storage.isBlocked(this.model.id)) { toast = new Whisper.BlockedToast(); }