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

com.amazonaws.athena.connector.lambda.serde.v2.LambdaFunctionExceptionSerDe Maven / Gradle / Ivy

Go to download

The Athena Query Federation SDK defines a set of interfaces and wire protocols that you can implement to enable Athena to delegate portions of it's query execution plan to code that you deploy/write.

There is a newer version: 2024.46.1
Show newest version
/*-
 * #%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.v2;

import com.amazonaws.athena.connector.lambda.serde.BaseDeserializer;
import com.amazonaws.services.lambda.invoke.LambdaFunctionException;
import com.fasterxml.jackson.core.JsonParser;
import com.fasterxml.jackson.databind.DeserializationContext;
import com.fasterxml.jackson.databind.JsonNode;
import com.google.common.base.Joiner;

import java.io.IOException;
import java.lang.reflect.Constructor;
import java.util.Iterator;
import java.util.LinkedList;
import java.util.List;

/**
 * Used strictly for deserialization only since we do not own {@link LambdaFunctionException} and never need-to/should serialize it.
 */
public class LambdaFunctionExceptionSerDe
{
    private static final Joiner COMMA_JOINER = Joiner.on(",");

    private static final String ERROR_TYPE_FIELD = "errorType";
    private static final String ERROR_MESSAGE_FIELD = "errorMessage";
    private static final String CAUSE_FIELD = "cause";
    private static final String STACK_TRACE_FIELD = "stackTrace";

    private LambdaFunctionExceptionSerDe() {}

    public static final class Deserializer extends BaseDeserializer
    {
        public Deserializer()
        {
            super(LambdaFunctionException.class);
        }

        @Override
        public LambdaFunctionException deserialize(JsonParser jparser, DeserializationContext ctxt)
                throws IOException
        {
            validateObjectStart(jparser.getCurrentToken());
            // readTree consumes object end token so skip validation
            return doDeserialize(jparser, ctxt);
        }

        @Override
        public LambdaFunctionException doDeserialize(JsonParser jparser, DeserializationContext ctxt)
                throws IOException
        {
            JsonNode root = jparser.getCodec().readTree(jparser);
            return recursiveParse(root);
        }

        private LambdaFunctionException recursiveParse(JsonNode root)
        {
            String errorType = getNullableStringValue(root, ERROR_TYPE_FIELD);
            String errorMessage = getNullableStringValue(root, ERROR_MESSAGE_FIELD);
            LambdaFunctionException cause = null;
            JsonNode causeNode = root.get(CAUSE_FIELD);
            if (causeNode != null) {
                cause = recursiveParse(causeNode);
            }
            List> stackTraces = new LinkedList<>();
            JsonNode stackTraceNode = root.get(STACK_TRACE_FIELD);
            if (stackTraceNode != null) {
                if (stackTraceNode.isArray()) {
                    Iterator elements = stackTraceNode.elements();
                    while (elements.hasNext()) {
                        List innerList = new LinkedList<>();
                        JsonNode element = elements.next();
                        if (element.isArray()) {
                            Iterator innerElements = element.elements();
                            while (innerElements.hasNext()) {
                                innerList.add(innerElements.next().asText());
                            }
                        }
                        else {
                            // emulate DeserializationFeature.ACCEPT_SINGLE_VALUE_AS_ARRAY
                            innerList.add(element.asText());
                        }
                        stackTraces.add(innerList);
                    }
                }
            }

            // HACK: LambdaFunctionException is only intended to be instantiated by Lambda server-side, so its constructors
            // are package-private or deprecated. Thus the need for reflection here. If the signature of the preferred
            // constructor does change, we fall back to the deprecated constructor (which requires us to append the stackTrace
            // to the errorMessage to not lose it). If the deprecated constructor is removed then this will not compile
            // and the appropriate adjustment can be made.
            try {
                Constructor constructor = LambdaFunctionException.class.getDeclaredConstructor(
                        String.class, String.class, LambdaFunctionException.class, List.class);
                constructor.setAccessible(true);
                return constructor.newInstance(errorType, errorMessage, cause, stackTraces);
            }
            catch (ReflectiveOperationException e) {
                return new LambdaFunctionException(appendStackTrace(errorMessage, stackTraces), false, errorType);
            }
        }

        private String getNullableStringValue(JsonNode parent, String field)
        {
            JsonNode child = parent.get(field);
            if (child != null) {
                return child.asText();
            }
            return null;
        }

        private String appendStackTrace(String errorMessage, List> stackTraces)
        {
            return errorMessage + ". Stack trace: " + stackTraces;
        }
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy