这是Jerry 2020年的第30篇文章,也是汪子熙公众号总共第213篇原创文章。


容器技术,Docker,虚拟化,这些名词诞生尽管有很长一段时间了,但是在云原生开发领域仍旧热度不减。甚至连SAP赖以成名的ABAP Netweaver,如今也踏上了容器化的上云探索之路,比如下面这张来自SAP社区的一篇博客:


Proof of Concept: Deploying ABAP in Kubernetes

https://blogs.sap.com/2020/02/06/proof-of-concept-deploying-abap-in-kubernetes/


在SAP云平台上部署和运行Docker应用_java


今天咱们暂时不聊ABAP的容器化,先从最简单的概念开始。网络上关于Docker容器的技术文章多如牛毛,Jerry也就不再重复了,我还是紧扣SAP技术和SAP解决方案来写。


如果有一个已经能够正常运行的Java应用,可以将其以Docker容器的方式,部署且运行在SAP云平台上吗?当然可以,而且只需要简单的几个命令行即可。


本文接下来的步骤是针对那些听说过Docker容器技术,但尚未动手实践过的朋友。通过阅读本文,可以了解将Java应用容器化并部署到SAP云平台运行的大致步骤。您也可以根据本文的步骤动手试一试,只需要在本地搭一个最简单的Java开发环境,安装Docker客户端和申请一个SAP云平台的trial账号即可。


(1) 首先得有一个能在本地正常运行的SpringBoot应用。如果没有,百度之然后自己动手做一个。


如果实在不想自己动手,从Jerry的github仓库上克隆一个下来也行:


https://github.com/wangzixi-diablo/mySpringBoot


本地配置好maven和JDK之后,直接用命令行启动它:


mvn spring-boot:run


你会看到Tomcat started on port 8000的提示,意思是该应用已经成功启动,监听在本地端口8000上。



浏览器里输入localhost:8000/product, 看到Hello World. 至此,我们就有了一个本地正常运行的SpringBoot应用了,下一步是将其打包成Docker镜像。



(2) 在SpringBoot项目里新建一个名叫Dockerfile的文件,输入以下内容:


FROM openjdk:8-jdk-alpine

VOLUME /tmp

VOLUME /log

EXPOSE 8080

ADD target/prolikeService.jar app.jar

ENV JAVA_OPTS="-Dserver.port=8080"

ENTRYPOINT exec java $JAVA_OPTS -jar /app.jar



这实际是一个指令文件,Docker客户端会根据里面的内容,制作一个Docker镜像。


简要介绍每一行内容的含义。


第一行:指定待制作的镜像是基于名叫openjdk:8-jdk-alpine的镜像,在Docker Hub网站上能找到更多关于该镜像的说明:


https://hub.docker.com/_/openjdk



第二行和第三行:定义两个名叫tmp和log的持久化存储。容器运行时产生的数据,随着容器的销毁而销毁,但有时我们又希望这些数据能够持久化保存,比如需要分析某个容器运行时产生的日志文件,这时可以使用Dockerfile里VOLUME关键字提供的容器持久化技术,创建所谓的“卷”,将容器应用运行时写入数据的目录,映射到宿主机上的某个目录下。


如果一个容器尚处在运行状态,我们可以用命令行进入容器内部,查看log文件夹里的运行日志文件


docker exec -it 8302db78f838 /bin/sh



如果该容器已经销毁,我们就可以到宿主机的/var/lib/docker/volumes目录下,查看持久化的日志文件:


在SAP云平台上部署和运行Docker应用_java_02


第五行:把当前项目通过maven打包而成的jar包拷贝到容器内部,重命名为app.jar

第六~七行:设置JVM启动参数,暴露8080端口给外部。


Dockerfile文件编写完毕后,使用命令行制作镜像:


docker build -t i042416/springbootexample:v1 .


v1代表该镜像的标签,命令行尾部的句号代表当前目录。


镜像制作完毕后,使用命令行将制作好的镜像推送到Docker Hub网站上(有点像我们本地用git客户端提交代码到Github上):


docker push i042416/springbootexample:v1


成功之后,能够在Docker Hub上看到推送好的镜像:


https://hub.docker.com/repository/docker/i042416/springbootexample


在SAP云平台上部署和运行Docker应用_java_03


这样,稍后SAP云平台就能从Docker Hub上拉取这个镜像了。


(3) 登录SAP云平台CloudFoundry环境,使用命令行部署, 用参数--docker-image指定我们刚刚上传到Docker Hub上的镜像名称,部署生成的应用名叫jerryjavadocker.


cf push jerryjavadocker --docker-image i042416/springbootexample:v6


因为我的容器镜像修改过好几次,所以标签从v1升到了v6. 


在SAP云平台上部署和运行Docker应用_java_04


成功部署,应用的状态显示成了running:


在SAP云平台上部署和运行Docker应用_java_05


在SAP云平台也能看到这个成功部署的应用,处于运行状态:


在SAP云平台上部署和运行Docker应用_java_06


运行这个成功部署好的Docker应用,和我们在本地mvn spring-boot:run的效果一样。至此这个SpringBoot应用的容器化和SAP云平台的部署就成功了。


在SAP云平台上部署和运行Docker应用_java_07


顺便说一句,昨天Jerry的文章 SAP新一代全栈开发工具:SAP Business Application Studio 推送之后,有一位朋友在文章下面留言:


在SAP云平台上部署和运行Docker应用_java_08


关于传统ABAP开发人员如何学习SAP Cloud Platform相关技术,Jerry后续如果有机会的话会专门写一篇短文介绍自己的做法,敬请期待。https://mp.weixin.qq.com/s/YkgOyeb4W83-USAu0pvJXA