7

我在从一个非常简单的 c# 或 c++ gui 程序编译为“windows 应用程序”(而不是“控制台应用程序”)执行时遇到 PsExec 挂起。在下面的 C) 部分下,我粘贴了重现问题的代码,在 D) 部分下,我粘贴了 c++ 代码以重现相同的问题。

当 psexec 挂起时,本地附加到 psexec 后的 windbg 输出粘贴在 B) 部分下。

在转储粘贴在 A) 部分下的输出后,我的程序挂起。

如果您将 psexec 命令替换为任何本地命令,则程序可以正常工作,例如 ProcessStartInfo("cmd.exe", "/c dir c:\windows\*.*");

我想知道是否有人经历过它并找到了解决方案。帮助将不胜感激。

谢谢,沙拉杰什


A) psexec 挂起时我的 c# 程序输出

PsExec v1.98 - 远程执行进程 版权所有 (C) 2001-2010 Mark Russinovich Sysinternals - www.sysinternals.com

驱动器 C 中的卷没有标签。


B) 挂起时 psexec 的 Windbg 输出

3 Id: 1614.15e4 Suspend: 1 Teb: 7efac000 Unfrozen ChildEBP RetAddr Args to Child
02a3fe68 75a6d0c5 00000180 00000000 00000000 ntdll!NtReadFile+0x15 (FPO: [9,0,0]) 02a3fecc 75cb18aa 00000180 02a3ff44 00010000 KERNELBASE!ReadFile+0x118 (FPO : [SEH]) 02a3ff14 00403bde 00000180 02a3ff44 00010000 kernel32!ReadFileImplementation+0xf0 (FPO: [SEH]) 警告:堆栈展开信息不可用。以下框架可能是错误的。02a3ff2c 00000000 00291e48 00000000 02a5ff80 psexec+0x3bde


C)重现问题的c#代码

using System;
using System.Windows.Forms;
using System.Diagnostics;

namespace WindowsFormsApplication1 {
  static class Program {
    static void DataReceiveHandler(object sender, DataReceivedEventArgs e) {
      Debug.WriteLine(e.Data);
    }

    public static void NotWorkingPsExec() {
      ProcessStartInfo startInfo = new ProcessStartInfo("psexec.exe",
        "\\\\raj-2k3-32 cmd.exe /c dir c:\\windows\\*.*");
      startInfo.UseShellExecute        = false;
      startInfo.CreateNoWindow         = true;
      startInfo.RedirectStandardOutput = true;
      startInfo.RedirectStandardError  = true;

      Process proc = new Process();
      proc.StartInfo           = startInfo;
      proc.ErrorDataReceived  += new DataReceivedEventHandler(DataReceiveHandler);
      proc.OutputDataReceived += new DataReceivedEventHandler(DataReceiveHandler);
      proc.Start();
      proc.BeginErrorReadLine();
      proc.BeginOutputReadLine();
      proc.WaitForExit();
      Debug.WriteLine("Exit code = {0}", proc.ExitCode);
    }

    public static void WorkingPsExec() {
      ProcessStartInfo startInfo = new ProcessStartInfo("psexec.exe", 
        "\\\\raj-2k3-32 cmd.exe /c dir c:\\windows\\*.*");
      startInfo.UseShellExecute = false;

      Process proc = new Process();
      proc.StartInfo = startInfo;
      proc.Start();
      proc.WaitForExit();
      Debug.WriteLine("Exit code = {0}", proc.ExitCode);
    }

    static void Main() {
      NotWorkingPsExec();
      //WorkingPsExec(); //If uncommented will work 
    }
  }
}

D)重现问题的c++代码

#include <windows.h>
#include <tchar.h>
#include <stdio.h>
#include <strsafe.h>

HANDLE g_hStdoutRd = NULL;
HANDLE g_hStdoutWr = NULL;

void StartCommand(TCHAR *szCmdline);
void ReadOutput();
void ErrorExit(PTSTR);

int APIENTRY _tWinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPTSTR lpCmdLine, int nCmdShow) {
  SECURITY_ATTRIBUTES saAttr;
  saAttr.nLength              = sizeof(SECURITY_ATTRIBUTES);
  saAttr.bInheritHandle       = TRUE;
  saAttr.lpSecurityDescriptor = NULL;
  if (!CreatePipe(&g_hStdoutRd, &g_hStdoutWr, &saAttr, 0))
    ErrorExit(TEXT("Stdout SetHandleInformation"));
  if (!SetHandleInformation(g_hStdoutRd, HANDLE_FLAG_INHERIT, 0))
    ErrorExit(TEXT("Stdout SetHandleInformation"));
  TCHAR szCmdline[] = TEXT("psexec.exe \\\\raj-2k3-32 cmd.exe /c dir /s c:\\windows\\*.*"); // Not Working
  //TCHAR szCmdline[] = TEXT("cmd.exe /c dir /s c:\\windows\\*.*"); // Working
  StartCommand(szCmdline);
  ReadOutput();
  return 0;
}

