poc: introduce AudioDownloader
This commit is contained in:
parent
b1ce4bc2b0
commit
3955a64d9f
|
@ -1 +1 @@
|
||||||
*.m4a
|
/backend/cache/
|
||||||
|
|
112
backend/main.go
112
backend/main.go
|
@ -3,8 +3,12 @@ package main
|
||||||
import (
|
import (
|
||||||
"bytes"
|
"bytes"
|
||||||
"encoding/binary"
|
"encoding/binary"
|
||||||
|
"fmt"
|
||||||
|
"io"
|
||||||
"log"
|
"log"
|
||||||
|
"os"
|
||||||
"os/exec"
|
"os/exec"
|
||||||
|
"strconv"
|
||||||
|
|
||||||
"github.com/kkdai/youtube/v2"
|
"github.com/kkdai/youtube/v2"
|
||||||
)
|
)
|
||||||
|
@ -12,49 +16,107 @@ import (
|
||||||
const (
|
const (
|
||||||
ItagM4AAudio = 140
|
ItagM4AAudio = 140
|
||||||
SizeOfInt16 = 2
|
SizeOfInt16 = 2
|
||||||
|
|
||||||
|
DefaultAudioCodec = "pcm_s16le"
|
||||||
|
DefaultFormat = "s16le"
|
||||||
|
DefaultSampleRate = 44100
|
||||||
)
|
)
|
||||||
|
|
||||||
// TODO: process as stream to avoid loading full buffer in memory
|
type byteToIntConverter struct {
|
||||||
|
callback func(p []uint16)
|
||||||
|
buf []uint16
|
||||||
|
}
|
||||||
|
|
||||||
func main() {
|
func (w byteToIntConverter) Write(p []byte) (int, error) {
|
||||||
var client youtube.Client
|
if len(p)%2 != 0 {
|
||||||
|
panic("need to handle odd number of bytes")
|
||||||
|
}
|
||||||
|
|
||||||
videoID := "s_oJYdRlrv0"
|
// grow the internal buffer if necessary:
|
||||||
|
exp := len(p) / SizeOfInt16
|
||||||
|
if len(w.buf) < exp {
|
||||||
|
w.buf = append(w.buf, make([]uint16, exp)...)
|
||||||
|
|
||||||
video, err := client.GetVideo(videoID)
|
}
|
||||||
|
for i := 0; i < len(p); i += SizeOfInt16 {
|
||||||
|
w.buf[i/2] = binary.LittleEndian.Uint16(p[i : i+SizeOfInt16])
|
||||||
|
}
|
||||||
|
|
||||||
|
w.callback(w.buf)
|
||||||
|
|
||||||
|
return len(p), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
type AudioDownloader struct {
|
||||||
|
videoID string
|
||||||
|
ytClient youtube.Client
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewAudioDownloader(videoID string) *AudioDownloader {
|
||||||
|
return &AudioDownloader{videoID: videoID}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *AudioDownloader) ReadInts(fn func(p []uint16)) error {
|
||||||
|
cachePath := fmt.Sprintf("cache/%s.raw", d.videoID)
|
||||||
|
|
||||||
|
var reader io.Reader
|
||||||
|
|
||||||
|
fetch := true
|
||||||
|
if _, err := os.Stat(cachePath); err == nil {
|
||||||
|
if fptr, err := os.Open(cachePath); err == nil {
|
||||||
|
defer fptr.Close()
|
||||||
|
reader = fptr
|
||||||
|
fetch = false
|
||||||
|
} else {
|
||||||
|
log.Printf("warning: error opening cache file: %v", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if fetch {
|
||||||
|
log.Println("fetching audio stream from youtube...")
|
||||||
|
|
||||||
|
video, err := d.ytClient.GetVideo(d.videoID)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatal(err)
|
return fmt.Errorf("error fetching video: %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
format := video.Formats.FindByItag(ItagM4AAudio)
|
format := video.Formats.FindByItag(ItagM4AAudio)
|
||||||
stream, _, err := client.GetStream(video, format)
|
stream, _, err := d.ytClient.GetStream(video, format)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatal(err)
|
return fmt.Errorf("error fetching stream: %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO: force 44.1khz
|
cacheFile, err := os.Create(cachePath)
|
||||||
cmd := exec.Command("ffmpeg", "-i", "-", "-f", "s16le", "-acodec", "pcm_s16le", "-")
|
if err != nil {
|
||||||
cmd.Stdin = stream
|
return fmt.Errorf("error creating cache file: %v", err)
|
||||||
|
}
|
||||||
var out bytes.Buffer
|
reader = io.TeeReader(stream, cacheFile)
|
||||||
cmd.Stdout = &out
|
}
|
||||||
|
|
||||||
var errOut bytes.Buffer
|
var errOut bytes.Buffer
|
||||||
|
cmd := exec.Command("ffmpeg", "-i", "-", "-f", DefaultFormat, "-ar", strconv.Itoa(DefaultSampleRate), "-acodec", DefaultAudioCodec, "-")
|
||||||
|
cmd.Stdin = reader
|
||||||
|
cmd.Stdout = byteToIntConverter{callback: fn}
|
||||||
cmd.Stderr = &errOut
|
cmd.Stderr = &errOut
|
||||||
|
|
||||||
if err = cmd.Run(); err != nil {
|
if err := cmd.Run(); err != nil {
|
||||||
log.Fatalf("err = %v, stdErr = %s", err, errOut.String())
|
log.Println(errOut.String())
|
||||||
|
return fmt.Errorf("error calling ffmpeg: %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
log.Printf("byteLen = %d", out.Len())
|
return nil
|
||||||
|
|
||||||
// TODO: reflect.Slice to avoid copying?
|
|
||||||
data := make([]uint16, 0, out.Len()/SizeOfInt16)
|
|
||||||
for i := 0; i < out.Len(); i += SizeOfInt16 {
|
|
||||||
v := binary.LittleEndian.Uint16(out.Bytes()[i : i+SizeOfInt16])
|
|
||||||
data = append(data, v)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
log.Printf("intLen = %d", len(data))
|
func main() {
|
||||||
log.Printf("ints = %+v", data[0:8192])
|
videoID := "s_oJYdRlrv0"
|
||||||
|
downloader := NewAudioDownloader(videoID)
|
||||||
|
err := downloader.ReadInts(func(p []uint16) {
|
||||||
|
// log.Printf("ints = %+v", p[0:16])
|
||||||
|
})
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
log.Fatalf("error calling downloader: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Println("done")
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue