Use payload & avoid transforming in clojure objects.

This commit includes a few performance fixes:
1) Pass a string payload instead of an hex encoded string, to avoid
unecessary conversion
2) Don't js->clj on messages, as that's fairly expensive and we can get
away without
3) Don't use `pr-str` `read-string`, rather convert to json

Signed-off-by: Andrea Maria Piana <andrea.maria.piana@gmail.com>
This commit is contained in:
Andrea Maria Piana 2019-10-23 09:09:11 +02:00
parent 57b1722863
commit 9cd891365a
No known key found for this signature in database
GPG Key ID: AA6CCA6DE0E06424
9 changed files with 654 additions and 626 deletions

View File

@ -42,6 +42,10 @@ var TopLevel = {
"charCodeAt" : function () {}, "charCodeAt" : function () {},
"checkAddressChecksum" : function () {}, "checkAddressChecksum" : function () {},
"checkVideoAuthorizationStatus" : function () {}, "checkVideoAuthorizationStatus" : function () {},
"author": function() {},
"encryptionId": function() {},
"chat": function() {},
"messageId": function() {},
"clear" : function () {}, "clear" : function () {},
"clearCookies" : function () {}, "clearCookies" : function () {},
"clearInterval" : function () {}, "clearInterval" : function () {},
@ -53,6 +57,10 @@ var TopLevel = {
"code" : function () {}, "code" : function () {},
"concat" : function () {}, "concat" : function () {},
"confirmMessagesProcessed" : function () {}, "confirmMessagesProcessed" : function () {},
"messages": function() {},
"discovery": function() {},
"negotiated": function() {},
"chatId": function() {},
"confirmMessagesProcessedByID" : function () {}, "confirmMessagesProcessedByID" : function () {},
"connectionChange" : function () {}, "connectionChange" : function () {},
"console" : function () {}, "console" : function () {},
@ -75,6 +83,8 @@ var TopLevel = {
"DateTimeFormat" : function () {}, "DateTimeFormat" : function () {},
"DEBUG" : function () {}, "DEBUG" : function () {},
"decay" : function () {}, "decay" : function () {},
"dedupId": function() {},
"DECIMAL_SEP" : function () {}, "DECIMAL_SEP" : function () {},
"GROUP_SEP" : function () {}, "GROUP_SEP" : function () {},
"PERCENT" : function () {}, "PERCENT" : function () {},
@ -162,6 +172,8 @@ var TopLevel = {
"getNode" : function () {}, "getNode" : function () {},
"getNodesFromContract" : function () {}, "getNodesFromContract" : function () {},
"getPublicKey" : function () {}, "getPublicKey" : function () {},
"publicKey": function() {},
"alias": function() {},
"getSecurityLevel" : function () {}, "getSecurityLevel" : function () {},
"getStateForAction" : function () {}, "getStateForAction" : function () {},
"getString" : function () {}, "getString" : function () {},
@ -246,6 +258,7 @@ var TopLevel = {
"MessagesSeenHandler" : function () {}, "MessagesSeenHandler" : function () {},
"messaging" : function () {}, "messaging" : function () {},
"method" : function () {}, "method" : function () {},
"metadata": function() {},
"minus" : function () {}, "minus" : function () {},
"plus" : function () {}, "plus" : function () {},
"mkdir" : function () {}, "mkdir" : function () {},
@ -510,6 +523,7 @@ var TopLevel = {
"toUtf8" : function () {}, "toUtf8" : function () {},
"toWei" : function () {}, "toWei" : function () {},
"translations" : function () {}, "translations" : function () {},
"type": function() {},
"Uint8Array" : function () {}, "Uint8Array" : function () {},
"unblockPin" : function () {}, "unblockPin" : function () {},
"unlink" : function () {}, "unlink" : function () {},

View File

@ -5,6 +5,7 @@
[clojure.string :as string] [clojure.string :as string]
[taoensso.timbre :as log] [taoensso.timbre :as log]
[re-frame.core :as re-frame] [re-frame.core :as re-frame]
[status-im.utils.types :as utils.types]
[status-im.ethereum.json-rpc :as json-rpc] [status-im.ethereum.json-rpc :as json-rpc]
[status-im.constants :as constants] [status-im.constants :as constants]
[status-im.utils.core :as utils])) [status-im.utils.core :as utils]))
@ -12,7 +13,7 @@
(defn prepare-content [content] (defn prepare-content [content]
(if (string? content) (if (string? content)
content content
(pr-str content))) (utils.types/clj->json content)))
(defn ->rpc [message] (defn ->rpc [message]
(-> message (-> message

View File

@ -2,6 +2,7 @@
(:require [status-im.ethereum.subscriptions :as ethereum.subscriptions] (:require [status-im.ethereum.subscriptions :as ethereum.subscriptions]
[status-im.i18n :as i18n] [status-im.i18n :as i18n]
[status-im.mailserver.core :as mailserver] [status-im.mailserver.core :as mailserver]
[clojure.string :as string]
[status-im.multiaccounts.login.core :as login] [status-im.multiaccounts.login.core :as login]
[status-im.multiaccounts.model :as multiaccounts.model] [status-im.multiaccounts.model :as multiaccounts.model]
[status-im.pairing.core :as pairing] [status-im.pairing.core :as pairing]
@ -42,19 +43,23 @@
(fx/defn process (fx/defn process
[cofx event-str] [cofx event-str]
(let [{:keys [type event]} (types/json->clj event-str)] ;; We only convert to clojure when strictly necessary or we know it
;; won't impact performance, as it is a fairly costly operation on large-ish
;; data structures
(let [data (.parse js/JSON event-str)
event-js (.-event data)
type (.-type data)]
(case type (case type
"node.login" (status-node-started cofx event) "node.login" (status-node-started cofx (js->clj event-js :keywordize-keys true))
"envelope.sent" (transport.message/update-envelopes-status cofx (:ids event) :sent) "envelope.sent" (transport.message/update-envelopes-status cofx (:ids (js->clj event-js :keywordize-keys true)) :sent)
"envelope.expired" (transport.message/update-envelopes-status cofx (:ids event) :not-sent) "envelope.expired" (transport.message/update-envelopes-status cofx (:ids (js->clj event-js :keywordize-keys true)) :not-sent)
"bundles.added" (pairing/handle-bundles-added cofx event) "mailserver.request.completed" (mailserver/handle-request-completed cofx (js->clj event-js :keywordize-keys true))
"mailserver.request.completed" (mailserver/handle-request-completed cofx event)
"mailserver.request.expired" (when (multiaccounts.model/logged-in? cofx) "mailserver.request.expired" (when (multiaccounts.model/logged-in? cofx)
(mailserver/resend-request cofx {:request-id (:hash event)})) (mailserver/resend-request cofx {:request-id (.-hash event-js)}))
"discovery.summary" (summary cofx event) "discovery.summary" (summary cofx (js->clj event-js :keywordize-keys true))
"subscriptions.data" (ethereum.subscriptions/handle-signal cofx event) "subscriptions.data" (ethereum.subscriptions/handle-signal cofx (js->clj event-js :keywordize-keys true))
"subscriptions.error" (ethereum.subscriptions/handle-error cofx event) "subscriptions.error" (ethereum.subscriptions/handle-error cofx (js->clj event-js :keywordize-keys true))
"whisper.filter.added" (transport.filters/handle-negotiated-filter cofx event) "whisper.filter.added" (transport.filters/handle-negotiated-filter cofx (js->clj event-js :keywordize-keys true))
"messages.new" (transport.message/receive-messages cofx event) "messages.new" (transport.message/receive-messages cofx event-js)
"wallet" (ethereum.subscriptions/new-wallet-event cofx event) "wallet" (ethereum.subscriptions/new-wallet-event cofx (js->clj event-js :keywordize-keys true))
(log/debug "Event " type " not handled" event)))) (log/debug "Event " type " not handled"))))

View File

@ -26,13 +26,19 @@
dedup-id is passed by status-go and is used to deduplicate messages at that layer. dedup-id is passed by status-go and is used to deduplicate messages at that layer.
Once a message has been successfuly processed, that id needs to be sent back Once a message has been successfuly processed, that id needs to be sent back
in order to stop receiving that message" in order to stop receiving that message"
[cofx now-in-s filter-chat-id message] [cofx now-in-s filter-chat-id message-js]
(let [blocked-contacts (get-in cofx [:db :contacts/blocked] #{}) (let [blocked-contacts (get-in cofx [:db :contacts/blocked] #{})
{{:keys [payload timestamp]} :message payload (.-payload message-js)
metadata :metadata timestamp (.-timestamp (.-message message-js))
raw-payload :raw-payload} (add-raw-payload message) metadata-js (.-metadata message-js)
metadata {:author {:publicKey (.-publicKey (.-author metadata-js))
:alias (.-alias (.-author metadata-js))
:identicon (.-identicon (.-author metadata-js))}
:dedupId (.-dedupId metadata-js)
:encryptionId (.-encryptionId metadata-js)
:messageId (.-messageId metadata-js)}
raw-payload {:raw-payload message-js}
status-message (-> payload status-message (-> payload
ethereum/hex-to-utf8
transit/deserialize) transit/deserialize)
sig (-> metadata :author :publicKey)] sig (-> metadata :author :publicKey)]
(when (and sig (when (and sig
@ -70,9 +76,12 @@
(apply fx/merge cofx receive-message-fxs)) (apply fx/merge cofx receive-message-fxs))
(log/error "Something went wrong" error messages))) (log/error "Something went wrong" error messages)))
(fx/defn receive-messages [cofx event] (fx/defn receive-messages [cofx event-js]
(let [fxs (keep (let [fxs (keep
(fn [{:keys [chat messages error]}] (fn [message-specs]
(let [chat (.-chat message-specs)
messages (.-messages message-specs)
error (.-error message-specs)]
(when (seq messages) (when (seq messages)
(receive-whisper-messages (receive-whisper-messages
error error
@ -80,11 +89,11 @@
;; For discovery and negotiated filters we don't ;; For discovery and negotiated filters we don't
;; set a chatID, and we use the signature of the message ;; set a chatID, and we use the signature of the message
;; to indicate which chat it is for ;; to indicate which chat it is for
(if (or (:discovery chat) (if (or (.-discovery chat)
(:negotiated chat)) (.-negotiated chat))
nil nil
(:chatId chat))))) (.-chatId chat))))))
(:messages event))] (.-messages event-js))]
(apply fx/merge cofx fxs))) (apply fx/merge cofx fxs)))
(fx/defn remove-hash (fx/defn remove-hash

View File

@ -1,7 +1,5 @@
(ns status-im.utils.core (ns status-im.utils.core
(:require [clojure.string :as str] (:require [clojure.string :as str]
#?(:cljs [cljs.tools.reader.edn :as edn]
:clj [clojure.tools.reader.edn :as edn])
[taoensso.timbre :as log])) [taoensso.timbre :as log]))
(defn truncate-str (defn truncate-str
@ -70,6 +68,6 @@
#?(:cljs #?(:cljs
(defn safe-read-message-content [content] (defn safe-read-message-content [content]
(try (try
(edn/read-string content) (js->clj (.parse js/JSON content) :keywordize-keys true)
(catch :default e (catch :default e
(log/warn "failed to transform message with " e))))) (log/warn "failed to transform message with " e)))))

View File

@ -2,7 +2,7 @@
"_comment": "DO NOT EDIT THIS FILE BY HAND. USE 'scripts/update-status-go.sh <tag>' instead", "_comment": "DO NOT EDIT THIS FILE BY HAND. USE 'scripts/update-status-go.sh <tag>' instead",
"owner": "status-im", "owner": "status-im",
"repo": "status-go", "repo": "status-go",
"version": "v0.34.0-beta.0", "version": "5f2544f1e8f2ee832bbab95c8d74e26ba0ab2481",
"commit-sha1": "94228bce2ee6a7a5f62917e28247a22c37b6699b", "commit-sha1": "5f2544f1e8f2ee832bbab95c8d74e26ba0ab2481",
"src-sha256": "0f62yh94vn6k4gqa50r3gxzscqabmgc047sng291g5wp6d755fmr" "src-sha256": "0dlkr6056mhyny9k895zs5012ywiz75qm6y7a6hypb3pfxr35931"
} }

View File

@ -27,7 +27,7 @@
:from from :from from
:chatId chat-id :chatId chat-id
:replyTo "id-2" :replyTo "id-2"
:content "{:chat-id \"chat-id\", :response-to-v2 \"id-2\", :text \"hta\"}" :content "{\"chat-id\":\"chat-id\",\"response-to-v2\":\"id-2\",\"text\":\"hta\"}"
:contentType "text/plain" :contentType "text/plain"
:messageType "public-group-user-message" :messageType "public-group-user-message"
:clockValue 2 :clockValue 2
@ -55,12 +55,12 @@
:whisperTimestamp 1 :whisperTimestamp 1
:from from :from from
:chatId chat-id :chatId chat-id
:content "{:chat-id \"chat-id\", :text \"hta\"}" :content "{\"chat-id\":\"chat-id\",\"text\":\"hta\"}"
:contentType "text/plain" :contentType "text/plain"
:messageType "public-group-user-message" :messageType "public-group-user-message"
:clockValue 2 :clockValue 2
:quotedMessage {:from "from" :quotedMessage {:from "from"
:content "{:chat-id \"chat-id\", :text \"reply\"}"} :content "{\"chat-id\":\"chat-id\",\"text\":\"reply\"}"}
:timestamp 3 :timestamp 3
:outgoingStatus "sending"}] :outgoingStatus "sending"}]
(is (= expected (m/<-rpc message)))))) (is (= expected (m/<-rpc message))))))

