Attached is a patch that adds a new (web json) module. Some may
remember that I submitted a patch back in 2015 (time flies, eh?) for
an (ice-9 json) module that never made it in. Well, 10 years is a long
time and Guile still doesn't have a built-in JSON module. Third party
libraries like guile-json and guile-sjson are available, the latter
being an adaptation of my original patch and the former remaining the
go-to library used by larger Guile projects like Guix. There's also
SRFI-180 (which sounds like a cool surfing trick!) which was published
in 2020 but the API is, in my opinion, overly complicated due to
generators and other things.  Anyway, JSON continues to be *the* data
interchange format of the web and Guile really ought to have a simple
API that can read/write JSON to/from a port using only Scheme data
types that have read syntax (i.e. no hash tables like guile-json).
This minimal, practical API is what my patch provides.  I've tried my
best to make it as efficient as possible.

I've settled on the following JSON<->Scheme data type mapping which is
nearly identical to SRFI-180 with the exception of object keys:

- true and false are #t and #f
- null is the symbol 'null
- numbers are either exact integers (fixnums and bignums) or inexact
reals (flonums, NaNs and infinities excluded)
- strings are strings
- arrays are vectors
- objects are association lists with string keys (SRFI-180 chose
symbols but JSON uses strings so strings feel the most honest)

Thanks in advance for the review,

- Dave
From 104b57e2a7b4ca47096cb524aff688b0ada49f94 Mon Sep 17 00:00:00 2001
From: David Thompson <dthomps...@worcester.edu>
Date: Sat, 12 Apr 2025 08:27:35 -0400
Subject: [PATCH] web: Add JSON module.

* module/web/json.scm: New file.
* am/bootstrap.am (SOURCES): Add it.
* test-suite/tests/json.test: New file.
* test-suite/Makefile.am (SCM_TESTS): Add it.
* doc/ref/web.texi ("JSON"): New subsection.
---
 am/bootstrap.am            |   3 +-
 doc/ref/web.texi           |  93 +++++++++++
 module/web/json.scm        | 308 +++++++++++++++++++++++++++++++++++++
 test-suite/Makefile.am     |   1 +
 test-suite/tests/json.test | 154 +++++++++++++++++++
 5 files changed, 558 insertions(+), 1 deletion(-)
 create mode 100644 module/web/json.scm
 create mode 100644 test-suite/tests/json.test

diff --git a/am/bootstrap.am b/am/bootstrap.am
index 96023d83d..6806fda5d 100644
--- a/am/bootstrap.am
+++ b/am/bootstrap.am
@@ -425,7 +425,8 @@ SOURCES =					\
   web/response.scm				\
   web/server.scm				\
   web/server/http.scm				\
-  web/uri.scm
+  web/uri.scm					\
+  web/json.scm
 
 ELISP_SOURCES =					\
   language/elisp/boot.el
diff --git a/doc/ref/web.texi b/doc/ref/web.texi
index 607c855b6..53ce14820 100644
--- a/doc/ref/web.texi
+++ b/doc/ref/web.texi
@@ -40,6 +40,7 @@ back.
 * Transfer Codings::            HTTP Transfer Codings.
 * Requests::                    HTTP requests.
 * Responses::                   HTTP responses.
+* JSON::                        The JavaScript Object Notation.
 * Web Client::                  Accessing web resources over HTTP.
 * Web Server::                  Serving HTTP to the internet.
 * Web Examples::                How to use this thing.
