zoukankan      html  css  js  c++  java
  • 深入SpringBoot:自定义Endpoint

    前言

    上一篇文章介绍了SpringBoot的PropertySourceLoader,自定义了Json格式的配置文件加载。这里再介绍下EndPoint,并通过自定EndPoint来介绍实现原理。

    Endpoint

    SpringBoot的Endpoint主要是用来监控应用服务的运行状况,并集成在Mvc中提供查看接口。内置的Endpoint比如HealthEndpoint会监控dist和db的状况,MetricsEndpoint则会监控内存和gc的状况。
    Endpoint的接口如下,其中invoke()是主要的方法,用于返回监控的内容,isSensitive()用于权限控制。

        public interface Endpoint<T> {
            String getId();
            boolean isEnabled();
            boolean isSensitive();
            T invoke();
        }

    Endpoint的加载还是依靠spring.factories实现的。spring-boot-actuator包下的META-INF/spring.factories配置了EndpointAutoConfiguration

    org.springframework.boot.autoconfigure.EnableAutoConfiguration=
    ...
    org.springframework.boot.actuate.autoconfigure.EndpointAutoConfiguration,
    ...

    EndpointAutoConfiguration就会注入必要的Endpoint。有些Endpoint需要外部的收集类,比如TraceEndpoint

        @Bean
        @ConditionalOnMissingBean
        public TraceEndpoint traceEndpoint() {
            return new TraceEndpoint(this.traceRepository);
        }

    TraceEndpoint会记录每次请求的Request和Response的状态,需要嵌入到Request的流程中,这里就主要用到了3个类。

    1. TraceRepository用于保存和获取Request和Response的状态。
       public interface TraceRepository {
           List<Trace> findAll();
           void add(Map<String, Object> traceInfo);
       }
    2. WebRequestTraceFilter用于嵌入web request,收集请求的状态并保存在TraceRepository中。
    3. TraceEndpointinvoke()方法直接调用TraceRepository保存的数据。
       public class TraceEndpoint extends AbstractEndpoint<List<Trace>> {
           private final TraceRepository repository;
           public TraceEndpoint(TraceRepository repository) {
               super("trace");
               Assert.notNull(repository, "Repository must not be null");
               this.repository = repository;
           }
           public List<Trace> invoke() {
               return this.repository.findAll();
           }
       }

    Endpoint的Mvc接口主要是通过EndpointWebMvcManagementContextConfiguration实现的,这个类的配置也放在spring.factories中。

    ...
    org.springframework.boot.actuate.autoconfigure.ManagementContextConfiguration=
    org.springframework.boot.actuate.autoconfigure.EndpointWebMvcManagementContextConfiguration,
    org.springframework.boot.actuate.autoconfigure.EndpointWebMvcHypermediaManagementContextConfiguration

    EndpointWebMvcManagementContextConfiguration注入EndpointHandlerMapping来实现Endpoint的Mvc接口。

        @Bean
        @ConditionalOnMissingBean
        public EndpointHandlerMapping endpointHandlerMapping() {
            Set<? extends MvcEndpoint> endpoints = mvcEndpoints().getEndpoints();
            CorsConfiguration corsConfiguration = getCorsConfiguration(this.corsProperties);
            EndpointHandlerMapping mapping = new EndpointHandlerMapping(endpoints,corsConfiguration);
            boolean disabled = this.managementServerProperties.getPort() != null && this.managementServerProperties.getPort() == -1;
            mapping.setDisabled(disabled);
            if (!disabled) {
                mapping.setPrefix(this.managementServerProperties.getContextPath());
            }
            if (this.mappingCustomizers != null) {
                for (EndpointHandlerMappingCustomizer customizer : this.mappingCustomizers) {
                    customizer.customize(mapping);
                }
            }
            return mapping;
        }

    自定义Endpoint

    自定义Endpoint也是类似的原理。这里自定义Endpoint实现应用内存的定时收集。完整的代码放在Github上了。

    1. 收集内存,MemStatus是内存的存储结构,MemCollector是内存的收集类,使用Spring内置的定时功能,每5秒收集当前内存。
       public static class MemStatus {
           public MemStatus(Date date, Map<String, Object> status) {
               this.date = date;
               this.status = status;
           }
           private Date date;
           private Map<String, Object> status;
           public Date getDate() {
               return date;
           }
           public Map<String, Object> getStatus() {
               return status;
           }
       }
       public static class MemCollector {
           private int maxSize = 5;
           private List<MemStatus> status;
           public MemCollector(List<MemStatus> status) {
               this.status = status;
           }
           @Scheduled(cron = "0/5 * *  * * ? ")
           public void collect() {
               Runtime runtime = Runtime.getRuntime();
               Long maxMemory = runtime.maxMemory();
               Long totalMemory = runtime.totalMemory();
               Map<String, Object> memoryMap = new HashMap<String, Object>(2, 1);
               Date date = Calendar.getInstance().getTime();
               memoryMap.put("maxMemory", maxMemory);
               memoryMap.put("totalMemory", totalMemory);
               if (status.size() > maxSize) {
                   status.remove(0);
                   status.add(new MemStatus(date, memoryMap));
               } else {
                   status.add(new MemStatus(date, memoryMap));
               }
           }
       }
    2. 自定义Endpoint,getIdEndPoint的唯一标识,也是Mvc接口对外暴露的路径。invoke方法,取出maxMemorytotalMemory和对应的时间。
       public static class MyEndPoint implements Endpoint {
           private List<MemStatus> status;
           public MyEndPoint(List<MemStatus> status) {
               this.status = status;
           }
           public String getId() {
               return "my";
           }
           public boolean isEnabled() {
               return true;
           }
           public boolean isSensitive() {
               return false;
           }
           public Object invoke() {
               if (status == null || status.isEmpty()) {
                   return "hello world";
               }
               Map<String, List<Map<String, Object>>> result = new HashMap<String, List<Map<String, Object>>>();
               for (MemStatus memStatus : status) {
                   for (Map.Entry<String, Object> entry : memStatus.status.entrySet()) {
                       List<Map<String, Object>> collectList = result.get(entry.getKey());
                       if (collectList == null) {
                           collectList = new LinkedList<Map<String, Object>>();
                           result.put(entry.getKey(), collectList);
                       }
                       Map<String, Object> soloCollect = new HashMap<String, Object>();
                       soloCollect.put("date", memStatus.getDate());
                       soloCollect.put(entry.getKey(), entry.getValue());
                       collectList.add(soloCollect);
                   }
               }
               return result;
           }
       }
    3. AutoConfig,注入了MyEndPoint,和MemCollector
       public static class EndPointAutoConfig {
           private List<MemStatus> status = new ArrayList<MemStatus>();
           @Bean
           public MyEndPoint myEndPoint() {
               return new MyEndPoint(status);
           }
           @Bean
           public MemCollector memCollector() {
               return new MemCollector(status);
           }
       }
    4. 程序入口,运行后访问http://localhost:8080/my 就可以看到了。

       @Configuration
       @EnableAutoConfiguration
       public class CustomizeEndPoint {
      
           public static void main(String[] args) {
               SpringApplication application = new SpringApplication(CustomizeEndPoint.class);
               application.run(args);
           }
       }

    结语

    Endpoint也是通过spring.factories实现扩展功能,注入了对应的Bean来实现应用监控的功能。



    文/wcong(简书作者)
    原文链接:http://www.jianshu.com/p/9fab4e81d7bb
    著作权归作者所有,转载请联系作者获得授权,并标注“简书作者”。
  • 相关阅读:
    JSP学习
    Maven
    Android开发环境搭建
    Java Spring MVC
    you don't know js -- Scope and Closures学习笔记——第五章(闭包) 下篇
    you don't know js -- Scope and Closures学习笔记——第五章(闭包) 上篇
    you don't know js -- Scope and Closures学习笔记——第四章(声明提升 Hoisting)
    you don't know js -- Scope and Closures学习笔记——第三章(函数VS块作用域)
    you don't know js -- Scope and Closures学习笔记——第二章(词法作用域)
    you don't know js -- Scope and Closures学习笔记——第一章(什么是作用域)
  • 原文地址:https://www.cnblogs.com/softidea/p/6249635.html
Copyright © 2011-2022 走看看