Java tutorial
/* * 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; } } }