.Net Core in Docker - 使用阿里云Codepipeline及阿里云容器镜像服务实现持续交付/部署(CD)
上一次演示了如何使用阿里云codepipeline,阿里云容器镜像服务实现ci,讲到这里我们push一下代码后就自动编译、自动跑单元测试、自动构建镜像、自动推送镜像到私仓。那么离我们最初设定的目标只差那么一小步了,那就是自动部署到测试/生产环境,这一步就是持续交付/部署(cd)。
cd其实是两个意思
(1)continuous delivery (持续交付) 指的是,频繁地将软件的新版本,交付给质量团队或者用户,以供评审。如果评审通过,代码就进入生产阶段。
(2)continuous deployment(持续部署) 指的是代码通过评审以后,自动部署到生产环境。
之前我一直以为cd只是持续部署的意思,最近仔细查资料才发其实是有两层意思。虽然是两层意思,但是其实也差不多,都是部署到某个可以运行起来的环境中,把程序跑起来。持续交付一般是部署到测试环境,供测试团队评审;持续部署是指通过测试评审后把程序部署到生产环境。既然差不多这里我就不细分了,因为都是部署,只是部署的位置不一样。
流程
上次的流程到把镜像推送到私仓(阿里云容器镜像服务)后就结束了,后面的流程需要手动跑shell脚本来完成。我们要把后面的流程串起来,让shell脚本自动运行起来,需要一个触发机制,比如webhook。幸好,阿里云容器镜像服务有这么一个功能,可以让我们把流程串起来,那就是触发器功能。这个触发器功能跟webhook其实差不多,当容器镜像服务收到新镜像后会对外发送一个http post请求。那么我们只需要在服务器上部署一个web服务,当接收到post请求的时候就运行服务器端的shell脚本,拉取镜像,运行容器,这样程序就部署起来了。
新建publishhook服务
上面已经说了为了接收容器镜像服务发出的post请求,需要一个web服务来接收处理请求。这个服务很简单,使用asp.net mvc都是杀鸡用牛刀,仅仅是监控一个请求而已。这里我使用另外一个*aserverhttps://github.com/kklldog/aserver 。
1. 新建一个控制台程序,取名publishhook
2. 使用nuget安装aserver
3. 修改program的main函数
using agile.frameworknetcore.log; using system; using system.diagnostics; namespace publishhook { class program { static void main(string[] args) { console.writeline("publishhook is running now !"); var server = new agile.aserver.server(); server.addhandler(new agile.aserver.httphandler() { method = "post", path = "/api/hook", handler = (req, resp) => { string shell_name = req.query.shell; if (!string.isnullorempty(shell_name)) { runshell(shell_name); } return resp.write("ok"); } }); server .setip("0.0.0.0") .setport(9000) .run(); console.read(); } static void runshell(string filename) { var processstartinfo = new processstartinfo(filename) { redirectstandardoutput = true }; var process = process.start(processstartinfo); if (process == null) { console.writeline("can not run shell ."); } else { using (var sr = process.standardoutput) { while (!sr.endofstream) { var str = sr.readline(); console.writeline(str); logger.info(str); } if (!process.hasexited) { process.kill(); } } } } } }
启动一个http server监听9000端口,添加一个http handler,接收请求,解析querystring获取脚本名称,然后运行脚本
运行publish_hook
sudo dotnet restore sudo dotnet publish
使用dotnet publish命令发布这个程序,然后复制到服务器上。
sudo dotnet publishhook.dll
使用dotnet命令在服务器上运行这个服务。注意:这个服务不能使用docker运行,因为它要运行shell脚本来操作宿主机的docker。如果这个服务跑在容器内,那么它执行的shell是相对于它的容器来说的,无法操作宿主机的docker环境。
复制上次新建的publish_cicd_test.sh脚本文件到publishhook程序目录并赋予权限
复制上次新建的publish_cicd_test.sh脚本文件到publishhook程序目录,一遍程序能够从根目录读取。
chmod +x publish_cicd_test.sh
使用chmod +x给shell脚本赋值可执行权限
在容器镜像服务新建触发器
点击创建触发器
在新建界面填写触发器名称,触发器url。这个url就是publishhook监听的地址
测试一下
配置好容器镜像服务的触发器后,我们的配置工作基本都完成了。让我们修改一下corecicdtest项目,然后push到gitee上,看push后能不能全自动的部署成功。
@{ viewdata["title"] = "home page"; } <h3> .net core cicd test -- v 3.0 </h3>
修改home/index首页,从v2.0改为v3.0,然后使用git push命令推送代码。等待一会后,访问一下corecicdtest的网址。
wow!可以看到我们的网址已经自动部署成功了,终于完成了我们一开始设定的目标。
总结
回顾整个过程,我们可以发现各个服务之间虽然是彼此独立,但是我们可以通过webhook功能串联起来。甚至最后我们自己定义了一个webhook的监听程序来替我们执行对应的脚步。其实通过这种思想我们可以把更多的流程串联起来,实现更多自动化流程。
这次我们顺利的使用阿里云的codepipeline、容器镜像服务,实现了最基本的cicd。现在各大云服务厂商基本都提供了很多基础功能,而且大部分是免费的,有效的利用这些服务可以节省宝贵的时间,开发者可以更专注在核心业务上面。
上一篇: 关于C#中Convert.ToInt32()是干什么用的
下一篇: 自制肉松这样做最好吃,不信你试试