PsExec在从非常简单的c#或c ++ gui程序执行时挂起,该程序被编译为“ Windows应用程序”

| 我在从非常简单的C#或C ++ gui程序执行时遇到PsExec挂起的情况,该程序被编译为“ Windows应用程序”(而不是“控制台应用程序”)。下 在下面的C)节中,我粘贴了代码以重现该问题,在下面的D)部分中,我粘贴了c ++代码以重现该问题。 当psexec挂起时,在本地附加到psexec之后的windbg输出将粘贴在B节下)。 转储根据A)节粘贴的输出后,我的程序挂起。 如果您将psexec命令替换为任何本地内容,则该程序将正常运行ProcessStartInfo(\“ cmd.exe \”,\“ / c目录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暂停:1 Teb:7efac000未冻结 ChildEBP RetAddr Args到子 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);
}
    
已邀请:
PSExec太随意地挂在我身上。我没有按照您的方式重新创建问题,但我使用了精神上看似值得继承的\“ PAExec \”来避免麻烦。 http://www.poweradmin.com/PAExec/     
我也有类似的问题,这可能是由于您的eula引起的:   可能的原因:      1)psiexec.exe在首次运行时显示EULA消息。      2)权限      3)dll功能可能需要用户会话。      为避免这些问题,请尝试以下情形:      1)带有\“-accepteula \”参数      2)使用\“-s \”参数      3)使用\“-i \”参数      4)> 2 + 3 5)2 + 3 + 1 请参阅:http://www.appdeploy.com/messageboards/tm.asp?m=72376&mpage=1&key=𑪸 即使我已经多次检查了EULA     
还请在startInfo上设置WorkingDirectory属性,因为Sysinternals实用程序使用文件的运行时解压缩,并且内核无法找到解压缩的(实际的)可执行文件。     
构建一个运行
PsExec
和ѭ3runs及其所有朋友的通用控制台应用程序。通过您的代码调用此ConsoleApp而不是调用
NotWorkingPsExec
方法,它将正常工作。     
我有一个简单的解决方案, 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 .
run.bat
是:
psexec.bat > %1           //redirect to a text file
exit
psexec.bat
是:
psexec.exe ..........................
 exit
    
流上的同步读取有效:
        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文件。拖延进程/文件会导致更多问题。     

要回复问题请先登录注册