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

org.embulk.spi.ColumnJacksonModule Maven / Gradle / Ivy

package org.embulk.spi;

import com.fasterxml.jackson.core.JsonGenerator;
import com.fasterxml.jackson.core.JsonParser;
import com.fasterxml.jackson.core.JsonProcessingException;
import com.fasterxml.jackson.databind.DeserializationContext;
import com.fasterxml.jackson.databind.JsonMappingException;
import com.fasterxml.jackson.databind.JsonNode;
import com.fasterxml.jackson.databind.JsonSerializer;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.SerializerProvider;
import com.fasterxml.jackson.databind.deser.std.StdNodeBasedDeserializer;
import com.fasterxml.jackson.databind.module.SimpleModule;
import com.fasterxml.jackson.databind.node.ObjectNode;
import java.io.IOException;
import java.util.Collections;
import java.util.HashMap;
import java.util.Map;
import org.embulk.spi.type.Type;
import org.embulk.spi.type.Types;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

public final class ColumnJacksonModule extends SimpleModule {
    public ColumnJacksonModule() {
        this.addSerializer(Column.class, new ColumnSerializer());
        this.addDeserializer(Column.class, new ColumnDeserializer());
    }

    private static class ColumnSerializer extends JsonSerializer {
        @Override
        public void serialize(
                final Column value,
                final JsonGenerator jsonGenerator,
                final SerializerProvider provider)
                throws IOException {
            final ObjectNode object = OBJECT_MAPPER.createObjectNode();
            object.put("index", value.getIndex());
            object.put("name", value.getName());
            object.put("type", value.getType().getName());
            jsonGenerator.writeTree(object);
        }
    }

    private static class ColumnDeserializer extends StdNodeBasedDeserializer {
        protected ColumnDeserializer() {
            super(Column.class);
        }

        @Override
        public Column convert(
                final JsonNode root,
                final DeserializationContext context)
                throws JsonProcessingException {
            return convertJsonNodeToColumn(root, context.getParser());
        }
    }

    static final Column convertJsonNodeToColumn(final JsonNode root, final JsonParser jsonParser) throws JsonProcessingException {
        if (root == null || !root.isObject()) {
            throw JsonMappingException.from(jsonParser, "Column expects a JSON Object node.");
        }
        final ObjectNode object = (ObjectNode) root;

        final JsonNode indexNode = object.get("index");
        final int index;
        if (indexNode == null) {
            logger.warn("Building Column from JSON without \"index\".",
                        JsonMappingException.from(jsonParser, "Building Column from JSON without \"index\"."));
            index = 0;
        } else {
            index = OBJECT_MAPPER.treeToValue(indexNode, int.class);
        }

        final JsonNode nameNode = object.get("name");
        if (nameNode == null) {
            throw JsonMappingException.from(jsonParser, "Building Column from JSON without \"name\".");
        }
        final String name = OBJECT_MAPPER.treeToValue(nameNode, String.class);

        final JsonNode typeNode = object.get("type");
        if (typeNode == null) {
            throw JsonMappingException.from(jsonParser, "Building Column from JSON without \"type\".");
        }
        final String typeString = OBJECT_MAPPER.treeToValue(typeNode, String.class);

        if (!STRING_TO_TYPE.containsKey(typeString)) {
            throw JsonMappingException.from(jsonParser, "Building Column from JSON with unexpected type: " + typeString);
        }
        final Type type = STRING_TO_TYPE.get(typeString);

        return new Column(index, name, type);
    }

    static {
        final HashMap builder = new HashMap<>();
        builder.put(Types.BOOLEAN.getName(), Types.BOOLEAN);
        builder.put(Types.LONG.getName(), Types.LONG);
        builder.put(Types.DOUBLE.getName(), Types.DOUBLE);
        builder.put(Types.STRING.getName(), Types.STRING);
        builder.put(Types.TIMESTAMP.getName(), Types.TIMESTAMP);
        builder.put(Types.JSON.getName(), Types.JSON);
        STRING_TO_TYPE = Collections.unmodifiableMap(builder);
    }

    private static final Logger logger = LoggerFactory.getLogger(ColumnJacksonModule.class);

    private static final ObjectMapper OBJECT_MAPPER = new ObjectMapper();

    private static final Map STRING_TO_TYPE;
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy