目的
之前写了一个mini-tcc的测试springboot,想到在自己的windows弄个docker试试,以前在centos上装过docker,很久没玩了。这次写的这个过程基本上从网上查方法实现的,没啥新意,本没必要写出来,不过在这个过程中也碰到一些坑,而且自己写笔记记录一遍印象深,所以写一篇操作过程。
整体过程:你提交代码到仓库,如果是getlab,使用webhook自动触发拉取,不是的话就配置cron表达式定时构建,jenkins拉取到代码后,需要进行构建,测试,打包,一系列操作,这是最基本的,姑且认为这是持续集成CI,然后需要将镜像push到镜像仓库,然后远程ssh执行k8s的sh脚本命令,删除新建相关组件,从镜像仓库拉取镜像,部署服务,这可以叫持续交付CD,整个过程你就提交了代码,整体自动化,这可以算是持续部署,整体思路就是这样
一、安装 docker toolbox
window装docker有两个选择,使用toolbox或者Docker for Windows。因为本人家里的电脑了10年历史了,所以果断找toolbox,这个要先装一个linux虚拟机,其上再运行docker,差别不表网上很多。toolbox安装后会安装一个VirtualBox虚拟机,一个Kitematic,这是GUI管理docker的工具
安装时从阿里镜像找安装包:
http://mirrors.aliyun.com/docker-toolbox/windows/
安装后,桌面产生三个快捷方式
Docker Quickstart
Terminal:应该是快速启动virtualBox,再虚拟一个default的linux,再安装docker,返回一个shell可以操作。Kitematic (Alpha):Kitematic是一个 Docker GUI
工具,进去可以看到里面加好的docker容器,可以看到控制台。Oracle VM
VirtualBox:VirtualBox号称是最强的免费虚拟机软件,进去可以看到有一个default的linux虚拟机在运行。你可以增加自己的各种window,linux操作系统。
给当前用户产生多个环境变量:DOCKER_HOST、DOCKER_MACHINE_NAME、DOCKER_CERT_PATH…
(用testContainer做测试也方便了,目前还是用的内置H2数据库)
问题
运行Docker Quickstart Terminal后,一直提示要升级,网上查,把安装路径下的boot2docker.iso拷贝到C:\Users{window用户}\.docker\machine\cache,就可以了。
运行后,可以使用docker-machine来管理docker,和在Docker Quickstart Terminal里差不多,不过当前用户不同。在命令行CMD中,输入docker-machine ssh default,可以管理这个default里的docker
Docker Quickstart Terminal用户是windows当前用户,~目录是 /c/Users/{Acer},而没有在etc/passwd中找到这个文件,也找不到docker用户。
docker-machine进入default后,当前用户是docker,可以找到其它的root之类的用户,可以看到/c/这个本地挂载的目录。
两个都可以操作docker images/container,而且操作结果是一样的,还是不太明白这两个的关系。
给docker添加镜像源配置,docker从哪个镜像库来上传,下载docker镜像。
sudo sed -i “s|EXTRA_ARGS=’|EXTRA_ARGS=’–registry-mirror={你的镜像源地址} |g” /var/lib/boot2docker/profile
二、springboot应用处理
开始是使用maven插件:docker-maven-plugin
建文件:src/main/decker/Dockerfile
打包后,copy到windows桌面,因为default默认挂载c/User 对应c:\Users,可以/usr/local中建一个文件夹,把jar与Dockerfile放在一起,再构建一个本地镜像。
后来,使用更新的插件:dockerfile-maven-plugin,一次性打包到镜像库中。使用阿里云的镜像库,事先建好,用户密码都设置好。
<properties>
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
<project.reporting.outputEncoding>UTF-8</project.reporting.outputEncoding>
<java.version>1.8</java.version>
<docker.image.prefix>mini-tcc</docker.image.prefix>
<dockerfile.maven.version>1.4.0</dockerfile.maven.version>
<docker.registry.name.prefix>registry.cn-hangzhou.aliyuncs.com/springtcc</docker.registry.name.prefix>
</properties>
<plugin>
<groupId>com.spotify</groupId>
<artifactId>dockerfile-maven-plugin</artifactId>
<version>${dockerfile.maven.version}</version>
<dependencies>
<dependency>
<groupId>javax.activation</groupId>
<artifactId>activation</artifactId>
<version>1.1.1</version>
</dependency>
</dependencies>
<executions>
<execution>
<id>default</id>
<goals>
<goal>build</goal>
<goal>push</goal>
</goals>
</execution>
</executions>
<configuration>
<repository>${docker.registry.name.prefix}/${project.artifactId}</repository>
<tag>${project.version}</tag>
<username>{你自己的用户名}</username>
<password>{你自己的密码}</password>
<buildArgs>
<JAR_FILE>target/${project.build.finalName}.jar</JAR_FILE>
</buildArgs>
</configuration>
</plugin>
问题
在maven打包时,之前的Dockerfile放到pom同级别的根目录中,但没修改地址“ADD target/”,找不到jar文件。
maven goal:clean package -Dmaven.test.skip=true dockerfile:push
FROM openjdk:8-jdk-alpine
VOLUME /tmp
ADD target/spring-boot-thymeleaf-0.0.1-SNAPSHOT.jar app.jar
ENTRYPOINT ["JAVA","-Djava.security.egd=file:/dev/./urandom","-jar","app.jar"]
其它
查看运行日志:docker logs -f --tail=100 {vigorous_hermann} container的名字,这里没有用 -v 映射主机目录,只在控制台输出。
三、总结
运行镜像,看到之前测试mini_tcc的日志。
docker@default:/$ docker container ls -a
CONTAINER ID IMAGE COMMAND CREATED STATUS PORTS NAMES
c423234453d5 docker2 "java -Djava.securit…" 19 hours ago Exited (143) About an hour ago vigorous_hermann
docker@default:/$ docker start vigorous_hermann
vigorous_hermann
docker@default:/$ docker container ls
CONTAINER ID IMAGE COMMAND CREATED STATUS PORTS NAMES
c423234453d5 docker2 "java -Djava.securit…" 19 hours ago Up 5 seconds 0.0.0.0:80->8080/tcp vigorous_hermann
docker@default:/$ docker logs -f --tail=100 vigorous_hermann
at org.hibernate.internal.SessionImpl.doFlush(SessionImpl.java:1436)
Caused by: org.h2.jdbc.JdbcSQLException: Value too long for column "ID CHAR(20) NOT NULL": "'7777788888999999999999' (22)"; SQL statement:
insert into staff (age, name, id) values (?, ?, ?) [22001-196]
at org.h2.message.DbException.getJdbcSQLException(DbException.java:345)
开始本地DB非事务...
Hibernate: select staffbo0_.id as id1_0_0_, staffbo0_.age as age2_0_0_, staffbo0_.name as name3_0_0_ from staff staffbo0_ where staffbo0_.id=?
Hibernate: insert into staff (age, name, id) values (?, ?, ?)
开始远程TCC...
Eat book...!
[bookEat] return value:null
house book...!
[bookHouse] return value:HS100083
plane booked!
[bookPlane] return value:null
【正常,需确认数】3
Eat confirm...!
house confirm...!HS100083
plane confirmed!
【threadLocal removed!】
四、后续k8s
标准方式还是jenkins,由gitlab源码触发hook,自动构建,产生镜像,再调用k8s脚本完成部署
- jenkins中添加ssh服务器,即安装k8s的服务器,ssh登录并执行下面的脚本
- cd /data/script && sh service_start.sh myservice-docker
- /data/script/目录下创建service_start.sh脚本如下:
#!/bin/bash
export service_name=$1
sleep_second=3
echo 'kubectl delete rc ${service_name}'
kubectl delete rc ${service_name}
echo 'kubectl delete service ${service_name}'
kubectl delete service ${service_name}
echo 'kubectl delete pods -l app=${service_name}'
kubectl delete pods -l app=${service_name}
sleep ${sleep_second}
echo 'create rc'
kubectl create -f /data/${service_name}-rc.yaml
echo 'create service'
kubectl create -f /data/${service_name}-svc.yaml
参考:
https://www.cnblogs.com/java-zhao/p/6065268.html