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