PsExec зависает при выполнении из очень простой программы на языке c # или c ++, скомпилированной как «приложение для Windows» - PullRequest
6 голосов
/ 13 апреля 2011

Я испытываю зависание PsExec при выполнении из очень простой программы на языке c # или c ++, скомпилированной как «приложение для Windows» (а не как «консольное приложение»).В разделе C) ниже я вставил код для воспроизведения проблемы, а в разделе D) я вставил код C ++ для воспроизведения той же проблемы.

Когда psexec зависает, вывод windbg после подключения к psexecлокально вставляется в раздел B).

Моя программа зависает после выгрузки вывода, вставленного в раздел A).

Программа работает нормально, если вы заменяете команду psexec на что-либо локальное, например ProcessStartInfo ("cmd.exe "," / c dir c: \ windows \ *. * ");

Мне было интересно, если кто-нибудь испытал это и нашел решение для этого.Помощь будет принята с благодарностью.

Спасибо, Шарраджеш


A) Вывод моей программы на c # при зависании psexec

PsExec v1.98 - Удаленное выполнение процессов Copyright (C)) 2001-2010 Марк Руссинович Sysinternals - www.sysinternals.com

Том на диске C не имеет метки.


B) Вывод Windbg для psexec при зависании

3 Id: 1614.15e4 Приостановка: 1 Teb: 7efac000 Неразбитый ChildEBP RetAddr Аргументы к ребенку
02a3fe68 75a6d0c5 00000180 00000000 00000000 ntdll! NtReadFile + 0x15 (FPO: [9,0,0]) 02a3fecc 75c0100aFeF01aB0aaa0x118 (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);
}

Ответы [ 6 ]

5 голосов
/ 16 октября 2012

PSExec зависал для меня слишком случайно.Я не прикладывал усилий к тому, чтобы воссоздать проблему по-вашему, но я избежал своих неприятностей, используя «PAExec», на вид достойного преемника по духу: http://www.poweradmin.com/PAExec/

2 голосов
/ 12 июля 2011

У меня была похожая проблема, связанная с 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=&#72376

Несмотря на то, что я несколько раз проверял лицензионное соглашение

0 голосов
/ 14 января 2013

У меня есть простое решение для этого,

  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
    
0 голосов
/ 18 августа 2011

Создайте универсальное консольное приложение, которое запускает PsExec & PsKill и всех их друзей.Вызовите это ConsoleApp через ваш код вместо вызова метода NotWorkingPsExec, и он будет работать просто отлично.

0 голосов
/ 14 апреля 2011

Синхронное чтение в потоке работает:

        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

Edit:

Примечание по очистке PSEXESVC: Удалите файл PSEXESVC.EXE в C: \ Windows (или C: \ Windows \ system32 или оба) после завершения зависшего процесса PSEXESVC. Затяжной процесс / файл вызывают больше проблем.

0 голосов
/ 13 апреля 2011

Пожалуйста, также установите свойство WorkingDirectory в startInfo, так как утилиты Sysinternals используют распаковку файлов во время выполнения, а ядро ​​не может найти распакованный (настоящий) исполняемый файл.

...