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

com.facebook.presto.verifier.checksum.ChecksumResult Maven / Gradle / Ivy

/*
 * Licensed 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 com.facebook.presto.verifier.checksum;

import com.facebook.presto.common.type.SqlVarbinary;
import com.fasterxml.jackson.annotation.JsonTypeInfo;
import com.fasterxml.jackson.core.JsonParser;
import com.fasterxml.jackson.core.JsonProcessingException;
import com.fasterxml.jackson.databind.DeserializationContext;
import com.fasterxml.jackson.databind.JsonNode;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.deser.std.StdDeserializer;
import com.fasterxml.jackson.databind.jsontype.impl.LaissezFaireSubTypeValidator;
import com.fasterxml.jackson.databind.module.SimpleModule;
import com.google.common.base.Suppliers;

import java.io.IOException;
import java.sql.ResultSet;
import java.sql.ResultSetMetaData;
import java.sql.SQLException;
import java.sql.Types;
import java.util.HashMap;
import java.util.Map;
import java.util.Optional;
import java.util.function.Supplier;

import static com.google.common.base.Preconditions.checkArgument;
import static com.google.common.base.Preconditions.checkState;
import static java.util.Collections.unmodifiableMap;
import static java.util.Objects.requireNonNull;

public class ChecksumResult
{
    private static final Supplier OBJECT_MAPPER_SUPPLIER = Suppliers.memoize(
            () -> {
                ObjectMapper mapper = new ObjectMapper();
                mapper.activateDefaultTyping(LaissezFaireSubTypeValidator.instance, ObjectMapper.DefaultTyping.NON_FINAL, JsonTypeInfo.As.WRAPPER_ARRAY);
                return mapper;
            });
    private static final String ROW_COUNT = "rowCount";
    private static final String CHECKSUMS = "checksums";
    private final long rowCount;
    private final Map checksums;

    public ChecksumResult(long rowCount, Map checksums)
    {
        this.rowCount = rowCount;
        this.checksums = unmodifiableMap(requireNonNull(checksums, "checksums is null"));
        for (Object checksum : checksums.values()) {
            checkArgument(
                    checksum == null || checksum instanceof Long || checksum instanceof Double || checksum instanceof SqlVarbinary,
                    "checksum can only be one of (null, Long, Double, SqlVarbinary), found %s",
                    checksum == null ? "null" : checksum.getClass().getName());
        }
    }

    public long getRowCount()
    {
        return rowCount;
    }

    public Map getChecksums()
    {
        return checksums;
    }

    public Object getChecksum(String columnName)
    {
        checkArgument(checksums.containsKey(columnName), "column does not exist: %s", columnName);
        return checksums.get(columnName);
    }

    public static Optional fromResultSet(ResultSet resultSet)
            throws SQLException
    {
        long rowCount = resultSet.getLong(1);
        ResultSetMetaData metaData = resultSet.getMetaData();
        Map checksums = new HashMap<>(metaData.getColumnCount());

        for (int i = 2; i <= metaData.getColumnCount(); i++) {
            String columnName = metaData.getColumnName(i);
            Object checksum = resultSet.getObject(i);
            if (checksum == null) {
                checksums.put(columnName, null);
                continue;
            }

            if (metaData.getColumnType(i) == Types.DOUBLE) {
                checkState(checksum instanceof Double, "Expecting double for column %s, found %s", columnName, checksum.getClass());
                checksums.put(columnName, checksum);
            }
            else if (metaData.getColumnType(i) == Types.BIGINT) {
                checkState(checksum instanceof Long, "Expecting bigint for column %s, found %s", columnName, checksum.getClass());
                checksums.put(columnName, checksum);
            }
            else {
                checkState(checksum instanceof byte[], "Expecting binary for column %s, found %s", columnName, checksum.getClass());
                checksums.put(columnName, new SqlVarbinary((byte[]) checksum));
            }
        }
        return Optional.of(new ChecksumResult(rowCount, checksums));
    }

    public static String toJson(ChecksumResult checksumResult)
    {
        requireNonNull(checksumResult, "checksumResult is null");
        try {
            String json = OBJECT_MAPPER_SUPPLIER.get()
                    .writerWithDefaultPrettyPrinter()
                    .writeValueAsString(checksumResult);
            return json;
        }
        catch (JsonProcessingException exception) {
            throw new RuntimeException(exception);
        }
    }

    public static ChecksumResult fromJson(String json)
    {
        try {
            ObjectMapper objectMapper = OBJECT_MAPPER_SUPPLIER.get();
            SimpleModule module = new SimpleModule();
            module.addDeserializer(ChecksumResult.class, new ChecksumResultDeserializer());
            module.addDeserializer(SqlVarbinary.class, new SqlVarbinaryDeserializer());
            objectMapper.registerModule(module);

            ChecksumResult checksumResult = objectMapper.readValue(json, ChecksumResult.class);

            return checksumResult;
        }
        catch (JsonProcessingException exception) {
            throw new RuntimeException(exception);
        }
    }

    static class SqlVarbinaryDeserializer
            extends StdDeserializer
    {
        public SqlVarbinaryDeserializer()
        {
            this(null);
        }

        public SqlVarbinaryDeserializer(Class vc)
        {
            super(vc);
        }

        @Override
        public SqlVarbinary deserialize(JsonParser jp, DeserializationContext ctxt)
                throws IOException
        {
            JsonNode node = jp.getCodec().readTree(jp);
            return new SqlVarbinary(jp.getCodec().treeToValue(node, byte[].class));
        }
    }

    static class ChecksumResultDeserializer
            extends StdDeserializer
    {
        public ChecksumResultDeserializer()
        {
            this(null);
        }

        public ChecksumResultDeserializer(Class vc)
        {
            super(vc);
        }

        @Override
        public ChecksumResult deserialize(JsonParser jp, DeserializationContext ctxt)
                throws IOException
        {
            JsonNode node = jp.getCodec().readTree(jp);
            return new ChecksumResult(node.get(ROW_COUNT).asLong(), jp.getCodec().treeToValue(node.get(CHECKSUMS), Map.class));
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy