io.github.stewseo.client.json.jsonb.JsonbJsonpMapper Maven / Gradle / Ivy
package io.github.stewseo.client.json.jsonb;
import io.github.stewseo.client.json.JsonpDeserializer;
import io.github.stewseo.client.json.JsonpDeserializerBase;
import io.github.stewseo.client.json.JsonpMapper;
import io.github.stewseo.client.json.JsonpMapperBase;
import io.github.stewseo.client.json.JsonpSerializable;
import io.github.stewseo.client.json.JsonpUtils;
import jakarta.json.bind.Jsonb;
import jakarta.json.bind.spi.JsonbProvider;
import jakarta.json.spi.JsonProvider;
import jakarta.json.stream.JsonGenerator;
import jakarta.json.stream.JsonParser;
import jakarta.json.stream.JsonParser.Event;
import java.io.CharArrayReader;
import java.io.CharArrayWriter;
import java.lang.reflect.Type;
import java.util.EnumSet;
public class JsonbJsonpMapper extends JsonpMapperBase {
private final JsonProvider jsonProvider;
private final Jsonb jsonb;
public JsonbJsonpMapper(JsonProvider jsonProvider, Jsonb jsonb) {
this.jsonProvider = jsonProvider;
this.jsonb = jsonb;
}
public JsonbJsonpMapper(JsonProvider jsonProvider, JsonbProvider jsonbProvider) {
this(jsonProvider, jsonbProvider.create().build());
}
public JsonbJsonpMapper() {
this(JsonpUtils.provider(), JsonbProvider.provider());
}
@Override
public JsonpMapper withAttribute(String name, T value) {
return new JsonbJsonpMapper(this.jsonProvider, this.jsonb).addAttribute(name, value);
}
@Override
protected JsonpDeserializer getDefaultDeserializer(Type type) {
return new Deserializer<>(type);
}
@Override
public void serialize(T value, JsonGenerator generator) {
if (value instanceof JsonpSerializable) {
((JsonpSerializable)value).serialize(generator, this);
return;
}
// JSON-B doesn't offer a way to serialize to a JSON generator, so we have to roundtrip via a string representation.
CharArrayWriter caw = new CharArrayWriter();
jsonb.toJson(value, caw);
CharArrayReader car = new CharArrayReader(caw.toCharArray());
JsonParser parser = jsonProvider.createParser(car);
transferAll(parser, generator);
}
@Override
public JsonProvider jsonProvider() {
return jsonProvider;
}
private class Deserializer extends JsonpDeserializerBase {
private final Type type;
Deserializer(Type type) {
super(EnumSet.allOf(Event.class));
this.type = type;
}
@Override
public T deserialize(JsonParser parser, JsonpMapper mapper, Event event) {
// TODO: Add a runtime check to use Yasson's JsonB extensions
// JsonB doesn't provide methods to deserialize from a JsonParser or a JsonValue. We therefore have
// to roundtrip through a string, which is far from efficient. Yasson addresses this with an additional
// `YassonJsonb` that extends the base JSON-B interface with additional mapping functions. We should check
// here at runtime if the mapper implements this interface and use it if present.
CharArrayWriter caw = new CharArrayWriter();
JsonGenerator generator = jsonProvider.createGenerator(caw);
transferAll(parser, event, generator);
generator.close();
CharArrayReader car = new CharArrayReader(caw.toCharArray());
return jsonb.fromJson(car, type);
}
}
private void transferAll(JsonParser from, JsonGenerator to) {
transferAll(from, from.next(), to);
}
/**
* Pipe a JSON parser to a JSON generator.
*/
private void transferAll(JsonParser from, Event event, JsonGenerator to) {
transferEvent(from, event, to);
switch (event) {
case START_OBJECT -> {
int depth = 1;
do {
event = from.next();
transferEvent(from, event, to);
switch (event) {
case START_OBJECT -> depth++;
case END_OBJECT -> depth--;
}
} while (!(event == Event.END_OBJECT && depth == 0));
}
case START_ARRAY -> {
int depth = 1;
do {
event = from.next();
transferEvent(from, event, to);
switch (event) {
case START_ARRAY -> depth++;
case END_ARRAY -> depth--;
}
} while (!(event == Event.END_ARRAY && depth == 0));
}
default -> {
}
// nothing more
}
}
/**
* Transfer a single event from a parser to a generator
*/
private void transferEvent(JsonParser from, Event event, JsonGenerator to) {
switch (event) {
case START_OBJECT:
to.writeStartObject();
break;
case START_ARRAY:
to.writeStartArray();
break;
case END_OBJECT:
case END_ARRAY:
to.writeEnd();
break;
case KEY_NAME:
to.writeKey(from.getString());
break;
case VALUE_STRING:
to.write(from.getString());
break;
case VALUE_NUMBER:
if (from.isIntegralNumber()) {
to.write(from.getLong());
} else {
to.write(from.getBigDecimal());
}
break;
case VALUE_NULL:
to.writeNull();
break;
case VALUE_TRUE:
to.write(true);
break;
case VALUE_FALSE:
to.write(false);
break;
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy