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

org.babyfish.jimmer.sql.runtime.ReaderManager Maven / Gradle / Ivy

There is a newer version: 0.8.180
Show newest version
package org.babyfish.jimmer.sql.runtime;

import org.apache.commons.lang3.ArrayUtils;
import org.babyfish.jimmer.DraftConsumerUncheckedException;
import org.babyfish.jimmer.impl.util.CollectionUtils;
import org.babyfish.jimmer.sql.Serialized;
import org.babyfish.jimmer.impl.util.PropCache;
import org.babyfish.jimmer.impl.util.TypeCache;
import org.babyfish.jimmer.meta.*;
import org.babyfish.jimmer.runtime.DraftSpi;
import org.babyfish.jimmer.sql.association.Association;
import org.babyfish.jimmer.sql.association.meta.AssociationType;
import org.babyfish.jimmer.sql.dialect.Dialect;
import org.babyfish.jimmer.sql.meta.ColumnDefinition;
import org.babyfish.jimmer.sql.meta.FormulaTemplate;
import org.babyfish.jimmer.sql.meta.SqlTemplate;
import org.babyfish.jimmer.sql.meta.Storage;

import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;
import java.math.BigDecimal;
import java.math.BigInteger;
import java.nio.ByteBuffer;
import java.sql.Blob;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.sql.Timestamp;
import java.time.*;
import java.util.*;

public class ReaderManager {

    private static final Map, Reader> BASE_READER_MAP;

    private static final Map, Reader> SIMPLE_LIST_READER_MAP;

    private final JSqlClientImplementor sqlClient;

    private final TypeCache> typeReaderCache =
            new TypeCache<>(this::createTypeReader, true);

    private final PropCache> propReaderCache =
            new PropCache<>(this::createPropReader, true);

    public ReaderManager(JSqlClientImplementor sqlClient) {
        this.sqlClient = sqlClient;
    }

    public Reader reader(Class type) {
        ImmutableType immutableType = ImmutableType.tryGet(type);
        return immutableType != null ? reader(immutableType) : scalarReader(type);
    }

    public Reader reader(ImmutableType type) {
        return typeReaderCache.get(type);
    }

    public Reader reader(ImmutableProp prop) {
        return propReaderCache.get(prop);
    }

    @SuppressWarnings("unchecked")
    private Reader createPropReader(ImmutableProp prop) {

        Storage storage = prop.getStorage(sqlClient.getMetadataStrategy());
        if (storage instanceof ColumnDefinition) {
            if (prop.isEmbedded(EmbeddedLevel.SCALAR)) {
                return new FixedEmbeddedReader(prop.getTargetType(), this);
            }
            if (prop.isReference(TargetLevel.ENTITY)) {
                return new ReferenceReader(prop, this);
            }
            return scalarReader(prop);
        } else if (prop.getDeclaringType().isEmbeddable()) {
            return scalarReader(prop);
        }
        SqlTemplate template = prop.getSqlTemplate();
        if (template instanceof FormulaTemplate) {
            return scalarReader(prop);
        }
        return null;
    }

    private Reader createTypeReader(ImmutableType immutableType) {
        if (immutableType.isEmbeddable()) {
            return new FixedEmbeddedReader(immutableType, this);
        }
        if (immutableType instanceof AssociationType) {
            return new AssociationReader((AssociationType) immutableType, this);
        }
        if (!immutableType.isEntity()) {
            return null;
        }
        Map> nonIdReaderMap = new LinkedHashMap<>();
        Reader idReader = null;
        for (ImmutableProp prop : immutableType.getSelectableProps().values()) {
            if (prop.isId()) {
                idReader = reader(prop);
            } else {
                nonIdReaderMap.put(prop, reader(prop));
            }
        }
        return new ObjectReader(immutableType, idReader, nonIdReaderMap);
    }

