Я пробую clojure (впервые) для простого проекта. Мне нужно обновить дерево xml с помощью файла csv. Я читаю CSV-файл построчно, извлекаю некоторые значения, зацикливаю узел с некоторыми значениями и вставляю дочерний узел с другим значением.
Это отлично работает при первой вставке элемента. Второй раз получаю исключение NullPointerException (без следа). Я ищу корень из возвращаемого значения, которое я получаю от insert-child, и передаю этот корневой узел в следующий цикл. Каким-то образом вторая вставка не работает в этом корневом элементе. Кто-нибудь видит, что здесь происходит не так? Или отзыв об этом коде в целом, так как это моя первая попытка написать что-нибудь на Clojure.
(require 'clojure.string)
(require '[clojure.java.io :as io])
(require '[clojure.xml :as xml])
(require '[clojure.zip :as zip])
(require '[clojure.data.zip.xml :as zf])
(def business-object-config (xml/parse "BusinessObject.config"))
(def zipped (zip/xml-zip business-object-config ))
(defn sql-table-name [table-name]
(second (re-matches #"(.*?)(Base|ExtensionBase|$)" table-name)))
(defn insert-sqlpropertyname-elem [loc name]
(zip/root (zip/insert-child loc {:tag :SqlPropertyName :content [name]})))
(defn get-entity-node [table-name crm-name business-objects]
(first (zf/xml-> business-objects :Entities
:Entity [:CrmName (zf/text= (clojure.string/lower-case (sql-table-name table-name)))]
:EntityItems
:EntityItem [:CrmPropertyName (zf/text= (clojure.string/lower-case crm-name))])))
(defn process-line [line business-objects]
(let [{crm-name 0 table-name 1 sql-name 6} (clojure.string/split line #";")
node (get-entity-node table-name crm-name business-objects)]
(insert-sqlpropertyname-elem node sql-name)))
(defn process-csv []
(with-open
[rdr (io/reader "input.csv")]
(loop [lines (vec (take 5 (rest (line-seq rdr))))
index (dec (count lines))
boc zipped]
(if (neg? index)
boc
(recur lines (dec index) (process-line (nth lines index) boc))))))
(spit "out.xml" (with-out-str (xml/emit (process-csv)) :pad true))