背景
最近项目部署任务,最终客户的环境是:飞腾CPU(ARM64)+ 银河麒麟(Kylin)+ 容器部署
现象
部署完成后发现:服务运行5-10分钟就会崩溃,报错日志很多,最有用的信息是下面这一行:guarantee(Rs != Rn && Rs != Rt) failed: unpredictable instruction
解决
根据报错内容猜测是Java服务使用的基础镜像的问题。当前用的openjdk:8-jdk-alpine
作为基础镜像,换成openjdk:8
作为基础镜像,重新部署后该问题解决,但是新的问题又出现了
我们知道SpringBoot读取配置是有一定的顺序的,详见:Spring Boot Features.external-config。所以我们将特殊配置都放进了容器的环境变量,结果问题来了:SpringBoot启动后不会读取环境变量!
然后我们在main方法中增加了代码:
public static void main(String[] args) {
LOGGER.warn("=============test=============");
for (Map.Entry<String, String> entry : System.getenv().entrySet()) {
LOGGER.warn("{}={}", entry.getKey(), entry.getValue());
}
LOGGER.warn("=============test=============");
SpringApplication.run(Application.class, args);
}
打印出来的结果也确实没有我们额外定义的环境变量。
后来偶然发现:如果执行/docker-entrypoint.sh
问题就存在,但是执行bash /docker-entrypoint.sh
就没问题。然后恍然大悟看了眼/docker-entrypoint.sh
的解释器:果然是#!/bin/sh
。这么写也是因为openjdk:8-jdk-alpine
的基础镜像里就没有bash
这么个东西!
总以为是代码问题,在代码层面纠结了好久。结果还是因为自己Linux知识比较渣...