@@ -1448,6 +1449,98 @@ Return @code{#t} if @var{type}, a symbol as returned by
 @end deffn
 
 
+@node JSON
+@subsection JSON
+
+@cindex json
+@cindex (web json)
+
+@example
+(use-modules (web json))
+@end example
+
+JavaScript Object Notation (JSON) is the most common data interchange
+format on the web.  It is ubiquitous in HTTP APIs and has found its way
+into many other domains beyond the web, as well.  The @code{(web json)}
+module makes it possible to convert a subset of Scheme data types to
+JSON text and vice versa.  For example, the JSON document:
+
+@example
+@verbatim
+{
+  "name": "Eva Luator",
+  "age": 24,
+  "schemer": true,
+  "hobbies": [
+    "hacking",
+    "cycling",
+    "surfing"
+  ]
+}
+@end verbatim
+@end example
+
+can be represented with the following Scheme expression:
+
+@example
+@verbatim
+'(("name" . "Eva Luator")
+  ("age" . 24)
+  ("schemer" . #t)
+  ("hobbies" . #("hacking" "cycling" "surfing")))
+@end verbatim
+@end example
+
+Strings, exact integers, inexact reals (excluding NaNs and infinities),
+@code{#t}, @code{#f}, the symbol @code{null}, vectors, and association
+lists may be serialized as JSON.  Association lists serialize as JSON
+objects and vectors serialize as JSON arrays.  The keys of association
+lists @emph{must} be strings.
+
+@deffn {Scheme Procedure} read-json [port]
+
+Parse a JSON-encoded value from @var{port} and return its Scheme
+representation.  If @var{port} is unspecified, the current input port is
+used.
+
+@example
+@verbatim
+(call-with-input-string "[true,false,null,42,\"foo\"]" read-json)
+;; => #(#t #f null 42 "foo")
+
+(call-with-input-string "{\"foo\":1,\"bar\":2}" read-json)
+;; => (("foo" . 1) ("bar" . 2))
+@end verbatim
+@end example
+
+@end deffn
+
+@deftp {Exception Type} &json-read-error
+An exception type denoting JSON read errors.
+@end deftp
+
+@deffn {Scheme Procedure} write-json exp [port]
+
+Serialize the expression @var{exp} as JSON-encoded text to @var{port}.
+If @var{port} is unspecified, the current output port is used.
+
+@example
+@verbatim
+(with-output-to-string (lambda () (write-json #(#t #f null 42 "foo"))))
+;; => "[true,false,null,42,\"foo\"]"
+
+(with-output-to-string (lambda () (write-json '(("foo" . 1) ("bar" . 2)))))
+;; => "{\"foo\":1,\"bar\":2}"
+@end verbatim
+@end example
+
+@end deffn
+
+@deftp {Exception Type} &json-write-error
+An exception type denoting JSON write errors.
+@end deftp
+
+
 @node Web Client
 @subsection Web Client
 
diff --git a/module/web/json.scm b/module/web/json.scm
new file mode 100644
index 000000000..41aac0e90
--- /dev/null
+++ b/module/web/json.scm
@@ -0,0 +1,308 @@
+;;;; json.scm --- JSON reader/writer (ECMA-404)
+;;;; Copyright (C) 2025 Free Software Foundation, Inc.
+;;;;
+;;;; This library is free software; you can redistribute it and/or
+;;;; modify it under the terms of the GNU Lesser General Public
+;;;; License as published by the Free Software Foundation; either
+;;;; version 3 of the License, or (at your option) any later version.
+;;;;
+;;;; This library is distributed in the hope that it will be useful,
+;;;; but WITHOUT ANY WARRANTY; without even the implied warranty of
+;;;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
+;;;; Lesser General Public License for more details.
+;;;;
+;;;; You should have received a copy of the GNU Lesser General Public
+;;;; License along with this library; if not, write to the Free Software
+;;;; Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
+;;;;
+
+(define-module (web json)
+  #:use-module (ice-9 exceptions)
+  #:use-module (ice-9 match)
+  #:use-module (ice-9 textual-ports)
+  #:export (&json-read-error
+            read-json
+
+            &json-write-error
+            write-json))
+
+(define-exception-type &json-read-error &error
+  make-json-read-error
+  json-read-error?)
+
+(define* (read-json #:optional (port (current-input-port)))
+  "Parse a JSON-encoded value from @var{port} and return its Scheme
+representation.  If @var{port} is unspecified, the current input port is
+used."
+  (define (fail message)
+    (raise-exception
+     (make-exception (make-json-read-error)
+                     (make-exception-with-origin 'read-json)
+                     (make-exception-with-message message)
+                     (make-exception-with-irritants (list port)))))
+  (define (consume-whitespace)
+    (case (peek-char port)
+      ((#\space #\tab #\return #\newline)
+       (read-char port)
+       (consume-whitespace))
+      (else (values))))
+  (define-syntax-rule (define-keyword-reader name str val)
+    (define (name)
+      (if (string=? (get-string-n port (string-length str)) str)
+          val
+          (fail "invalid keyword"))))
+  (define-keyword-reader read-true "true" #t)
+  (define-keyword-reader read-false "false" #f)
+  (define-keyword-reader read-null "null" 'null)
+  (define (read-hex-digit)
+    (case (peek-char port)
+      ((#\0 #\1 #\2 #\3 #\4 #\5 #\6 #\7 #\8 #\9)
+       (- (char->integer (read-char port)) (char->integer #\0)))
+      ((#\a #\b #\c #\d #\e #\f)
+       (+ 10 (- (char->integer (read-char port)) (char->integer #\a))))
+      ((#\A #\B #\C #\D #\E #\F)
+       (+ 10 (- (char->integer (read-char port)) (char->integer #\A))))
+      (else (fail "invalid hex digit"))))
+  (define (read-utf16-character)
+    (let* ((a (read-hex-digit))
+           (b (read-hex-digit))
+           (c (read-hex-digit))
+           (d (read-hex-digit)))
+      (integer->char (+ (* a (expt 16 3)) (* b (expt 16 2)) (* c 16) d))))
+  (define (read-escape-character)
+    (case (read-char port)
+      ((#\") #\")
+      ((#\\) #\\)
+      ((#\/) #\/)
+      ((#\b) #\backspace)
+      ((#\f) #\page)
+      ((#\n) #\newline)
+      ((#\r) #\return)
+      ((#\t) #\tab)
+      ((#\u) (read-utf16-character))
+      (else (fail "invalid escape character"))))
+  (define (read-string)
+    (read-char port)
+    (list->string
+     (let lp ()
+       (match (read-char port)
+         ((? eof-object?) (fail "EOF while reading string"))
+         (#\" '())
+         (#\\ (cons (read-escape-character) (lp)))
+         (char (cons char (lp)))))))
+  (define (read-digit-maybe)
+    (case (peek-char port)
+      ((#\0 #\1 #\2 #\3 #\4 #\5 #\6 #\7 #\8 #\9)
+       (- (char->integer (read-char port))
+          (char->integer #\0)))
+      (else #f)))
+  (define (read-integer)
+    (let ((x (read-digit-maybe)))
+      (and x
+           (let lp ((x x))
+             (match (read-digit-maybe)
+               (#f x)
+               (y (lp (+ (* x 10) y))))))))
+  (define (read-fraction)
+    (case (peek-char port)
+      ((#\.)
+       (read-char port)
+       (let lp ((mag 10))
+         (let ((n (read-digit-maybe)))
+           (if n (+ (/ n mag) (lp (* mag 10))) 0))))
+      (else 0)))
+  (define (read-exponent)
+    (case (peek-char port)
+      ((#\e #\E)
+       (read-char port)
+       (case (peek-char port)
+         ((#\-)
+          (read-char port)
+          (expt 10 (- (read-integer))))
+         ((#\+)
+          (read-char port)
+          (expt 10 (read-integer)))
+         (else
+          (expt 10 (read-integer)))))
+      (else 1)))
+  (define (read-positive-number)
+    (let ((n (read-integer)))
+      (and n
+           (let* ((f (read-fraction))
+                  (e (read-exponent))
+                  (x (* (+ n f) e)))
+             (if (exact-integer? x) x (exact->inexact x))))))
+  (define (read-negative-number)
+    (read-char port)
+    (let ((x (read-positive-number)))
+      (if x (- x) (fail "invalid number"))))
+  (define (read-leading-zero-number)
+    (read-char port)
+    (case (peek-char port)
+      ;; Extraneous zeroes are not allowed.  A single leading zero
+      ;; can only be followed by a decimal point.
+      ((#\0 #\1 #\2 #\3 #\4 #\5 #\6 #\7 #\8 #\9 #\e #\E)
+       (fail "extraneous leading zero"))
+      ;; Fractional number.
+      ((#\.)
+       (let* ((d (read-fraction))
+              (e (read-exponent)))
+         (exact->inexact (* d e))))
+      ;; Just plain zero.
+      (else 0)))
+  (define (read-key+value-pair)
+    (let ((key (read-string)))
+      (consume-whitespace)
+      (case (read-char port)
+        ((#\:)
+         (consume-whitespace)
+         (cons key (read-value)))
+        (else (fail "invalid key/value pair delimiter")))))
+  (define (read-object)
+    (read-char port)
+    (consume-whitespace)
+    (case (peek-char port)
+      ;; Empty object.
+      ((#\})
+       (read-char port)
+       '())
+      (else
+       ;; Read first key/value pair, then all subsequent pairs delimited
+       ;; by commas.
+       (cons (read-key+value-pair)
+             (let lp ()
+               (consume-whitespace)
+               (case (peek-char port)
+                 ((#\,)
+                  (read-char port)
+                  (consume-whitespace)
+                  (cons (read-key+value-pair) (lp)))
+                 ;; End of object.
+                 ((#\})
+                  (read-char port)
+                  '())
+                 (else (fail "invalid object delimiter"))))))))
+  (define (read-array)
+    (read-char port)
+    (consume-whitespace)
+    (case (peek-char port)
+      ;; Empty array.
+      ((#\])
+       (read-char port)
+       #())
+      (else
+       (list->vector
+        ;; Read the first element, then all subsequent elements
+        ;; delimited by commas.
+        (cons (read-value)
+              (let lp ()
+                (consume-whitespace)
+                (case (peek-char port)
+                  ;; Elements are comma delimited.
+                  ((#\,)
+                   (read-char port)
+                   (consume-whitespace)
+                   (cons (read-value) (lp)))
+                  ;; End of array.
+                  ((#\])
+                   (read-char port)
+                   '())
+                  (else (fail "invalid array delimiter")))))))))
+  (define (read-value)
+    (consume-whitespace)
+    (case (peek-char port)
+      ((#\") (read-string))
+      ((#\{) (read-object))
+      ((#\[) (read-array))
+      ((#\t) (read-true))
+      ((#\f) (read-false))
+      ((#\n) (read-null))
+      ((#\-) (read-negative-number))
+      ((#\0) (read-leading-zero-number))
+      ((#\1 #\2 #\3 #\4 #\5 #\6 #\7 #\8 #\9) (read-positive-number))
+      (else (fail "invalid value"))))
+  (read-value))
+
+(define-exception-type &json-write-error &error
+  make-json-write-error
+  json-write-error?)
+
+(define* (write-json exp #:optional (port (current-output-port)))
+  "Serialize the expression @var{exp} as JSON-encoded text to @var{port}.
+If @var{port} is unspecified, the current output port is used."
+  (define (fail message x)
+    (raise-exception
+     (make-exception (make-json-write-error)
+                     (make-exception-with-origin 'write-json)
+                     (make-exception-with-message message)
+                     (make-exception-with-irritants (list x)))))
+  (define (write-char/escape char)
+    (match char
+      (#\" (put-string port "\\\""))
+      (#\\ (put-string port "\\\\"))
+      (#\/ (put-string port "\\/"))
+      (#\backspace (put-string port "\\b"))
+      (#\page (put-string port "\\f"))
+      (#\newline (put-string port "\\n"))
+      (#\return (put-string port "\\r"))
+      (#\tab (put-string port "\\t"))
+      (_ (put-char port char))))
+  (define (write-string str)
+    (let ((in (open-input-string str)))
+      (put-char port #\")
+      (string-for-each write-char/escape str)
+      (put-char port #\")))
+  (define (write-pair x)
+    (match x
+      (((? string? key) . value)
+       (write-string key)
+       (put-char port #\:)
+       (write-value value))
+      (_ (fail "invalid key/value pair" x))))
+  (define (write-object obj)
+    (put-char port #\{)
+    (match obj
+      ((head . rest)
+       (write-pair head)
+       (let lp ((obj rest))
+         (match obj
+           (() (values))
+           ((head . rest)
+            (put-char port #\,)
+            (write-pair head)
+            (lp rest))
+           (_ (fail "invalid object" obj))))))
+    (put-char port #\}))
+  (define (write-array v)
+    (put-char port #\[)
+    (match (vector-length v)
+      (0 (values))
+      (n
+       (write-value (vector-ref v 0))
+       (do ((i 1 (1+ i)))
+           ((= i n))
+         (put-char port #\,)
+         (write-value (vector-ref v i)))))
+    (put-char port #\]))
+  (define (write-number x)
+    (if (or (exact-integer? x)
+            (and (real? x)
+                 (inexact? x)
+                 ;; NaNs and infinities are not allowed.
+                 (not (or (nan? x) (inf? x)))))
+        ;; Scheme's string representations of exact integers and floats
+        ;; are compatible with JSON.
+        (put-string port (number->string x))
+        (fail "invalid number" x)))
+  (define (write-value x)
+    (match x
+      (#t (put-string port "true"))
+      (#f (put-string port "false"))
+      ('null (put-string port "null"))
+      (() (put-string port "{}"))
+      ((? pair?) (write-object x))
+      ((? vector?) (write-array x))
+      ((? string?) (write-string x))
+      ((? number?) (write-number x))
+      (_ (fail "invalid value" x))))
+  (write-value exp))
diff --git a/test-suite/Makefile.am b/test-suite/Makefile.am
index 6014b1f1f..00afea142 100644
--- a/test-suite/Makefile.am
+++ b/test-suite/Makefile.am
@@ -73,6 +73,7 @@ SCM_TESTS = tests/00-initial-env.test		\
 	    tests/iconv.test			\
 	    tests/import.test			\
 	    tests/interp.test			\
+	    tests/json.test			\
 	    tests/keywords.test			\
 	    tests/list.test			\
 	    tests/load.test			\
diff --git a/test-suite/tests/json.test b/test-suite/tests/json.test
new file mode 100644
index 000000000..f92eeccec
--- /dev/null
+++ b/test-suite/tests/json.test
@@ -0,0 +1,154 @@
+;;;; json.test --- test JSON reader/writer     -*- scheme -*-
+;;;;
+;;;; Copyright (C) 2015 Free Software Foundation, Inc.
+;;;;
+;;;; This library is free software; you can redistribute it and/or
+;;;; modify it under the terms of the GNU Lesser General Public
+;;;; License as published by the Free Software Foundation; either
+;;;; version 3 of the License, or (at your option) any later version.
+;;;;
+;;;; This library is distributed in the hope that it will be useful,
+;;;; but WITHOUT ANY WARRANTY; without even the implied warranty of
+;;;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
+;;;; Lesser General Public License for more details.
+;;;;
+;;;; You should have received a copy of the GNU Lesser General Public
+;;;; License along with this library; if not, write to the Free Software
+;;;; Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
+
+(define-module (test-suite test-json)
+  #:use-module (test-suite lib)
+  #:use-module (web json))
+
+;;;
+;;; Reader
+;;;
+
+(define (read-json-string str)
+  (call-with-input-string str read-json))
+
+(define (json-read=? str x)
+  (= x (read-json-string str)))
+
+(define (json-read-eq? str x)
+  (eq? x (read-json-string str)))
+
+(define (json-read-equal? str x)
+  (equal? x (read-json-string str)))
+
+(define (json-read-string=? str x)
+  (string=? x (read-json-string str)))
+
+(with-test-prefix "read-json"
+  ;; Keywords
+  (pass-if (json-read-eq? "true" #t))
+  (pass-if (json-read-eq? "false" #f))
+  (pass-if (json-read-eq? "null" 'null))
+  ;; Numbers
+  (pass-if (json-read=? "0" 0))
+  (pass-if (json-read=? "-0" 0))
+  (pass-if (json-read=? "0.0" 0.0))
+  (pass-if (json-read=? "-0.0" -0.0))
+  (pass-if (json-read=? "0.1" 0.1))
+  (pass-if (json-read=? "1.234" 1.234))
+  (pass-if (json-read=? "1" 1))
+  (pass-if (json-read=? "-1" -1))
+  (pass-if (json-read=? "1.1" 1.1))
+  (pass-if (json-read=? "1e2" 1e2))
+  (pass-if (json-read=? "1.1e2" 1.1e2))
+  (pass-if (json-read=? "1.1e-2" 1.1e-2))
+  (pass-if (json-read=? "1.1e+2" 1.1e2))
+  ;; Extraneous zeroes in fraction
+  (pass-if (json-read=? "1.000" 1))
+  (pass-if (json-read=? "1.5000" 1.5))
+  ;; Extraneous zeroes in exponent
+  (pass-if (json-read=? "1.1e000" 1.1))
+  (pass-if (json-read=? "1.1e-02" 1.1e-2))
+  (pass-if (json-read=? "1.1e+02" 1.1e2))
+  ;; Strings
+  (pass-if (json-read-string=? "\"foo\"" "foo"))
+  ;; Escape codes
+  (pass-if (json-read-string=? "\"\\\"\"" "\""))
+  (pass-if (json-read-string=? "\"\\\\\"" "\\"))
+  (pass-if (json-read-string=? "\"\\/\"" "/"))
+  (pass-if (json-read-string=? "\"\\b\"" "\b"))
+  (pass-if (json-read-string=? "\"\\f\"" "\f"))
+  (pass-if (json-read-string=? "\"\\n\"" "\n"))
+  (pass-if (json-read-string=? "\"\\r\"" "\r"))
+  (pass-if (json-read-string=? "\"\\t\"" "\t"))
+  ;; Unicode in hexadecimal format
+  (pass-if (json-read-string=? "\"\\u12ab\"" "\u12ab"))
+  ;; Objects
+  (pass-if (json-read-equal? "{}" '()))
+  (pass-if (json-read-equal? "{ \"foo\": \"bar\", \"baz\": \"frob\"}"
+                             '(("foo" . "bar") ("baz" . "frob"))))
+  ;; Nested objects
+  (pass-if (json-read-equal? "{\"foo\":{\"bar\":\"baz\"}}"
+                             '(("foo" . (("bar" . "baz"))))))
+  ;; Arrays
+  (pass-if (json-read-equal? "[]" #()))
+  (pass-if (json-read-equal? "[1, 2, \"foo\"]"
+                             #(1 2 "foo")))
+  ;; Nested arrays
+  (pass-if (json-read-equal? "[1, 2, [\"foo\", \"bar\"]]"
+                             #(1 2 #("foo" "bar"))))
+  ;; Arrays and objects nested within each other
+  (pass-if (json-read-equal? "{\"foo\":[{\"bar\":true},{\"baz\":[1,2,3]}]}"
+                             '(("foo" . #((("bar" . #t))
+                                          (("baz" . #(1 2 3))))))))
+  ;; Leading whitespace
+  (pass-if (json-read-eq? "\t\r\n true" #t)))
+
+;;;
+;;; Writer
+;;;
+
+(define (write-json-string exp)
+  (call-with-output-string
+   (lambda (port)
+     (write-json exp port))))
+
+(define (json-write-string=? exp str)
+  (string=? str (write-json-string exp)))
+
+(with-test-prefix "write-json"
+  ;; Keywords
+  (pass-if (json-write-string=? #t "true"))
+  (pass-if (json-write-string=? #f "false"))
+  (pass-if (json-write-string=? 'null "null"))
+  ;; Numbers
+  (pass-if (json-write-string=? 0 "0"))
+  (pass-if (json-write-string=? 0.0 "0.0"))
+  (pass-if (json-write-string=? 0.1 "0.1"))
+  (pass-if (json-write-string=? 1 "1"))
+  (pass-if (json-write-string=? -1 "-1"))
+  (pass-if (json-write-string=? 1.1 "1.1"))
+  ;; Strings
+  (pass-if (json-write-string=? "foo" "\"foo\""))
+  ;; Escape codes
+  (pass-if (json-write-string=? "\"" "\"\\\"\""))
+  (pass-if (json-write-string=? "\\" "\"\\\\\""))
+  (pass-if (json-write-string=? "/" "\"\\/\""))
+  (pass-if (json-write-string=? "\b" "\"\\b\""))
+  (pass-if (json-write-string=? "\f" "\"\\f\""))
+  (pass-if (json-write-string=? "\n" "\"\\n\""))
+  (pass-if (json-write-string=? "\r" "\"\\r\""))
+  (pass-if (json-write-string=? "\t" "\"\\t\""))
+  ;; Objects
+  (pass-if (json-write-string=? '() "{}"))
+  (pass-if (json-write-string=? '(("foo" . "bar") ("baz" . "frob"))
+                                "{\"foo\":\"bar\",\"baz\":\"frob\"}"))
+  ;; Nested objects
+  (pass-if (json-write-string=? '(("foo" . (("bar" . "baz"))))
+                                "{\"foo\":{\"bar\":\"baz\"}}"))
+  ;; Arrays
+  (pass-if (json-write-string=? #() "[]"))
+  (pass-if (json-write-string=? #(1 2 "foo")
+                                "[1,2,\"foo\"]"))
+  ;; Nested arrays
+  (pass-if (json-write-string=? #(1 2 #("foo" "bar"))
+                                "[1,2,[\"foo\",\"bar\"]]"))
+  ;; Arrays and objects nested in each other
+  (pass-if (json-write-string=? '(("foo" . #((("bar" . #t))
+                                             (("baz" . #(1 2))))))
+                                "{\"foo\":[{\"bar\":true},{\"baz\":[1,2]}]}")))
-- 
2.47.1

Reply via email to