深入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
著作权归作者所有,转载请联系作者获得授权,并标注“简书作者”。

 

时间: 2024-10-28 13:53:49

深入SpringBoot:自定义Endpoint的相关文章

spring-boot自定义启动端口

Spring boot 自定义端口 前言 spring boot本身内置tomcat,我们不需要进行tomcat的配置,只需要引入tomcat的依赖即可. <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-tomcat</artifactId> </dependency> 自定义端口 方法① 1.spring

Spring Actuator源码分析

Actuator Endpoint Actuator模块通过Endpoint暴露一些接口,可以是Rest方式,也可以是JMX等其他方式. 如果使用Rest方式,通常SpringMVC是使用@RequestMapping,以及@Controller标注一个控制器方法,如果不使用SpringMVC,即没引入SpringMVC的包,那么Springboot就会出错.所以为了不走正常的SpringMVC机制,Actuator用EndpointHandlerMapping重写了RequestMapping

spring-boot 速成(4) 自定义配置

spring-boot 提供了很多默认的配置项,但是开发过程中,总会有一些业务自己的配置项,下面示例了,如何添加一个自定义的配置: 一.写一个自定义配置的类 package com.example.config; import lombok.Data; import org.springframework.boot.context.properties.ConfigurationProperties; import org.springframework.stereotype.Component

SpringBoot编写自定义的starter

在之前的文章中,我们分析过SpringBoot内部的自动化配置原理和自动化配置注解开关原理. 我们先简单分析一下mybatis starter的编写,然后再编写自定义的starter. mybatis中的autoconfigure模块中使用了一个叫做MybatisAutoConfiguration的自动化配置类. 这个MybatisAutoConfiguration需要在这些Condition条件下才会执行: @ConditionalOnClass({ SqlSessionFactory.cla

SpringBoot中如何自定义静态资源路径及映射

差不多写完一个日志模板了, 明天集成到主程序,测试一下, 下周一就可以上线了. 今天遇到最后一个问题, 在将SPRINGBOOT与DJANGO,UWSGI,NGINX作集成时, SPRINGBOOT的静态资源目录不能像测试环境一样,直接用static下面的js,css,img等目录. 必须在nginx下作一下location,定位到springboot专属的目录位置. 这时,就涉及自定义静态资源路径及映射. 举例: 如果我想在thymeleaf中用 <link rel="styleshee

SpringBoot之退出服务(exit)时调用自定义的销毁方法

我们在工作中有时候可能会遇到这样场景,需要在退出容器的时候执行某些操作.SpringBoot中有两种方法可以供我们来选择(其实就是spring中我们常用的方式.只是destory-method是在XML中配置的,SpringBoot是去配置化.所以这里就不提这种方式了),一种是实现DisposableBean接口,一种是使用@PreDestroy注解.OK,下面我写两个例子看一下: DisposableBean接口 我们可以通过实现这个接口来在容器退出的时候执行某些操作.例子如下: packag

SpringBoot笔记一

1 开始 1.1 spring介绍 Spring Boot使开发独立的,产品级别的基于Spring的应用变得非常简单,你只需"just run". 我们为Spring平台及第三方库提供开箱即用的设置,这样你就可以有条不紊地开始.多数Spring Boot应用需要很少的Spring配置. 你可以使用Spring Boot创建Java应用,并使用java -jar启动它或采用传统的war部署方式. 1.2 系统要求 默认情况下,Spring Boot 1.3.0.BUILD-SNAPSHO

Spring自定义配置Schema可扩展(一)_java

简述 本教程主要介绍如何扩展Spring的xml配置,让Spring能够识别我们自定义的Schema和Annotation. 这里我们要实现的功能如下,首先让Spring能够识别下面的配置. <std:annotation-endpoint /> 这个配置的要实现的功能是,配置完后能够让Spring扫描我们自定义的@Endpoint注解.并且根据注解自动发布WebService服务.功能未完全实现,作为扩展Spring的教程,起一个抛砖引玉的作用. 创建项目 首先需要创建一个Java项目,这里

Spring自定义配置Schema可扩展(二)_java

命名空间支持 要实现命名空间支持,需要继承自NamespaceHandlerSupport. package com.codestd.spring.cxf.config.schema; import org.springframework.beans.factory.xml.NamespaceHandlerSupport; import com.codestd.spring.cxf.config.EndpointBeanProcessor; /** * 处理命名空间 * @author jaun