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

com.amazonaws.athena.connector.lambda.serde.BaseDeserializer Maven / Gradle / Ivy

/*-
 * #%L
 * Amazon Athena Query Federation SDK
 * %%
 * Copyright (C) 2019 - 2020 Amazon Web Services
 * %%
 * 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.
 * #L%
 */
package com.amazonaws.athena.connector.lambda.serde;

import com.fasterxml.jackson.core.JsonGenerator;
import com.fasterxml.jackson.core.JsonParser;
import com.fasterxml.jackson.core.JsonToken;
import com.fasterxml.jackson.databind.DeserializationContext;
import com.fasterxml.jackson.databind.deser.std.StdDeserializer;
import com.fasterxml.jackson.databind.jsontype.TypeDeserializer;

import java.io.IOException;
import java.util.ArrayList;
import java.util.Collection;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

import static com.amazonaws.athena.connector.lambda.serde.BaseSerializer.TYPE_FIELD;

public abstract class BaseDeserializer extends StdDeserializer implements VersionedSerDe.Deserializer
{
    protected BaseDeserializer(Class clazz)
    {
        super(clazz);
    }

    @Override
    public T deserialize(JsonParser jparser, DeserializationContext ctxt)
            throws IOException
    {
        if (jparser.nextToken() != JsonToken.VALUE_NULL) {
            validateObjectStart(jparser.getCurrentToken());
            T result = doDeserialize(jparser, ctxt);
            validateObjectEnd(jparser);
            return result;
        }
        else {
            return null;
        }
    }

    @Override
    public Object deserializeWithType(JsonParser jp, DeserializationContext ctxt, TypeDeserializer typeDeserializer)
            throws IOException
    {
        // TODO leverage TypeDeserializer if it simplifies things
        return deserialize(jp, ctxt);
    }

    public abstract T doDeserialize(JsonParser jparser, DeserializationContext ctxt)
            throws IOException;

    /**
     * Helper used to extract named String fields from the json parser in a streaming fashion.
     *
     * @param jparser The parser to use for extraction.
     * @param expectedFieldName The expected name of the next field in the stream.
     * @return The string representation of the requested field.
     * @throws IOException If there is an error parsing the field.
     */
    protected String getNextStringField(JsonParser jparser, String expectedFieldName)
            throws IOException
    {
        assertFieldName(jparser, expectedFieldName);

        //move to the value token
        jparser.nextToken();
        return jparser.getValueAsString();
    }

    /**
     * Helper used to extract named boolean fields from the json parser in a streaming fashion.
     *
     * @param jparser The parser to use for extraction.
     * @param expectedFieldName The expected name of the next field in the stream.
     * @return The boolean representation of the requested field.
     * @throws IOException If there is an error parsing the field.
     */
    protected boolean getNextBoolField(final JsonParser jparser, final String expectedFieldName)
            throws IOException
    {
        assertFieldName(jparser, expectedFieldName);

        //move to the value token
        jparser.nextToken();
        return jparser.getValueAsBoolean();
    }

    /**
     * Helper used to extract named byte array fields from the json parser in a streaming fashion.
     *
     * @param jparser The parser to use for extraction.
     * @param expectedFieldName The expected name of the next field in the stream.
     * @return The byte array representation of the requested field.
     * @throws IOException If there is an error parsing the field.
     */
    protected byte[] getNextBinaryField(final JsonParser jparser, final String expectedFieldName)
            throws IOException
    {
        assertFieldName(jparser, expectedFieldName);

        //move to the value token
        jparser.nextToken();
        return jparser.getBinaryValue();
    }

    /**
     * Helper used to extract named integer fields from the json parser in a streaming fashion.
     *
     * @param jparser The parser to use for extraction.
     * @param expectedFieldName The expected name of the next field in the stream.
     * @return The integer representation of the requested field.
     * @throws IOException If there is an error parsing the field.
     */
    protected int getNextIntField(JsonParser jparser, String expectedFieldName)
            throws IOException
    {
        assertFieldName(jparser, expectedFieldName);

        //move to the value token
        jparser.nextToken();
        return jparser.getValueAsInt();
    }

    /**
     * Helper used to extract named long fields from the json parser in a streaming fashion.
     *
     * @param jparser The parser to use for extraction.
     * @param expectedFieldName The expected name of the next field in the stream.
     * @return The integer representation of the requested field.
     * @throws IOException If there is an error parsing the field.
     */
    protected long getNextLongField(JsonParser jparser, String expectedFieldName)
            throws IOException
    {
        assertFieldName(jparser, expectedFieldName);

        //move to the value token
        jparser.nextToken();
        return jparser.getValueAsLong();
    }

    /**
     * Helper used to help serialize a list of strings.
     *
     * @param jgen The json generator to use.
     * @param fieldname The name to associated to the resulting json array.
     * @param values The values to populate the array with.
     * @throws IOException If an error occurs while writing to the generator.
     */
    protected void writeStringArray(JsonGenerator jgen, String fieldname, Collection values)
            throws IOException
    {
        jgen.writeArrayFieldStart(fieldname);

        for (String nextElement : values) {
            jgen.writeString(nextElement);
        }

        jgen.writeEndArray();
    }