    @SuppressWarnings("unchecked")
    private Reader scalarReader(ImmutableProp prop) {
        ImmutableType immutableType = prop.getTargetType();
        if (immutableType != null && immutableType.isEmbeddable()) {
            return new FixedEmbeddedReader(immutableType, this);
        }
        ScalarProvider scalarProvider = sqlClient.getScalarProvider(prop);
        if (scalarProvider != null) {
            Class sqlType = scalarProvider.getSqlType();
            Reader reader = baseReader(sqlType);
            if (reader == null) {
                reader = unknownSqlTypeReader(sqlType, scalarProvider, sqlClient.getDialect());
            }
            reader = new CustomizedScalarReader<>(
                    scalarProvider,
                    (Reader) reader
            );
            return reader;
        }
        if (sqlClient.getDialect().isArraySupported()) {
            Class returnClass = prop.getReturnClass();
            if (prop.getAnnotation(Serialized.class) == null && (returnClass == List.class || returnClass == Collection.class)) {
                Type genericType = prop.getGenericType();
                if (genericType instanceof ParameterizedType) {
                    Type argumentType = ((ParameterizedType) genericType).getActualTypeArguments()[0];
                    if (argumentType instanceof Class) {
                        Reader reader = SIMPLE_LIST_READER_MAP.get((Class) argumentType);
                        if (reader != null) {
                            return reader;
                        }
                    }
                }
            }
        }
        return scalarReader(prop.getReturnClass());
    }

    @SuppressWarnings("unchecked")
    private Reader scalarReader(Class type) {
        ScalarProvider scalarProvider = sqlClient.getScalarProvider(type);
        if (scalarProvider != null) {
            Class sqlType = scalarProvider.getSqlType();
            Reader reader = baseReader(sqlType);
            if (reader == null) {
                reader = unknownSqlTypeReader(sqlType, scalarProvider, sqlClient.getDialect());
            }
            reader = new CustomizedScalarReader<>(
                    (ScalarProvider) scalarProvider,
                    (Reader) reader
            );
            return reader;
        }
        ImmutableType immutableType = ImmutableType.tryGet(type);
        if (immutableType != null && immutableType.isEmbeddable()) {
            return new FixedEmbeddedReader(immutableType, this);
        }
        Reader reader = baseReader(type);
        if (reader == null) {
            throw new IllegalArgumentException(
                    "No scalar provider for customized scalar type \"" +
                            type.getName() +
                            "\""
            );
        }
        return reader;
    }

    private Reader baseReader(Class type) {
        if (type.isArray()) {
            return type == byte[].class || sqlClient.getDialect().isArraySupported() ?
                    BASE_READER_MAP.get(type) :
                    null;
        }
        return BASE_READER_MAP.get(type);
    }

    private static Reader unknownSqlTypeReader(
            Class sqlType,
            ScalarProvider provider,
            Dialect dialect
    ) {
        Reader reader = provider.reader();
        if (reader == null) {
            reader = dialect.unknownReader(sqlType);
            if (reader == null) {
                throw new IllegalStateException(
                        "There is no reader for unknown type \"" +
                                sqlType.getName() +
                                "\" in both \"" +
                                ScalarProvider.class.getName() +
                                "\" and \"" +
                                dialect.getClass().getName() +
                                "\""
                );
            }
        }
        return reader;
    }

    private static class ByteArrayReader implements Reader {

        @Override
        public byte[] read(ResultSet rs, Context ctx) throws SQLException {
            return rs.getBytes(ctx.col());
        }
    }

    private static class BoxedByteArrayReader implements Reader {

        @Override
        public Byte[] read(ResultSet rs, Context ctx) throws SQLException {
            return ctx.getDialect().getArray(rs, ctx.col(), Byte[].class);
        }
    }

    private static class ByteListReader implements Reader> {

        @Override
        public List read(ResultSet rs, Context ctx) throws SQLException {
            return CollectionUtils.toListOrNull(ctx.getDialect().getArray(rs, ctx.col(), Byte[].class));
        }
    }

    private static class ShortArrayReader implements Reader {

        @Override
        public short[] read(ResultSet rs, Context ctx) throws SQLException {
            return ArrayUtils.toPrimitive(ctx.getDialect().getArray(rs, ctx.col(), Short[].class));
        }
    }

    private static class BoxedShortArrayReader implements Reader {

        @Override
        public Short[] read(ResultSet rs, Context ctx) throws SQLException {
            return ctx.getDialect().getArray(rs, ctx.col(), Short[].class);
        }
    }

    private static class ShortListReader implements Reader> {

        @Override
        public List read(ResultSet rs, Context ctx) throws SQLException {
            return CollectionUtils.toListOrNull(ctx.getDialect().getArray(rs, ctx.col(), Short[].class));
        }
    }

    private static class IntArrayReader implements Reader {

        @Override
        public int[] read(ResultSet rs, Context ctx) throws SQLException {
            return ArrayUtils.toPrimitive(ctx.getDialect().getArray(rs, ctx.col(), Integer[].class));
        }
    }

    private static class BoxedIntArrayReader implements Reader {

        @Override
        public Integer[] read(ResultSet rs, Context ctx) throws SQLException {
            return ctx.getDialect().getArray(rs, ctx.col(), Integer[].class);
        }
    }

    private static class IntListReader implements Reader> {

        @Override
        public List read(ResultSet rs, Context ctx) throws SQLException {
            return CollectionUtils.toListOrNull(ctx.getDialect().getArray(rs, ctx.col(), Integer[].class));
        }
    }

    private static class LongArrayReader implements Reader {

        @Override
        public long[] read(ResultSet rs, Context ctx) throws SQLException {
            return ArrayUtils.toPrimitive(ctx.getDialect().getArray(rs, ctx.col(), Long[].class));
        }
    }

    private static class BoxedLongArrayReader implements Reader {

        @Override
        public Long[] read(ResultSet rs, Context ctx) throws SQLException {
            return ctx.getDialect().getArray(rs, ctx.col(), Long[].class);
        }
    }

    private static class LongListReader implements Reader> {

        @Override
        public List read(ResultSet rs, Context ctx) throws SQLException {
            return CollectionUtils.toListOrNull(ctx.getDialect().getArray(rs, ctx.col(), Long[].class));
        }
    }

    private static class FloatArrayReader implements Reader {

        @Override
        public float[] read(ResultSet rs, Context ctx) throws SQLException {
            return ArrayUtils.toPrimitive(ctx.getDialect().getArray(rs, ctx.col(), Float[].class));
        }
    }

    private static class BoxedFloatArrayReader implements Reader {

        @Override
        public Float[] read(ResultSet rs, Context ctx) throws SQLException {
            return ctx.getDialect().getArray(rs, ctx.col(), Float[].class);
        }
    }

    private static class FloatListReader implements Reader> {

        @Override
        public List read(ResultSet rs, Context ctx) throws SQLException {
            return CollectionUtils.toListOrNull(ctx.getDialect().getArray(rs, ctx.col(), Float[].class));
        }
    }

    private static class DoubleArrayReader implements Reader {

        @Override
        public double[] read(ResultSet rs, Context ctx) throws SQLException {
            return ArrayUtils.toPrimitive(ctx.getDialect().getArray(rs, ctx.col(), Double[].class));
        }
    }

    private static class BoxedDoubleArrayReader implements Reader {

        @Override
        public Double[] read(ResultSet rs, Context ctx) throws SQLException {
            return ctx.getDialect().getArray(rs, ctx.col(), Double[].class);
        }
    }

    private static class DoubleListReader implements Reader> {

        @Override
        public List read(ResultSet rs, Context ctx) throws SQLException {
            return CollectionUtils.toListOrNull(ctx.getDialect().getArray(rs, ctx.col(), Double[].class));
        }
    }

    private static class StringArrayReader implements Reader {

        @Override
        public String[] read(ResultSet rs, Context ctx) throws SQLException {
            return ctx.getDialect().getArray(rs, ctx.col(), String[].class);
        }
    }

