Wie kommuniziere ich mit einer Schachengine in Python?

Auf Win 7 kann ich über die Kommandozeile mit einer Schachengine kommunizieren. Kleine Beispielsession mitStockfisch auf Win 7:

C:\run\Stockfish>stockfish-x64.exe
Stockfish 2.2.2 JA SSE42 by Tord Romstad, Marco Costalba and Joona Kiiski
quit

C:\run\Stockfish>

Die erste Zeile wurde von der Engine ausgegeben und das 'Verlassen' war das, was ich eingegeben habe, um die Engine zu verlassenandere Dinge, die ich tun kann, aber das ist mir klar).

Jetzt möchte ich mit dieser Engine von Python aus kommunizieren:

import subprocess
engine = subprocess.Popen(
    'stockfish-x64.exe',
    stdin=subprocess.PIPE,
    stdout=subprocess.PIPE,
)
for line in engine.stdout:
    print(line.strip())
engine.stdin.write('quit\n')

und ich verstehe

C:\run\Stockfish>communicate.py
b'Stockfish 2.2.2 JA SSE42 by Tord Romstad, Marco Costalba and Joona Kiiski'

Die Engine wird jedoch nicht beendet (keine Eingabeaufforderung C: \ run \ Stockfish>), sondern es wird auf Eingaben gewartet. Ich muss das Fenster von Hand schließen. Es scheint, als würde meine Beendigungsnachricht (letzte Zeile des Python-Skripts) nicht nach stdin geschrieben.

Mit anderen Worten, ich kann von stdout lesen, aber wenn ich an stdin schreibe, passiert nichts.

Was mache ich falsch und wie mache ich es richtig?

Edit: ok, dank der Hilfe von Larsmans habe ich es gelöst:

Beispiel-Python-Skript:

import subprocess, time

engine = subprocess.Popen(
    'stockfish-x64.exe',
    universal_newlines=True,
    stdin=subprocess.PIPE,
    stdout=subprocess.PIPE,
)

def put(command):
    print('\nyou:\n\t'+command)
    engine.stdin.write(command+'\n')

def get():
    # using the 'isready' command (engine has to answer 'readyok')
    # to indicate current last line of stdout
    engine.stdin.write('isready\n')
    print('\nengine:')
    while True:
        text = engine.stdout.readline().strip()
        if text == 'readyok':
            break
        if text !='':
            print('\t'+text)

get()
put('uci')
get()
put('setoption name Hash value 128')
get()
put('ucinewgame')
get()
put('position startpos moves e2e4 e7e5 f2f4')
get()
put('go infinite')
time.sleep(3)
get()
put('stop')
get()
put('quit')

Ausgabe:

C:\run\Stockfish>1-communicate.py

engine:
        Stockfish 2.2.2 JA SSE42 by Tord Romstad, Marco Costalba and Joona Kiiski

you:
        uci

engine:
        id name Stockfish 2.2.2 JA SSE42
        id author Tord Romstad, Marco Costalba and Joona Kiiski
        option name Use Search Log type check default false
        option name Search Log Filename type string default SearchLog.txt
        ... etc ...
        uciok

you:
        setoption name Hash value 128

engine:

you:
        ucinewgame

engine:

you:
        position startpos moves e2e4 e7e5 f2f4

engine:

you:
        go infinite

engine:
        info depth 1 seldepth 1 score cp 56 nodes 62 nps 1675 time 37 multipv 1 pv e5f4
        info depth 2 seldepth 2 score cp 48 nodes 804 nps 21157 time 38 multipv 1 pv b8c6 g1h3
        info depth 3 seldepth 3 score cp 64 nodes 1409 nps 37078 time 38 multipv 1 pv b8c6 b1c3 e5f4
        ... etc ...

you:
        stop

engine:
        bestmove e5f4 ponder g1f3

you:
        quit

C:\run\Stockfish>

Antworten auf die Frage(1)

Ihre Antwort auf die Frage