将内容翻译为中文: 写入到ffpmeg的标准输入会导致程序冻结。

huangapple go评论96阅读模式
英文:

Writing to ffpmeg stdin freezes program

问题

我正在尝试使用ffmpeg将内存中的文件转换为另一种格式,通过使用stdin和stdout,但每次我尝试写入stdin时,我的ffmpeg命令就会卡住。

package main

import (
	"bufio"
	"fmt"
	"io"
	"os"
	"os/exec"
)

func test(bytes []byte) ([]byte, error) {

	cmd := exec.Command(
		"ffmpeg",
		"-i", "pipe:0", // 从stdin读取
		"-vcodec", "copy",
		"-acodec", "copy",
		"-f", "matroska",
		"pipe:1",
	)

	in, err := cmd.StdinPipe()
	if err != nil {
		panic(err)
	}
	out, err := cmd.StdoutPipe()
	if err != nil {
		panic(err)
	}

	fmt.Println("starting")
	err = cmd.Start()
	if err != nil {
		panic(err)
	}

	fmt.Println("writing")
	w := bufio.NewWriter(in)
	_, err = w.Write(bytes)
	if err != nil {
		panic(err)
	}

	err = w.Flush()
	if err != nil {
		panic(err)
	}

	err = in.Close()
	if err != nil {
		panic(err)
	}

	fmt.Println("reading")
	outBytes, err := io.ReadAll(out)
	if err != nil {
		panic(err)
	}

	fmt.Println("waiting")
	err = cmd.Wait()
	if err != nil {
		panic(err)
	}

	return outBytes, nil
}

func main() {
	dat, err := os.ReadFile("speech.mp4")
	if err != nil {
		panic(err)
	}

	out, err := test(dat)
	if err != nil {
		panic(err)
	}

	err = os.WriteFile("test.m4v", out, 0644)
	if err != nil {
		panic(err)
	}
}

它打印出:

starting
writing

然后卡住了。我尝试使用grep运行类似的代码,一切正常工作,所以这似乎是一些特定于ffmpeg的问题。

我尝试运行:

cat speech.mp4 | ffmpeg -i pipe:0 -vcodec copy -acodec copy -f matroska pipe:1 | cat > test.mkv

这个命令可以正常工作,所以不是ffmpeg的问题,而是我在处理数据的方式上出了问题。

我的speech.mp4文件大小约为2MB。

英文:

I'm trying to convert a file in memory using ffmpeg to another format by using stdin and stdout, but everytime I try to write to stdin, of my ffmpeg command, it just freezes there.

package main

import (
	"bufio"
	"fmt"
	"io"
	"os"
	"os/exec"
)

func test(bytes []byte) ([]byte, error) {

	cmd := exec.Command(
		"ffmpeg",
		"-i", "pipe:0", // read from stdin
		"-vcodec", "copy",
		"-acodec", "copy",
		"-f", "matroska",
		"pipe:1",
	)

	in, err := cmd.StdinPipe()
	if err != nil {
		panic(err)
	}
	out, err := cmd.StdoutPipe()
	if err != nil {
		panic(err)
	}

	fmt.Println("starting")
	err = cmd.Start()
	if err != nil {
		panic(err)
	}

	fmt.Println("writing")
	w := bufio.NewWriter(in)
	_, err = w.Write(bytes)
	if err != nil {
		panic(err)
	}

	err = w.Flush()
	if err != nil {
		panic(err)
	}

	err = in.Close()
	if err != nil {
		panic(err)
	}

	fmt.Println("reading")
	outBytes, err := io.ReadAll(out)
	if err != nil {
		panic(err)
	}

	fmt.Println("waiting")
	err = cmd.Wait()
	if err != nil {
		panic(err)
	}

	return outBytes, nil
}

func main() {
	dat, err := os.ReadFile("speech.mp4")
	if err != nil {
		panic(err)
	}

	out, err := test(dat)
	if err != nil {
		panic(err)
	}

	err = os.WriteFile("test.m4v", out, 0644)
	if err != nil {
		panic(err)
	}
}


It prints

starting
writing

and gets stuck there. I tried similar code with grep, and the everything worked fine, so this seems to be some ffmpeg specific problem.

I tried running

cat speech.mp4 | ffmpeg -i pipe:0 -vcodec copy -acodec copy -f matroska pipe:1 | cat > test.mkv

and that works fine, so it's not an ffmpeg problem, but some problem with how I'm piping/reading/writing my data.

My speech.mp4 file is around 2MB.

答案1

得分: 1

所以秘密就在于在将字节转储到stdin时读取stdout,因为写入管道会阻塞。感谢@JimB帮助我找出这个问题。

你只需要在写入时进行读取:

cmd := exec.Command(
	"ffmpeg",
	"-i", "pipe:0", // 从stdin读取
	"-vcodec", "copy",
	"-acodec", "copy",
	"-f", "matroska",
	"pipe:1",
)

out, err := cmd.StdoutPipe()
if err != nil {
	panic(err)
}

in, err := cmd.StdinPipe()
writer := bufio.NewWriter(in)
if err != nil {
	panic(err)
}

fmt.Println("开始")
err = cmd.Start()
if err != nil {
	panic(err)
}

go func() {
	defer writer.Flush()
	defer in.Close()
	fmt.Println("写入")
	_, err = writer.Write(bytes)
	if err != nil {
		panic(err)
	}
}()

var outBytes []byte

defer out.Close()
fmt.Println("读取")
outBytes, err = io.ReadAll(out)
if err != nil {
	panic(err)
}
fmt.Println("等待")
err = cmd.Wait()
if err != nil {
	panic(err)
}

return outBytes, nil
英文:

So the secret lied in reading stdout as you dumped the bytes into stdin, since writing to the pipe blocks. Thanks @JimB for helping me figure this out.

You just have to read as you write:

cmd := exec.Command(
	"ffmpeg",
	"-i", "pipe:0", // read from stdin
	"-vcodec", "copy",
	"-acodec", "copy",
	"-f", "matroska",
	"pipe:1",
)

out, err := cmd.StdoutPipe()
if err != nil {
	panic(err)
}

in, err := cmd.StdinPipe()
writer := bufio.NewWriter(in)
if err != nil {
	panic(err)
}

fmt.Println("starting")
err = cmd.Start()
if err != nil {
	panic(err)
}

go func() {
	defer writer.Flush()
	defer in.Close()
	fmt.Println("writing")
	_, err = writer.Write(bytes)
	if err != nil {
		panic(err)
	}
}()

var outBytes []byte

defer out.Close()
fmt.Println("reading")
outBytes, err = io.ReadAll(out)
if err != nil {
	panic(err)
}
fmt.Println("waiting")
err = cmd.Wait()
if err != nil {
	panic(err)
}

return outBytes, nil

huangapple
  • 本文由 发表于 2021年12月24日 02:28:24
  • 转载请务必保留本文链接:https://go.coder-hub.com/70466094.html
匿名

发表评论

匿名网友

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

确定