ASP.NET Core使用TopShelf部署Windows服务
asp.net core很大的方便了跨平台的开发者,linux的开发者可以使用apache和nginx来做反向代理,windows上可以用iis进行反向代理。
反向代理可以提供很多特性,固然很好。但是还有复杂性,我们也可以使用windows service来直接启动kestrel。
asp.net core官方网站提供了一种基于windows服务部署的方法:在 windows 服务中托管 asp.net core
这种方式需要修改代码,然后部署的时候,使用命令行创建、安装服务,然后再启动。
感觉还是不够爽快,我们可以使用topshelf改造一下。
topshelf
topshelf可以很便捷地将一个windows console程序改造成windows service,只需要稍微修改一下代码结构,然后通过nuget包就可以简单操作了。安装与部署也是极其方便,而且,topshelf在调试的时候,直接是作为console程序,极其便于调试。
topshelf项目地址:
步骤
首先引用nuget包:
install-package topshelf
然后改造一下program.cs
public class program { public static void main(string[] args) { var rc = hostfactory.run(x => //1 { x.service<mainservice>(s => //2 { s.constructusing(name => new mainservice(args)); //3 s.whenstarted(tc => tc.start()); //4 s.whenstopped(tc => tc.stop()); //5 }); x.runaslocalsystem(); //6 x.setdescription("jwtapiservice"); //7 x.setdisplayname("jwtapiservice"); //8 x.setservicename("jwtapiservice"); //9 }); //10 var exitcode = (int)convert.changetype(rc, rc.gettypecode()); //11 environment.exitcode = exitcode; //createwebhostbuilder(args).build().runasservice(); } }
这里指定服务程序的内容在mainservice
这个类里面,并通过代码指定了服务的名称和描述等行为。以前的启动createwebhostbuilder方法转移到了这个类中:
public class mainservice { private string[] args; public mainservice(string[] vs) { args = vs; } public void start() { var isservice = !(debugger.isattached || args.contains("--console")); var builder = createwebhostbuilder(args.where(arg => arg != "--console").toarray()); if (isservice) { var pathtoexe = process.getcurrentprocess().mainmodule.filename; var pathtocontentroot = path.getdirectoryname(pathtoexe); builder.usecontentroot(pathtocontentroot); } var host = builder.build(); host.run(); } public void stop() { } public static iwebhostbuilder createwebhostbuilder(string[] args) { var config = new configurationbuilder() // .setbasepath(directory.getcurrentdirectory()) .addjsonfile("config.json", optional: true, reloadonchange: true) .build(); return webhost.createdefaultbuilder(args) .usekestrel() .useconfiguration(config) .usestartup<startup>(); } }
start方法指定服务启动时,服务的执行不需要依赖于microsoft.aspnetcore.hosting.windowsservices这个nuget包。
另外contentroot需要注意,使用windows服务进行提供服务,getcurrentdirectory的根目录是system32,而不是asp.net core的dll的目录。使用appsettings.json时,可能会引起问题,最好使用自定义的程序配置(例如这里通过config.json进行设置)。
运行
- 确定是否存在 windows 运行时标识符 (rid),或将其添加到包含目标框架的
中:
<propertygroup> <targetframework>netcoreapp2.1</targetframework> <runtimeidentifier>win7-x64</runtimeidentifier> </propertygroup>
- 发布,最终可以得到可执行程序。直接双击运行,程序就可以以console的形式启动,方便调试。
- 命令行运行xxxx.exe install就而可以安装服务,然后服务就可以自动启动。
- 命令行运行xxxx.exe uninstall就可以卸载服务。整个过程不需要新建用户与策略。
后记
吐槽:直接使用topshelf,调试windows服务的过程变得不那么痛苦了,想起附加调试器的过程,简直了。
p.s. 需要最新版本的topshelf才可以支持asp.net core的服务部署。
推荐阅读
-
ASP.NET Core部署前期准备 使用Hyper-V安装Ubuntu Server 16.10
-
使用Topshelf组件构建简单的Windows服务
-
.Net Core WebApi(二)在Windows服务器上部署
-
.net core3.0部署Linux服务器 使用Docker容器和Nginx反代理教程
-
使用 Topshelf 组件一步一步创建 Windows 服务 (2) 使用Quartz.net 调度
-
ASP.NET Core Web程序托管到Windows 服务
-
ASP.NET Core 3.0 : 二十八. 在Docker中的部署以及docker-compose的使用
-
ASP.NET Core部署前期准备 使用Hyper-V安装Ubuntu Server 16.10
-
使用Topshelf组件构建简单的Windows服务
-
.Net Core 在Linux服务器下部署程序--(1). Windows 连接 Linux服务器