    private static class StringListReader implements Reader> {

        @Override
        public List read(ResultSet rs, Context ctx) throws SQLException {
            return CollectionUtils.toListOrNull(ctx.getDialect().getArray(rs, ctx.col(), String[].class));
        }
    }

    private static class UUIDArrayReader implements Reader {

        @Override
        public UUID[] read(ResultSet rs, Context ctx) throws SQLException {
            return ctx.getDialect().getArray(rs, ctx.col(), UUID[].class);
        }
    }

    private static class UUIDListReader implements Reader> {

        @Override
        public List read(ResultSet rs, Context ctx) throws SQLException {
            return CollectionUtils.toListOrNull(ctx.getDialect().getArray(rs, ctx.col(), UUID[].class));
        }
    }

    private static class BooleanReader implements Reader {

        @Override
        public Boolean read(ResultSet rs, Context ctx) throws SQLException {
            boolean value = rs.getBoolean(ctx.col());
            if (!value && rs.wasNull()) {
                return null;
            }
            return value;
        }
    }

    private static class CharReader implements Reader {

        @Override
        public Character read(ResultSet rs, Context ctx) throws SQLException {
            String str = rs.getString(ctx.col());
            return str != null ? str.charAt(0) : null;
        }
    }

    private static class ByteReader implements Reader {

        @Override
        public Byte read(ResultSet rs, Context ctx) throws SQLException {
            byte value = rs.getByte(ctx.col());
            if (value == 0 && rs.wasNull()) {
                return null;
            }
            return value;
        }
    }

    private static class ShortReader implements Reader {

        @Override
        public Short read(ResultSet rs, Context ctx) throws SQLException {
            short value = rs.getShort(ctx.col());
            if (value == 0 && rs.wasNull()) {
                return null;
            }
            return value;
        }
    }

    private static class IntReader implements Reader {

        @Override
        public Integer read(ResultSet rs, Context ctx) throws SQLException {
            int value = rs.getInt(ctx.col());
            if (value == 0 && rs.wasNull()) {
                return null;
            }
            return value;
        }
    }

    private static class LongReader implements Reader {

        @Override
        public Long read(ResultSet rs, Context ctx) throws SQLException {
            long value = rs.getLong(ctx.col());
            if (value == 0 && rs.wasNull()) {
                return null;
            }
            return value;
        }
    }

    private static class FloatReader implements Reader {

        @Override
        public Float read(ResultSet rs, Context ctx) throws SQLException {
            float value = rs.getFloat(ctx.col());
            if (value == 0 && rs.wasNull()) {
                return null;
            }
            return value;
        }
    }

    private static class DoubleReader implements Reader {

        @Override
        public Double read(ResultSet rs, Context ctx) throws SQLException {
            double value = rs.getDouble(ctx.col());
            if (value == 0 && rs.wasNull()) {
                return null;
            }
            return value;
        }
    }

    private static class BigIntegerReader implements Reader {

        @Override
        public BigInteger read(ResultSet rs, Context ctx) throws SQLException {
            BigDecimal decimal = rs.getBigDecimal(ctx.col());
            return decimal != null ? decimal.toBigInteger() : null;
        }
    }

    private static class BigDecimalReader implements Reader {

        @Override
        public BigDecimal read(ResultSet rs, Context ctx) throws SQLException {
            return rs.getBigDecimal(ctx.col());
        }
    }

    private static class StringReader implements Reader {

        @Override
        public String read(ResultSet rs, Context ctx) throws SQLException {
            return rs.getString(ctx.col());
        }
    }

    private static class UUIDReader implements Reader {

        @Override
        public UUID read(ResultSet rs, Context ctx) throws SQLException {
            Object obj = rs.getObject(ctx.col());
            if (obj == null) {
                return null;
            }
            if (obj instanceof byte[]) {
                ByteBuffer byteBuffer = ByteBuffer.wrap((byte[]) obj);
                long high = byteBuffer.getLong();
                long low = byteBuffer.getLong();
                return new UUID(high, low);
            }
            return UUID.fromString(obj.toString());
        }
    }

