Files
logseq/src/main/frontend/worker/rtc/malli_schema.cljs

411 lines
13 KiB
Clojure

(ns frontend.worker.rtc.malli-schema
"Malli schema for rtc"
(:require [lambdaisland.glogi :as log]
[logseq.db.frontend.malli-schema :as db-malli-schema]
[logseq.db.frontend.schema :as db-schema]
[malli.core :as m]
[malli.transform :as mt]
[malli.util :as mu]))
(def block-pos-schema
[:catn
[:parent-uuid [:maybe :uuid]]
[:order [:maybe db-malli-schema/block-order]]])
(def av-schema
[:cat
:keyword
[:or
:uuid ;; reference type
:string ;; all other type value convert to string by transit
]
:int ;; t
:boolean ;; add(true) or retract
])
(def to-ws-op-schema
[:multi {:dispatch first :decode/string #(update % 0 keyword)}
[:update-kv-value
[:cat :keyword
[:map
[:db-ident :keyword]
[:value :string]]]]
[:rename-db-ident
[:cat :keyword
[:map
[:db-ident-or-block-uuid [:or :keyword :uuid]]
[:new-db-ident :keyword]]]]
[:move
[:cat :keyword
[:map
[:block-uuid :uuid]
[:pos block-pos-schema]]]]
[:remove
[:cat :keyword
[:map
[:block-uuids [:sequential :uuid]]]]]
[:update-page
[:cat :keyword
[:map
[:block-uuid :uuid]
[:db/ident {:optional true} :keyword]
[:page-name :string]
[:block/title :string]]]]
[:remove-page
[:cat :keyword
[:map
[:block-uuid :uuid]]]]
[:update
[:cat :keyword
[:map
[:block-uuid :uuid]
[:db/ident {:optional true} :keyword]
[:pos block-pos-schema]
[:av-coll [:sequential av-schema]]
[:card-one-attrs {:optional true} [:sequential :keyword]]]]]
[:update-schema
[:cat :keyword
[:map
[:block-uuid :uuid]
[:db/ident :keyword]
[:db/valueType :keyword]
[:db/cardinality {:optional true} :keyword]
[:db/index {:optional true} :boolean]]]]])
(comment
(def to-ws-ops-validator (m/validator [:sequential to-ws-op-schema])))
(def to-ws-ops-decoder (m/decoder [:sequential to-ws-op-schema] mt/string-transformer))
(defn- with-shared-schema-attrs
[shared-map-keys malli-schema]
(let [[head api-schema-seq] (split-at 2 malli-schema)]
(vec
(concat
head
(map
(fn [api-schema]
(let [[api-name [type']] api-schema]
(if (and (some? api-name) (= :map type'))
[api-name (vec (concat (second api-schema) shared-map-keys))]
api-schema)))
api-schema-seq)))))
(def ^:private extra-attr-map-schema
[:map-of
:keyword
[:or
[:or :uuid :string]
[:sequential [:or :uuid :string]]]])
(def data-from-ws-schema-fallback
"TODO: split this mix schema to multiple ones"
[:map
[:req-id :string]
[:profile {:optional true} :map]
[:t {:optional true} :int]
[:t-before {:optional true} :int]
[:s3-presign-url {:optional true} :string]
[:server-builtin-db-idents {:optional true} [:set :keyword]]
[:server-only-db-ident-blocks {:optional true} [:maybe :string] ;;transit
]
[:users {:optional true} [:sequential
[:map {:closed true}
[:user/uuid :uuid]
[:user/name :string]
[:user/email :string]
[:user/online? :boolean]
[:user/avatar {:optional true} :string]
[:graph<->user/user-type :keyword]]]]
[:online-users {:optional true} [:sequential
[:map {:closed true}
[:user/uuid :uuid]
[:user/name :string]
[:user/email :string]
[:user/avatar {:optional true} :string]]]]
[:asset-uuid->url {:optional true} [:map-of :uuid :string]]
[:uploaded-assets {:optional true} [:map-of :uuid :map]]
[:ex-data {:optional true} [:map [:type :keyword]]]
[:ex-message {:optional true} :string]])
(def ^:private apply-ops-response-schema
[:map
[:t :int]
[:t-before :int]
[:t-query-end {:optional true} :int] ;TODO: remove 'optional' later, be compatible with old-clients for now
[:affected-blocks
[:map-of :uuid
[:multi {:dispatch :op :decode/string #(update % :op keyword)}
[:move
[:map {:closed true}
[:op :keyword]
[:self :uuid]
[:parents [:sequential :uuid]]
[:block/order {:optional true} db-malli-schema/block-order]
[:hash {:optional true} :int]
[:db/ident {:optional true} :keyword]]]
[:remove
[:map
[:op :keyword]
[:block-uuid :uuid]]]
[:update-attrs
[:map
[:op :keyword]
[:self :uuid]
[:parents {:optional true} [:sequential :uuid]]
[:block/order {:optional true} db-malli-schema/block-order]
[:hash {:optional true} :int]
[:db/ident {:optional true} :keyword]
[::m/default extra-attr-map-schema]]]
[:move+update-attrs
[:map
[:op :keyword]
[:self :uuid]
[:parents {:optional true} [:sequential :uuid]]
[:block/order {:optional true} db-malli-schema/block-order]
[:hash {:optional true} :int]
[:db/ident {:optional true} :keyword]
[::m/default extra-attr-map-schema]]]
[:update-page
[:map
[:op :keyword]
[:self :uuid]
[:block/title :string]
[:db/ident {:optional true} :keyword]
[:block/order {:optional true} db-malli-schema/block-order]
[::m/default extra-attr-map-schema]]]
[:remove-page
[:map
[:op :keyword]
[:block-uuid :uuid]]]]]]
[:refed-blocks
[:maybe
[:sequential
[:map
[:block/uuid :uuid]
[:db/ident {:optional true} :keyword]
[:block/order {:optional true} db-malli-schema/block-order]
[:block/parent {:optional true} :uuid]
[::m/default extra-attr-map-schema]]]]]
[:failed-ops {:optional true} ;DEPRECATED
[:maybe [:sequential to-ws-op-schema]]]
[:not-found-target-ops {:optional true}
[:maybe [:sequential to-ws-op-schema]]]])
(def data-from-ws-schema
(with-shared-schema-attrs
[[:req-id :string]]
[:multi {:dispatch #(or (when (:ex-data %) :exception) (:action %))}
[:exception
[:map
[:ex-message :string]
[:ex-data [:map [:type :keyword]]]]]
["init-request"
(concat [:map
[:max-remote-schema-version {:optional true} :string]
[:server-schema-version {:optional true} :string]
[:server-builtin-db-idents {:optional true} [:set :keyword]]]
(m/children apply-ops-response-schema))]
["register-graph-updates"
[:map
[:t :int]
[:max-remote-schema-version {:optional true} :string]]]
["apply-ops" apply-ops-response-schema]
["push-asset-block-updates" apply-ops-response-schema]
["branch-graph"
[:map
[:graph-uuid :uuid]
[:schema-version db-schema/major-schema-version-string-schema]]]
["delete-graph" [:map]]
["download-graph"
[:map
[:graph-uuid :uuid]
[:schema-version db-schema/major-schema-version-string-schema]
[:download-info-uuid :string]]]
["upload-graph"
[:map
[:graph-uuid :uuid]
[:schema-version db-schema/major-schema-version-string-schema]]]
["download-info-list"
[:map
[:download-info-list [:sequential :any]]]]
["grant-access" [:map]]
["get-graph-skeleton"
[:map
[:server-schema-version :string]
[:server-builtin-db-idents [:set :keyword]]]]
["presign-put-temp-s3-obj" [:map]]
["get-users-info"
[:map
[:users
[:sequential [:map
[:user/uuid :uuid]
[:user/name :string]
[:user/email :string]
[:user/avatar {:optional true} :string]
[:graph<->user/user-type :keyword]
[:user/online? :boolean]]]]]]
["inject-users-info" [:map]]
;; keys manage
["fetch-user-rsa-key-pair"
[:map
[:public-key [:maybe :string]]
[:encrypted-private-key [:maybe :string]]]]
["fetch-graph-encrypted-aes-key"
[:map
[:encrypted-aes-key [:maybe :string]]]]
["fetch-user-rsa-public-key"
[:map
[:public-key [:maybe :string]]]]
["upload-user-rsa-key-pair"
[:map
[:public-key :string]
[:encrypted-private-key :string]]]
["reset-user-rsa-key-pair"
[:map
[:public-key :string]
[:encrypted-private-key :string]]]
[nil data-from-ws-schema-fallback]]))
(def data-from-ws-coercer (m/coercer data-from-ws-schema mt/string-transformer nil
#(m/-fail! ::data-from-ws-schema (select-keys % [:value]))))
(def data-from-ws-validator (m/validator data-from-ws-schema))
(def ^:large-vars/data-var data-to-ws-schema
(mu/closed-schema
(with-shared-schema-attrs
[[:req-id :string]
[:action :string]
[:profile {:optional true} :boolean]]
[:multi {:dispatch :action}
["list-graphs"
[:map]]
["init-request"
[:map
[:graph-uuid :uuid]
[:api-version :string]
[:schema-version db-schema/major-schema-version-string-schema]
[:t-before :int]
[:get-graph-skeleton :boolean]]]
["register-graph-updates"
[:map
[:graph-uuid :uuid]
[:schema-version db-schema/major-schema-version-string-schema]]]
["apply-ops"
[:or
[:map
[:req-id :string]
[:action :string]
[:api-version :string]
[:profile {:optional true} :boolean]
[:graph-uuid :uuid]
[:schema-version db-schema/major-schema-version-string-schema]
[:ops [:sequential to-ws-op-schema]]
[:t-before :int]]
[:map
[:req-id :string]
[:action :string]
[:profile {:optional true} :boolean]
[:s3-key :string]]]]
["presign-put-temp-s3-obj"
[:map]]
["upload-graph"
[:map
[:s3-key :string]
[:graph-name :string]
[:schema-version db-schema/major-schema-version-string-schema]
[:encrypted-aes-key {:optional true} :string]]]
["branch-graph"
[:map
[:s3-key :string]
[:graph-uuid :uuid]
[:schema-version db-schema/major-schema-version-string-schema]]]
["download-graph"
[:map
[:graph-uuid :uuid]
[:schema-version db-schema/major-schema-version-string-schema]]]
["download-info-list"
[:map
[:graph-uuid :uuid]
[:schema-version db-schema/major-schema-version-string-schema]]]
["grant-access"
[:map
[:graph-uuid :uuid]
[:target-user-email+encrypted-aes-key-coll
[:sequential
[:map
[:user/email :string]
[:encrypted-aes-key [:maybe :string]]]]]]]
["get-users-info"
[:map
[:graph-uuid :uuid]]]
["inject-users-info"
[:map
[:graph-uuid :uuid]
[:schema-version db-schema/major-schema-version-string-schema]]]
["delete-graph"
[:map
[:graph-uuid :uuid]
[:schema-version db-schema/major-schema-version-string-schema]]]
["query-block-content-versions"
[:map
[:graph-uuid :uuid]
[:block-uuids [:sequential :uuid]]]]
["calibrate-graph-skeleton"
[:map
[:graph-uuid :uuid]
[:t :int]
[:schema-version :int]
[:db-ident-blocks [:sequential
[:map
[:db/ident :keyword]
[::m/default extra-attr-map-schema]]]]]]
["get-graph-skeleton"
[:map
[:graph-uuid :uuid]
[:schema-version db-schema/major-schema-version-string-schema]]]
["get-assets-upload-urls"
[:map
[:graph-uuid :uuid]
[:asset-uuid->metadata [:map-of :uuid [:map-of :string :string]]]]]
["get-assets-download-urls"
[:map
[:graph-uuid :uuid]
[:asset-uuids [:sequential :uuid]]]]
["delete-assets"
[:map
[:graph-uuid :uuid]
[:schema-version db-schema/major-schema-version-string-schema]
[:asset-uuids [:sequential :uuid]]]]
;; ================================================================
["upload-user-rsa-key-pair"
[:map
[:user-uuid :uuid]
[:public-key {:optional true} :string]
[:encrypted-private-key :string]
[:reset-private-key {:optional true} :boolean]]]
["reset-user-rsa-key-pair"
[:map
[:user-uuid :uuid]
[:public-key :string]
[:encrypted-private-key :string]]]
["fetch-user-rsa-key-pair"
[:map
[:user-uuid :uuid]]]
["fetch-graph-encrypted-aes-key"
[:map
[:graph-uuid :uuid]]]
["fetch-user-rsa-public-key"
[:map
[:user/email :string]]]])))
(def data-to-ws-encoder (m/encoder data-to-ws-schema (mt/transformer
mt/string-transformer
(mt/key-transformer {:encode m/-keyword->string}))))
(def data-to-ws-coercer (m/coercer data-to-ws-schema mt/string-transformer nil
#(do
(log/error ::data-to-ws-schema %)
(m/-fail! ::data-to-ws-schema (select-keys % [:value])))))