mirror of
https://github.com/teacat/chaturbate-dvr.git
synced 2025-10-29 16:59:59 +00:00
almost done
This commit is contained in:
@@ -0,0 +1,9 @@
|
||||
# Chaturbate DVR
|
||||
|
||||
The program helps you to watching a specified Chaturbate channel and save the streaming in real-time when the channel goes online.
|
||||
|
||||
## Usage
|
||||
|
||||
```
|
||||
$ chaturbate-dvr -u mychannelname -q high
|
||||
```
|
||||
5
go.mod
Normal file
5
go.mod
Normal file
@@ -0,0 +1,5 @@
|
||||
module github.com/YamiOdymel/chaturbate-dvr
|
||||
|
||||
go 1.12
|
||||
|
||||
require github.com/urfave/cli/v2 v2.1.1 // indirect
|
||||
12
go.sum
Normal file
12
go.sum
Normal file
@@ -0,0 +1,12 @@
|
||||
github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU=
|
||||
github.com/cpuguy83/go-md2man/v2 v2.0.0-20190314233015-f79a8a8ca69d h1:U+s90UTSYgptZMwQh2aRr3LuazLJIa+Pg3Kc1ylSYVY=
|
||||
github.com/cpuguy83/go-md2man/v2 v2.0.0-20190314233015-f79a8a8ca69d/go.mod h1:maD7wRr/U5Z6m/iR4s+kqSMx2CaBsrgA7czyZG/E6dU=
|
||||
github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
|
||||
github.com/russross/blackfriday/v2 v2.0.1 h1:lPqVAte+HuHNfhJ/0LC98ESWRz8afy9tM/0RK8m9o+Q=
|
||||
github.com/russross/blackfriday/v2 v2.0.1/go.mod h1:+Rmxgy9KzJVeS9/2gXHxylqXiyQDYRxCVz55jmeOWTM=
|
||||
github.com/shurcooL/sanitized_anchor_name v1.0.0 h1:PdmoCO6wvbs+7yrJyMORt4/BmY5IYyJwS/kOiWx8mHo=
|
||||
github.com/shurcooL/sanitized_anchor_name v1.0.0/go.mod h1:1NzhyTcUVG4SuEtjjoZeVRXNmyL/1OwPU0+IJeTBvfc=
|
||||
github.com/urfave/cli/v2 v2.1.1 h1:Qt8FeAtxE/vfdrLmR3rxR6JRE0RoVmbXu8+6kZtYU4k=
|
||||
github.com/urfave/cli/v2 v2.1.1/go.mod h1:SE9GqnLQmjVa0iPEY0f1w3ygNIYcIJ0OKPMoW2caLfQ=
|
||||
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
|
||||
gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
|
||||
93
main.go
93
main.go
@@ -5,6 +5,7 @@ import (
|
||||
"errors"
|
||||
"fmt"
|
||||
"log"
|
||||
"net/http"
|
||||
"os"
|
||||
"regexp"
|
||||
"strconv"
|
||||
@@ -13,6 +14,7 @@ import (
|
||||
|
||||
"github.com/grafov/m3u8"
|
||||
"github.com/parnurzeal/gorequest"
|
||||
"github.com/urfave/cli"
|
||||
)
|
||||
|
||||
// chaturbateURL is the base url of the website.
|
||||
@@ -32,7 +34,8 @@ var bucket []string
|
||||
|
||||
//
|
||||
var (
|
||||
errInternal = errors.New("err")
|
||||
errInternal = errors.New("err")
|
||||
errNoUsername = errors.New("chaturbate-dvr: channel username required with `-u [username]` argument")
|
||||
)
|
||||
|
||||
// roomDossier is the struct to parse the HLS source from the content body.
|
||||
@@ -98,7 +101,13 @@ func parseHLSSource(url string, baseURL string) string {
|
||||
|
||||
//
|
||||
func parseM3U8Source(url string) (chunks []*m3u8.MediaSegment, wait float64, err error) {
|
||||
_, body, _ := gorequest.New().Get(url).End()
|
||||
resp, body, errs := gorequest.New().Get(url).End()
|
||||
if len(errs) > 0 {
|
||||
return nil, 3, errInternal
|
||||
}
|
||||
if resp.StatusCode == http.StatusForbidden {
|
||||
return nil, 3, errInternal
|
||||
}
|
||||
|
||||
//
|
||||
p, listType, _ := m3u8.DecodeFrom(strings.NewReader(body), true)
|
||||
@@ -107,7 +116,7 @@ func parseM3U8Source(url string) (chunks []*m3u8.MediaSegment, wait float64, err
|
||||
}
|
||||
|
||||
media := p.(*m3u8.MediaPlaylist)
|
||||
wait = media.TargetDuration
|
||||
wait = media.TargetDuration / 2
|
||||
|
||||
// Only fill with the real segments.
|
||||
for _, v := range media.Segments {
|
||||
@@ -125,14 +134,16 @@ func start(username string) {
|
||||
for {
|
||||
// Check again after a while if the user is currently not online.
|
||||
if !getOnlineStatus(username) {
|
||||
log.Printf("%s is not online, check again after 3 minutes...", username)
|
||||
<-time.After(time.Minute * 3)
|
||||
log.Printf("%s is offlined, check again after 1 minutes...", username)
|
||||
<-time.After(time.Minute * 1)
|
||||
start(username)
|
||||
return
|
||||
}
|
||||
|
||||
log.Printf("%s is online! Fetching the stream...", username)
|
||||
|
||||
// Define the video filename by current time.
|
||||
filename := time.Now().String() + ".mp4"
|
||||
filename := time.Now().String() + ".ts"
|
||||
// Get the channel page content body.
|
||||
body := getBody(username)
|
||||
// Get the master playlist URL from extracting the channel body.
|
||||
@@ -144,7 +155,16 @@ func start(username string) {
|
||||
if err != nil {
|
||||
panic(err)
|
||||
}
|
||||
defer f.Close()
|
||||
defer func() {
|
||||
for {
|
||||
if len(buffer) == 0 {
|
||||
f.Close()
|
||||
return
|
||||
}
|
||||
log.Printf("Waiting for the buffer to be cleaned...")
|
||||
<-time.After(2 * time.Second)
|
||||
}
|
||||
}()
|
||||
|
||||
go comsumer(f, baseURL)
|
||||
|
||||
@@ -163,10 +183,14 @@ func start(username string) {
|
||||
//
|
||||
retriesAfterOnlined++
|
||||
// Wait to fetch the next playlist.
|
||||
<-time.After(time.Duration(wait) * time.Second)
|
||||
<-time.After(time.Duration(wait*1000) * time.Millisecond)
|
||||
continue
|
||||
}
|
||||
}
|
||||
if retriesAfterOnlined != 0 {
|
||||
log.Printf("%s is backed online!", username)
|
||||
retriesAfterOnlined = 0
|
||||
}
|
||||
for _, v := range chunks {
|
||||
var ignore bool
|
||||
for _, j := range bucket {
|
||||
@@ -182,13 +206,7 @@ func start(username string) {
|
||||
log.Printf("%s (%d in buffer)", v.URI, len(buffer))
|
||||
buffer <- v
|
||||
}
|
||||
//
|
||||
//buffer = append(buffer, chunks...)
|
||||
//log.Printf("Storing chunks.")
|
||||
// Append the chunks to the video file.
|
||||
//appendChunks(f, baseURL, chunks)
|
||||
// Wait to fetch the next playlist.
|
||||
<-time.After(time.Duration(wait) * time.Second)
|
||||
<-time.After(time.Duration(wait*1000) * time.Millisecond)
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -204,21 +222,34 @@ func comsumer(file *os.File, baseURL string) {
|
||||
}
|
||||
}
|
||||
|
||||
// appendChunks appends the streaming chunks data into a single video file.
|
||||
//func appendChunks(file *os.File, baseURL string, chunks []*m3u8.MediaSegment) {
|
||||
// for _, v := range chunks {
|
||||
// _, body, _ := gorequest.New().Get(fmt.Sprintf("%s%s", baseURL, v.URI)).EndBytes()
|
||||
//
|
||||
// log.Println(v.URI)
|
||||
//
|
||||
// if _, err := file.Write(body); err != nil {
|
||||
// panic(err)
|
||||
// }
|
||||
// }
|
||||
//}
|
||||
|
||||
func main() {
|
||||
username := "yesonee"
|
||||
|
||||
start(username)
|
||||
app := &cli.App{
|
||||
Flags: []cli.Flag{
|
||||
&cli.StringFlag{
|
||||
Name: "username",
|
||||
Aliases: []string{"u"},
|
||||
Value: "",
|
||||
Usage: "channel username to watching",
|
||||
},
|
||||
&cli.StringFlag{
|
||||
Name: "quality",
|
||||
Aliases: []string{"q"},
|
||||
Value: "",
|
||||
Usage: "video quality with `high`, `medium` and `low`",
|
||||
},
|
||||
},
|
||||
Name: "chaturbate-dvr",
|
||||
Usage: "watching a specified chaturbate channel and auto saved to local file",
|
||||
Action: func(c *cli.Context) error {
|
||||
if c.String("username") == "" {
|
||||
log.Fatal(errNoUsername)
|
||||
}
|
||||
start(c.String("username"))
|
||||
return nil
|
||||
},
|
||||
}
|
||||
err := app.Run(os.Args)
|
||||
if err != nil {
|
||||
log.Fatal(err)
|
||||
}
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user