2024-02-20 14:19:23 +11:00
|
|
|
(ns parse
|
|
|
|
(:require [clojure.spec.alpha :as s]
|
|
|
|
[clojure.walk :as walk]))
|
|
|
|
|
2024-02-20 18:26:45 +11:00
|
|
|
(s/def ::token (s/+ (set "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ01234567890-:")))
|
|
|
|
|
|
|
|
(s/def ::number (s/+ (set "01234567890-.")))
|
2024-02-20 14:19:23 +11:00
|
|
|
|
|
|
|
(s/def ::quoted-token (s/cat
|
|
|
|
:_ #{\"}
|
2024-02-21 18:08:49 +11:00
|
|
|
:token (s/+ (set "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ01234567890-:./% ()"))
|
2024-02-20 14:19:23 +11:00
|
|
|
:_ #{\"}))
|
|
|
|
|
|
|
|
(s/def ::whitespace (s/+ #{\space}))
|
|
|
|
|
|
|
|
(s/def ::meta
|
|
|
|
(s/cat :_ ::whitespace
|
|
|
|
:key ::token
|
|
|
|
:_ #{\:}
|
|
|
|
:_ ::whitespace
|
2024-02-20 14:37:54 +11:00
|
|
|
:value ::quoted-token
|
2024-02-20 14:19:23 +11:00
|
|
|
:_ #{\newline}))
|
|
|
|
|
|
|
|
(s/def ::posting
|
|
|
|
(s/cat :_ ::whitespace
|
|
|
|
:account ::token
|
|
|
|
:_ ::whitespace
|
2024-02-20 18:26:45 +11:00
|
|
|
:amount ::number
|
2024-02-20 14:19:23 +11:00
|
|
|
:_ ::whitespace
|
|
|
|
:currency ::token
|
|
|
|
:_ #{\newline}
|
|
|
|
:meta (s/* ::meta)))
|
|
|
|
|
|
|
|
(s/def ::transaction
|
|
|
|
(s/cat
|
|
|
|
:date ::token
|
|
|
|
:_ ::whitespace
|
|
|
|
:_ ::token
|
|
|
|
:_ ::whitespace
|
2024-02-21 18:08:49 +11:00
|
|
|
:payee (s/? ::quoted-token)
|
|
|
|
:_ (s/? ::whitespace)
|
2024-02-20 14:19:23 +11:00
|
|
|
:desc ::quoted-token
|
|
|
|
:_ #{\newline}
|
|
|
|
:meta (s/* ::meta)
|
|
|
|
:postings (s/* ::posting)
|
|
|
|
:_ (s/* #{\newline})))
|
|
|
|
|
2024-02-20 14:55:14 +11:00
|
|
|
(s/def ::transactions (s/+ ::transaction))
|
2024-02-20 14:19:23 +11:00
|
|
|
|
|
|
|
(defn kv->map
|
2024-02-20 14:55:14 +11:00
|
|
|
"Convert vector [{:key \"x\" :value \"y\"}] to {:x y}"
|
2024-02-20 14:19:23 +11:00
|
|
|
[vec]
|
|
|
|
(into {} (for [{:keys [key value]} vec]
|
|
|
|
[(keyword key) value])))
|
|
|
|
|
|
|
|
(defn convert-parse-tree
|
|
|
|
"Reformat the parse tree into a similar data structure as used during import."
|
|
|
|
[tree]
|
|
|
|
(walk/postwalk #(cond
|
|
|
|
;; vector of chars to string
|
|
|
|
(and (vector? %)
|
|
|
|
(= (type (first %)) java.lang.Character)) (apply str %)
|
|
|
|
;; posting amount to bigdec
|
|
|
|
(and (= (type %) clojure.lang.PersistentArrayMap)
|
|
|
|
(contains? % :amount)) (dissoc (update % :amount bigdec) :_)
|
2024-02-20 14:37:54 +11:00
|
|
|
;; flatten quoted-tokens
|
2024-02-20 14:19:23 +11:00
|
|
|
(and (= (type %) clojure.lang.PersistentArrayMap)
|
2024-02-20 14:37:54 +11:00
|
|
|
(contains? % :token)) (:token %)
|
2024-02-20 14:19:23 +11:00
|
|
|
;; drop :_ keys
|
|
|
|
(= (type %) clojure.lang.PersistentArrayMap) (dissoc % :_)
|
2024-02-20 14:55:14 +11:00
|
|
|
;; convert vector of :key/:value maps to a map
|
2024-02-20 14:19:23 +11:00
|
|
|
(and (vector? %)
|
|
|
|
(= (type (first %)) clojure.lang.PersistentArrayMap)
|
|
|
|
(contains? (first %) :key)) (kv->map %)
|
|
|
|
:else %)
|
|
|
|
tree))
|
|
|
|
|
|
|
|
(defn parse
|
|
|
|
"Parse a Beancount transaction.
|
|
|
|
Use so we can convert the data generated during import with the hand-generated
|
|
|
|
Beancount transactions."
|
|
|
|
[text]
|
|
|
|
(let [tree (s/conform ::transactions (conj (vec text) \newline))]
|
|
|
|
(convert-parse-tree tree)))
|