query_custom.cljs 3.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778
  1. (ns frontend.db.query-custom
  2. "Handles executing custom queries a.k.a. advanced queries"
  3. (:require [frontend.state :as state]
  4. [frontend.db.query-react :as query-react]
  5. [frontend.db.query-dsl :as query-dsl]
  6. [frontend.db.model :as model]
  7. [logseq.db.frontend.rules :as rules]
  8. [frontend.util.datalog :as datalog-util]
  9. [clojure.walk :as walk]
  10. [frontend.config :as config]))
  11. ;; FIXME: what if users want to query other attributes than block-attrs?
  12. (defn- replace-star-with-block-attrs!
  13. [l]
  14. (let [block-attrs (butlast model/block-attrs)]
  15. (walk/postwalk
  16. (fn [f]
  17. (if (and (list? f)
  18. (= 'pull (first f))
  19. (= '?b (second f))
  20. (= '[*] (nth f 2)))
  21. `(~'pull ~'?b ~block-attrs)
  22. f))
  23. l)))
  24. (defn- add-rules-to-query
  25. "Searches query's :where for rules and adds them to query if used"
  26. [{:keys [query] :as query-m} {:keys [db-graph?]}]
  27. (let [{:keys [where in]} (datalog-util/query-vec->map query)
  28. query-dsl-rules (if db-graph? rules/db-query-dsl-rules rules/query-dsl-rules)
  29. rules-found (datalog-util/find-rules-in-where where (-> query-dsl-rules keys set))]
  30. (if (seq rules-found)
  31. (if (and (= '% (last in)) (vector? (last (:inputs query-m))))
  32. ;; Add to existing :inputs rules
  33. (update query-m
  34. :inputs
  35. (fn [inputs]
  36. (assoc (vec inputs)
  37. ;; last position is rules
  38. (dec (count inputs))
  39. (->> (rules/extract-rules query-dsl-rules rules-found)
  40. (into (last inputs))
  41. ;; user could give rules that we already have
  42. distinct
  43. vec))))
  44. ;; Add new rules
  45. (-> query-m
  46. (update :query
  47. (fn [q]
  48. (if (contains? (set q) :in)
  49. ;; only add '% if not already present
  50. (if (not (contains? (set q) '%))
  51. (datalog-util/add-to-end-of-query-section q :in ['%])
  52. q)
  53. (into q [:in '$ '%]))))
  54. (update :rules
  55. (fn [rules]
  56. (into (or rules [])
  57. (rules/extract-rules query-dsl-rules rules-found))))))
  58. query-m)))
  59. (defn custom-query
  60. "Executes a datalog query through query-react, given either a regular datalog
  61. query or a simple query"
  62. ([query]
  63. (custom-query query {}))
  64. ([query query-opts]
  65. (custom-query (state/get-current-repo) query query-opts))
  66. ([repo query query-opts]
  67. (let [query' (replace-star-with-block-attrs! query)
  68. query-opts (if (:query-string query-opts) query-opts
  69. (assoc query-opts :query-string (str query)))]
  70. (if (or (list? (:query query'))
  71. (not= :find (first (:query query')))) ; dsl query
  72. (query-dsl/custom-query repo query' query-opts)
  73. (query-react/react-query repo
  74. (add-rules-to-query query' {:db-graph? (config/db-based-graph? repo)})
  75. query-opts)))))