All Downloads are FREE. Search and download functionalities are using the official Maven repository.

groovy.json.JsonBuilder Maven / Gradle / Ivy

There is a newer version: 3.0.22
Show newest version
/*
 *  Licensed to the Apache Software Foundation (ASF) under one
 *  or more contributor license agreements.  See the NOTICE file
 *  distributed with this work for additional information
 *  regarding copyright ownership.  The ASF licenses this file
 *  to you under the Apache License, Version 2.0 (the
 *  "License"); you may not use this file except in compliance
 *  with the License.  You may obtain a copy of the License at
 *
 *    http://www.apache.org/licenses/LICENSE-2.0
 *
 *  Unless required by applicable law or agreed to in writing,
 *  software distributed under the License is distributed on an
 *  "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
 *  KIND, either express or implied.  See the License for the
 *  specific language governing permissions and limitations
 *  under the License.
 */
package groovy.json;

import groovy.lang.Closure;
import groovy.lang.GroovyObjectSupport;
import groovy.lang.Writable;

import java.io.IOException;
import java.io.Writer;
import java.util.*;

/**
 * A builder for creating JSON payloads.
 * 

* This builder supports the usual builder syntax made of nested method calls and closures, * but also some specific aspects of JSON data structures, such as list of values, etc. * Please make sure to have a look at the various methods provided by this builder * to be able to learn about the various possibilities of usage. *

* Example: *


 *       def builder = new groovy.json.JsonBuilder()
 *       def root = builder.people {
 *           person {
 *               firstName 'Guillame'
 *               lastName 'Laforge'
 *               // Named arguments are valid values for objects too
 *               address(
 *                       city: 'Paris',
 *                       country: 'France',
 *                       zip: 12345,
 *               )
 *               married true
 *               // a list of values
 *               conferences 'JavaOne', 'Gr8conf'
 *           }
 *       }
 *
 *       // creates a data structure made of maps (Json object) and lists (Json array)
 *       assert root instanceof Map
 *
 *       assert builder.toString() == '{"people":{"person":{"firstName":"Guillame","lastName":"Laforge","address":{"city":"Paris","country":"France","zip":12345},"married":true,"conferences":["JavaOne","Gr8conf"]}}}'
 * 
* * @author Guillaume Laforge * @author Andrey Bloshetsov * @since 1.8.0 */ public class JsonBuilder extends GroovyObjectSupport implements Writable { private Object content; /** * Instantiates a JSON builder. */ public JsonBuilder() { } /** * Instantiates a JSON builder with some existing data structure. * * @param content a pre-existing data structure */ public JsonBuilder(Object content) { this.content = content; } public Object getContent() { return content; } /** * Named arguments can be passed to the JSON builder instance to create a root JSON object *

* Example: *


     * def json = new JsonBuilder()
     * json name: "Guillaume", age: 33
     *
     * assert json.toString() == '{"name":"Guillaume","age":33}'
     * 
* * @param m a map of key / value pairs * @return a map of key / value pairs */ public Object call(Map m) { content = m; return content; } /** * A list of elements as arguments to the JSON builder creates a root JSON array *

* Example: *


     * def json = new JsonBuilder()
     * def result = json([1, 2, 3])
     *
     * assert result instanceof List
     * assert json.toString() == "[1,2,3]"
     * 
* * @param l a list of values * @return a list of values */ public Object call(List l) { content = l; return content; } /** * Varargs elements as arguments to the JSON builder create a root JSON array *

* Example: *


     * def json = new JsonBuilder()
     * def result = json 1, 2, 3
     *
     * assert result instanceof List
     * assert json.toString() == "[1,2,3]"
     * 
* * @param args an array of values * @return a list of values */ public Object call(Object... args) { List listContent = new ArrayList(); for (Object it : args) { listContent.add(it); } content = listContent; return content; } /** * A collection and closure passed to a JSON builder will create a root JSON array applying * the closure to each object in the collection *

* Example: *


     * class Author {
     *      String name
     * }
     * def authors = [new Author (name: "Guillaume"), new Author (name: "Jochen"), new Author (name: "Paul")]
     *
     * def json = new JsonBuilder()
     * json authors, { Author author ->
     *      name author.name
     * }
     *
     * assert json.toString() == '[{"name":"Guillaume"},{"name":"Jochen"},{"name":"Paul"}]'
     * 
* @param coll a collection * @param c a closure used to convert the objects of coll * @return a list of values */ public Object call(Collection coll, Closure c) { List listContent = new ArrayList(); if (coll != null) { for (Object it : coll) { listContent.add(JsonDelegate.curryDelegateAndGetContent(c, it)); } } content = listContent; return content; } /** * A closure passed to a JSON builder will create a root JSON object *

* Example: *


     * def json = new JsonBuilder()
     * def result = json {
     *      name "Guillaume"
     *      age 33
     * }
     *
     * assert result instanceof Map
     * assert json.toString() == '{"name":"Guillaume","age":33}'
     * 
* * @param c a closure whose method call statements represent key / values of a JSON object * @return a map of key / value pairs */ public Object call(Closure c) { content = JsonDelegate.cloneDelegateAndGetContent(c); return content; } /** * A method call on the JSON builder instance will create a root object with only one key * whose name is the name of the method being called. * This method takes as arguments: *
    *
  • a closure
  • *
  • a map (ie. named arguments)
  • *
  • a map and a closure
  • *
  • or no argument at all
  • *
*

* Example with a classicala builder-style: *


     * def json = new JsonBuilder()
     * def result = json.person {
     *      name "Guillaume"
     *      age 33
     * }
     *
     * assert result instanceof Map
     * assert json.toString() == '{"person":{"name":"Guillaume","age":33}}'
     * 
* * Or alternatively with a method call taking named arguments: *

     * def json = new JsonBuilder()
     * json.person name: "Guillaume", age: 33
     *
     * assert json.toString() == '{"person":{"name":"Guillaume","age":33}}'
     * 
* * If you use named arguments and a closure as last argument, * the key/value pairs of the map (as named arguments) * and the key/value pairs represented in the closure * will be merged together — * the closure properties overriding the map key/values * in case the same key is used. *

     * def json = new JsonBuilder()
     * json.person(name: "Guillaume", age: 33) { town "Paris" }
     *
     * assert json.toString() == '{"person":{"name":"Guillaume","age":33,"town":"Paris"}}'
     * 
* * The empty args call will create a key whose value will be an empty JSON object: *

     * def json = new JsonBuilder()
     * json.person()
     *
     * assert json.toString() == '{"person":{}}'
     * 
* * @param name the single key * @param args the value associated with the key * @return a map with a single key */ public Object invokeMethod(String name, Object args) { if (args != null && Object[].class.isAssignableFrom(args.getClass())) { Object[] arr = (Object[]) args; if (arr.length == 0) { return setAndGetContent(name, new HashMap()); } else if (arr.length == 1) { if (arr[0] instanceof Closure) { return setAndGetContent(name, JsonDelegate.cloneDelegateAndGetContent((Closure) arr[0])); } else if (arr[0] instanceof Map) { return setAndGetContent(name, arr[0]); } } else if (arr.length == 2) { if (arr[0] instanceof Map && arr[1] instanceof Closure) { Map subMap = new LinkedHashMap(); subMap.putAll((Map) arr[0]); subMap.putAll(JsonDelegate.cloneDelegateAndGetContent((Closure) arr[1])); return setAndGetContent(name, subMap); } else if (arr[0] instanceof Collection && arr[1] instanceof Closure) { List> list = new ArrayList>(); for (Object it : (Collection) arr[0]) { list.add(JsonDelegate.curryDelegateAndGetContent((Closure) arr[1], it)); } return setAndGetContent(name, list); } } throw new JsonException("Expected no arguments, a single map, a single closure, or a map and closure as arguments."); } else { return setAndGetContent(name, new HashMap()); } } private Object setAndGetContent(String name, Object value) { Map contentMap = new LinkedHashMap(); contentMap.put(name, value); content = contentMap; return content; } /** * Serializes the internal data structure built with the builder to a conformant JSON payload string *

* Example: *


     * def json = new JsonBuilder()
     * json { temperature 37 }
     *
     * assert json.toString() == '{"temperature":37}'
     * 
* * @return a JSON output */ public String toString() { return JsonOutput.toJson(content); } /** * Pretty-prints and formats the JSON payload. *

* This method calls the JsonLexer to parser the output of the builder, * so this may not be an optimal method to call, * and should be used mainly for debugging purpose * for a human-readable output of the JSON content. * * @return a pretty printed JSON output */ public String toPrettyString() { return JsonOutput.prettyPrint(toString()); } /** * The JSON builder implements the Writable interface, * so that you can have the builder serialize itself the JSON payload to a writer. *

* Example: *


     * def json = new JsonBuilder()
     * json { temperature 37 }
     *
     * def out = new StringWriter()
     * out << json
     *
     * assert out.toString() == '{"temperature":37}'
     * 
* * @param out a writer on which to serialize the JSON payload * @return the writer */ public Writer writeTo(Writer out) throws IOException { return out.append(toString()); } }