| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172 | (ns query  "A script that queries any db graph from the commandline e.g.  $ yarn -s nbb-logseq script/query.cljs db-name '[:find (pull ?b [:block/name :block/title]) :where [?b :block/created-at]]'"  (:require ["child_process" :as child-process]            [babashka.cli :as cli]            [clojure.edn :as edn]            [clojure.pprint :as pprint]            [datascript.core :as d]            [logseq.db.common.sqlite-cli :as sqlite-cli]            [logseq.db.frontend.rules :as rules]            [nbb.core :as nbb]))(defn- sh  "Run shell cmd synchronously and print to inherited streams by default. Aims  to be similar to babashka.tasks/shell"  [cmd opts]  (child-process/spawnSync (first cmd)                           (clj->js (rest cmd))                           (clj->js (merge {:stdio "inherit"} opts))))(def spec  "Options spec"  {:help {:alias :h          :desc "Print help"}   :verbose {:alias :v             :desc "Print more info"}   :raw {:alias :r         :desc "Print results plainly. Useful when piped to bb"}   :additional-graphs {:alias :a                       :coerce []                       :desc "Additional graphs to query"}   :entity {:alias :e            :coerce []            :desc "Lookup entities instead of query"}})(defn query-graph [graph args'' options]  (let [conn (apply sqlite-cli/open-db! (sqlite-cli/->open-db-args graph))        results (if (:entity options)                  (map #(when-let [ent (d/entity @conn                                                 (if (string? %) (edn/read-string %) %))]                          (cond-> (into {:db/id (:db/id ent)} ent)                            (seq (:block/properties ent))                            (update :block/properties (fn [props] (map (fn [m] (into {} m)) props)))))                       (:entity options))                  ;; assumes no :in are in queries                  (let [query (into (edn/read-string (first args'')) [:in '$ '%])                        res (d/q query @conn (rules/extract-rules rules/db-query-dsl-rules))]                    ;; Remove nesting for most queries which just have one :find binding                    (if (= 1 (count (first res))) (mapv first res) res)))]    (when (:verbose options) (println "DB contains" (count (d/datoms @conn :eavt)) "datoms"))    (if (:raw options)      (prn results)      (if (zero? (.-status (child-process/spawnSync "which" #js ["puget"])))        (sh ["puget"] {:input (pr-str results) :stdio ["pipe" "inherit" "inherit"]})        (pprint/pprint results)))))(defn -main [args]  (let [{options :opts args' :args} (cli/parse-args args {:spec spec})        [graph-dir & args''] args'        _ (when (or (nil? graph-dir) (:help options))            (println (str "Usage: $0 GRAPH-NAME [& ARGS] [OPTIONS]\nOptions:\n"                          (cli/format-opts {:spec spec})))            (js/process.exit 1))        graph-dirs (cond-> [graph-dir]                     (:additional-graphs options)                     (into (:additional-graphs options)))]    (doseq [graph-dir graph-dirs]      (query-graph graph-dir args'' options))))(when (= nbb/*file* (nbb/invoked-file))  (-main *command-line-args*))
 |