    private static class BlobReader implements Reader {

        @Override
        public Blob read(ResultSet rs, Context ctx) throws SQLException {
            return rs.getBlob(ctx.col());
        }
    }

    private static class SqlDateReader implements Reader {

        @Override
        public java.sql.Date read(ResultSet rs, Context ctx) throws SQLException {
            return rs.getDate(ctx.col());
        }
    }

    private static class SqlTimeReader implements Reader {

        @Override
        public java.sql.Time read(ResultSet rs, Context ctx) throws SQLException {
            return rs.getTime(ctx.col());
        }
    }

    private static class SqlTimestampReader implements Reader {

        @Override
        public Timestamp read(ResultSet rs, Context ctx) throws SQLException {
            return rs.getTimestamp(ctx.col());
        }
    }

    private static class DateReader implements Reader {
        @Override
        public java.util.Date read(ResultSet rs, Context ctx) throws SQLException {
            Timestamp timestamp = rs.getTimestamp(ctx.col());
            return timestamp != null ?
                    java.util.Date.from(timestamp.toInstant()) :
                    null;
        }
    }

    private static class LocalDateReader implements Reader {

        @Override
        public LocalDate read(ResultSet rs, Context ctx) throws SQLException {
            Timestamp timestamp = rs.getTimestamp(ctx.col());
            return timestamp != null ?
                    LocalDateTime.ofInstant(timestamp.toInstant(), ctx.getZoneId()).toLocalDate() :
                    null;
        }
    }

    private static class LocalTimeReader implements Reader {

        @Override
        public LocalTime read(ResultSet rs, Context ctx) throws SQLException {
            Timestamp timestamp = rs.getTimestamp(ctx.col());
            return timestamp != null ?
                    LocalDateTime.ofInstant(timestamp.toInstant(), ctx.getZoneId()).toLocalTime() :
                    null;
        }
    }

    private static class LocalDateTimeReader implements Reader {

        @Override
        public LocalDateTime read(ResultSet rs, Context ctx) throws SQLException {
            Timestamp timestamp = rs.getTimestamp(ctx.col());
            return timestamp != null ?
                    LocalDateTime.ofInstant(timestamp.toInstant(), ctx.getZoneId()) :
                    null;
        }
    }

    private static class OffsetDateTimeReader implements Reader {

        @Override
        public OffsetDateTime read(ResultSet rs, Context ctx) throws SQLException {
            Timestamp timestamp = rs.getTimestamp(ctx.col());
            return timestamp != null ?
                    OffsetDateTime.ofInstant(timestamp.toInstant(), ctx.getZoneId()) :
                    null;
        }
    }

    private static class ZonedDateTimeReader implements Reader {

        @Override
        public ZonedDateTime read(ResultSet rs, Context ctx) throws SQLException {
            Timestamp timestamp = rs.getTimestamp(ctx.col());
            return timestamp != null ?
                    ZonedDateTime.ofInstant(timestamp.toInstant(), ctx.getZoneId()) :
                    null;
        }
    }

    private static class InstantReader implements Reader {

        @Override
        public Instant read(ResultSet rs, Context ctx) throws SQLException {
            Timestamp timestamp = rs.getTimestamp(ctx.col());
            return timestamp != null ? timestamp.toInstant() : null;
        }
    }

    private static class CustomizedScalarReader implements Reader {

        private final ScalarProvider scalarProvider;

        private final Reader sqlReader;

        CustomizedScalarReader(ScalarProvider scalarProvider, Reader sqlReader) {
            this.scalarProvider = scalarProvider;
            this.sqlReader = sqlReader;
        }

