001/*
002 * acme4j - Java ACME client
003 *
004 * Copyright (C) 2015 Richard "Shred" Körber
005 *   http://acme4j.shredzone.org
006 *
007 * Licensed under the Apache License, Version 2.0 (the "License");
008 * you may not use this file except in compliance with the License.
009 *
010 * This program is distributed in the hope that it will be useful,
011 * but WITHOUT ANY WARRANTY; without even the implied warranty of
012 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
013 */
014package org.shredzone.acme4j.toolbox;
015
016import static org.shredzone.acme4j.toolbox.AcmeUtils.base64UrlEncode;
017
018import java.security.Key;
019import java.security.PublicKey;
020import java.time.Duration;
021import java.time.Instant;
022import java.time.format.DateTimeFormatter;
023import java.util.Collection;
024import java.util.Collections;
025import java.util.LinkedHashMap;
026import java.util.Map;
027import java.util.Objects;
028
029import edu.umd.cs.findbugs.annotations.Nullable;
030import org.jose4j.json.JsonUtil;
031
032/**
033 * Builder for JSON structures.
034 * <p>
035 * Example:
036 * <pre>
037 * JSONBuilder cb = new JSONBuilder();
038 * cb.put("foo", 123).put("bar", "hello world");
039 * cb.object("sub").put("data", "subdata");
040 * cb.array("array", 123, 456, 789);
041 * </pre>
042 */
043public class JSONBuilder {
044
045    private final Map<String, Object> data = new LinkedHashMap<>();
046
047    /**
048     * Puts a property. If a property with the key exists, it will be replaced.
049     *
050     * @param key
051     *            Property key
052     * @param value
053     *            Property value
054     * @return {@code this}
055     */
056    public JSONBuilder put(String key, @Nullable Object value) {
057        data.put(Objects.requireNonNull(key, "key"), value);
058        return this;
059    }
060
061    /**
062     * Puts an {@link Instant} to the JSON. If a property with the key exists, it will be
063     * replaced.
064     *
065     * @param key
066     *            Property key
067     * @param value
068     *            Property {@link Instant} value
069     * @return {@code this}
070     */
071    public JSONBuilder put(String key, @Nullable Instant value) {
072        if (value == null) {
073            put(key, (Object) null);
074            return this;
075        }
076
077        put(key, DateTimeFormatter.ISO_INSTANT.format(value));
078        return this;
079    }
080
081    /**
082     * Puts a {@link Duration} to the JSON. If a property with the key exists, it will be
083     * replaced.
084     *
085     * @param key
086     *            Property key
087     * @param value
088     *            Property {@link Duration} value
089     * @return {@code this}
090     * @since 2.3
091     */
092    public JSONBuilder put(String key, @Nullable Duration value) {
093        if (value == null) {
094            put(key, (Object) null);
095            return this;
096        }
097
098        put(key, value.getSeconds());
099        return this;
100    }
101
102    /**
103     * Puts binary data to the JSON. The data is base64 url encoded.
104     *
105     * @param key
106     *            Property key
107     * @param data
108     *            Property data
109     * @return {@code this}
110     */
111    public JSONBuilder putBase64(String key, byte[] data) {
112        return put(key, base64UrlEncode(data));
113    }
114
115    /**
116     * Puts a {@link Key} into the claim. The key is serializied as JWK.
117     *
118     * @param key
119     *            Property key
120     * @param publickey
121     *            {@link PublicKey} to serialize
122     * @return {@code this}
123     */
124    public JSONBuilder putKey(String key, PublicKey publickey) {
125        Objects.requireNonNull(publickey, "publickey");
126
127        data.put(key, JoseUtils.publicKeyToJWK(publickey));
128        return this;
129    }
130
131    /**
132     * Creates an object for the given key.
133     *
134     * @param key
135     *            Key of the object
136     * @return Newly created {@link JSONBuilder} for the object.
137     */
138    public JSONBuilder object(String key) {
139        var subBuilder = new JSONBuilder();
140        data.put(key, subBuilder.data);
141        return subBuilder;
142    }
143
144    /**
145     * Puts an array.
146     *
147     * @param key
148     *            Property key
149     * @param values
150     *            Collection of property values
151     * @return {@code this}
152     */
153    public JSONBuilder array(String key, Collection<?> values) {
154        data.put(key, values);
155        return this;
156    }
157
158    /**
159     * Returns a {@link Map} representation of the current state.
160     *
161     * @return {@link Map} of the current state
162     */
163    public Map<String, Object> toMap() {
164        return Collections.unmodifiableMap(data);
165    }
166
167    /**
168     * Returns a {@link JSON} representation of the current state.
169     *
170     * @return {@link JSON} of the current state
171     */
172    public JSON toJSON() {
173        return JSON.parse(toString());
174    }
175
176    /**
177     * Returns a JSON string representation of the current state.
178     */
179    @Override
180    public String toString() {
181        return JsonUtil.toJson(data);
182    }
183
184}