PythonパイプのサブプロセスI/O overソケット

私はそこに同様の質問があることを知っていますが、私はこの具体的な例に問題があり、良い答えを見つけていません。私は、darのリモートバックアップサーバを設定しようとしています。これらの行に沿って 。 subprocess.Popenでnetcatを呼び出すことでこれを行うことについて別の質問をお願いしましたが、可能であれば、Pythonですべてのパイプ処理を行います。いくつかのギグが転送されるので、私はすべての入力を最初に読んでからそれを渡すことはできません。

問題は、サーバーがデータを読み取っていないように見えることです。

現時点では、私は次のコードを持っています:

from socket import socket, AF_INET, SOCK_STREAM
import sys
import SocketServer
import subprocess

class DarHandler(SocketServer.BaseRequestHandler):
    def handle(self):
        print('entering handler')
        data = self.request.recv(1024).strip()
        print('got: ' + data)
        if data == 'xform':
            s = socket(AF_INET, SOCK_STREAM)
            s.bind(('',0))
            myaddr, myport = s.getsockname()
            print('bound new socket to {0}:{1}'.format(myaddr, myport))
            self.request.send(str(myport))
            s.listen(1)
            conn, remoteaddr = s.accept()
            print('accepted connection from {0}:{1}'.format(*remoteaddr))
            xform_input = conn.makefile('rb',0)
            proc = subprocess.Popen(
                ['/usr/bin/dar_xform', '-s', '10k', '-', 'archives/sockbackup',],
                 stdin=xform_input
            )
            return_code = proc.wait()
            print('dar_xform returned {0}'.format(return_code))
            conn.close()
            self.request.send(str(return_code))
        else:
            self.request.send('bad request')
        print('send result, exiting handler')

server_address = ('localhost', 18010)
def server():
    server = SocketServer.TCPServer(server_address, DarHandler)
    print('listening')
    server.serve_forever()

def client():
    sock = socket(AF_INET, SOCK_STREAM)
    print('connecting to server')
    sock.connect(('localhost', 18010))
    print('connected, sending request')
    sock.send('xform')
    print('waiting for response')
    port = sock.recv(1024)
    print('got: ' + port)
    s = socket(AF_INET, SOCK_STREAM)
    s.connect(('localhost', int(port)))
    print('connected to dar_xform port')
    dar_output = s.makefile('wb',0)
    return_code = subprocess.call(
          ['/usr/bin/dar', '-B', 'config/test.dcf', '-c', '-',], 
          stdout=dar_output
    )
    print('dar returned {0}'.format(return_code))
    s.close()
    result = sock.recv(1024)
    print('received: ' + result)
    sock.close()
    print('socket closed, exiting')

if __name__ == "__main__":
    if sys.argv[1].startswith('serv'):
        server()
    else:
        client()

クライアントを実行すると、サーバー側から次のような出力が得られます。

listening
entering handler
got: xform
bound new socket to 0.0.0.0:41658
accepted connection from 127.0.0.1:42440

それからちょうどそこに座っています。一方、 dar はクライアントで実行され、クライアントはサーバーからの応答を待っていません:

connecting to server
connected, sending request
waiting for response
got: 41300
connected to dar_xform port


 --------------------------------------------
 53 inode(s) saved
 with 0 hard link(s) recorded
 0 inode(s) changed at the moment of the backup
 0 inode(s) not saved (no inode/file change)
 0 inode(s) failed to save (filesystem error)
 1 inode(s) ignored (excluded by filters)
 0 inode(s) recorded as deleted from reference backup
 --------------------------------------------
 Total number of inodes considered: 54
 --------------------------------------------
 EA saved for 0 inode(s)
 --------------------------------------------
dar returned 0
1

1 答え

サーバー上のdar_xformコマンドが失敗しているようです...

dar_xform コマンドのフルパスを subprocess.Popen(...)に指定して、問題が引き続き発生するかどうかを確認してください。あなたが説明したことは、特にシェルプロンプトで動作するので、 PATH のような疑いがあるようです。

0
追加された
これはおそらく愚かな質問ですが、 dar_xform /usr/bin にあることを確認してください。
追加された 著者 larsks,
私はあなたがそれが働いてうれしいです。
追加された 著者 larsks,
良い考えですが、それはまだ失敗します。私はそれらを含めるために質問を編集しました。
追加された 著者 Aryeh Leib Taurog,
はい。すぐ問題はカンマがないことでした。これは、Popenにshell = Trueを追加して捕まえました。私はそれが2つの文字列を連結して得た '/ usr/bin/dar_xform-s'を探しているのを見ました。 13組の目がそれを逃した。私は質問を更新しました。
追加された 著者 Aryeh Leib Taurog,
エラーを正しく診断したので、私はあなたの答えを受け入れています。残りの質問はこのトピックに関する私の他の質問と全く同じです。
追加された 著者 Aryeh Leib Taurog,