Browse Source

fix: db import of basic block properties

Some FIXMEs will be fixed soon. Part of LOG-3176
Gabriel Horner 1 year ago
parent
commit
bd5f43d233

+ 150 - 76
deps/graph-parser/src/logseq/graph_parser/exporter.cljs

@@ -20,7 +20,10 @@
             [logseq.db.frontend.class :as db-class]
             [logseq.common.util.page-ref :as page-ref]
             [promesa.core :as p]
-            [logseq.db.frontend.order :as db-order]))
+            [cljs.pprint]
+            [logseq.db.frontend.order :as db-order]
+            [logseq.db.frontend.db-ident :as db-ident]
+            [logseq.db.frontend.property.build :as db-property-build]))
 
 (defn- get-pid
   "Get a property's id (name or uuid) given its name. For db graphs"
@@ -212,11 +215,23 @@
                            remaining-text)]
     (some? non-ref-char)))
 
+(defn- create-property-ident [db all-idents property-name]
+  (let [db-ident (->> (db-property/create-user-property-ident-from-name (name property-name))
+                      ;; TODO: Detect new ident conflicts within same page
+                      (db-ident/ensure-unique-db-ident db))]
+    (swap! all-idents assoc property-name db-ident)))
+
+(defn- get-ident [all-idents kw]
+  (if (and (qualified-keyword? kw) (db-property/logseq-property? kw))
+    kw
+    (or (get all-idents kw)
+        (throw (ex-info (str "No ident found for " (pr-str kw)) {})))))
+
 (defn- infer-property-schema-and-get-property-change
   "Infers a property's schema from the given _user_ property value and adds new ones to
   the property-schemas atom. If a property's :type changes, returns a map of
   the schema attribute changed and how it changed e.g. `{:type {:from :default :to :url}}`"
-  [prop-val prop prop-val-text refs property-schemas macros]
+  [db prop-val prop prop-val-text refs {:keys [property-schemas all-idents]} macros]
   ;; Explicitly fail an unexpected case rather than cause silent downstream failures
   (when (and (coll? prop-val) (not (every? string? prop-val)))
     (throw (ex-info (str "Import cannot infer schema of unknown property value " (pr-str prop-val))
@@ -233,7 +248,9 @@
                         (db-property-type/infer-property-type-from-value
                          (macro-util/expand-value-if-macro prop-val macros)))
         prev-type (get-in @property-schemas [prop :type])]
-    (when-not prev-type
+    ;; Create new property
+    (when-not (get @property-schemas prop)
+      (create-property-ident db all-idents prop)
       (let [schema (cond-> {:type prop-type}
                      (#{:page :date} prop-type)
                      ;; Assume :many for now as detecting that detecting property values across files are consistent
@@ -340,35 +357,50 @@
        (map :name)
        set))
 
-(defn- update-properties
-  "Updates block property names and values"
-  [props db page-names-to-uuids
+(defn- build-properties-and-values
+  "For given block properties, builds property values tx and returns a map with
+  updated properties in :block-properties and any property values tx in :pvalues-tx"
+  [props db _page-names-to-uuids
    {:block/keys [properties-text-values] :as block}
-   {:keys [whiteboard? import-state] :as options}]
-  (let [prop-name->uuid (if whiteboard?
-                          (fn prop-name->uuid [k]
-                            (or (get-pid db k)
-                                (throw (ex-info (str "No uuid found for page " (pr-str k))
-                                                {:page k}))))
-                          (fn prop-name->uuid [k]
-                            (cached-prop-name->uuid db page-names-to-uuids k)))
+   {:keys [_whiteboard? import-state] :as options}]
+  (let [;; FIXME: Whiteboard
+        ;; prop-name->uuid (if whiteboard?
+        ;;                   (fn prop-name->uuid [k]
+        ;;                     (or (get-pid db k)
+        ;;                         (throw (ex-info (str "No uuid found for page " (pr-str k))
+        ;;                                         {:page k}))))
+        ;;                   (fn prop-name->uuid [k]
+        ;;                     (cached-prop-name->uuid db page-names-to-uuids k)))
+        {:keys [all-idents property-schemas]} import-state
+        get-ident' #(get-ident @all-idents %)
         user-properties (apply dissoc props built-in-property-names)]
-    (when (seq user-properties)
-      (swap! (:block-properties-text-values import-state)
-             assoc
+    ;; FIXME: Fix block-properties-text-values
+    #_(when (seq user-properties)
+        (swap! (:block-properties-text-values import-state)
+               assoc
              ;; For pages, valid uuid is in page-names-to-uuids, not in block
-             (if (:block/name block) (get page-names-to-uuids (:block/name block)) (:block/uuid block))
-             properties-text-values))
+               (if (:block/name block) (get page-names-to-uuids (:block/name block)) (:block/uuid block))
+               properties-text-values))
     ;; TODO: Add import support for :template. Ignore for now as they cause invalid property types
     (if (contains? props :template)
       {}
-      (-> (update-built-in-property-values
-           (select-keys props built-in-property-names)
-           db
-           (:ignored-properties import-state)
-           (select-keys block [:block/name :block/content]))
-          (merge (update-user-property-values user-properties prop-name->uuid properties-text-values import-state options))
-          (update-keys prop-name->uuid)))))
+      (let [props' (-> (update-built-in-property-values
+                        (select-keys props built-in-property-names)
+                        db
+                        (:ignored-properties import-state)
+                        (select-keys block [:block/name :block/content]))
+                       (merge (update-user-property-values user-properties get-ident' properties-text-values import-state options)))
+            pvalue-tx-m (->> props'
+                             (map (fn [[k v]]
+                                    (let [property-map {:db/ident (get-ident @all-idents k)
+                                                        :original-property-id k
+                                                        :block/schema (get @property-schemas k)}]
+                                      [property-map v])))
+                             (db-property-build/build-property-values-tx-m block))
+            block-properties (-> (merge props' (db-property-build/build-properties-with-ref-values pvalue-tx-m))
+                                 (update-keys get-ident'))]
+        {:block-properties block-properties
+         :pvalues-tx (into (mapcat #(if (set? %) % [%]) (vals pvalue-tx-m)))}))))
 
 (def ignored-built-in-properties
   "Ignore built-in properties that are already imported or not supported in db graphs"
@@ -405,7 +437,8 @@
          (into {}))))
 
 (defn- handle-page-and-block-properties
-  "Handles modifying :block/properties, updating classes from property-classes
+  "Returns a map of :block with updated block and :properties-tx with any properties tx.
+   Handles modifying :block/properties, updating classes from property-classes
   and removing any deprecated property related attributes. Before updating most
   :block/properties, their property schemas are inferred as that can affect how
   a property is updated. Only infers property schemas on user properties as
@@ -425,24 +458,32 @@
               (->> properties-to-infer
                    (keep (fn [[prop val]]
                            (when-let [property-change
-                                      (infer-property-schema-and-get-property-change val prop (get (:block/properties-text-values block) prop) refs (:property-schemas import-state) macros)]
+                                      (infer-property-schema-and-get-property-change db val prop (get (:block/properties-text-values block) prop) refs import-state macros)]
                              [prop property-change])))
                    (into {}))
               ;; _ (when (seq property-changes) (prn :prop-changes property-changes))
-              options' (assoc options :property-changes property-changes)]
-          (cond-> (assoc-in block [:block/properties]
-                            (update-properties properties' db page-names-to-uuids
-                                               (select-keys block [:block/properties-text-values :block/name :block/content :block/uuid])
-                                               options'))
-            (seq classes-from-properties)
+              options' (assoc options :property-changes property-changes)
+              {:keys [block-properties pvalues-tx]}
+              (build-properties-and-values properties' db page-names-to-uuids
+                                           (select-keys block [:block/properties-text-values :block/name :block/content :block/uuid])
+                                           options')]
+          ;; (prn :handle-props (:all-idents import-state) properties')
+          ;; (prn pvalues-tx)
+          ;; (prn block-properties)
+          {:block
+           (cond-> (dissoc block :block/properties)
+             true
+             (merge block-properties)
+             (seq classes-from-properties)
             ;; Add a map of {:block.temp/new-class TAG} to be processed later
-            (update :block/tags
-                    (fnil into [])
-                    (map #(hash-map :block.temp/new-class %
-                                    :block/uuid (or (get-pid db %) (d/squuid)))
-                         classes-from-properties))))
-        block)
-      (dissoc :block/properties-text-values :block/properties-order :block/invalid-properties)))
+             (update :block/tags
+                     (fnil into [])
+                     (map #(hash-map :block.temp/new-class %
+                                     :block/uuid (or (get-pid db %) (d/squuid)))
+                          classes-from-properties)))
+           :properties-tx pvalues-tx})
+        {:block block :properties-tx []})
+      (update :block dissoc :block/properties-text-values :block/properties-order :block/invalid-properties)))
 
 (defn- handle-page-properties
   [{:block/keys [properties] :as block} db page-names-to-uuids refs
@@ -464,18 +505,23 @@
                        :block/uuid (or (get-pid db new-class) (d/squuid))
                        :block/name (common-util/page-name-sanity-lc new-class)})))))
         block)
-      (handle-page-and-block-properties db page-names-to-uuids refs options)))
+      (handle-page-and-block-properties db page-names-to-uuids refs options)
+      ;; FIXME: Handle page properties-tx
+      :block))
 
 (defn- handle-block-properties
   "Does everything page properties does and updates a couple of block specific attributes"
-  [block db page-names-to-uuids refs {:keys [property-classes] :as options}]
-  (cond-> (handle-page-and-block-properties block db page-names-to-uuids refs options)
-    (and (seq property-classes) (seq (:block/refs block)))
-    ;; remove unused, nonexistent property page
-    (update :block/refs (fn [refs] (remove #(property-classes (keyword (:block/name %))) refs)))
-    (and (seq property-classes) (seq (:block/path-refs block)))
-    ;; remove unused, nonexistent property page
-    (update :block/path-refs (fn [refs] (remove #(property-classes (keyword (:block/name %))) refs)))))
+  [block* db page-names-to-uuids refs {:keys [property-classes] :as options}]
+  (let [{:keys [block properties-tx]} (handle-page-and-block-properties block* db page-names-to-uuids refs options)]
+    {:block
+     (cond-> block
+       (and (seq property-classes) (seq (:block/refs block*)))
+       ;; remove unused, nonexistent property page
+       (update :block/refs (fn [refs] (remove #(property-classes (keyword (:block/name %))) refs)))
+       (and (seq property-classes) (seq (:block/path-refs block*)))
+       ;; remove unused, nonexistent property page
+       (update :block/path-refs (fn [refs] (remove #(property-classes (keyword (:block/name %))) refs))))
+     :properties-tx properties-tx}))
 
 (defn- update-block-refs
   "Updates the attributes of a block ref as this is where a new page is defined. Also
@@ -494,6 +540,7 @@
            (mapv (fn [ref]
                    (if (ref-to-ignore? ref)
                      ref
+                     ;; FIXME: Strip down just to just uuid
                      (merge (assoc ref :block/format :markdown)
                             (when-let [schema (get new-property-schemas (keyword (:block/name ref)))]
                               {:block/type "property"
@@ -541,36 +588,36 @@
     (assoc :block/parent {:block/uuid (cached-prop-name->uuid db page-names-to-uuids (:block/name (:block/parent block)))})))
 
 (defn- build-block-tx
-  [db block pre-blocks page-names-to-uuids {:keys [import-state tag-classes] :as options}]
+  [db block* pre-blocks page-names-to-uuids {:keys [import-state tag-classes] :as options}]
   ;; (prn ::block-in block)
-  (let [old-property-schemas @(:property-schemas import-state)]
-    (-> block
-        (fix-pre-block-references pre-blocks)
-        (fix-block-name-lookup-ref db page-names-to-uuids)
-        (update-block-macros db page-names-to-uuids)
+  (let [old-property-schemas @(:property-schemas import-state)
         ;; needs to come before update-block-refs to detect new property schemas
-        (handle-block-properties db page-names-to-uuids (:block/refs block) options)
-        (update-block-refs page-names-to-uuids old-property-schemas options)
-        (update-block-tags tag-classes page-names-to-uuids)
-        (update-block-marker db options)
-        (update-block-priority db options)
-        (update-block-deadline db options)
-        add-missing-timestamps
-        ;; ((fn [x] (prn :block-out x) x))
-        ;; TODO: org-mode content needs to be handled
-        (assoc :block/format :markdown))))
+        {:keys [block properties-tx]}
+        (handle-block-properties block* db page-names-to-uuids (:block/refs block*) options)
+        block' (-> block
+                   (fix-pre-block-references pre-blocks)
+                   (fix-block-name-lookup-ref db page-names-to-uuids)
+                   (update-block-macros db page-names-to-uuids)
+                   (update-block-refs page-names-to-uuids old-property-schemas options)
+                   (update-block-tags tag-classes page-names-to-uuids)
+                   (update-block-marker db options)
+                   (update-block-priority db options)
+                   (update-block-deadline db options)
+                   add-missing-timestamps
+                   ;; ((fn [x] (prn :block-out x) x))
+                   ;; TODO: org-mode content needs to be handled
+                   (assoc :block/format :markdown))]
+    ;; Order matters as properties are referenced in block
+    (concat properties-tx [block'])))
 
 (defn- build-new-page
-  [m new-property-schemas tag-classes page-names-to-uuids page-tags-uuid]
+  [m tag-classes page-names-to-uuids page-tags-uuid]
   (-> m
       ;; Fix pages missing :block/original-name. Shouldn't happen
       ((fn [m']
          (if-not (:block/original-name m')
            (assoc m' :block/original-name (:block/name m'))
            m')))
-      (merge (when-let [schema (get new-property-schemas (keyword (:block/name m)))]
-               {:block/type "property"
-                :block/schema schema}))
       add-missing-timestamps
       ;; TODO: org-mode content needs to be handled
       (assoc :block/format :markdown)
@@ -614,7 +661,7 @@
                                 (update-page-tags tag-classes page-names-to-uuids page-tags-uuid)
                                 schema
                                 (assoc :block/type "property" :block/schema schema))))
-                          (build-new-page % new-property-schemas tag-classes page-names-to-uuids page-tags-uuid))
+                          (build-new-page % tag-classes page-names-to-uuids page-tags-uuid))
                        all-pages')]
     {:pages-tx pages-tx
      :page-names-to-uuids page-names-to-uuids}))
@@ -663,7 +710,7 @@
     []))
 
 (defn new-import-state
-  "New import state that is used in add-file-to-db-graph. State is atom per
+  "New import state that is used for import of one graph. State is atom per
    key to make code more readable and encourage local mutations"
   []
   {;; Vec of maps with keys :property, :value, :schema and :location.
@@ -673,6 +720,8 @@
    ;; Map of property names (keyword) and their current schemas (map).
    ;; Used for adding schemas to properties and detecting changes across a property's usage
    :property-schemas (atom {})
+   ;; Map of property or class names (keyword) to db-ident keywords
+   :all-idents (atom {})
    ;; Map of block uuids to their :block/properties-text-values value.
    ;; Used if a property value changes to :default
    :block-properties-text-values (atom {})})
@@ -692,6 +741,23 @@
                               (set (map (comp keyword string/lower-case) (:property-parent-classes user-options)))
                               built-in-property-names)}))
 
+(defn- split-pages-and-properties-tx
+  "Separates new pages from new properties tx in preparation for properties to
+  be transacted separatedly. Also rebuilds properties tx"
+  [pages-tx old-properties import-state]
+  (let [new-properties (set/difference (set (keys @(:property-schemas import-state))) (set old-properties))
+        _ (prn :new-properties new-properties)
+        [properties-tx pages-tx'] ((juxt filter remove)
+                                   #(contains? new-properties (keyword (:block/name %))) pages-tx)
+        properties-tx' (map (fn [{:block/keys [original-name uuid]}]
+                              (let [db-ident (get @(:all-idents import-state) (keyword original-name))]
+                                (sqlite-util/build-new-property db-ident
+                                                                (get @(:property-schemas import-state) (keyword original-name))
+                                                                {:original-name original-name :block-uuid uuid})))
+                            properties-tx)
+        #_(cljs.pprint/pprint properties-tx')]
+    [pages-tx' properties-tx']))
+
 (defn add-file-to-db-graph
   "Parse file and save parsed data to the given db graph. Options available:
 
@@ -726,6 +792,7 @@
               :else
               (notify-user {:msg (str "Skipped file since its format is not supported: " file)}))
         tx-options (build-tx-options options)
+        old-properties (keys @(get-in options [:import-state :property-schemas]))
         ;; Build page and block txs
         {:keys [pages-tx page-names-to-uuids]} (build-pages-tx conn pages blocks tx-options)
         whiteboard-pages (->> pages-tx
@@ -739,8 +806,14 @@
         pre-blocks (->> blocks (keep #(when (:block/pre-block? %) (:block/uuid %))) set)
         blocks-tx (->> blocks
                        (remove :block/pre-block?)
-                       (mapv #(build-block-tx @conn % pre-blocks page-names-to-uuids
-                                              (assoc tx-options :whiteboard? (some? (seq whiteboard-pages))))))
+                       (mapcat #(build-block-tx @conn % pre-blocks page-names-to-uuids
+                                                (assoc tx-options :whiteboard? (some? (seq whiteboard-pages)))))
+                       vec)
+
+        [pages-tx' properties-tx] (split-pages-and-properties-tx pages-tx old-properties (:import-state options))
+        ;; Necessary to transact new property entities first so that block+page properties can be transacted next
+        _ (d/transact! conn properties-tx)
+
         upstream-properties-tx (build-upstream-properties-tx
                                 @conn
                                 page-names-to-uuids
@@ -749,7 +822,7 @@
                                 blocks-tx
                                 log-fn)
         ;; Build indices
-        pages-index (map #(select-keys % [:block/uuid]) pages-tx)
+        pages-index (map #(select-keys % [:block/uuid]) pages-tx')
         block-ids (map (fn [block] {:block/uuid (:block/uuid block)}) blocks-tx)
         block-refs-ids (->> (mapcat :block/refs blocks-tx)
                             (filter (fn [ref] (and (vector? ref)
@@ -760,8 +833,9 @@
         block-ids (set/union (set block-ids) (set block-refs-ids))
         ;; Order matters as upstream-properties-tx can override some blocks-tx and indices need
         ;; to come before their corresponding tx
-        tx (concat whiteboard-pages pages-index pages-tx block-ids blocks-tx upstream-properties-tx)
+        tx (concat whiteboard-pages pages-index pages-tx' block-ids blocks-tx upstream-properties-tx)
         tx' (common-util/fast-remove-nils tx)
+        ;; _ (cljs.pprint/pprint {:tx tx'})
         result (d/transact! conn tx')]
     result))
 

+ 31 - 3
deps/graph-parser/test/logseq/graph_parser/exporter_test.cljs

@@ -10,7 +10,17 @@
             [logseq.db.frontend.schema :as db-schema]
             [logseq.db.frontend.validate :as db-validate]
             [logseq.db.sqlite.create-graph :as sqlite-create-graph]
-            [logseq.graph-parser.exporter :as gp-exporter]))
+            [logseq.graph-parser.exporter :as gp-exporter]
+            [logseq.db.frontend.malli-schema :as db-malli-schema]
+            [logseq.db.frontend.property :as db-property]))
+
+(defn- find-block-by-content [db content]
+  (->> content
+       (d/q '[:find [(pull ?b [*]) ...]
+              :in $ ?content
+              :where [?b :block/content ?content]]
+            db)
+       first))
 
 (defn- build-graph-files
   "Given a file graph directory, return all files including assets and adds relative paths
@@ -88,5 +98,23 @@
              (ffirst (d/q '[:find ?content :where [?b :file/path "logseq/custom.js"] [?b :file/content ?content]] @conn)))))
 
     (testing "user content"
-      (is (= 2 (count (d/q '[:find ?b :where [?b :block/type "journal"]] @conn))))
-      (is (= 1 (count @assets))))))
+      (is (= 3 (count (d/q '[:find ?b :where [?b :block/type "journal"]] @conn))))
+      (is (= 1 (count @assets)))
+
+      (testing "user properties"
+        (is (= #{{:db/ident :user.property/prop-bool :block/schema {:type :checkbox}}
+                 {:db/ident :user.property/prop-string :block/schema {:type :default}}
+                 {:db/ident :user.property/prop-num :block/schema {:type :number}}}
+               (->> @conn
+                    (d/q '[:find [(pull ?b [:db/ident :block/schema]) ...]
+                           :where [?b :block/type "property"]])
+                    (remove #(db-malli-schema/internal-ident? (:db/ident %)))
+                    set))
+            "properties defined correctly")
+        (is (= {:user.property/prop-bool true
+                :user.property/prop-num 5
+                :user.property/prop-string "woot"}
+               (update-vals (db-property/properties (find-block-by-content @conn "b1"))
+                            (fn [ref]
+                              (db-property/ref->property-value-content @conn ref))))
+            "Basic block has correct properties")))))

+ 4 - 0
deps/graph-parser/test/resources/exporter-test-graph/journals/2024_01_17.md

@@ -0,0 +1,4 @@
+- b1
+  prop-string:: woot
+  prop-num:: 5
+  Prop-bool:: true