как вызвать событие getche () из process.Popen () - не отслеживает stdin

Я запускаю двоичный файл в Windows, используя:

process = subprocess.Popen(cmd, stderr = subprocess.PIPE, stdin = subprocess.PIPE, stdout = ch.input)

ch.input comes from:

ch = InputStreamChunker('\n')
ch.daemon = True
ch.start()

это был классный неблокирующий метод чтения из стандартного вывода. Это был принятый ответ на этот вопрос SO: Как я могу прочитать все доступные данные из subprocess.Popen.stdout (без блокировки)?

Это основная часть моего скрипта, который пытается запустить / контролировать процесс:

import subprocess
import time
from e import *
import msvcrt

ch = InputStreamChunker('\n')
ch.daemon = True
ch.start()


cmd = "C:\TEST.EXE"
process = subprocess.Popen(cmd, stderr = subprocess.PIPE, stdin = subprocess.PIPE, stdout = ch.input)
i = process.stdin
answers = []
waitforresults(ch, answers, expect = 5)
an = 1
for i in answers:
    print "ANSWER # " + str(an) + ":" + i.read()
    an= an+1

answers = []    
time.sleep(5)
msvcrt.ungetch('s')
ch.flush()
waitforresults(ch, answers, expect = 1)
for a in answers:
    print a.getvalue()

process.terminate()
ch.stop()
del process, ch
print "DONE"

Во всяком случае, я могу читать из процесса, используя этот метод нормально. Когда я пытаюсь написать в процесс, используя:

i = process.stdin i.write ("s \ n") или i.write ("s")

Вывод печатается в окне консоли, но не фиксируется двоичным файлом. Я взглянул на двоичный файл в IDA Pro и обнаружил, что он использует нестандартный метод сбора пользовательского ввода во время его работы.

Процесс представляет собой интерактивный инструмент cli, который ожидает S, R, P или Q (статус, возобновление, пауза, выход) *** и фиксирует ввод, при этом пользователь нажимает клавишу ввода *** r. Это достигается с помощью getche (). Я бросил двоичный файл в IDA Pro, чтобы подтвердить:

v7 = getche();
if ( (unsigned int)dword_41F060 > 1 )
  sub_4030A0(&unk_41C111, v5);
WaitForSingleObject((HANDLE)dword_41F040, 0xFFFFFFFFu);
if ( v7 == 113 )
{
  if ( dword_41F060 != 1 )
  {
    if ( dword_41F060 )
      dword_41F060 = 6;
  }
}
else
{
  if ( v7 <= 113 )
  {
    if ( v7 == 112 )
    {
      if ( dword_41F060 == 2 )
        QueryPerformanceCounter((LARGE_INTEGER *)&qword_41F128);
      dword_41F060 = 3;
      sub_4030A0("Paused", v5);
    }
  }
  else
  {
    if ( v7 == 114 )
    {
      if ( dword_41F060 == 3 )
      {
        QueryPerformanceFrequency((LARGE_INTEGER *)&v9);
        QueryPerformanceCounter((LARGE_INTEGER *)&v8);
        v3 = (double)(v8 - qword_41F128);
        LODWORD(v4) = sub_413FF0(v9, v10, 1000, 0);
        dbl_41F138 = v3 / (double)v4 + dbl_41F138;
      }
      dword_41F060 = 2;
      sub_4030A0("Resumed", v5);
    }
    else
    {
      if ( v7 == 115 )
        sub_40AFC0();
    }
  }
}

Кто-нибудь знает, как вызвать это событие? Это выглядело многообещающим: http://docs.python.org/library/msvcrt.html и я попытался использовать:

msvcrt.ungetch ("s") "Заставляет символ char быть" возвращенным "в буфер консоли; это будет следующий символ, прочитанный getch () или getche ()."

который ДЕЙСТВИТЕЛЬНО подтолкнул букву «s» к консоли, но не сработал мою точку останова на getche (). Нажатие буквы S вручную действительно работает и заставляет IDA pro попасть в точку останова

халп? :)

РЕДАКТИРОВАТЬ:

Я создал два небольших консольных приложения для Windows, чтобы продемонстрировать, что работает, а что нет, и убедиться, что мой питон работает нормально.

В первом случае я НЕ могу распознать ввод, записав его на стандартный ввод, второй - могу.

#include "stdafx.h"
#include <conio.h>


int _tmain(int argc, _TCHAR* argv[])
{
    char response;
    printf("Enter \"s\":\n");
    response = _getch();
    printf("You entered %c\n", response);
    return 0;
}

можете написать этому:

#include "stdafx.h"
#include <conio.h>


int _tmain(int argc, _TCHAR* argv[])
{
    char response [2];
    printf("Enter \"s\":\n");
    gets(response);
    printf("You entered %s", response);
    return 0;
}

РЕДАКТИРОВАТЬ 2. Я также пробовал: import subprocess import time from e import * import msvcrt

ch = InputStreamChunker('\n')
ch.daemon = True
ch.start()


cmd = ["test-getch.exe"]
process = subprocess.Popen(cmd, stderr = subprocess.PIPE, stdin = subprocess.PIPE, stdout = ch.input)
i = process.stdin
answers = []
msvcrt.putch('s')
ch.flush
waitforresults(ch, answers, expect = 1)
for answer in answers:
    print answer.getvalue()
i.close()
time.sleep(3)
#process.terminate()
ch.stop()
del process, ch
print "DONE"

person python_crazy    schedule 07.11.2011    source источник
comment
вы пробовали msvcrt.putch()?   -  person jfs    schedule 07.11.2011
comment
Я действительно пробовал .. C: \ code ›C: \ Python27 \ python.exe te.py sОсновной разговор: получено 0 ответов, ожидается 1 ..... как вы можете видеть s переносится на консоль, но не выбирается вверх по text.exe   -  person python_crazy    schedule 07.11.2011
comment
Я бы задал getche() как отдельный вопрос, используя только простые subprocess функции (без InputStreamChunker(), например, Popen(..stdin=PIPE).communicate('s')) и простую программу C. Добавьте тег Windows.   -  person jfs    schedule 07.11.2011
comment
ungetch () предназначен для резервного копирования, пока вы читаете поток. Обычно гарантируется резервное копирование только одним символом, и он не предназначен для вывода. В любом случае, чтобы писать в пайп, вам не придется проходить через все эти искажения. Вы пробовали установить bufsize=0 в Popen?   -  person alexis    schedule 06.03.2012


Ответы (1)


Выдержки из MSDN:

Эти подпрограммы читают и записывают на вашу консоль. Подпрограммы ввода-вывода консоли несовместимы с подпрограммами потокового ввода-вывода или низкоуровневыми библиотеками ввода-вывода. В операционных системах Windows вывод этих функций всегда направляется на консоль и не может быть перенаправлен.

Итак, я вижу 2 возможных варианта, и оба они не перспективны:

  1. Создайте псевдо-консоль Windows и запустите на ней свою программу. Тогда вы получите там свой результат. (Аналог pty в UNIX)
  2. Как-то получить данные из существующей консоли Windows. Console2 http://sourceforge.net/projects/console/ делает что-то подобное.

Я не знаю, как это сделать и возможно ли это вообще.

В любом случае я не вижу практического решения, в основном говоря, что вам, на мой взгляд, не повезло.

person Alx_s    schedule 21.03.2012