aboutsummaryrefslogtreecommitdiff
path: root/stdlib/source/lux/data/format/json.lux
diff options
context:
space:
mode:
Diffstat (limited to 'stdlib/source/lux/data/format/json.lux')
-rw-r--r--stdlib/source/lux/data/format/json.lux216
1 files changed, 129 insertions, 87 deletions
diff --git a/stdlib/source/lux/data/format/json.lux b/stdlib/source/lux/data/format/json.lux
index 7fae80334..a5611a7c3 100644
--- a/stdlib/source/lux/data/format/json.lux
+++ b/stdlib/source/lux/data/format/json.lux
@@ -1,34 +1,36 @@
(.module: {#.doc (.doc "Functionality for reading and writing values in the JSON format."
"For more information, please see: http://www.json.org/")}
[lux #*
+ ["." meta (#+ monad with-gensyms)]
[abstract
- ["." monad (#+ do)]
[equivalence (#+ Equivalence)]
- codec]
+ [codec (#+ Codec)]
+ [predicate (#+ Predicate)]
+ ["." monad (#+ do)]]
[control
pipe
["." try (#+ Try)]
["<>" parser ("#@." monad)
- ["<t>" text (#+ Parser)]]]
+ ["<.>" text (#+ Parser)]]]
[data
["." bit]
["." maybe]
["." product]
+ ["." text ("#@." equivalence monoid)]
[number
["n" nat]
["f" frac ("#@." decimal)]]
- ["." text ("#@." equivalence monoid)]
[collection
["." list ("#@." fold functor)]
["." row (#+ Row row) ("#@." monad)]
["." dictionary (#+ Dictionary)]]]
- ["." meta (#+ monad with-gensyms)]
[macro
[syntax (#+ syntax:)]
["." code]]])
(template [<name> <type>]
- [(type: #export <name> <type>)]
+ [(type: #export <name>
+ <type>)]
[Null Any]
[Boolean Bit]
@@ -45,22 +47,28 @@
(#Object (Dictionary String JSON)))
(template [<name> <type>]
- [(type: #export <name> <type>)]
+ [(type: #export <name>
+ <type>)]
[Array (Row JSON)]
[Object (Dictionary String JSON)]
)
+(def: #export null?
+ (Predicate JSON)
+ (|>> (case> #Null true
+ _ false)))
+
(def: #export object
(-> (List [String JSON]) JSON)
(|>> (dictionary.from-list text.hash) #..Object))
(syntax: #export (json token)
{#.doc (doc "A simple way to produce JSON literals."
+ (json #null)
(json #1)
(json +123.456)
- (json "Some text")
- (json #null)
+ (json "this is a string")
(json ["this" "is" "an" "array"])
(json {"this" "is"
"an" "object"}))}
@@ -98,7 +106,7 @@
_
(wrap (list token)))))
-(def: #export (get-fields json)
+(def: #export (fields json)
{#.doc "Get all the fields in a JSON object."}
(-> JSON (Try (List String)))
(case json
@@ -198,7 +206,9 @@
############################################################
############################################################
-(def: (format-null _) (-> Null Text) "null")
+(def: (format-null _)
+ (-> Null Text)
+ "null")
(def: format-boolean
(-> Boolean Text)
@@ -209,29 +219,56 @@
(def: format-number
(-> Number Text)
(|>> (case>
- +0.0 "0.0"
- -0.0 "0.0"
+ (^or +0.0 -0.0) "0.0"
value (let [raw (:: f.decimal encode value)]
(if (f.< +0.0 value)
raw
(|> raw (text.split 1) maybe.assume product.right))))))
-(def: format-string (-> String Text) text.encode)
+(def: escape "\")
+(def: escaped-dq (text@compose ..escape text.double-quote))
-(def: (format-array format elems)
- (-> (-> JSON Text) (-> Array Text))
- ($_ text@compose "["
- (|> elems (row@map format) row.to-list (text.join-with ","))
- "]"))
+(def: format-string
+ (-> String Text)
+ (|>> (text.replace-all text.double-quote ..escaped-dq)
+ (text.enclose [text.double-quote text.double-quote])))
-(def: (format-object format object)
+(template [<token> <name>]
+ [(def: <name>
+ Text
+ <token>)]
+
+ ["," separator]
+ [":" entry-separator]
+
+ ["[" open-array]
+ ["]" close-array]
+
+ ["{" open-object]
+ ["}" close-object]
+ )
+
+(def: (format-array format)
+ (-> (-> JSON Text) (-> Array Text))
+ (|>> (row@map format)
+ row.to-list
+ (text.join-with ..separator)
+ (text.enclose [..open-array ..close-array])))
+
+(def: (format-kv format [key value])
+ (-> (-> JSON Text) (-> [String JSON] Text))
+ ($_ text@compose
+ (..format-string key)
+ ..entry-separator
+ (format value)
+ ))
+
+(def: (format-object format)
(-> (-> JSON Text) (-> Object Text))
- ($_ text@compose "{"
- (|> object
- dictionary.entries
- (list@map (function (_ [key value]) ($_ text@compose (format-string key) ":" (format value))))
- (text.join-with ","))
- "}"))
+ (|>> dictionary.entries
+ (list@map (..format-kv format))
+ (text.join-with ..separator)
+ (text.enclose [..open-object ..close-object])))
(def: #export (format json)
(-> JSON Text)
@@ -239,61 +276,66 @@
(^template [<tag> <format>]
[(<tag> value)
(<format> value)])
- ([#Null format-null]
- [#Boolean format-boolean]
- [#Number format-number]
- [#String format-string]
- [#Array (format-array format)]
- [#Object (format-object format)])
+ ([#Null ..format-null]
+ [#Boolean ..format-boolean]
+ [#Number ..format-number]
+ [#String ..format-string]
+ [#Array (..format-array format)]
+ [#Object (..format-object format)])
))
############################################################
############################################################
############################################################
-(def: space~
+(def: parse-space
(Parser Text)
- (<t>.some <t>.space))
+ (<text>.some <text>.space))
-(def: data-sep
+(def: parse-separator
(Parser [Text Any Text])
- ($_ <>.and space~ (<t>.this ",") space~))
+ ($_ <>.and
+ ..parse-space
+ (<text>.this ..separator)
+ ..parse-space))
-(def: null~
+(def: parse-null
(Parser Null)
(do <>.monad
- [_ (<t>.this "null")]
+ [_ (<text>.this "null")]
(wrap [])))
(template [<name> <token> <value>]
[(def: <name>
(Parser Boolean)
(do <>.monad
- [_ (<t>.this <token>)]
+ [_ (<text>.this <token>)]
(wrap <value>)))]
- [true~ "true" #1]
- [false~ "false" #0]
+ [parse-true "true" #1]
+ [parse-false "false" #0]
)
-(def: boolean~
+(def: parse-boolean
(Parser Boolean)
- (<>.either true~ false~))
+ ($_ <>.either
+ ..parse-true
+ ..parse-false))
-(def: number~
+(def: parse-number
(Parser Number)
(do {! <>.monad}
- [signed? (<>.parses? (<t>.this "-"))
- digits (<t>.many <t>.decimal)
+ [signed? (<>.parses? (<text>.this "-"))
+ digits (<text>.many <text>.decimal)
decimals (<>.default "0"
(do !
- [_ (<t>.this ".")]
- (<t>.many <t>.decimal)))
+ [_ (<text>.this ".")]
+ (<text>.many <text>.decimal)))
exp (<>.default ""
(do !
- [mark (<t>.one-of "eE")
- signed?' (<>.parses? (<t>.this "-"))
- offset (<t>.many <t>.decimal)]
+ [mark (<text>.one-of "eE")
+ signed?' (<>.parses? (<text>.this "-"))
+ offset (<text>.many <text>.decimal)]
(wrap ($_ text@compose mark (if signed?' "-" "") offset))))]
(case (f@decode ($_ text@compose (if signed? "-" "") digits "." decimals exp))
(#try.Failure message)
@@ -302,77 +344,77 @@
(#try.Success value)
(wrap value))))
-(def: escaped~
+(def: parse-escaped
(Parser Text)
($_ <>.either
- (<>.after (<t>.this "\t")
+ (<>.after (<text>.this "\t")
(<>@wrap text.tab))
- (<>.after (<t>.this "\b")
+ (<>.after (<text>.this "\b")
(<>@wrap text.back-space))
- (<>.after (<t>.this "\n")
+ (<>.after (<text>.this "\n")
(<>@wrap text.new-line))
- (<>.after (<t>.this "\r")
+ (<>.after (<text>.this "\r")
(<>@wrap text.carriage-return))
- (<>.after (<t>.this "\f")
+ (<>.after (<text>.this "\f")
(<>@wrap text.form-feed))
- (<>.after (<t>.this (text@compose "\" text.double-quote))
+ (<>.after (<text>.this (text@compose "\" text.double-quote))
(<>@wrap text.double-quote))
- (<>.after (<t>.this "\\")
+ (<>.after (<text>.this "\\")
(<>@wrap "\"))))
-(def: string~
+(def: parse-string
(Parser String)
- (<| (<t>.enclosed [text.double-quote text.double-quote])
+ (<| (<text>.enclosed [text.double-quote text.double-quote])
(loop [_ []])
(do {! <>.monad}
- [chars (<t>.some (<t>.none-of (text@compose "\" text.double-quote)))
- stop <t>.peek])
+ [chars (<text>.some (<text>.none-of (text@compose "\" text.double-quote)))
+ stop <text>.peek])
(if (text@= "\" stop)
(do !
- [escaped escaped~
+ [escaped parse-escaped
next-chars (recur [])]
(wrap ($_ text@compose chars escaped next-chars)))
(wrap chars))))
-(def: (kv~ json~)
+(def: (parse-kv parse-json)
(-> (Parser JSON) (Parser [String JSON]))
(do <>.monad
- [key string~
- _ space~
- _ (<t>.this ":")
- _ space~
- value json~]
+ [key ..parse-string
+ _ ..parse-space
+ _ (<text>.this ..entry-separator)
+ _ ..parse-space
+ value parse-json]
(wrap [key value])))
(template [<name> <type> <open> <close> <elem-parser> <prep>]
- [(def: (<name> json~)
+ [(def: (<name> parse-json)
(-> (Parser JSON) (Parser <type>))
(do <>.monad
- [_ (<t>.this <open>)
- _ space~
- elems (<>.sep-by data-sep <elem-parser>)
- _ space~
- _ (<t>.this <close>)]
+ [_ (<text>.this <open>)
+ _ parse-space
+ elems (<>.sep-by ..parse-separator <elem-parser>)
+ _ parse-space
+ _ (<text>.this <close>)]
(wrap (<prep> elems))))]
- [array~ Array "[" "]" json~ row.from-list]
- [object~ Object "{" "}" (kv~ json~) (dictionary.from-list text.hash)]
+ [parse-array Array ..open-array ..close-array parse-json row.from-list]
+ [parse-object Object ..open-object ..close-object (parse-kv parse-json) (dictionary.from-list text.hash)]
)
-(def: json~
+(def: parse-json
(Parser JSON)
(<>.rec
- (function (_ json~)
+ (function (_ parse-json)
($_ <>.or
- null~
- boolean~
- number~
- string~
- (array~ json~)
- (object~ json~)))))
+ parse-null
+ parse-boolean
+ parse-number
+ parse-string
+ (parse-array parse-json)
+ (parse-object parse-json)))))
(structure: #export codec
(Codec Text JSON)
(def: encode ..format)
- (def: decode (<t>.run json~)))
+ (def: decode (<text>.run parse-json)))