我正在尝试将.ps1文件转换为Windows服务运行。这需要作为业务连续性的要求作为服务运行(计划任务不是一个选择)。我一直使用NSSM来包装.ps1,因为它随后将通过NSSM作为exe运行。
这适用于Windows Server 2012中的不同脚本,但是此脚本略有不同,我需要让此服务才能在Windows Server 2016上运行。脚本本身连接到大量其他服务器(总计将具有3个服务-Windows服务/ Windows进程/ Linux进程),这些服务仅在PowerShell中运行时都可以工作。
下面是脚本开始的示例,因此您可以了解脚本的工作原理(可能不相关);
while ($test = 1)
{
[string]$query
[string]$dbServer = "DBSERVER" # DB Server (either IP or hostname)
[string]$dbName = "DBNAME" # Name of the database
[string]$dbUser = "CONNECTIONUSER" # User we'll use to connect to the database/server
[string]$dbPass = "CONNECTIONPASSWORD" # Password for the $dbUser
$conn = New-Object System.Data.Odbc.OdbcConnection
$conn.ConnectionString = "Driver={PostgreSQL Unicode(x64)};Server=$dbServer;Port=5432;Database=$dbName;Uid=$dbUser;Pwd=$dbPass;"
$conn.open()
$cmd = New-object System.Data.Odbc.OdbcCommand("select * from DBNAME.TABLENAME where typeofcheck = 'Windows Service' and active = 'Yes'",$conn)
$ds = New-Object system.Data.DataSet
(New-Object system.Data.odbc.odbcDataAdapter($cmd)).fill($ds) | out-null
$conn.close()
$results = $ds.Tables[0]
$Output = @()
foreach ($result in $results)
{
$Hostone = $Result.hostone
$Hosttwo = $Result.hosttwo
$Hostthree = $Result.hostthree
$Hostfour = $Result.hostfour
Write-Output "checking DB ID $($result.id)"
#Host One Check
if (!$result.hostone)
{
$hostonestatus = 17
$result.hostone = ""
}
else
{
try
{
if(Test-Connection -ComputerName $result.hostone -quiet -count 1)
{
$hostoneres = GET-SERVICE -COMPUTERNAME $result.hostone -NAME $result.ServiceName -ErrorAction Stop
$hostonestatus = $hostoneres.status.value__
$Result.HostOneCheckTime = "Last checked from $env:COMPUTERNAME at $(Get-date)"
}
else
{
$hostonestatus = 0
$result.hostonestatus = "Failed"
$Result.HostOneCheckTime = "Last checked from $env:COMPUTERNAME at $(Get-date)"
}
}
catch
{
$hostonestatus = 0
$result.hostonestatus = "Failed"
$Result.HostOneCheckTime = "Last checked from $env:COMPUTERNAME at $(Get-date) Errors Found"
}
if ($hostonestatus -eq 4)
{
$result.hostonestatus = "Running"
}
if ($hostonestatus -eq 1)
{
$result.hostonestatus = "Stopped"
}
elseif ($hostonestatus -eq 0)
{
$result.hostonestatus = "Failed"
}
}
如上所述,运行独立的确切脚本可以无缝运行。
将其作为服务运行的最佳方法是什么,或者与Windows Server 2016一起使用时,NSSM是否存在任何已知问题?
我还发现下面的内容可能指向正确的方向,因为我断断续续地在日志中找到了这些内容;
这里是Windows sysadmin。
从纯粹的面向服务的角度,安静地完成此操作的几种方法。
--- 1 ---
[如果您使用的是Server 2016,我相信Powershell命令'New-Service'可能是最干净的方法之一。看看下面的语法,如果它适合您的用例-
https://support.microsoft.com/en-au/help/137890/how-to-create-a-user-defined-service
此CMDlet带有凭据参数,因此根据您的用例,可能会很好地访问其他外部计算机上的资源。
--- 2 ---
另一种方法是在Windows中使用旧的可信任内置SC.exe实用程序。
SC CREATE <servicename> Displayname= "<servicename>" binpath= "srvstart.exe <servicename> -c <path to srvstart config file>" start= <starttype>
示例-
SC CREATE Plex Displayname= "Plex" binpath= "srvstart.exe Plex -c C:PlexService.ini" start= auto
据我所知,这将创建一个将在本地系统上下文中执行的服务。有关更多信息,请查看以下内容:
https://support.microsoft.com/en-au/help/251192/how-to-create-a-windows-service-by-using-sc-exe
https://www.howtogeek.com/50786/using-srvstart-to-run-any-application-as-a-windows-service/
--- 3 ---
[您可能要考虑手动注入一些注册表项来创建自己的服务(这实际上是SC.exe在后台执行的操作。]
尽管很遗憾,我目前无法提供样板代码,但我鼓励您查看以下资源:
https://www.osronline.com/article.cfm%5Eid=170.htm
注意-您需要提供所有必需的子键才能使它起作用。
与任何注册表更改一样,在进行任何更改之前,请备份您的注册表并自行承担修改的风险。我只能建议在实现生产之前在备用/测试VM上尝试此操作。