(local entity-replacements {"&" "&" ; must be first! "<" "<" ">" ">" "\"" """}) (local entity-search (.. "[" (table.concat (icollect [k (pairs entity-replacements)] k)) "]")) (fn escape [s] (assert (= (type s) :string)) (s:gsub entity-search entity-replacements)) (fn tag [tag-name attrs] (assert (= (type attrs) "table") (.. "Missing attrs table: " tag-name)) (let [attr-str (table.concat (icollect [k v (pairs attrs)] (if (= v true) k (.. k "=\"" (escape v)"\""))) " ")] (.. "<" tag-name " " attr-str">"))) (fn render [document allow-no-escape?] (if (= (type document) :string) (escape document) (and allow-no-escape? (= (. document 1) :NO-ESCAPE)) (. document 2) (let [[tag-name attrs & body] document] (.. (tag tag-name attrs) (table.concat (icollect [_ element (ipairs body)] (render element allow-no-escape?)) " ") "")))) { :render render }