.NET遇上Docker - Docker集成Cron定时运行.NETCore(ConsoleApp)程序.md
配置项目的Docker支持
对于VS中Docker的配置,依旧重复一些废话。
给项目添加Docker支持,VS2015可以直接使用Docker for VS插件,VS2017在安装时选择容器支持。VS配置好容器支持后,右键点击项目,添加菜单中就可看到Docker Support选项。
VS2015的Docker for VS插件会把Dockerfile加入到package.json的publishOptions中。这样一次dotnet publish操作就可以获得含有Dockerfile可以直接用于生成Image的输出。
而VS2017中,使用“添加-Docker支持”菜单添加的Dockerfile文件不会自动配置为输出到输出目录,可以选择Dockerfile并右击属性,在属性选项卡中,选择“如果较新则复制”。
在Linux使用cron定时运行.NETCore App
一个坑
如果直接在crontab -e中通过下面的方法添加任务:
*/2 * * * * dotnet /publish/your.app.dll #2分钟运行一次,用于测试
肯定是没有办法执行的,通过tail -f /var/log/cron可以看到如下错误输出(CentOS):
(root) CMDOUT (/bin/sh: dotnet: 未找到命令)
爬坑
我们需要使用如下脚本来帮助运行.NetCore应用:
#!/bin/sh
export PATH=$PATH:/publish:/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/root/bin
cd /publish
dotnet your.app.dll >> /var/log/cron.log 2>&1 #将应用的输出重定向(可选),默认输出到`/var/log/cron`
exit;
EOF
一些已知限制,应用程序目录中不能含有
.
为避免在cron输出日志时遇到
No MTA installed, discarding output这样的错误,建议重定向应用的输出到指定的文件
将这个文件保存为rundotnet.sh并设置输出到发布文件中。这个文件将被cron调用定时来运行.NetCore程序。
将使用crontab -e修改cron任务如下(添加后可使用crontab -l验证添加结果):
*/2 * * * * /publish/rundotnet.sh
注意修改rundotnet.sh的权限使其具有执行权限
另一种方式是直接将应用发布为可执行程序。
亲测使用VS2017发布.NetCore App(.NETCore版本1.0.4),可以在安装有.NetCore SDK 1.0.3的Linux上使用下面的cron配置正常执行。
*/2 * * * * /publish/your.app
同样需要赋予your.app文件执行权限
而VS2015发布.NetCore App(.NETCore版本1.0.0),在安装有.NetCore SDK 1.0.1的Linux上就不能正常执行。
整合到Docker中
创建一个名为crontab的文件用于保存cron命令,内容就是我们之前使用crontab -e添加的任务:
*/2 * * * root /publish/rundotnet.sh # 用户名root在这里不可少,根据需要修改定时执行的时间
# 合法的cron文件需要留一行空行
可能有眼尖的园友发现为什么这里有
root这个用户名而前文没有,前文的测试是在一台RedHat系(CentOS7)上进行的。而微软的.NETCore Docker Iamge基于Debian系系统,这里必须有用户名,不然会报类似“crontab文件格式不合法“类似的错误。
创建好文件后将其配置为发布到输出目录,以便可以在dotnet pulish后被docker build所使用
然后添加一个文件runcron.sh,这将作为docker container的执行入口
rsyslogd
cron
touch /var/log/cron.log
tail -F /var/log/syslog /var/log/cron.log
最后一行分别输出系统日志(cron执行情况,在syslog中)和dotnet app的输出(在cron.log中,前提是前面添加了输出的重定向)
tail命令将保证container一致执行,不会退出
也可以用docker的--restart=always参数控制容器的运行
修改VS创建的Dockerfile文件中的内容为类似如下的样子:
FROM microsoft/dotnet:1.0-runtime
COPY . /publish
WORKDIR /publish
# 开始配置cron
# microsoft/dotnet:1.0-runtime镜像不包含cron,先安装,顺便安装rsyslog输出日志
RUN apt-get update \
&& apt-get install -y --no-install-recommends \
cron \
rsyslog \
&& rm -rf /var/lib/apt/lists/*
# 将我们编辑的crontab任务文件添加到cron配置目录
ADD crontab /etc/cron.d/dotnet-cron
# 赋予crontab文件读权限(rw/r/r)
RUN chmod 0644 /etc/cron.d/dotnet-cron
# 创建日志文件,以便执行tail命令
RUN touch /var/log/cron.log
# 赋予runcron.sh和rundotnet.sh执行权限
RUN chmod +x /publish/rundotnet.sh
RUN chmod +x /publish/runcron.sh
# 执行入口脚本runcron.sh
CMD ["bash","/publish/runcron.sh"]
在Dockerfile中我们向基础容器添加了cron和rsyslog,在debian/ubuntu上(dotnet官方镜像的基础系统)cron需要靠rsyslog来输出日志
最终输出目录要包含下列文件:
"appsettings.json",
"Dockerfile",
"crontab",
"rundotnet.sh",
"runcron.sh"
使用如下命令publish项目
dotnet publish --framework netcoreapp1.0 --configuration release --output publish
将publish中文件传到安装有docker的环境中准备生成docker image
生成Docker映像,运行Container
我们将生成运行(包含清理旧映像)的脚本整合为一个buildhelper.sh:
#!/bin/bash
contName=$1
contName=${contName:="containerName"}
imagName=$2
imagName=${imagName:="orgname/projname:tag"}
docker stop ${contName}
docker rm -f ${contName}
docker rmi ${imagName}
docker ps -a
docker images
sudo docker build --rm -t orgname/projname:tag .
sudo docker run --name containerName -it orgname/projname:tag
依然配置此文件发布到输出到发布目录(在Linux上要赋予执行权限)。
容器运行后可以连接到container的bash查看运行情况
docker exec -it containerName /bin/bash
如果感觉测试时,每次Docker Image都会进行apt-get安装cron很麻烦,可以先生成一个包含cron以及rsyslog的镜像,并以此作为这个例子的基础镜像。(按照博主提供的这些基本可以一次成功)
提示
所有shell脚本文件(xxx.sh),如果是UTF8格式,都要使用无BOM的UTF8编码。或者使用ASCII。不然bash无法正常运行。
一键去BOM的方法: grep -r -I -l $'^\xEF\xBB\xBF' /path | xargs sed -i 's/^\xEF\xBB\xBF//g'
如果在Windows里编辑过sh文件,在linux种可能出现错误:“/bin/bash^M:损坏的解释器: 没有那个文件或目录”
这是由于两种系统换行符不同,Windows为\n\r,Linux为\n,而\r会被显示为^M。
一键替换 sed -i 's/\r$//g' filename
大致有如下文件需要处理:
sed -i 's/\r$//g' crontab
sed -i 's/\r$//g' buildhelper.sh
sed -i 's/\r$//g' rundotnet.sh
sed -i 's/\r$//g' runcron.sh
crontab处理换行很重要,不然在linux会导致必须存在的空行无法被识别而无法正茬加载。
rsyslogd的如下错误提示可以安全忽略:
rsyslogd: imklog: cannot open kernel log(/proc/kmsg): Operation not permitted.
而错误:
Could not open output pipe '/dev/xconsole': No such file or directory
暂时不知道有无影响(实测不会对rsyslog的工作产生影响)
定时任务时间 对于定时任务,要注意如之前使用的基础镜像microsoft/dotnet:1.0.1-core的时区都是UTC时间,对于我们UTC+8的地方要自行在配置cron的时间时加上8个小时。
好办法 在生成镜像的时候直接修改好时区,Dockerfile中加入 RUN cp /usr/share/zoneinfo/Asia/Shanghai /etc/localtime
可能会有小朋友质疑,用Docker运行这种定时任务有必要吗,楼主感觉Docker最主要的还是屏蔽不同底层系统的差异。顺便可以让不同的应用的定时任务可以分离。本文也是提供一种方法,供有需要的童鞋参考。
.NET遇上Docker - Docker集成Cron定时运行.NETCore(ConsoleApp)程序.md的更多相关文章
- (转)Permission denied: win7下面eclipse上传本地文件到DFS && 运行M/R程序时出现的同样的错误解决方法
原文地址: http://mntms.iteye.com/blog/2095651 hadoopeclipse远程控制权限 情景一: 当在win7下面的eclipse装好插件,首次运行M/R程序的时 ...
- Docker学习笔记-磁盘挂载运行.netcore
前言: 环境:centos7.5 64 位 正文: 首先我们在宿主机上安装 .NET Core SDK sudo rpm --import https://packages.microsoft.com ...
- raspberrypi(树莓派)上安装mono和jexus,运行asp.net程序
参考网址: http://www.linuxdot.net/ http://www.cnblogs.com/mayswind/p/3279380.html http://www.raspberrypi ...
- 当DOCKER遇上ESXI
特别是你要为DOCKER窗口设置静态IP,且和公司局域网打成一片的时候, 苦逼的测试就会开始,我差不多前前后后测试了四五天,一百多个容器报废. NETNS,NSENTER,PIPWORK,各种镜像合下 ...
- .NET遇上Docker - 使用Docker Compose组织Ngnix和.NETCore运行
本文工具准备: Docker for Windows Visual Studio 2015 与 Visual Studio Tools for Docker 或 Visual Studio 2017 ...
- .NET遇上Docker - Harbor的安装与基本使用
Harbor是一个开源企业级Docker注册中心,可以用于搭建私有的Docker Image仓库.可以实现权限控制等. 安装Harbor 首先,需要安装Docker和Docker Compose,参考 ...
- 从零开始搭建Jenkins+Docker自动化集成环境
本文只简单标记下大概的步骤,具体搭建各个部分的细节,还请自行搜索.第一.二部分只是对Jenkins和Docker的简单介绍,熟悉的同学请直接跳到第三部分. 一.关于Jenkins Jenkins简介 ...
- Jenkins + Docker 持续集成
Jenkins介绍 Jenkins是一个开源软件项目,是基于Java开发的一种持续集成工具,用于监控持续重复的工作,旨在提供一个开放易用的软件平台,使软件的持续集成变成可能. 安装部署Jenkins ...
- Jenkins+Docker持续集成
本节内容: Jenkins介绍 安装部署Jenkins Jenkins构建maven风格的job Jenkins邮件通知设置 Sonar Jenkins与Docker结合 一.Jenkins介绍 Je ...
随机推荐
- 1901: Zju2112 Dynamic Rankings
1901: Zju2112 Dynamic Rankings Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 5268 Solved: 2207[Su ...
- Postgresql 经纬度
增加一列: add column `location` geometry default p.p1 { margin: 0.0px 0.0px 0.0px 0.0px; font: 12.0px &q ...
- gcc下c++的对象模型 (1)
所有示例代码在如下环境中执行 ubuntu 16.04.4 (64位) gcc version 5.4.0 开启std11 gdb version 7.11.1 1. 空类的大小 定义一个空类A,实例 ...
- Unity文档总结(2)-Understanding Automatic Memory Management
当一个对象.字符串.数组被创建的时候,从中间池申请分配需要存储的内存称为堆.当该部分不在使用时,一度占用的内存被释放收回,用于别的事物.在过去,它通常由开发人员分配和释放这些堆内存块,明确相应的功能调 ...
- lvm的vg扩容
本次扩容的目的是要扩展 / 的整体容量,具体操作如下: 1.首先查看是否存在未分配的磁盘 [root@NH-Test-44 ~]# fdisk -l Disk /dev/vda: 53.7 GB, 5 ...
- nfs服务端配置文件详解
配置参数语法 配置文件路径 /etc/exports 可以使用#进行注释说明 一个完整的配置实例至少需要3个参数 1.被共享的目录 必须是存在的目录 2.向谁共享 可以指定主机或网络范围,支持通配符* ...
- iOS截取http/https流量
0x01.Why? 做移动测试的同学经常会在app和server中间架设一个代理(例如charles或者fiddler等),由经代理,app和server之间的交互及交互内容变得可视化,使得我们不再摸 ...
- python 基础知识(待补充)
数据结构: python数据结构共包含四大类: 列表 列表是可变元素,支持增删改查操作 字典 字典是遍历,key:value 格式. 元组 元组数据不可变元素,仅仅有两个操作 index, coun ...
- Project下载提示检索 COM 类工厂中 CLSID 为 {36D27C48-A1E8-11D3-BA55-00C04F72F325} 的组件失败
做后台系统导出Project时,部署到服务器提示:检索 COM 类工厂中 CLSID 为 {36D27C48-A1E8-11D3-BA55-00C04F72F325} 的组件失败,原因是出现以下错误: ...
- (转)POPTEST联合创始人李爱然的“IT培训创业的随想"
IT教育行业最大的问题是缺少像互联网行业一样的产品经理. 大多数IT教育机构在早期依靠个人或者一套课程开创了一定的局面,随着机构的壮大,机构把市场营销提到至高点,销售至上,而把产品(培训产品)放在后面 ...