前言

上一篇文章介绍了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
著作权归作者所有,转载请联系作者获得授权,并标注“简书作者”。

最新文章

  1. 数据结构:优先队列 基于list实现(python版)
  2. 基于ArcGIS API for Javascript的地图编辑工具
  3. iOS_XCode7_Launch Image 的初使用
  4. 《精通C#》委托与事件(10章)
  5. Iterator遍历器 调用Symbol.Iterator属性,遍历器对象。
  6. ACM/ICPC 之 SPFA练习两道(ZOJ3088-ZOJ3103)
  7. python学习之字符串变量
  8. WindowsServer问题总结
  9. 基于 Koa平台Node.js开发的KoaHub.js的控制器,模型,帮助方法自动加载
  10. JAVA_SE基础——63.String类的常用方法
  11. Android八门神器(一): OkHttp框架源码解析
  12. window下为kibana安装x-pack时候出现Plugin installation was unsuccessful due to error &quot;No valid url specified.&quot;错误的解决方案
  13. Python 线程和进程
  14. Docker Kubernetes 创建管理 Deployment
  15. html/css/js-个人容易忘的一些属性
  16. 函数式编程之 Python
  17. Hibernate进行对象的增删改查
  18. 安装express并创建工程
  19. 读取Apache访问日志,查看每一个独立客户端连接获得的字节数
  20. 使用 Selenium 实现基于 Web 的自动化测试

热门文章

  1. BZOJ1012[JSOI2008]最大数maxnumber 题解
  2. 20145308刘昊阳 《Java程序设计》第2周学习总结
  3. js获取屏幕大小
  4. 关于Nodejs的多进程模块Cluster
  5. nodeType的返回
  6. 细说jQuery原型的创建和实现原理,并用实例简单模仿
  7. zk编程语言: 如何改变datebox框值的大小及高度
  8. Canvas 属性,方法
  9. spring security动态管理资源结合自定义登录页面
  10. 讨论一下hibernate如何动态注册一个动态生成的实体类