Java tutorial
/* * Copyright 2013-2014 the original author or authors. * * 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 org.springframework.cloud.netflix.hystrix; import java.io.IOException; import java.util.Arrays; import java.util.HashSet; import java.util.Map; import java.util.Set; import org.apache.catalina.core.ApplicationContext; import org.apache.commons.logging.Log; import org.apache.commons.logging.LogFactory; import org.springframework.beans.factory.DisposableBean; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.boot.actuate.metrics.GaugeService; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnExpression; import org.springframework.boot.autoconfigure.condition.ConditionalOnWebApplication; import org.springframework.context.SmartLifecycle; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import org.springframework.context.annotation.ImportAware; import org.springframework.core.Ordered; import org.springframework.core.annotation.AnnotationAttributes; import org.springframework.core.type.AnnotationMetadata; import org.springframework.util.Assert; import com.fasterxml.jackson.databind.ObjectMapper; import com.netflix.hystrix.Hystrix; import com.netflix.hystrix.contrib.javanica.aop.aspectj.HystrixCommandAspect; import com.netflix.hystrix.contrib.metrics.eventstream.HystrixMetricsPoller; import com.netflix.hystrix.contrib.metrics.eventstream.HystrixMetricsPoller.MetricsAsJsonPollerListener; /** * @author Spencer Gibb * @author Christian Dupuis */ @Configuration public class HystrixConfiguration implements ImportAware { private AnnotationAttributes enableHystrix; @Bean public HystrixCommandAspect hystrixCommandAspect() { return new HystrixCommandAspect(); } @Bean public HystrixShutdownHook hystrixShutdownHook() { return new HystrixShutdownHook(); } @Configuration @ConditionalOnExpression("${hystrix.stream.endpoint.enabled:true}") @ConditionalOnWebApplication protected static class HystrixWebConfiguration { @Bean public HystrixStreamEndpoint hystrixStreamEndpoint() { return new HystrixStreamEndpoint(); } } @Override public void setImportMetadata(AnnotationMetadata importMetadata) { this.enableHystrix = AnnotationAttributes .fromMap(importMetadata.getAnnotationAttributes(EnableHystrix.class.getName(), false)); Assert.notNull(this.enableHystrix, "@EnableHystrix is not present on importing class " + importMetadata.getClassName()); } @Configuration @ConditionalOnClass(GaugeService.class) protected static class HystrixMetricsPollerConfiguration implements SmartLifecycle { private static Log logger = LogFactory.getLog(HystrixMetricsPollerConfiguration.class); @Autowired(required = false) private GaugeService gauges; private ObjectMapper mapper = new ObjectMapper(); private HystrixMetricsPoller poller; private Set<String> reserved = new HashSet<String>(Arrays.asList("group", "name", "type", "currentTime")); @Override public void start() { if (gauges == null) { return; } MetricsAsJsonPollerListener listener = new MetricsAsJsonPollerListener() { @Override public void handleJsonMetric(String json) { try { @SuppressWarnings("unchecked") Map<String, Object> map = mapper.readValue(json, Map.class); if (map != null && map.containsKey("type")) { addMetrics(map, "hystrix."); } } catch (IOException e) { // ignore } } }; poller = new HystrixMetricsPoller(listener, 2000); // start polling and it will write directly to the listener poller.start(); logger.info("Starting poller"); } private void addMetrics(Map<String, Object> map, String root) { StringBuilder prefixBuilder = new StringBuilder(root); if (map.containsKey("type")) { prefixBuilder.append((String) map.get("type")); if (map.containsKey("group")) { prefixBuilder.append(".").append(map.get("group")); } prefixBuilder.append(".").append(map.get("name")); } String prefix = prefixBuilder.toString(); for (String key : map.keySet()) { Object value = map.get(key); if (!reserved.contains(key)) { if (value instanceof Number) { String name = prefix + "." + key; gauges.submit(name, ((Number) value).doubleValue()); } else if (value instanceof Map) { @SuppressWarnings("unchecked") Map<String, Object> sub = (Map<String, Object>) value; addMetrics(sub, prefix); } } } } @Override public void stop() { if (poller != null) { poller.shutdown(); } } @Override public boolean isRunning() { return poller != null ? poller.isRunning() : false; } @Override public int getPhase() { return Ordered.LOWEST_PRECEDENCE; } @Override public boolean isAutoStartup() { return true; } @Override public void stop(Runnable callback) { if (poller != null) { poller.shutdown(); } callback.run(); } } /** * {@link DisposableBean} that makes sure that Hystrix internal state is cleared when * {@link ApplicationContext} shuts down. */ private class HystrixShutdownHook implements DisposableBean { @Override public void destroy() throws Exception { // Just call Hystrix to reset thread pool etc. Hystrix.reset(); } } }