        @Override
        public T read(ResultSet rs, Context ctx) throws SQLException {
            S sqlValue = sqlReader.read(rs, ctx);
            try {
                return sqlValue != null ? scalarProvider.toScalar(sqlValue) : null;
            } catch (Exception ex) {
                throw new ExecutionException(
                        "Cannot convert \"" +
                                sqlValue +
                                "\" to the jvm type \"" +
                                scalarProvider.getScalarType() +
                                "\" by scalar provider \"" +
                                scalarProvider +
                                "\"",
                        ex
                );
            }
        }
    }

    private static class ReferenceReader implements Reader {

        private final ImmutableType targetType;

        private final Reader foreignKeyReader;

        private ReferenceReader(ImmutableProp prop, ReaderManager readerManager) {
            this.targetType = prop.getTargetType();
            this.foreignKeyReader = readerManager.scalarReader(targetType.getIdProp());
        }

        @Override
        public Object read(ResultSet rs, Context ctx) throws SQLException {
            Object fk = foreignKeyReader.read(rs, ctx);
            if (fk == null) {
                return null;
            }
            DraftSpi spi = (DraftSpi) targetType.getDraftFactory().apply(ctx.draftContext(), null);
            try {
                spi.__set(targetType.getIdProp().getId(), fk);
            } catch (Throwable ex) {
                throw DraftConsumerUncheckedException.rethrow(ex);
            }
            return ctx.resolve(spi);
        }
    }

    private static class AssociationReader implements Reader> {

        private final Reader sourceReader;

        private final Reader targetReader;

        AssociationReader(AssociationType associationType, ReaderManager readerManager) {
            sourceReader = new ReferenceReader(associationType.getSourceProp(), readerManager);
            targetReader = new ReferenceReader(associationType.getTargetProp(), readerManager);
        }

        @Override
        public Association read(ResultSet rs, Context ctx) throws SQLException {
            Object source = sourceReader.read(rs, ctx);
            Object target = targetReader.read(rs, ctx);
            return new Association<>(source, target);
        }
    }

    private static class FixedEmbeddedReader implements Reader {

        private static final ImmutableProp[] EMPTY_PROPS = new ImmutableProp[0];

        private static final Reader[] EMPTY_READERS = new Reader[0];

        private final ImmutableType targetType;

        private ImmutableProp[] props;

        private Reader[] readers;

        FixedEmbeddedReader(ImmutableType targetType, ReaderManager readerManager) {
            this.targetType = targetType;
            Map> map = new LinkedHashMap<>();
            for (ImmutableProp childProp : targetType.getProps().values()) {
                if (childProp.isEmbedded(EmbeddedLevel.SCALAR)) {
                    map.put(childProp, new FixedEmbeddedReader(childProp.getTargetType(), readerManager));
                } else if (!childProp.isFormula()) {
                    assert childProp.getSqlTemplate() == null; // SQL formula is not supported by embeddable
                    map.put(childProp, readerManager.scalarReader(childProp));
                }
            }
            props = map.keySet().toArray(EMPTY_PROPS);
            readers = map.values().toArray(EMPTY_READERS);
        }

        @Override
        public Object read(ResultSet rs, Context ctx) throws SQLException {
            DraftSpi spi = (DraftSpi) targetType.getDraftFactory().apply(ctx.draftContext(), null);
            boolean hasNoNull = false;
            boolean hasRequiredNull = false;
            try {
                int size = readers.length;
                for (int i = 0; i < size; i++) {
                    Object value = readers[i].read(rs, ctx);
                    if (hasRequiredNull) {
                        continue;
                    }
                    ImmutableProp prop = props[i];
                    if (value == null) {
                        if (prop.isNullable()) {
                            spi.__set(prop.getId(), null);
                        } else {
                            hasRequiredNull = true;
                        }
                    } else {
                        spi.__set(prop.getId(), value);
                        hasNoNull = true;
                    }
                }
            } catch (Throwable ex) {
                return DraftConsumerUncheckedException.rethrow(ex);
            }
            return hasNoNull && !hasRequiredNull ? ctx.resolve(spi) : null;
        }
    }

    public static boolean isStandardScalarType(Class type) {
        return BASE_READER_MAP.containsKey(type);
    }

