com.netflix.spinnaker.clouddriver.aws.security.sdkclient.AmazonClientInvocationHandler.java Source code

Java tutorial

Introduction

Here is the source code for com.netflix.spinnaker.clouddriver.aws.security.sdkclient.AmazonClientInvocationHandler.java

Source

/*
 * Copyright 2016 Netflix, Inc.
 *
 * 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.netflix.spinnaker.clouddriver.aws.security.sdkclient;

import com.amazonaws.AmazonServiceException;
import com.amazonaws.AmazonWebServiceRequest;
import com.amazonaws.services.autoscaling.model.*;
import com.amazonaws.services.cloudwatch.model.DescribeAlarmsRequest;
import com.amazonaws.services.cloudwatch.model.DescribeAlarmsResult;
import com.amazonaws.services.cloudwatch.model.MetricAlarm;
import com.amazonaws.services.ec2.model.*;
import com.amazonaws.services.ec2.model.Instance;
import com.amazonaws.services.elasticloadbalancing.model.DescribeLoadBalancersRequest;
import com.amazonaws.services.elasticloadbalancing.model.DescribeLoadBalancersResult;
import com.amazonaws.services.elasticloadbalancing.model.LoadBalancerDescription;
import com.amazonaws.services.elasticloadbalancingv2.AmazonElasticLoadBalancing;
import com.amazonaws.services.elasticloadbalancingv2.model.*;
import com.amazonaws.services.elasticloadbalancingv2.model.TargetGroup;
import com.fasterxml.jackson.annotation.JsonCreator;
import com.fasterxml.jackson.annotation.JsonProperty;
import com.fasterxml.jackson.databind.JavaType;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.google.common.collect.ImmutableMap;
import com.netflix.spectator.api.Id;
import com.netflix.spectator.api.Registry;
import com.netflix.spinnaker.clouddriver.aws.security.EddaTimeoutConfig;
import org.apache.http.HttpEntity;
import org.apache.http.HttpResponse;
import org.apache.http.HttpStatus;
import org.apache.http.client.HttpClient;
import org.apache.http.client.config.RequestConfig;
import org.apache.http.client.methods.HttpGet;
import org.apache.http.util.EntityUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.io.IOException;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.*;
import java.util.concurrent.TimeUnit;

public class AmazonClientInvocationHandler implements InvocationHandler {

    private static final Logger log = LoggerFactory.getLogger(AmazonClientInvocationHandler.class);

    public static final ThreadLocal<Long> lastModified = new ThreadLocal<>();

    private final String edda;
    private final HttpClient httpClient;
    private final Object delegate;
    private final String serviceName;
    private final ObjectMapper objectMapper;
    private final EddaTimeoutConfig eddaTimeoutConfig;
    private final Registry registry;
    private final Map<String, String> metricTags;

    public AmazonClientInvocationHandler(Object delegate, String serviceName, String edda, HttpClient httpClient,
            ObjectMapper objectMapper, EddaTimeoutConfig eddaTimeoutConfig, Registry registry,
            Map<String, String> metricTags) {
        this.edda = edda;
        this.httpClient = httpClient;
        this.objectMapper = objectMapper;
        this.delegate = delegate;
        this.serviceName = serviceName;
        this.eddaTimeoutConfig = eddaTimeoutConfig == null ? EddaTimeoutConfig.DEFAULT : eddaTimeoutConfig;
        this.registry = registry;
        this.metricTags = ImmutableMap.copyOf(metricTags);
    }

    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        final Id id = registry.createId("awsClientProxy.invoke", metricTags).withTag("method", method.getName());
        final long startTime = System.nanoTime();
        boolean wasDelegated = false;

        try {
            if (!eddaTimeoutConfig.getAlbEnabled()
                    && method.getDeclaringClass().equals(AmazonElasticLoadBalancing.class)) {
                throw new NoSuchMethodException();
            }
            Method thisMethod = this.getClass().getMethod(method.getName(),
                    args != null && args.length > 0 ? getClassArgs(args) : new Class[0]);
            return thisMethod.invoke(this, args);
        } catch (NoSuchMethodException e) {
            wasDelegated = true;
            try {
                return method.invoke(delegate, args);
            } catch (InvocationTargetException ite) {
                throw ite.getCause();
            }
        } finally {
            registry.timer(id.withTag("requestMode", wasDelegated ? "sdkClient" : "edda"))
                    .record(System.nanoTime() - startTime, TimeUnit.NANOSECONDS);
        }
    }

    static Class[] getClassArgs(Object[] args) {
        List<Class> classes = new ArrayList<>();
        for (Object object : args) {
            classes.add(object.getClass());
        }
        return classes.toArray(new Class[classes.size()]);
    }

    ////////////////////////////////////
    //
    // AmazonAutoScaling
    //
    ////////////////////////////////////
    public DescribeAutoScalingGroupsResult describeAutoScalingGroups() {
        return describeAutoScalingGroups(null);
    }

    public DescribeAutoScalingGroupsResult describeAutoScalingGroups(DescribeAutoScalingGroupsRequest request) {
        return new DescribeAutoScalingGroupsResult().withAutoScalingGroups(
                describe(request, "autoScalingGroupNames", "autoScalingGroups", AutoScalingGroup.class));
    }

    ////////////////////////////////////
    //
    // AmazonCloudWatch
    //
    ////////////////////////////////////
    public DescribeAlarmsResult describeAlarms() {
        return describeAlarms(null);
    }

    public DescribeAlarmsResult describeAlarms(DescribeAlarmsRequest request) {
        return new DescribeAlarmsResult()
                .withMetricAlarms(describe(request, "alarmNames", "alarms", MetricAlarm.class));
    }

    public DescribeScheduledActionsResult describeScheduledActions() {
        return describeScheduledActions(null);
    }

    public DescribeScheduledActionsResult describeScheduledActions(DescribeScheduledActionsRequest request) {
        return new DescribeScheduledActionsResult().withScheduledUpdateGroupActions(
                describe(request, "scheduledActionNames", "scheduledActions", ScheduledUpdateGroupAction.class));
    }

    public DescribePoliciesResult describePolicies() {
        return describePolicies(null);
    }

    public DescribePoliciesResult describePolicies(DescribePoliciesRequest request) {
        return new DescribePoliciesResult()
                .withScalingPolicies(describe(request, "policyNames", "scalingPolicies", ScalingPolicy.class));
    }

    ////////////////////////////////////
    //
    // AmazonEC2
    //
    ////////////////////////////////////
    public DescribeImagesResult describeImages() {
        return describeImages(null);
    }

    public DescribeImagesResult describeImages(DescribeImagesRequest request) {
        return new DescribeImagesResult().withImages(describe(request, "imageIds", "images", Image.class));
    }

    public DescribeInstancesResult describeInstances() {
        return describeInstances(null);
    }

    public DescribeInstancesResult describeInstances(DescribeInstancesRequest request) {
        return new DescribeInstancesResult().withReservations(new Reservation().withReservationId("1234")
                .withInstances(describe(request, "instanceIds", "../view/instances", Instance.class)));
    }

    public DescribeLaunchConfigurationsResult describeLaunchConfigurations() {
        return describeLaunchConfigurations(null);
    }

    public DescribeLaunchConfigurationsResult describeLaunchConfigurations(
            DescribeLaunchConfigurationsRequest request) {
        return new DescribeLaunchConfigurationsResult().withLaunchConfigurations(
                describe(request, "launchConfigurationNames", "launchConfigurations", LaunchConfiguration.class));
    }

    public DescribeReservedInstancesOfferingsResult describeReservedInstancesOfferings() {
        return describeReservedInstancesOfferings(null);
    }

    public DescribeReservedInstancesOfferingsResult describeReservedInstancesOfferings(
            DescribeReservedInstancesOfferingsRequest request) {
        return new DescribeReservedInstancesOfferingsResult().withReservedInstancesOfferings(describe(request,
                "reservedInstancesOfferingIds", "reservedInstancesOfferings", ReservedInstancesOffering.class));
    }

    public DescribeSecurityGroupsResult describeSecurityGroups() {
        return describeSecurityGroups(null);
    }

    public DescribeSecurityGroupsResult describeSecurityGroups(DescribeSecurityGroupsRequest request) {
        return new DescribeSecurityGroupsResult()
                .withSecurityGroups(describe(request, "groupIds", "securityGroups", SecurityGroup.class));
    }

    public DescribeSubnetsResult describeSubnets() {
        return describeSubnets(null);
    }

    public DescribeSubnetsResult describeSubnets(DescribeSubnetsRequest request) {
        return new DescribeSubnetsResult().withSubnets(describe(request, "subnetIds", "subnets", Subnet.class));
    }

    public DescribeVpcsResult describeVpcs() {
        return describeVpcs(null);
    }

    public DescribeVpcsResult describeVpcs(DescribeVpcsRequest request) {
        return new DescribeVpcsResult().withVpcs(describe(request, "vpcIds", "vpcs", Vpc.class));
    }

    public DescribeVpcClassicLinkResult describeVpcClassicLink() {
        return describeVpcClassicLink(null);
    }

    public DescribeVpcClassicLinkResult describeVpcClassicLink(DescribeVpcClassicLinkRequest request) {
        return new DescribeVpcClassicLinkResult()
                .withVpcs(describe(request, "vpcIds", "vpcClassicLinks", VpcClassicLink.class));
    }

    public DescribeClassicLinkInstancesResult describeClassicLinkInstances() {
        return describeClassicLinkInstances(null);
    }

    public DescribeClassicLinkInstancesResult describeClassicLinkInstances(
            DescribeClassicLinkInstancesRequest request) {
        return new DescribeClassicLinkInstancesResult()
                .withInstances(describe(request, "instanceIds", "classicLinkInstances", ClassicLinkInstance.class));
    }

    ////////////////////////////////////
    //
    // AmazonElasticLoadBalancing
    //
    ////////////////////////////////////
    public DescribeLoadBalancersResult describeLoadBalancers() {
        return describeLoadBalancers((DescribeLoadBalancersRequest) null);
    }

    public DescribeLoadBalancersResult describeLoadBalancers(DescribeLoadBalancersRequest request) {
        return new DescribeLoadBalancersResult().withLoadBalancerDescriptions(
                describe(request, "loadBalancerNames", "loadBalancers", LoadBalancerDescription.class));
    }

    // Cannot have overloaded method with same parameters and different return types, for now, no calls to this parameter-less function, so commenting out for now
    // public com.amazonaws.services.elasticloadbalancingv2.model.DescribeLoadBalancersResult describeLoadBalancers() {
    //   return describeLoadBalancers((com.amazonaws.services.elasticloadbalancingv2.model.DescribeLoadBalancersRequest)null);
    // }

    public com.amazonaws.services.elasticloadbalancingv2.model.DescribeLoadBalancersResult describeLoadBalancers(
            com.amazonaws.services.elasticloadbalancingv2.model.DescribeLoadBalancersRequest request) {
        return new com.amazonaws.services.elasticloadbalancingv2.model.DescribeLoadBalancersResult()
                .withLoadBalancers(describe(request, "names", "appLoadBalancers", LoadBalancer.class));
    }

    public DescribeTargetGroupsResult describeTargetGroups() {
        return describeTargetGroups(null);
    }

    public DescribeTargetGroupsResult describeTargetGroups(DescribeTargetGroupsRequest request) {
        return new DescribeTargetGroupsResult()
                .withTargetGroups(describe(request, "names", "targetGroups", TargetGroup.class));
    }
    ////////////////////////////////////

    private <T> List<T> describe(AmazonWebServiceRequest request, String idKey, final String object,
            final Class<T> singleType) {
        lastModified.set(null);
        final Map<String, String> metricTags = new HashMap<>(this.metricTags);
        metricTags.put("collection", object);
        try {
            final Collection<String> ids = getRequestIds(request, idKey);
            metricTags.put("collectionMode", ids.isEmpty() ? "full" : "byId");
            final JavaType singleMeta = objectMapper.getTypeFactory().constructParametrizedType(Metadata.class,
                    Metadata.class, singleType);
            Long mtime = null;
            final List<T> results = new ArrayList<>();

            final Id deserializeJsonTimer = registry.createId("edda.deserializeJson", metricTags);
            final Id resultSizeCounter = registry.createId("edda.resultSize", metricTags);
            if (ids.isEmpty()) {
                HttpEntity entity = getHttpEntity(metricTags, object, null);
                try {
                    final JavaType listMeta = objectMapper.getTypeFactory().constructParametrizedType(List.class,
                            List.class, singleMeta);
                    final List<Metadata<T>> metadataResults = registry.timer(deserializeJsonTimer)
                            .record(() -> objectMapper.readValue(entity.getContent(), listMeta));
                    for (Metadata<T> meta : metadataResults) {
                        mtime = mtime == null ? meta.mtime : Math.min(mtime, meta.mtime);
                        results.add(meta.data);
                    }
                } finally {
                    EntityUtils.consume(entity);
                }
            } else {
                for (String id : ids) {
                    HttpEntity entity = getHttpEntity(metricTags, object, id);
                    try {
                        final Metadata<T> result = registry.timer(deserializeJsonTimer)
                                .record(() -> objectMapper.readValue(entity.getContent(), singleMeta));
                        mtime = mtime == null ? result.mtime : Math.min(mtime, result.mtime);
                        results.add(result.data);
                    } finally {
                        EntityUtils.consume(entity);
                    }
                }
            }
            registry.counter(resultSizeCounter).increment(results.size());
            lastModified.set(mtime);
            return results;
        } catch (Exception e) {
            log.error(e.getMessage() + " (retries exhausted)");

            registry.counter(registry.createId("edda.failures", metricTags)).increment();
            final AmazonServiceException ex = new AmazonServiceException(
                    "400 Bad Request -- Edda could not find one of the managed objects requested.", e);
            ex.setStatusCode(400);
            ex.setServiceName(serviceName);
            ex.setErrorType(AmazonServiceException.ErrorType.Unknown);
            throw ex;
        }
    }

    private static Collection<String> getRequestIds(AmazonWebServiceRequest request, String idFieldName) {
        if (request == null) {
            return Collections.emptySet();
        }
        try {
            Field field = request.getClass().getDeclaredField(idFieldName);
            field.setAccessible(true);
            Collection<String> collection = (Collection<String>) field.get(request);
            return collection == null ? Collections.emptySet() : collection;
        } catch (NoSuchFieldException | IllegalAccessException e) {
            throw new RuntimeException(e);
        }
    }

    private HttpEntity getHttpEntity(Map<String, String> metricTags, String objectName, String key)
            throws IOException {
        final String url = edda + "/REST/v2/aws/" + objectName + (key == null ? ";_expand" : "/" + key) + ";_meta";
        final HttpGet get = new HttpGet(url);
        get.setConfig(RequestConfig.custom().setConnectTimeout(eddaTimeoutConfig.getConnectTimeout())
                .setConnectionRequestTimeout(eddaTimeoutConfig.getConnectionRequestTimeout())
                .setSocketTimeout(eddaTimeoutConfig.getSocketTimeout()).build());

        long retryDelay = eddaTimeoutConfig.getRetryBase();
        int retryAttempts = 0;
        String lastException = "";
        String lastUrl = "";
        Random r = new Random();
        Exception ex;

        final Id httpExecuteTime = registry.createId("edda.httpExecute", metricTags);
        final Id httpErrors = registry.createId("edda.errors", metricTags).withTag("errorType", "http");
        final Id networkErrors = registry.createId("edda.errors", metricTags).withTag("errorType", "network");
        final Id retryDelayMillis = registry.createId("edda.retryDelayMillis", metricTags);
        final Id retries = registry.createId("edda.retries", metricTags);
        while (retryAttempts < eddaTimeoutConfig.getMaxAttempts()) {
            ex = null;
            HttpEntity entity = null;

            try {
                final HttpResponse response = registry.timer(httpExecuteTime).record(() -> httpClient.execute(get));
                final int statusCode = response.getStatusLine().getStatusCode();
                entity = response.getEntity();
                if (statusCode != HttpStatus.SC_OK) {
                    lastException = response.getProtocolVersion().toString() + " "
                            + response.getStatusLine().getStatusCode() + " "
                            + response.getStatusLine().getReasonPhrase();
                    registry.counter(httpErrors.withTag("statusCode", Integer.toString(statusCode))).increment();
                } else {
                    return entity;
                }
            } catch (Exception e) {
                lastException = e.getClass().getSimpleName() + ": " + e.getMessage();
                ex = e;
                registry.counter(networkErrors.withTag("exceptionType", e.getClass().getSimpleName())).increment();
            } finally {
                lastUrl = url;
            }

            // ensure that the content stream is closed on a non-200 response from edda
            EntityUtils.consumeQuietly(entity);

            final String exceptionFormat = "Edda request {} failed with {}";
            if (ex == null) {
                log.warn(exceptionFormat, url, lastException);
            } else {
                log.warn(exceptionFormat, url, lastException, ex);
            }
            try {
                registry.counter(retryDelayMillis).increment(retryDelay);
                Thread.sleep(retryDelay);
            } catch (InterruptedException inter) {
                break;
            }
            registry.counter(retries).increment();
            retryAttempts++;
            retryDelay += r.nextInt(eddaTimeoutConfig.getBackoffMillis());
        }
        throw new IOException("Edda request " + lastUrl + " failed with " + lastException);
    }

    private static class Metadata<T> {
        final Long mtime;
        final T data;

        @JsonCreator
        public Metadata(@JsonProperty("mtime") Long mtime, @JsonProperty("data") T data) {
            this.mtime = mtime;
            this.data = data;
        }
    }
}