https://github.com/zq2599/blog_demos
首先,SpringBoot为kubernetes提供了两个actuator项,但是那些并未部署在kubernetes的SringBoot应用呢?用不上这两项也要对外暴露这两个服务地址吗?
其次,就绪探针是什么时候开始返回200返回码的?应用启动阶段,业务服务可能需要一段时间才能正常工作,就绪探针要是提前返回了200,那k8s就认为容器可以正常工作了,这时候把外部请求调度过来是无法正常响应的,所以搞清楚就绪探针的状态变化逻辑很重要;
最后,也是最重要的一点:有的场景下,例如外部依赖服务异常、本地全局异常等情况下,业务不想对外提供服务,等到问题解决后业务又可以对外提供服务了,如果此时我们能自己写代码控制就绪探针的返回码,那就做到了控制kubernetes是否将外部请求调度到此容器上,这可是个很实用的功能!
本篇就是为了解决上述问题而作,这些问题解决后才能用好探针技术,让它在容器环境带来更大价值;
解决上述问题的关键集中在以下几个知识点:
接下来挨个学习这些知识点;
官方文档如下图所示,SpringBoot判断是否是kubernetes环境的逻辑很简单:是否有_SERVICE_HOST和_SERVICE_PORT这两个环境变量:
熟悉kubernetes的读者看到_SERVICE_HOST" 和_SERVICE_PORT,应该会想起KUBERNETES_SERVICE_HOST和KUBERNETES_SERVICE_PORT,这是k8s给pod中配置的环境变量,看来SpringBoot也是针对k8s的这个规则来判定是否是容器环境的(如果将来k8s的某个版本不给pod设置这个环境变量,那些原本可以正常运行的pod岂不是有危险了?);
接下来通过实践来验证上述规则是否有效;
创建一个SpringBoot-2.3.0.RELEASE的应用,其pom.xml中的parent信息如下:
<parent> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-parent</artifactId> <version>2.3.0.RELEASE</version> <relativePath/> </parent>
<dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-actuator</artifactId> </dependency>
/actuator/health/liveness和/actuator/health/readiness在kubernetes环境才会开启,但是一般情况下,在开发阶段SpringBoot应用可能运行在自己的电脑上,此时如果想查看这两个接口的返回值有两种方式:
第一种,就是前面提到的添加_SERVICE_HOST和_SERVICE_PORT这两个环境变量,让SpringBoot以为当前环境是kubernetes环境;
第二种,是按照官方指导添加属性,如下图红框所示:
另外,上图的since注解显示这两个枚举是从2.3.0版本开始生效的;
小小八卦一下,上述两个枚举的作者Brian Clozel,坐标法国里昂,目前在sringboot的提交次数排第8名:
如果业务应用想获取当前的存活和就绪状态,将ApplicationAvailability接口autowire进来即可,下一篇[《实战篇》]会有详细的使用方式,这里看下关键代码:
得益于Spring完整的事件发布和订阅机制,业务应用通过EventListener注解就能监听到存活和就绪状态的变化,在EventListener注解修饰的方法中写入必要的业务代码即可实现状态监听,下一篇[《实战篇》]会有详细的使用方式,这里看下关键代码:
重要的事情一定要强调:咱们修改状态的最终目的,不是为了取得applicationAvailability.getReadinessState()返回新的枚举对象,而是要改变/actuator/health/readiness接口的返回码(就绪是200,未就绪是503),这是kubernetes的探针规则要用到的;