Нужен способ программно проверить статус службы Windows
Вот ситуация:
Мне было предложено работать с InstallAnywhere 8, созданной на основе Java IDE для установки, которая позволяет запускать и останавливать службы Windows, но не имеет встроенного метода для запроса своих состояний. К счастью, он позволяет создавать пользовательские действия в Java, которые могут быть вызваны в любое время в процессе установки (в качестве того, что я считаю довольно запутанным API).
Мне просто нужно что-то, что скажет мне, запущена или остановлена конкретная служба.
IDE также позволяет вызывать пакетные скрипты, так что это также вариант, хотя после запуска script почти невозможно проверить, что это удалось, поэтому я стараюсь избегать этого.
Любые предложения или критика приветствуются.
Ответы
Ответ 1
вот что я должен был сделать. Это уродливо, но прекрасно работает.
String STATE_PREFIX = "STATE : ";
String s = runProcess("sc query \""+serviceName+"\"");
// check that the temp string contains the status prefix
int ix = s.indexOf(STATE_PREFIX);
if (ix >= 0) {
// compare status number to one of the states
String stateStr = s.substring(ix+STATE_PREFIX.length(), ix+STATE_PREFIX.length() + 1);
int state = Integer.parseInt(stateStr);
switch(state) {
case (1): // service stopped
break;
case (4): // service started
break;
}
}
runProcess
- частный метод, который запускает заданную строку как процесс командной строки и возвращает полученный результат. Как я уже сказал, уродливо, но работает. Надеюсь, это поможет.
Ответ 2
Вы можете создать небольшую VBS на лету, запустить ее и захватить код возврата.
import java.io.File;
import java.io.FileWriter;
public class VBSUtils {
private VBSUtils() { }
public static boolean isServiceRunning(String serviceName) {
try {
File file = File.createTempFile("realhowto",".vbs");
file.deleteOnExit();
FileWriter fw = new java.io.FileWriter(file);
String vbs = "Set sh = CreateObject(\"Shell.Application\") \n"
+ "If sh.IsServiceRunning(\""+ serviceName +"\") Then \n"
+ " wscript.Quit(1) \n"
+ "End If \n"
+ "wscript.Quit(0) \n";
fw.write(vbs);
fw.close();
Process p = Runtime.getRuntime().exec("wscript " + file.getPath());
p.waitFor();
return (p.exitValue() == 1);
}
catch(Exception e){
e.printStackTrace();
}
return false;
}
public static void main(String[] args){
//
// DEMO
//
String result = "";
msgBox("Check if service 'Themes' is running (should be yes)");
result = isServiceRunning("Themes") ? "" : " NOT ";
msgBox("service 'Themes' is " + result + " running ");
msgBox("Check if service 'foo' is running (should be no)");
result = isServiceRunning("foo") ? "" : " NOT ";
msgBox("service 'foo' is " + result + " running ");
}
public static void msgBox(String msg) {
javax.swing.JOptionPane.showConfirmDialog((java.awt.Component)
null, msg, "VBSUtils", javax.swing.JOptionPane.DEFAULT_OPTION);
}
}
Ответ 3
Я занимаюсь установщиками в течение многих лет, и трюк заключается в том, чтобы создать свой собственный EXE и вызвать его при настройке. Это обеспечивает хорошую гибкость, например, отображение точных сообщений об ошибках в случае возникновения ошибки и наличие возвращаемых значений на основе успеха, чтобы ваш установщик знал о том, что произошло.
Здесь, как запускать, останавливать и запрашивать состояния для служб Windows (С++):
http://msdn.microsoft.com/en-us/library/ms684941(VS.85).aspx
(VB и С# предлагают аналогичные функции)
Ответ 4
Основываясь на других ответах, я построил следующий код для проверки состояния службы Windows:
public void checkService() {
String serviceName = "myService";
try {
Process process = new ProcessBuilder("C:\\Windows\\System32\\sc.exe", "query" , serviceName ).start();
InputStream is = process.getInputStream();
InputStreamReader isr = new InputStreamReader(is);
BufferedReader br = new BufferedReader(isr);
String line;
String scOutput = "";
// Append the buffer lines into one string
while ((line = br.readLine()) != null) {
scOutput += line + "\n" ;
}
if (scOutput.contains("STATE")) {
if (scOutput.contains("RUNNING")) {
System.out.println("Service running");
} else {
System.out.println("Service stopped");
}
} else {
System.out.println("Unknown service");
}
} catch (IOException e) {
e.printStackTrace();
}
}
Ответ 5
У меня была некоторая удача в прошлом с помощью Java Service Wrapper. В зависимости от вашей ситуации вам может потребоваться заплатить, чтобы использовать его. Но он предлагает чистое решение, которое поддерживает Java и может быть использовано в среде InstallAnywhere с (я думаю) небольшими проблемами. Это также позволит вам поддерживать службы в блоках Unix.
http://wrapper.tanukisoftware.org/doc/english/download.jsp
Ответ 6
Снимок в темноте, но посмотрите на свою документацию java Install Anywhere.
В частности,
/javadoc/com/installshield/wizard/platform/win32/Win32Service.html
Класс:
com.installshield.wizard.platform.win32
Interface Win32Service
All Superinterfaces:
Service
Метод:
public NTServiceStatus queryNTServiceStatus(String name)
throws ServiceException
Calls the Win32 QueryServiceStatus to retrieve the status of the specified service. See the Win32 documentation for this API for more information.
Parameters:
name - The internal name of the service.
Throws:
ServiceException
Ответ 7
Здесь выполняется решение straignt С#/P/Invoke.
/// <summary>
/// Returns true if the specified service is running, or false if it is not present or not running.
/// </summary>
/// <param name="serviceName">Name of the service to check.</param>
/// <returns>Returns true if the specified service is running, or false if it is not present or not running.</returns>
static bool IsServiceRunning(string serviceName)
{
bool rVal = false;
try
{
IntPtr smHandle = NativeMethods.OpenSCManager(null, null, NativeMethods.ServiceAccess.ENUMERATE_SERVICE);
if (smHandle != IntPtr.Zero)
{
IntPtr svHandle = NativeMethods.OpenService(smHandle, serviceName, NativeMethods.ServiceAccess.ENUMERATE_SERVICE);
if (svHandle != IntPtr.Zero)
{
NativeMethods.SERVICE_STATUS servStat = new NativeMethods.SERVICE_STATUS();
if (NativeMethods.QueryServiceStatus(svHandle, servStat))
{
rVal = servStat.dwCurrentState == NativeMethods.ServiceState.Running;
}
NativeMethods.CloseServiceHandle(svHandle);
}
NativeMethods.CloseServiceHandle(smHandle);
}
}
catch (System.Exception )
{
}
return rVal;
}
public static class NativeMethods
{
[DllImport("AdvApi32")]
public static extern IntPtr OpenSCManager(string machineName, string databaseName, ServiceAccess access);
[DllImport("AdvApi32")]
public static extern IntPtr OpenService(IntPtr serviceManagerHandle, string serviceName, ServiceAccess access);
[DllImport("AdvApi32")]
public static extern bool CloseServiceHandle(IntPtr serviceHandle);
[DllImport("AdvApi32")]
public static extern bool QueryServiceStatus(IntPtr serviceHandle, [Out] SERVICE_STATUS status);
[Flags]
public enum ServiceAccess : uint
{
ALL_ACCESS = 0xF003F,
CREATE_SERVICE = 0x2,
CONNECT = 0x1,
ENUMERATE_SERVICE = 0x4,
LOCK = 0x8,
MODIFY_BOOT_CONFIG = 0x20,
QUERY_LOCK_STATUS = 0x10,
GENERIC_READ = 0x80000000,
GENERIC_WRITE = 0x40000000,
GENERIC_EXECUTE = 0x20000000,
GENERIC_ALL = 0x10000000
}
public enum ServiceState
{
Stopped = 1,
StopPending = 3,
StartPending = 2,
Running = 4,
Paused = 7,
PausePending =6,
ContinuePending=5
}
[StructLayout(LayoutKind.Sequential, Pack = 1)]
public class SERVICE_STATUS
{
public int dwServiceType;
public ServiceState dwCurrentState;
public int dwControlsAccepted;
public int dwWin32ExitCode;
public int dwServiceSpecificExitCode;
public int dwCheckPoint;
public int dwWaitHint;
};
}
Ответ 8
При запуске создайте файл с File.deleteOnExit().
Проверьте наличие файла в ваших сценариях.