此片文章目标是将 .Net Core 发布到 Docker 上,并且连接到在 Docker上的 Redis 、上传文件到本机文件夹和连接 sqlserver 数据库。
创建项目就不用说了,我是用得 vs2019 创建的 net core 3.1 的项目,把上传文件、连接redis和sqlserver的代码写好,因为我是在 window 环境下开发的,所以如果我们需要发布到 linux 系统的话需要注意所有用到路径的地方,比如我遇到的这几个点:
1.保存文件地址
如下在 window 上保存文件的话地址是如下使用 '' 的
//新文件名(包括路径)
filename = Environment.CurrentDirectory + @"\upload\" +filename;
在 linux 系统下就需要用 '/' 否则可保存不了呀
//新文件名(包括路径)
filename = Environment.CurrentDirectory + @"/upload/" +filename;
2.加载程序集
我是通过如下方法动态加载实体映射类型的,看下面标黄的代码,在获取的路径前加了一个 “/” 是因为在 linux 中需要使用绝对路径,像是 /app/xxx 这种,不加的话就会报错
protected override voidOnModelCreating(ModelBuilder modelBuilder)
{
//通过反射获取继承IEntityTypeConfiguration的实体类型
string assembleFileName = Assembly.GetExecutingAssembly().CodeBase.Replace("Capricorn.Db.SqlServer.dll", "Capricorn.Entity.Mapping.dll").Replace("file:///", "");
Assembly asm = Assembly.LoadFile("/" +assembleFileName);
//.....省略其他代码}
报错内容,看到这个就要注意路径问题了
Absolute path information is required. (Parameter 'path')
我们还要在 Program.cs 文件下加上如下标黄的代码,设置端口
public static IWebHostBuilder CreateWebHostBuilder(string[] args) =>WebHost.CreateDefaultBuilder(args)
.UseUrls("http://*:80")
.UseStartup<Startup>();
发布代码前需要先添加 Docker 支持,右键项目->添加-》Docker 支持 后就会让你选择发布的目标系统如下,我这里选择 linux
选择完成后会添加一个 Dockerfile 文件,会帮我们生成相应的构建镜像的代码,但是我们可以简化到如下
FROM mcr.microsoft.com/dotnet/core/aspnet:3.1-buster-slim AS baseRUN sed -i 's/TLSv1.2/TLSv1.0/g' /etc/ssl/openssl.cnf
WORKDIR /app
EXPOSE 80COPY . .
ENTRYPOINT ["dotnet", "Capricorn.dll"]
注意 Dockerfile 文件中的第二行 RUN 的代码因为需要更新镜像环境的 TLS 版本,否则在连接 sqlserver 的时候可能会报如下错误:
A connection was successfully established with the server, but then an error occurred during the pre-login handshake. (provider: TCP Provider, error: 35 - An internal exception was caught
之后就发布为文件系统就行如下,我发布到 D:\website
之后我们到发布的目标位置下,打开 PowerShell 执行如下打包代码
PS D:\website> docker build -t myweb .
等待 Dockerfile 中的基础镜像下载完成,后就会完成我们打包的镜像了
备注
其实我们还可以通过 vs 进行打包,我们如下图所示,在工具栏出选择 Docker,之后执行就会自动进行打包
但是会有个问题,你可能会发现打包了半天都打包不完,一直显示容器预热中,然后在输出视图中看到容器工具中输出停止在如下代码,这三个内容很关键
Info: Using vsdbg version '16.8.11013.1'Info: Using Runtime ID 'linux-x64'Info: C:\Users\Xu\vsdbg\vs2017u5 exists, deleting.
要解决这个问题我们可以手动拼接下载路径,下载路径如下
https://vsdebugger.azureedge.net/vsdbg-(你的版本号 .号换成-号)/vsdbg-(你的Runtime ID).zip
比如按照我的输出中的内容就需要如下的下载链接
https://vsdebugger.azureedge.net/vsdbg-16-8-11013-1/vsdbg-linux-x64.zip
下载完成后在我们在需要的路径下直接解压文件,再创建两个文本文件 success_version.txt 和 success_rid.txt 。在 success_version.txt 文件中添加我们的版本号(我这是16.8.11013.1),在 success_rid.txt 中添加我们的 Runtime ID (我这是linux-x64),之后取消执行,重启项目,然后在重新生成就行啦。
我们因为需要连接 redis 所以需要一个网络使容器可以互相通信,建立网络代码如下
PS D:\website> docker network create -d bridge test-net
之后我们运行项目的容器做如下操作
1.绑定刚刚创建的网络 test-net,可以与后面的 redis 通信
2.挂载容器中的路径 /app 到本机的发布路径 d:\website 下,这样我们如果代码需要重新发布,在发布完成后直接重启容器就可以生效了
3.绑定容器内的 80 端口到本机的 5000 端口,可以通过访问本地端口访问网站
PS D:\website> docker run -itd --name myweb --network test-net -v d:\website:/app -p 5000:80 myweb
运行Redis容器,我们直接用 Docker Hub 中的镜像
PS D:\website> docker run -itd --name redis_myweb --network test-net redis
如果数据库用的也是容器那和 redis 的操作差不多,不过我这里使用的是本机上的 sqlserver 数据库,具体在 appsetting.json 的配置内容如下
{
"ConnectionStrings": {
"BaseDb": {
"ProviderName": "System.Data.SqlClient",
"ConnectionString": "Data Source=host.docker.internal,1433;Database=****;User ID=sa;Password=*********"},
"Redis": {
"ConnectionString": "redis_myweb",
"Prefix": "Cap_"}
}
}
可以看到,由于 redis 是在容器中运行的,所以连接字符串为运行的容器的名称。而数据库因为连接本机,所以地址使用了 host.docker.internal 作为地址。host.docker.internal 表示主机的地址,在主机为 windows 的机器中才能用,如果 redis 想使用本机的,那连接字符串也可以用 host.docker.internal 。
我们访问地址 http://localhost:5000/ 可以看到访问成功了
之后上传文件,也可以看到我们本机挂载的文件夹下多了一个 upload 文件夹,里面多了我们上传的文件
访问数据库可以看到成功返回了数据
设置缓存也成功啦
版权说明 : 本文为转载文章, 版权归原作者所有 版权申明
原文链接 : https://www.cnblogs.com/xwc1996/p/14169758.html
内容来源于网络,如有侵权,请联系作者删除!