java-jar jar包带环境变量(参数)启动

需求

java工程我们可以编译成jar也可以翻译成war,一般地,war包我会丢到tomcat容器里,启动tomcat来访问服务,端口、SSL证书、日志等等,都托给tomcat。

如果打的是jar包,我通常会用nohup启动,比如生产环境的一些db、redis、第三方secret等不会配置到项目里,今天要记录的便是用nohup java -jar 启动jar包时如何加载环境变量配置的问题。

java语言开发的jar包启动时可以按照如下方式加启动参数。

方式一:-DpropName=propValue

这种方式应该很快都能找到

-DpropName=propValue

比如:

java -jar -DdatabaseUrl="mysql://localhost:3306/pdb?user=root&password=root"  -Dapp.key="123" -Dapp.secret="xxx"  demo.jar

多个参数也可以。

方式二:参数直接跟在命令后面,多个参数之间用空格隔开

java -jar demo.jar JOURNAL_TREENODE_DATA-20190404174502.txt processType=1

这种方式参数就是jar包里主启动类中main方法的args参数,按顺序来

方式三:使用springboot的方式,--propName=propValue方式

java -jar demo.jar  --spring.profiles.active=dev  --server.port=8181

注意:
运行jar包时指定端口:java -jar xxx.jar --server.port=8088
若命令行传入的server.port没有作用,服务仍然使用8081端口启动,原因是spring-cloud-config会覆盖命令行传入的参数,这是有意为之,

办法是在web-prod.yml中做点小改动,让“配置”变得“可配置”:加一对花括符

server.port={port:8081}

用clojure开发的jar里有惊喜

在clojure上面的配置就变得诡异了,猜猜下面的配置能不能生效呢?

java -jar -Ddatabase-url="mysql://localhost:3306/pdb?user=root&password=root"  -Dapp.key="123" -Dapp.secret="xxx"  demo.jar

如果你用cider-conect通过nrepl的端口连接上你的服务,你会发现,这个配置导致database-url的值确实已经改了,但是后面两个没有。

究其原因,我们java从classpath里获取参数使用的是properties形式的,也就是json的格式。这不难理解,spring有它的办法,clojure当然也有自己的方式。

"app": {
    "key": "123",
    "secret":"xxx"
}

虽然json和我们的edn里map是很像的,但是毕竟是不同,于是要分析下现在的edn里的配置信息他是怎么读取的呢?

代码里的env

config这个namespace里找到了env

(defstate env
  :start
  (load-config
   :merge
   [(args)
    (source/from-system-props)
    (source/from-env)]))

cprop加载配置

cprop.source这个文件就是用来加载edn文件的

(defn- env->path [k]
  (k->path k "_" #"__"))

(defn read-system-env
  ([]
   (read-system-env {}))
  ([opts]
   (->> (System/getenv)
        (map (fn [[k v]] [(env->path k)
                          (str->value v opts)]))
        (into {}))))

从获取的过程看,应该是会把_做为单元节点断开,因此需要改成这样

java -jar -Ddatabase-url="mysql://localhost:3306/pdb?user=root&password=root"  -Dapp_key="123" -Dapp_secret="xxx"  demo.jar

对应到edn里应该是

{:app
    {:key "123"
     :secret "xxx"}}

如果仔细看看cprop这个库就不难理解了。
cprop加载配置文件的顺序是 :

By default cprop will merge all configurations it can find in the following order:
classpath resource config
file on a file system (pointed by a conf system property or by (load-config :file <path>))
custom configurations, maps from various sources, etc.
System properties
ENV variables

对于ENV的加载也有明确说明

ENV variables lack structure. The only way to mimic the structure is via use of an underscore character. The _ is converted to - by cprop, so instead, to identify nesting, two underscores can be used.

了解更多详情,请阅读cprop介绍

我的期望

像上面的命令如果有20个参数需要在启东时指定,估计看那个命令就疯了,那能不能在启东时指定一个配置文件,比如prod-config.edn之类的呢?我没有找到,如果有就更方便了。

参考

cprop

作者:小马将过河
链接:https://www.jianshu.com/p/fed7a174bfb8
来源:简书
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。


版权声明:本文为zhongguowangzhan原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
THE END
< <上一篇
下一篇>>