void StartCommand(TCHAR *szCmdline) {
  PROCESS_INFORMATION piProcInfo  = {0};
  STARTUPINFO         siStartInfo = {0};
  siStartInfo.cb         = sizeof(STARTUPINFO);
  siStartInfo.hStdError  = g_hStdoutWr;
  siStartInfo.hStdOutput = g_hStdoutWr;
  siStartInfo.dwFlags   |= STARTF_USESTDHANDLES;
  BOOL bSuccess = CreateProcess(NULL, szCmdline, NULL, NULL, TRUE, 0, NULL, NULL, &siStartInfo, &piProcInfo);
  if (!bSuccess)
    ErrorExit(TEXT("CreateProcess"));
  else {
    CloseHandle(piProcInfo.hProcess);
    CloseHandle(piProcInfo.hThread);
  }
}

void ReadOutput() {
  if (!CloseHandle(g_hStdoutWr))
    ErrorExit(TEXT("StdOutWr CloseHandle"));
  for (;; ) {
    CHAR    chBuf[4096] = {0};
    DWORD   dwRead;
    BOOLEAN bSuccess    = ReadFile(g_hStdoutRd, chBuf, ARRAYSIZE(chBuf), &dwRead, NULL);
    if (!bSuccess || dwRead == 0)
      break;
    OutputDebugStringA(chBuf);
  }
}

void ErrorExit(PTSTR lpszFunction) {
  OutputDebugString(lpszFunction);
  ExitProcess(1);
}
4

6 回答 6

7

PSExec 对我来说太随意了。我没有努力以您的方式重新创建问题,但我通过使用“PAExec”避免了我的麻烦,这是一个看似值得的精神继承者: http ://www.poweradmin.com/PAExec/

于 2012-10-16T14:27:55.627 回答
2

我有一个类似的问题,这是由于 eula 造成的,这可能是你的问题:

可能的原因:

1) psiexec.exe 在首次运行时显示 EULA 消息。

2) 权限

3) dll 函数可能需要用户会话。

为避免这些问题,请尝试以下方案:

1) 带有“-accepteula”参数

2) 带有“-s”参数

3) 带有“-i”参数

4) > 2 + 3 5) 2 + 3 + 1

见:

0

我有一个简单的解决方案,

  1. C# 运行如下过程:

    Process.Start("start run.bat xx.txt");    //call it async
    
    //and then we make some code juse wait xx.txt appear and finish written .
    
  2. run.bat是 :

    psexec.bat > %1           //redirect to a text file
    exit
    
  3. psexec.bat是:

    psexec.exe ..........................
     exit
    
于 2013-01-14T04:34:30.760 回答
0

还请在 startInfo 上设置 WorkingDirectory 属性,因为 Sysinternals 实用程序使用文件的运行时解包,并且内核无法找到解包的(真正的)可执行文件。

于 2011-04-13T19:19:28.533 回答
0

构建一个运行PsExec&PsKill和他们所有朋友的通用控制台应用程序。通过您的代码调用此 ConsoleApp 而不是调用NotWorkingPsExec方法,它会正常工作。

于 2011-08-17T20:38:57.063 回答
-1

流上的同步读取工作:

        ProcessStartInfo startInfo = new ProcessStartInfo("psexec.exe", @"\\localhost cmd.exe /c dir c:\windows\*.*");
        startInfo.UseShellExecute = false;
        startInfo.CreateNoWindow = true;
        startInfo.RedirectStandardOutput = true;
        //startInfo.RedirectStandardError = true;
        //startInfo.RedirectStandardInput = true;

        Process proc = new Process();
        proc.StartInfo = startInfo;
        //proc.ErrorDataReceived += new DataReceivedEventHandler(DataReceiveHandler);
        //proc.OutputDataReceived += new DataReceivedEventHandler(DataReceiveHandler);
        proc.Start();
        //proc.BeginErrorReadLine();
        //proc.BeginOutputReadLine();
        string output = proc.StandardOutput.ReadToEnd();
        proc.WaitForExit();
        Console.WriteLine(output);
        Console.WriteLine("Exit code = {0}", proc.ExitCode);

即使在这里,请注意 ReadToEnd() 应该在 WaitForExit() 之前完成。

我相信 PSExec 总是有这样的问题。在Java服务下运行时,我们曾经将输出重定向到nul,无法得到运行进程的输出,但可以得到PSExec本身的输出。

参考下面给出的讨论:

http://forum.sysinternals.com/psexec-always-hangs-when-run-from-java_topic5013.html

http://forum.sysinternals.com/unusual-problem-with-psexecexe_topic6655.html

编辑:

PSEXESVC 清理注意事项:在终止挂起的 PSEXESVC 进程后,删除 C:\Windows(或 C:\Windows\system32 或两者)中的 PSEXESVC.EXE 文件。挥之不去的进程/文件会导致更多问题。

于 2011-04-13T22:27:35.703 回答