我想执行系统命令(git clone)并通过Phoenix中的通道将输出流式传输给用户?
我可以以某种方式使System.cmd流结果而不是等到它完成?
或者,我可以将输出写入文件并从那里流式传输内容,因为它正在追加?
我建议使用瓷器。具体检查https://github.com/alco/porcelain#messages。
您可以使用Porcelain.spawn_shell
使用out: {:send, self()}
选项运行命令并实现匹配的handle_info
回调。
可以使用以下方式覆盖输出:
System.cmd "git", ["clone", "YOUR_GIT"], into: IO.stream(:stdio, :line)
结果:
克隆到'YOUR_GIT'...远程:计数对象:1665,完成。 remote:压缩对象:0%(1/979)remote:压缩对象:100%(979/979),完成。接收物体:0%(1/166删除:总计1665(delta 855),重复使用1331(delta 597)接收物体:92%(接收物体:100%(1665/1665),5.25 MiB | 376.00 KiB / s,完成。解决delResolving增量:100%(855/855),完成。检查连接...完成。 {%IO.Stream {device :: standard_io,line_or_bytes :: line,raw:false},0}
为了帮助您完成特定任务,将本地标准流重定向到外部标准流,有一个库porcelain,handles it perfectly。
如果你不想使用具有go可执行依赖性的Porcelain,你可以使用port,如下所示:
def long do
port = Port.open({:spawn_executable, /path/to/command"}, [:stderr_to_stdout, :binary, :exit_status, args: ["arg1"]])
stream_output(port)
end
defp stream_output(port) do
receive do
{^port, {:data, data}} ->
Logger.info(data) # send to phoenix channel
stream_output(port)
{^port, {:exit_status, 0}} ->
Logger.info("Command success")
{^port, {:exit_status, status}} ->
Logger.info("Command error, status #{status}")
end
end