コマンドを実行し、データをそのstdinにフィードし、そのstdoutから読み取ろうとしています。MacRubyで公開されているNSTaskだけでなくRubyのOpen3#popen3を使ってみました。私が書いているプログラムのソースはここにあります。私はこれをXcodeとMacRubyで行っています。
ここにいくつかの選択コードがあります:
エントリポイント。2つの方法を簡単に切り替えることができます。
def do_gpg_cmd cmd
do_gpg_cmd_nstask cmd
end
Open3#popen3を使用したルビーの方法。
def do_gpg_cmd_ruby cmd
gpg = "#{@gpg_path} --no-tty "
cmd_output = ''
logg "executing [#{cmd}]"
Dispatch::Queue.concurrent.async do
logg "new thread starting"
Open3.popen3(gpg + cmd) do |stdin, stdout, stderr|
stdin.write input_text
stdin.close
cmd_output = stdout.read
output_text cmd_output
stdout.close
logg stderr.read
stderr.close
end
end
return cmd_output
end
このアプローチでは、アプリケーションがフリーズします(実行中のアプリの[署名]ボタンをクリックしてテストしていますgpg --clearsign --local-user $key
)。
アプリケーションを強制終了すると、Xcodeは自動的に表示されるスレッド診断にこれを表示します。
libsystem_kernel.dylib`__psynch_cvwait:
0x7fff84b390f0: movl $33554737, %eax
0x7fff84b390f5: movq %rcx, %r10
0x7fff84b390f8: syscall
0x7fff84b390fa: jae 0x7fff84b39101 ; __psynch_cvwait + 17 ; THIS LINE IS HIGHLIGHTED
0x7fff84b390fc: jmpq 0x7fff84b3a4d4 ; cerror_nocancel
0x7fff84b39101: ret
0x7fff84b39102: nop
0x7fff84b39103: nop
NSTaskを使用したCocoaの方法。
def do_gpg_cmd_nstask cmd
Dispatch::Queue.concurrent.async do
fcmd = "--no-tty " + cmd
task = NSTask.alloc.init
task.setLaunchPath(@gpg_path)
task.setArguments(fcmd.split(" ") << nil)
task.arguments.each {|a| puts "ARG: [#{a}]" }
inpipe = NSPipe.pipe
outpipe = NSPipe.pipe
errpipe = NSPipe.pipe
task.setStandardOutput(outpipe)
task.setStandardInput(inpipe)
task.setStandardError(errpipe)
output = outpipe.fileHandleForReading
errput = errpipe.fileHandleForReading
input = inpipe.fileHandleForWriting
task.launch
input.writeData input_text.dataUsingEncoding(NSUTF8StringEncoding)
input.closeFile
outdata = output.readDataToEndOfFile
errdata = errput.readDataToEndOfFile
output.closeFile
errput.closeFile
outstring = NSString.alloc.initWithData(outdata, encoding: NSUTF8StringEncoding)
errstring = NSString.alloc.initWithData(errdata, encoding: NSUTF8StringEncoding)
output_text outstring
logg errstring
end
end
これを実行すると、Xcodeデバッグ出力でこのエラーが発生します。私は明らかにARGパーツを超ダムロギングとして自分で出力しています。サブプロセスは実行されません。
ARG: [--no-tty]
ARG: [--clearsign]
ARG: [--local-user]
ARG: [0xC2808780]
ARG: []
2013-03-12 23:27:39.305 GPGBoard[84924:3503] -[NSNull fileSystemRepresentation]: unrecognized selector sent to instance 0x7fff75b05310
*** Dispatch block exited prematurely because of an uncaught exception:
/Users/colin/Library/Developer/Xcode/DerivedData/GPGBoard-bradukgmaegxvmbukhwehepzyxcv/Build/Products/Debug/GPGBoard.app/Contents/Resources/AppDelegate.rb:81:in `block': NSInvalidArgumentException: -[NSNull fileSystemRepresentation]: unrecognized selector sent to instance 0x7fff75b05310 (RuntimeError)
どちらのアプローチの問題も相互に排他的であると思われます。Open3#popen3の問題は読み取りのブロックに関連している可能性がありますが、NSTaskの問題はパイプの問題に関連しています。