(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 self-closing?] (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 (if self-closing? " />" ">")))) (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 self-closing? (= 0 (# body))] (.. (tag tag-name attrs self-closing?) (table.concat (icollect [_ element (ipairs body)] (render element allow-no-escape?)) " ") (if (not self-closing?) (.. "") ""))))) { :render render }