Self review #1
|
@ -0,0 +1 @@
|
|||
/segmento
|
|
@ -0,0 +1,120 @@
|
|||
package main
|
||||
|
||||
import (
|
||||
"flag"
|
||||
"fmt"
|
||||
"io"
|
||||
"log"
|
||||
"net/http"
|
||||
"os"
|
||||
"time"
|
||||
|
||||
"github.com/tcolgate/mp3"
|
||||
)
|
||||
|
||||
// TODO what should be the behaviour when adding a Frame to a Segment causes it to over-shoot
|
||||
// the TargetDuration?
|
||||
// should the frame be partially added, or not added at all?
|
||||
// typical frame duration 36ms
|
||||
type Segment struct {
|
||||
TargetDuration time.Duration
|
||||
Duration time.Duration
|
||||
Data []byte
|
||||
}
|
||||
|
||||
func newSegment(capacity int) *Segment {
|
||||
return &Segment{
|
||||
Data: make([]byte, 0, capacity),
|
||||
}
|
||||
}
|
||||
|
||||
func (s *Segment) Write(p []byte) (n int, err error) {
|
||||
fmt.Println("write", len(p), "bytes")
|
||||
return len(p), nil
|
||||
}
|
||||
|
||||
type MP3HTTPSegmenter struct {
|
||||
decoder *mp3.Decoder
|
||||
}
|
||||
|
||||
func (s *MP3HTTPSegmenter) Segment(r io.Reader) (chan Segment, error) {
|
||||
c := make(chan Segment)
|
||||
|
||||
go func() {
|
||||
d := mp3.NewDecoder(r)
|
||||
|
||||
var (
|
||||
v mp3.Frame
|
||||
skipped int
|
||||
)
|
||||
|
||||
for {
|
||||
if err := d.Decode(&v, &skipped); err != nil {
|
||||
log.Fatal(err)
|
||||
}
|
||||
|
||||
fmt.Println("decoded frame of", v.Size(), "bytes, duration", v.Duration(), ", skipped", skipped, "bytes")
|
||||
}
|
||||
}()
|
||||
|
||||
return c, nil
|
||||
}
|
||||
|
||||
func newMP3HTTPSegmenter() *MP3HTTPSegmenter {
|
||||
return &MP3HTTPSegmenter{}
|
||||
}
|
||||
|
||||
// TODO
|
||||
|
||||
// A Segmenter class which allows the passing in of a Reader
|
||||
// i.e.
|
||||
// Segment(r io.Reader) (chan *Segment, error)
|
||||
// This will create an mp3.NewDecoder(r) and store the returned decoder.
|
||||
// As the Reader is read, the Segmenter will publish a stream of SegmentEvents
|
||||
|
||||
func main() {
|
||||
// TODO accept some flags with:
|
||||
// URL - source of stream
|
||||
// TargetLength - length of segments in seconds
|
||||
// Output
|
||||
// -d some_dir/ => output playlist and chunks to this directory, cleaning up old files from time to time.
|
||||
// -b 0.0.0.0:3000 => serve playlist and chunks from an HTTP server bound to this address
|
||||
|
||||
var url string
|
||||
flag.StringVar(&url, "url", "", "URL of MP3 stream")
|
||||
|
||||
flag.Parse()
|
||||
|
||||
if url == "" {
|
||||
log.Println("Invalid arguments")
|
||||
flag.PrintDefaults()
|
||||
os.Exit(-1)
|
||||
}
|
||||
|
||||
log.Println("Open URL:", url)
|
||||
|
||||
client := &http.Client{}
|
||||
resp, err := client.Get(url)
|
||||
|
||||
if err != nil {
|
||||
log.Fatal(err)
|
||||
}
|
||||
|
||||
defer resp.Body.Close()
|
||||
|
||||
segmenter := newMP3HTTPSegmenter()
|
||||
segments, err := segmenter.Segment(resp.Body)
|
||||
if err != nil {
|
||||
log.Fatal(err)
|
||||
}
|
||||
|
||||
go func() {
|
||||
for segment := range segments {
|
||||
log.Println("got segment", segment)
|
||||
}
|
||||
}()
|
||||
|
||||
time.Sleep(10 * time.Second)
|
||||
|
||||
log.Println("exiting")
|
||||
}
|
Loading…
Reference in New Issue