    /**
     * Helper used to read a list of strings.
     *
     * @param jparser The parser to read from.
     * @param expectedFieldName The expected name of the field containing the string array.
     * @return The contents of the array as a list.
     * @throws IOException If an error occurs while reading from the parser.
     */
    protected List getNextStringArray(JsonParser jparser, String expectedFieldName)
            throws IOException
    {
        assertFieldName(jparser, expectedFieldName);

        validateArrayStart(jparser);

        List result = new ArrayList<>();
        while (jparser.nextToken() != JsonToken.END_ARRAY) {
            result.add(jparser.getValueAsString());
        }

        return result;
    }

    /**
     * Helper used to read a Map of strings to strings.
     *
     * @param jparser The parser to read from.
     * @param expectedFieldName The expected name of the field containing the string map.
     * @return The contents of the array as a map.
     * @throws IOException If an error occurs while reading from the parser.
     */
    protected Map getNextStringMap(JsonParser jparser, String expectedFieldName)
            throws IOException
    {
        assertFieldName(jparser, expectedFieldName);

        validateObjectStart(jparser);

        Map result = new HashMap<>();
        while (jparser.nextToken() != JsonToken.END_OBJECT) {
            result.put(jparser.getCurrentName(), jparser.getValueAsString());
        }

        return result;
    }

    /**
     * Helper used to validate an expected field name.
     *
     * @param jparser The parser to read from.
     * @param expectedFieldName The expected name of the next field.
     * @throws IOException If an error occurs while reading from the parser.
     */
    protected void assertFieldName(JsonParser jparser, String expectedFieldName)
            throws IOException
    {
        jparser.nextToken();
        if (jparser.getCurrentToken() != JsonToken.FIELD_NAME) {
            throw new IllegalStateException("Expected field name token but got " + jparser.getCurrentToken());
        }
        String fieldNameToken = jparser.getCurrentName();
        if (expectedFieldName != null && !expectedFieldName.equals(fieldNameToken)) {
            throw new IllegalStateException("Unexpected field[" + fieldNameToken + "] while expecting[" + expectedFieldName + "]");
        }
    }

    /**
     * Helper used to validate that an object is starting (open brace) at the provided {@link JsonToken}.
     *
     * @param token The token expected to be an object start.
     * @throws IOException If an error occurs while reading from the parser.
     */
    protected void validateObjectStart(JsonToken token)
            throws IOException
    {
        if (!JsonToken.START_OBJECT.equals(token)) {
            throw new IllegalStateException("Expected " + JsonToken.START_OBJECT + " found " + token.asString());
        }
    }

    /**
     * Helper used to validate that an object is ending (close brace) at the next token.
     *
     * @param jparser The parser to read from.
     * @throws IOException If an error occurs while reading from the parser.
     */
    protected void validateObjectEnd(JsonParser jparser)
            throws IOException
    {
        if (!JsonToken.END_OBJECT.equals(jparser.nextToken())) {
            throw new IllegalStateException("Expected " + JsonToken.END_OBJECT + " found " + jparser.getText());
        }
    }

    /**
     * Helper used to validate that an array is starting (open bracket) at the next token.
     *
     * @param jparser The parser to read from.
     * @throws IOException If an error occurs while reading from the parser.
     */
    protected void validateArrayStart(JsonParser jparser)
            throws IOException
    {
        if (!JsonToken.START_ARRAY.equals(jparser.nextToken())) {
            throw new IllegalStateException("Expected " + JsonToken.START_ARRAY + " found " + jparser.getText());
        }
    }

    /**
     * Helper used to validate that an object is starting (open brace) at the next token.
     *
     * @param jparser The parser to read from.
     * @throws IOException If an error occurs while reading from the parser.
     */
    protected void validateObjectStart(JsonParser jparser)
            throws IOException
    {
        if (!JsonToken.START_OBJECT.equals(jparser.nextToken())) {
            throw new IllegalStateException("Expected " + JsonToken.START_OBJECT + " found " + jparser.getText());
        }
    }

    /**
     * Helper used to parse the type of the object (usually the class name).
     *
     * @param jparser The parser to use for extraction.
     * @return The type name of the object being deserialized.
     * @throws IOException If there is an error parsing the field.
     */
    protected String getType(JsonParser jparser)
            throws IOException
    {
        assertFieldName(jparser, TYPE_FIELD);
        jparser.nextToken();
        return jparser.getValueAsString();
    }

    /**
     * Helper used to skip to the end of the current object.  Useful for forwards compatibility.
     *
     * @param jparser The parser to use for extraction.
     * @throws IOException If there is an error parsing.
     */
    protected void ignoreRestOfObject(JsonParser jparser)
            throws IOException
    {
        if (jparser.getCurrentToken() == JsonToken.END_OBJECT) {
            return;
        }

        int open = 1;
        /* Since proper matching of start/end markers is handled
         * by nextToken(), we'll just count nesting levels here
         */
        while (true) {
            JsonToken t = jparser.nextToken();
            if (t == null) {
                throw new IllegalStateException("Expected " + JsonToken.END_OBJECT + " found " + jparser.getText());
            }
            if (t.isStructStart()) {
                ++open;
            }
            else if (t == JsonToken.END_OBJECT) {
                if (--open == 0) {
                    return;
                }
            }
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy