1
0
Эх сурвалжийг харах

enhance: add malli schema for frontend.fs.sync

rcmerci 2 жил өмнө
parent
commit
dafb2a4aff

+ 33 - 17
.clj-kondo/metosin/malli-types/config.edn

@@ -2,27 +2,24 @@
  {:unresolved-symbol {:exclude [(malli.core/=>)]},
   :type-mismatch
   {:namespaces
-   {frontend.util
-    {safe-re-find {:arities {2 {:args [:any :string], :ret :any}}},
-     check-password-strength
+   {frontend.fs.sync
+    {sync-state
      {:arities
-      {1
-       {:args [:string],
+      {0
+       {:args [],
         :ret
         {:op :keys,
          :req
-         {:contains :sequential,
-          :length :int,
-          :id :int,
-          :value :string},
-         :nilable true}}}},
-     uuid-string? {:arities {1 {:args [:string], :ret :boolean}}},
-     safe-parse-int {:arities {1 {:args [:any], :ret :int}}},
-     safe-sanitize-file-name
-     {:arities {1 {:args [:string], :ret :string}}},
-     safe-parse-float {:arities {1 {:args [:any], :ret :double}}}},
-    frontend.fs.sync
-    {<update-graphs-txid-only-txid!
+         {:full-local->remote-files :set,
+          :history :sequential,
+          :queued-local->remote-files :set,
+          :state :keyword,
+          :current-remote->local-files :set,
+          :current-local->remote-files :set,
+          :full-remote->local-files :set,
+          :recent-remote->local-files :set,
+          :current-syncing-graph-uuid :nilable/string}}}}},
+     <update-graphs-txid-only-txid!
      {:arities {2 {:args [:int :string], :ret :any}}},
      read-graphs-txid
      {:arities
@@ -48,5 +45,24 @@
            :work-dir :string}}
          :string],
         :ret :any}}}},
+    frontend.util
+    {safe-re-find {:arities {2 {:args [:any :string], :ret :any}}},
+     check-password-strength
+     {:arities
+      {1
+       {:args [:string],
+        :ret
+        {:op :keys,
+         :req
+         {:contains :sequential,
+          :length :int,
+          :id :int,
+          :value :string},
+         :nilable true}}}},
+     uuid-string? {:arities {1 {:args [:string], :ret :boolean}}},
+     safe-parse-int {:arities {1 {:args [:any], :ret :int}}},
+     safe-sanitize-file-name
+     {:arities {1 {:args [:string], :ret :string}}},
+     safe-parse-float {:arities {1 {:args [:any], :ret :double}}}},
     frontend.state
     {pub-event! {:arities {1 {:args [:vector], :ret :any}}}}}}}}

+ 38 - 132
src/main/frontend/fs/sync.cljs

@@ -10,7 +10,6 @@
                                                poll! timeout]]
             [cljs.core.async.impl.channels]
             [cljs.core.async.interop :refer [p->c]]
-            [cljs.spec.alpha :as s]
             [clojure.pprint :as pp]
             [clojure.set :as set]
             [clojure.string :as string]
@@ -23,6 +22,11 @@
             [frontend.encrypt :as encrypt]
             [frontend.fs :as fs]
             [frontend.fs.capacitor-fs :as capacitor-fs]
+            [frontend.fs.sync-schema :refer [diff-schema state-schema
+                                             sync-state-schema
+                                             sync-local->remote-all-files!-result-schema
+                                             sync-remote->local!-result-schema
+                                             sync-local->remote!-result-schema]]
             [frontend.handler.notification :as notification]
             [frontend.handler.user :as user]
             [frontend.mobile.util :as mobile-util]
@@ -34,6 +38,7 @@
             [goog.string :as gstring]
             [lambdaisland.glogi :as log]
             [logseq.graph-parser.util :as gp-util]
+            [malli.core :as m]
             [medley.core :refer [dedupe-by]]
             [promesa.core :as p]
             [rum.core :as rum]))
@@ -65,104 +70,6 @@
 ;; TODO: a remote delete-diff cause local related-file deleted, then trigger a `FileChangeEvent`,
 ;;       and re-produce a new same-file-delete diff.
 
-;;; ### specs
-(s/def ::state #{;; do following jobs when ::starting:
-                 ;; - wait seconds for file-change-events from file-watcher
-                 ;; - drop redundant file-change-events
-                 ;; - setup states in `frontend.state`
-                 ::starting
-                 ::need-password
-                 ::idle
-                 ;; sync local-changed files
-                 ::local->remote
-                 ;; sync remote latest-transactions
-                 ::remote->local
-                 ;; local->remote full sync
-                 ::local->remote-full-sync
-                 ;; remote->local full sync
-                 ::remote->local-full-sync
-                 ;; snapshot state when switching between apps on iOS
-                 ::pause
-                 ::stop})
-(s/def ::path string?)
-(s/def ::time t/date?)
-(s/def ::remote->local-type #{:delete :update
-                              ;; :rename=:delete+:update
-                              })
-(s/def ::current-syncing-graph-uuid (s/or :nil nil? :graph-uuid string?))
-(s/def ::recent-remote->local-file-item (s/keys :req-un [::remote->local-type ::checksum ::path]))
-(s/def ::current-local->remote-files (s/coll-of ::path :kind set?))
-(s/def ::current-remote->local-files (s/coll-of ::path :kind set?))
-(s/def ::recent-remote->local-files (s/coll-of ::recent-remote->local-file-item :kind set?))
-(s/def ::history-item (s/keys :req-un [::path ::time]))
-(s/def ::history (s/coll-of ::history-item :kind seq?))
-(s/def ::sync-state (s/keys :req-un [::current-syncing-graph-uuid
-                                     ::state
-                                     ::current-local->remote-files
-                                     ::current-remote->local-files
-                                     ::queued-local->remote-files
-                                     ;; Downloading files from remote will trigger filewatcher events,
-                                     ;; causes unreasonable information in the content of ::queued-local->remote-files,
-                                     ;; use ::recent-remote->local-files to filter such events
-                                     ::recent-remote->local-files
-                                     ::history]))
-
-;; diff
-(s/def ::TXId pos-int?)
-(s/def ::TXType #{"update_files" "delete_files" "rename_file"})
-(s/def ::TXContent-to-path string?)
-(s/def ::TXContent-from-path (s/or :some string? :none nil?))
-(s/def ::TXContent-checksum (s/or :some string? :none nil?))
-(s/def ::TXContent-item (s/tuple ::TXContent-to-path
-                                 ::TXContent-from-path
-                                 ::TXContent-checksum))
-(s/def ::TXContent (s/coll-of ::TXContent-item))
-(s/def ::diff (s/keys :req-un [::TXId ::TXType ::TXContent]))
-
-(s/def ::succ-map #(= {:succ true} %))
-(s/def ::unknown-map (comp some? :unknown))
-(s/def ::stop-map #(= {:stop true} %))
-(s/def ::pause-map #(= {:pause true} %))
-(s/def ::need-sync-remote #(= {:need-sync-remote true} %))
-(s/def ::graph-has-been-deleted #(= {:graph-has-been-deleted true} %))
-
-(s/def ::sync-local->remote!-result
-  (s/or :stop ::stop-map
-        :succ ::succ-map
-        :pause ::pause-map
-        :need-sync-remote ::need-sync-remote
-        :graph-has-been-deleted ::graph-has-been-deleted
-        :unknown ::unknown-map))
-
-(s/def ::sync-remote->local!-result
-  (s/or :succ ::succ-map
-        :need-remote->local-full-sync
-        #(= {:need-remote->local-full-sync true} %)
-        :stop ::stop-map
-        :pause ::pause-map
-        :unknown ::unknown-map))
-
-(s/def ::sync-local->remote-all-files!-result
-  (s/or :succ ::succ-map
-        :stop ::stop-map
-        :need-sync-remote ::need-sync-remote
-        :graph-has-been-deleted ::graph-has-been-deleted
-        :unknown ::unknown-map))
-
-;; sync-event type
-(s/def ::event #{:created-local-version-file
-                 :finished-local->remote
-                 :finished-remote->local
-                 :start
-                 :pause
-                 :resume
-                 :exception-decrypt-failed
-                 :remote->local-full-sync-failed
-                 :local->remote-full-sync-failed
-                 :get-remote-graph-failed
-                 :get-deletion-logs-failed})
-
-(s/def ::sync-event (s/keys :req-un [::event ::data]))
 
 (defonce download-batch-size 100)
 (defonce upload-batch-size 20)
@@ -584,7 +491,7 @@
 
 (defn- filepath+checksum->diff
   [index {:keys [relative-path checksum user-uuid graph-uuid]}]
-  {:post [(s/valid? ::diff %)]}
+  {:post [(util/validate diff-schema %)]}
   {:TXId (inc index)
    :TXType "update_files"
    :TXContent [[(util/string-join-path [user-uuid graph-uuid relative-path]) nil checksum]]})
@@ -2123,10 +2030,10 @@
   [graph-uuid]
   (swap! *resume-state dissoc graph-uuid))
 
+(m/=> sync-state [:=> [:cat] sync-state-schema])
 (defn sync-state
   "create a new sync-state"
   []
-  {:post [(s/valid? ::sync-state %)]}
   {:current-syncing-graph-uuid  nil
    :state                       ::starting
    :full-local->remote-files    #{}
@@ -2139,66 +2046,65 @@
 
 (defn- sync-state--update-current-syncing-graph-uuid
   [sync-state graph-uuid]
-  {:pre  [(s/valid? ::sync-state sync-state)]
-   :post [(s/valid? ::sync-state %)]}
+  {:post [(util/validate sync-state-schema %)]}
   (assoc sync-state :current-syncing-graph-uuid graph-uuid))
 
 (defn- sync-state--update-state
   [sync-state next-state]
-  {:pre  [(s/valid? ::state next-state)]
-   :post [(s/valid? ::sync-state %)]}
+  {:post [(util/validate sync-state-schema %)]}
   (assoc sync-state :state next-state))
 
 (defn sync-state--add-current-remote->local-files
   [sync-state paths]
-  {:post [(s/valid? ::sync-state %)]}
+  {:post [(util/validate sync-state-schema %)]}
   (update sync-state :current-remote->local-files into paths))
 
 (defn sync-state--add-current-local->remote-files
   [sync-state paths]
-  {:post [(s/valid? ::sync-state %)]}
+  {:post [(util/validate sync-state-schema %)]}
   (update sync-state :current-local->remote-files into paths))
 
 (defn sync-state--add-queued-local->remote-files
   [sync-state event]
-  {:post [(s/valid? ::sync-state %)]}
+  {:post [(util/validate sync-state-schema %)]}
   (update sync-state :queued-local->remote-files
           (fn [o event]
-            (->> (concat o [event])
-                 (util/distinct-by-last-wins (fn [e] (.-path e))))) event))
+            (->> (conj o event)
+                 seq
+                 (util/distinct-by-last-wins (fn [e] (.-path e)))
+                 set)) event))
 
 (defn sync-state--remove-queued-local->remote-files
   [sync-state event]
-  {:post [(s/valid? ::sync-state %)]}
+  {:post [(util/validate sync-state-schema %)]}
   (update sync-state :queued-local->remote-files
           (fn [o event]
-            (remove #{event} o)) event))
+            (set (remove #{event} o))) event))
 
 (defn sync-state-reset-queued-local->remote-files
   [sync-state]
-  {:post [(s/valid? ::sync-state %)]}
-  (assoc sync-state :queued-local->remote-files nil))
+  {:post [(util/validate sync-state-schema %)]}
+  (assoc sync-state :queued-local->remote-files #{}))
 
 (defn sync-state--add-recent-remote->local-files
   [sync-state items]
-  {:pre [(s/valid? (s/coll-of ::recent-remote->local-file-item) items)]
-   :post [(s/valid? ::sync-state %)]}
+  {:post [(util/validate sync-state-schema %)]}
   (update sync-state :recent-remote->local-files (partial apply conj) items))
 
 (defn sync-state--remove-recent-remote->local-files
   [sync-state items]
-  {:post [(s/valid? ::sync-state %)]}
+  {:post [(util/validate sync-state-schema %)]}
   (update sync-state :recent-remote->local-files set/difference items))
 
-(defn sync-state-reset-full-local->remote-files
+(defn- sync-state-reset-full-local->remote-files
   [sync-state events]
-  {:post [(s/valid? ::sync-state %)]}
-  (assoc sync-state :full-local->remote-files events))
+  {:post [(util/validate sync-state-schema %)]}
+  (assoc sync-state :full-local->remote-files (set events)))
 
 (defn sync-state-reset-full-remote->local-files
   [sync-state events]
-  {:post [(s/valid? ::sync-state %)]}
-  (assoc sync-state :full-remote->local-files events))
+  {:post [(util/validate sync-state-schema %)]}
+  (assoc sync-state :full-remote->local-files (set events)))
 
 (defn- add-history-items
   [history paths now]
@@ -2214,7 +2120,7 @@
 
 (defn sync-state--remove-current-remote->local-files
   [sync-state paths add-history?]
-  {:post [(s/valid? ::sync-state %)]}
+  {:post [(util/validate sync-state-schema %)]}
   (let [now (t/now)]
     (cond-> sync-state
       true         (update :current-remote->local-files set/difference paths)
@@ -2222,7 +2128,7 @@
 
 (defn sync-state--remove-current-local->remote-files
   [sync-state paths add-history?]
-  {:post [(s/valid? ::sync-state %)]}
+  {:post [(util/validate sync-state-schema %)]}
   (let [now (t/now)]
     (cond-> sync-state
       true         (update :current-local->remote-files set/difference paths)
@@ -2231,12 +2137,12 @@
 (defn sync-state--stopped?
   "Graph syncing is stopped"
   [sync-state]
-  {:pre [(s/valid? ::sync-state sync-state)]}
+  {:pre [(util/validate sync-state-schema sync-state)]}
   (= ::stop (:state sync-state)))
 
 (defn sync-state--valid-to-accept-filewatcher-event?
   [sync-state]
-  {:pre [(s/valid? ::sync-state sync-state)]}
+  {:pre [(util/validate sync-state-schema sync-state)]}
   (contains? #{::idle ::local->remote ::remote->local ::local->remote-full-sync ::remote->local-full-sync}
              (:state sync-state)))
 
@@ -2707,7 +2613,7 @@
                           {:succ true}
                           (let [{:keys [succ need-sync-remote graph-has-been-deleted unknown stop] :as r}
                                 (<! (<sync-local->remote! this (first es-partitions)))]
-                            (s/assert ::sync-local->remote!-result r)
+                            (assert (util/validate sync-local->remote!-result-schema r))
                             (cond
                               succ
                               (recur (next es-partitions))
@@ -2730,7 +2636,7 @@
               private-remote->local-full-sync-chan private-pause-resume-chan]
   Object
   (schedule [this next-state args reason]
-    {:pre [(s/valid? ::state next-state)]}
+    {:pre [(util/validate state-schema next-state)]}
     (println "[SyncManager" graph-uuid "]"
              (and state (name state)) "->" (and next-state (name next-state)) :reason reason :local-txid @*txid :now (tc/to-string (t/now)))
     (set! state next-state)
@@ -2821,7 +2727,7 @@
     [this]
     (go
       (let [next-state (<! (<loop-ensure-pwd&keys graph-uuid (state/get-current-repo) *stopped?))]
-        (assert (s/valid? ::state next-state) next-state)
+        (assert (util/validate state-schema next-state))
         (when (= next-state ::idle)
           (<! (<ensure-set-env&keys graph-uuid *stopped?)))
         (if @*stopped?
@@ -2897,7 +2803,7 @@
     (go
       (let [{:keys [succ need-sync-remote graph-has-been-deleted unknown stop] :as r}
             (<! (<sync-local->remote-all-files! local->remote-syncer))]
-        (s/assert ::sync-local->remote-all-files!-result r)
+        (assert (util/validate sync-local->remote-all-files!-result-schema r))
         (cond
           succ
           (do
@@ -2971,7 +2877,7 @@
         (let [origin-txid @*txid
               {:keys [succ unknown stop pause need-remote->local-full-sync] :as r}
               (<! (<sync-remote->local! remote->local-syncer))]
-          (s/assert ::sync-remote->local!-result r)
+          (assert (util/validate sync-remote->local!-result-schema r))
           (cond
             need-remote->local-full-sync
             (do (util/drain-chan ops-chan)
@@ -3022,7 +2928,7 @@
                 :else
                 (let [{:keys [succ need-sync-remote graph-has-been-deleted pause unknown stop] :as r}
                       (<! (<sync-local->remote! local->remote-syncer (first es-partitions)))]
-                  (s/assert ::sync-local->remote!-result r)
+                  (assert (util/validate sync-local->remote!-result-schema r))
                   (cond
                     succ
                     (recur (next es-partitions))

+ 97 - 0
src/main/frontend/fs/sync_schema.cljs

@@ -0,0 +1,97 @@
+(ns frontend.fs.sync-schema
+  "malli schema for frontend.fs.sync"
+  (:require [cljs-time.core :as t]))
+
+
+(def graph-uuid-schema
+  [:string {:min 36 :max 36}])
+
+(def state-schema
+  [:enum
+   :frontend.fs.sync/starting
+   :frontend.fs.sync/need-password
+   :frontend.fs.sync/idle
+   :frontend.fs.sync/local->remote
+   :frontend.fs.sync/remote->local
+   :frontend.fs.sync/local->remote-full-sync
+   :frontend.fs.sync/remote->local-full-sync
+   :frontend.fs.sync/pause
+   :frontend.fs.sync/stop])
+
+(def recent-remote->local-file-item-schema
+  [:map {:closed true}
+   [:remote->local-type [:enum :delete :update]]
+   [:checksum :string]
+   [:path :string]])
+
+(def history-item-schema
+  [:and
+   [:map {:closed true}
+    [:path :string]
+    [:time :any]]
+   [:fn #(t/date? (:time %))]])
+
+(def ^:private file-change-event-schema
+  [:fn #(= "frontend.fs.sync/FileChangeEvent" (type->str (type %)))])
+
+(def ^:private file-metadata-schema
+  [:fn #(= "frontend.fs.sync/FileMetadata" (type->str (type %)))])
+
+(def sync-state-schema
+  [:map {:closed true}
+   [:current-syncing-graph-uuid [:maybe graph-uuid-schema]]
+   [:state state-schema]
+   [:full-local->remote-files [:set file-change-event-schema]]
+   [:full-remote->local-files [:set file-metadata-schema]]
+   [:current-local->remote-files [:set :string]]
+   [:current-remote->local-files [:set :string]]
+   [:queued-local->remote-files [:set file-change-event-schema]]
+   ;; Downloading files from remote will trigger filewatcher events,
+   ;; causes unreasonable information in the content of :queued-local->remote-files,
+   ;; use :recent-remote->local-files to filter such events
+   [:recent-remote->local-files [:set recent-remote->local-file-item-schema]]
+   [:history [:sequential history-item-schema]]])
+
+(def diff-schema
+  [:map {:closed true}
+   [:TXId [:int {:min 1}]]
+   [:TXType [:enum "update_files" "delete_files" "rename_file"]]
+   [:TXContent [:sequential
+                [:catn
+                 [:to-path :string]
+                 [:from-path [:maybe :string]]
+                 [:checksum [:maybe :string]]]]]])
+
+
+
+(def sync-local->remote!-result-schema
+  [:or
+   [:enum {:desc "add first map to avoid {:succ true} be treated as the property map "}
+    {:succ true}
+    {:stop true}
+    {:pause true}
+    {:need-sync-remote true}
+    {:graph-has-been-deleted true}]
+   [:map {:closed true}
+    [:unknown :some]]])
+
+
+(def sync-remote->local!-result-schema
+  [:or
+   [:enum {:desc "add first map to avoid {:succ true} be treated as the property map "}
+    {:succ true}
+    {:stop true}
+    {:pause true}
+    {:need-remote->local-full-sync true}]
+   [:map {:closed true}
+    [:unknown :some]]])
+
+(def sync-local->remote-all-files!-result-schema
+  [:or
+   [:enum {:desc "add first map to avoid {:succ true} be treated as the property map "}
+    {:succ true}
+    {:stop true}
+    {:need-sync-remote true}
+    {:graph-has-been-deleted true}]
+   [:map {:closed true}
+    [:unknown :some]]])

+ 5 - 4
src/main/frontend/state.cljs

@@ -3,13 +3,13 @@
   cursors"
   (:require [cljs-bean.core :as bean]
             [cljs.core.async :as async :refer [<!]]
-            [cljs.spec.alpha :as s]
             [clojure.string :as string]
             [dommy.core :as dom]
             [electron.ipc :as ipc]
+            [frontend.fs.sync-schema :as sync-schema]
             [frontend.mobile.util :as mobile-util]
-            [frontend.storage :as storage]
             [frontend.spec.storage :as storage-spec]
+            [frontend.storage :as storage]
             [frontend.util :as util]
             [frontend.util.cursor :as cursor]
             [goog.dom :as gdom]
@@ -1992,8 +1992,9 @@ Similar to re-frame subscriptions"
                :file-sync/progress]
               nil))
 
-(defn set-file-sync-state [graph-uuid v]
-  (when v (s/assert :frontend.fs.sync/sync-state v))
+(defn set-file-sync-state
+  [graph-uuid v]
+  {:pre [(util/validate [:maybe sync-schema/sync-state-schema] v)]}
   (set-state! [:file-sync/graph-state graph-uuid :file-sync/sync-state] v))
 
 (defn get-current-file-sync-graph-uuid

+ 40 - 28
src/main/frontend/util.cljc

@@ -2,34 +2,35 @@
   "Main ns for utility fns. This ns should be split up into more focused namespaces"
   #?(:clj (:refer-clojure :exclude [format]))
   #?(:cljs (:require-macros [frontend.util]))
-  #?(:cljs (:require
-            ["/frontend/selection" :as selection]
-            ["/frontend/utils" :as utils]
-            ["@capacitor/status-bar" :refer [^js StatusBar Style]]
-            ["@hugotomazi/capacitor-navigation-bar" :refer [^js NavigationBar]]
-            ["grapheme-splitter" :as GraphemeSplitter]
-            ["remove-accents" :as removeAccents]
-            ["sanitize-filename" :as sanitizeFilename]
-            ["check-password-strength" :refer [passwordStrength]]
-            ["path-complete-extname" :as pathCompleteExtname]
-            [frontend.loader :refer [load]]
-            [cljs-bean.core :as bean]
-            [cljs-time.coerce :as tc]
-            [cljs-time.core :as t]
-            [clojure.pprint]
-            [dommy.core :as d]
-            [frontend.mobile.util :as mobile-util]
-            [logseq.graph-parser.util :as gp-util]
-            [goog.dom :as gdom]
-            [goog.object :as gobj]
-            [goog.string :as gstring]
-            [goog.userAgent]
-            [promesa.core :as p]
-            [rum.core :as rum]
-            [clojure.core.async :as async]
-            [cljs.core.async.impl.channels :refer [ManyToManyChannel]]
-            [medley.core :as medley]
-            [frontend.pubsub :as pubsub]))
+  #?(:cljs (:require ["/frontend/selection" :as selection]
+                     ["/frontend/utils" :as utils]
+                     ["@capacitor/status-bar" :refer [^js StatusBar Style]]
+                     ["@hugotomazi/capacitor-navigation-bar" :refer [^js NavigationBar]]
+                     ["check-password-strength" :refer [passwordStrength]]
+                     ["grapheme-splitter" :as GraphemeSplitter]
+                     ["path-complete-extname" :as pathCompleteExtname]
+                     ["remove-accents" :as removeAccents]
+                     ["sanitize-filename" :as sanitizeFilename]
+                     [cljs-bean.core :as bean]
+                     [cljs-time.coerce :as tc]
+                     [cljs-time.core :as t]
+                     [cljs.core.async.impl.channels :refer [ManyToManyChannel]]
+                     [clojure.core.async :as async]
+                     [clojure.pprint]
+                     [dommy.core :as d]
+                     [frontend.loader :refer [load]]
+                     [frontend.mobile.util :as mobile-util]
+                     [frontend.pubsub :as pubsub]
+                     [goog.dom :as gdom]
+                     [goog.object :as gobj]
+                     [goog.string :as gstring]
+                     [goog.userAgent]
+                     [logseq.graph-parser.util :as gp-util]
+                     [malli.core :as m]
+                     [malli.util :as mu]
+                     [medley.core :as medley]
+                     [promesa.core :as p]
+                     [rum.core :as rum]))
   #?(:cljs (:import [goog.async Debouncer]))
   (:require
    [clojure.pprint]
@@ -1521,3 +1522,14 @@ Arg *stop: atom, reset to true to stop the loop"
   "Vector version of remove. non-lazy"
   [pred coll]
   `(vec (remove ~pred ~coll)))
+
+
+#?(:cljs
+   (defn validate
+     "call malli.core/validate, if failed, then explain it,
+  return the boolean value"
+     [malli-schema value]
+     (let [v (m/validate malli-schema value)]
+       (when-not v
+         (js/console.error :validate-err (mu/explain-data malli-schema value)))
+       v)))