View File

@ -46,8 +46,9 @@
(def sig "0x04325367620ae20dd878dbb39f69f02c567d789dd21af8a88623dc5b529827c2812571c380a2cd8236a2851b8843d6486481166c39debf60a5d30b9099c66213e4") (def sig "0x04325367620ae20dd878dbb39f69f02c567d789dd21af8a88623dc5b529827c2812571c380a2cd8236a2851b8843d6486481166c39debf60a5d30b9099c66213e4")
(def messages [{:id "someid" (def messages [(clj->js {:id "someid"
:metadata {:author {:publicKey sig}} :metadata {:author {:publicKey sig}}
:payload "[\"~#c4\",[\"F\",\"text/plain\",\"~:public-group-user-message\",152769201438101,1527692014377]]"
:message {:sig sig :message {:sig sig
:ttl 10 :ttl 10
:timestamp 1527692015 :timestamp 1527692015
@ -55,7 +56,7 @@
:payload "0x5b227e236334222c5b2246222c22746578742f706c61696e222c227e3a7075626c69632d67726f75702d757365722d6d657373616765222c3135323736393230313433383130312c313532373639323031343337375d5d" :payload "0x5b227e236334222c5b2246222c22746578742f706c61696e222c227e3a7075626c69632d67726f75702d757365722d6d657373616765222c3135323736393230313433383130312c313532373639323031343337375d5d"
:padding "0xbf06347cc7f9aa18b4a846032264a88f559d9b14079975d14b10648847c0543a77a80624e101c082d19b502ae3b4f97958d18abf59eb0a82afc1301aa22470495fac739a30c2f563599fa8d8e09363a43d39311596b7f119dee7b046989c08224f1ef5cdc385" :padding "0xbf06347cc7f9aa18b4a846032264a88f559d9b14079975d14b10648847c0543a77a80624e101c082d19b502ae3b4f97958d18abf59eb0a82afc1301aa22470495fac739a30c2f563599fa8d8e09363a43d39311596b7f119dee7b046989c08224f1ef5cdc385"
:pow 0.002631578947368421 :pow 0.002631578947368421
:hash "0x220ef9994a4fae64c112b27ed07ef910918159cbe6fcf8ac515ee2bf9a6711a0"}}]) :hash "0x220ef9994a4fae64c112b27ed07ef910918159cbe6fcf8ac515ee2bf9a6711a0"}})])
(deftest receive-whisper-messages-test (deftest receive-whisper-messages-test
(testing "an error is reported" (testing "an error is reported"