    static {
        Map, Reader> baseReaderMap = new HashMap<>();
        baseReaderMap.put(boolean.class, new BooleanReader());
        baseReaderMap.put(Boolean.class, new BooleanReader());
        baseReaderMap.put(char.class, new CharReader());
        baseReaderMap.put(Character.class, new CharReader());
        baseReaderMap.put(byte.class, new ByteReader());
        baseReaderMap.put(Byte.class, new ByteReader());
        baseReaderMap.put(byte[].class, new ByteArrayReader());
        baseReaderMap.put(Byte[].class, new BoxedByteArrayReader());
        baseReaderMap.put(short.class, new ShortReader());
        baseReaderMap.put(Short.class, new ShortReader());
        baseReaderMap.put(short[].class, new ShortArrayReader());
        baseReaderMap.put(Short[].class, new BoxedShortArrayReader());
        baseReaderMap.put(int.class, new IntReader());
        baseReaderMap.put(Integer.class, new IntReader());
        baseReaderMap.put(int[].class, new IntArrayReader());
        baseReaderMap.put(Integer[].class, new BoxedIntArrayReader());
        baseReaderMap.put(long.class, new LongReader());
        baseReaderMap.put(Long.class, new LongReader());
        baseReaderMap.put(long[].class, new LongArrayReader());
        baseReaderMap.put(Long[].class, new BoxedLongArrayReader());
        baseReaderMap.put(float.class, new FloatReader());
        baseReaderMap.put(Float.class, new FloatReader());
        baseReaderMap.put(float[].class, new FloatArrayReader());
        baseReaderMap.put(Float[].class, new BoxedFloatArrayReader());
        baseReaderMap.put(double.class, new DoubleReader());
        baseReaderMap.put(Double.class, new DoubleReader());
        baseReaderMap.put(double[].class, new DoubleArrayReader());
        baseReaderMap.put(Double[].class, new BoxedDoubleArrayReader());
        baseReaderMap.put(BigInteger.class, new BigIntegerReader());
        baseReaderMap.put(BigDecimal.class, new BigDecimalReader());
        baseReaderMap.put(String.class, new StringReader());
        baseReaderMap.put(String[].class, new StringArrayReader());
        baseReaderMap.put(UUID.class, new UUIDReader());
        baseReaderMap.put(UUID[].class, new UUIDArrayReader());
        baseReaderMap.put(Blob.class, new BlobReader());
        baseReaderMap.put(java.sql.Date.class, new SqlDateReader());
        baseReaderMap.put(java.sql.Time.class, new SqlTimeReader());
        baseReaderMap.put(java.sql.Timestamp.class, new SqlTimestampReader());
        baseReaderMap.put(java.util.Date.class, new DateReader());
        baseReaderMap.put(LocalDate.class, new LocalDateReader());
        baseReaderMap.put(LocalTime.class, new LocalTimeReader());
        baseReaderMap.put(LocalDateTime.class, new LocalDateTimeReader());
        baseReaderMap.put(OffsetDateTime.class, new OffsetDateTimeReader());
        baseReaderMap.put(ZonedDateTime.class, new ZonedDateTimeReader());
        baseReaderMap.put(Instant.class, new InstantReader());
        BASE_READER_MAP = baseReaderMap;

        Map, Reader> simpleListReaderMap = new HashMap<>();
        simpleListReaderMap.put(Byte.class, new ByteListReader());
        simpleListReaderMap.put(Short.class, new ShortListReader());
        simpleListReaderMap.put(Integer.class, new IntListReader());
        simpleListReaderMap.put(Long.class, new LongListReader());
        simpleListReaderMap.put(Float.class, new FloatListReader());
        simpleListReaderMap.put(Double.class, new DoubleListReader());
        simpleListReaderMap.put(String.class, new StringListReader());
        simpleListReaderMap.put(UUID.class, new UUIDListReader());
        SIMPLE_LIST_READER_MAP = simpleListReaderMap;
    }
}