深入SpringBoot:自定義Endpoint

前言

上一篇文章介紹了SpringBoot的PropertySourceLoader,自定義了Json格式的配置文件加載。這里再介紹下EndPoint,并通過自定EndPoint來介紹實(shí)現(xiàn)原理。

Endpoint

SpringBoot的Endpoint主要是用來監(jiān)控應(yīng)用服務(wù)的運(yùn)行狀況,并集成在Mvc中提供查看接口。內(nèi)置的Endpoint比如HealthEndpoint會(huì)監(jiān)控dist和db的狀況,MetricsEndpoint則會(huì)監(jiān)控內(nèi)存和gc的狀況。
Endpoint的接口如下,其中invoke()是主要的方法,用于返回監(jiān)控的內(nèi)容,isSensitive()用于權(quán)限控制。

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

Endpoint的加載還是依靠spring.factories實(shí)現(xiàn)的。spring-boot-actuator包下的META-INF/spring.factories配置了EndpointAutoConfiguration。

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

EndpointAutoConfiguration就會(huì)注入必要的Endpoint。有些Endpoint需要外部的收集類,比如TraceEndpoint

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

TraceEndpoint會(huì)記錄每次請(qǐng)求的Request和Response的狀態(tài),需要嵌入到Request的流程中,這里就主要用到了3個(gè)類。

  1. TraceRepository用于保存和獲取Request和Response的狀態(tài)。
    public interface TraceRepository {
        List<Trace> findAll();
        void add(Map<String, Object> traceInfo);
    }
  1. WebRequestTraceFilter用于嵌入web request,收集請(qǐng)求的狀態(tài)并保存在TraceRepository中。
  2. TraceEndpoint,invoke()方法直接調(diào)用TraceRepository保存的數(shù)據(jù)。
    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實(shí)現(xiàn)的,這個(gè)類的配置也放在spring.factories中。

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

EndpointWebMvcManagementContextConfiguration注入EndpointHandlerMapping來實(shí)現(xiàn)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實(shí)現(xiàn)應(yīng)用內(nèi)存的定時(shí)收集。完整的代碼放在Github上了。

  1. 收集內(nèi)存,MemStatus是內(nèi)存的存儲(chǔ)結(jié)構(gòu),MemCollector是內(nèi)存的收集類,使用Spring內(nèi)置的定時(shí)功能,每5秒收集當(dāng)前內(nèi)存。
    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));
            }
        }
    }
  1. 自定義Endpoint,getIdEndPoint的唯一標(biāo)識(shí),也是Mvc接口對(duì)外暴露的路徑。invoke方法,取出maxMemorytotalMemory和對(duì)應(yīng)的時(shí)間。
    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;
        }
    }
  1. 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);
        }
    }
  1. 程序入口,運(yùn)行后訪問http://localhost:8080/my 就可以看到了。
    @Configuration
    @EnableAutoConfiguration
    public class CustomizeEndPoint {

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

結(jié)語

Endpoint也是通過spring.factories實(shí)現(xiàn)擴(kuò)展功能,注入了對(duì)應(yīng)的Bean來實(shí)現(xiàn)應(yīng)用監(jiān)控的功能。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理,服務(wù)發(fā)現(xiàn),斷路器,智...
    卡卡羅2017閱讀 136,569評(píng)論 19 139
  • 1. Java基礎(chǔ)部分 基礎(chǔ)部分的順序:基本語法,類相關(guān)的語法,內(nèi)部類的語法,繼承相關(guān)的語法,異常的語法,線程的語...
    子非魚_t_閱讀 34,728評(píng)論 18 399
  • 背景 一年多以前我在知乎上答了有關(guān)LeetCode的問題, 分享了一些自己做題目的經(jīng)驗(yàn)。 張土汪:刷leetcod...
    土汪閱讀 12,922評(píng)論 0 33
  • 哆啦小姐一直都是個(gè)自卑的短發(fā)女生,上學(xué)時(shí)曾看過很多瑪麗蘇的言情小說,卻一直覺得自己以后會(huì)是一個(gè)浪跡天涯的俠客,雖然...
    壹人留閱讀 811評(píng)論 4 6
  • 商家私鑰 ALIPAY_RSA_PRIVATE_KEY = <<-EOF-----BEGIN RSA PRIVAT...
    hatch_win閱讀 949評(píng)論 0 0

友情鏈接更多精彩內(nèi)容