2018-02-13 18:47:05 +01:00
|
|
|
package sources
|
|
|
|
|
|
|
|
import (
|
|
|
|
"net/http"
|
|
|
|
"os"
|
|
|
|
"regexp"
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
2018-08-23 00:28:31 +02:00
|
|
|
"sync"
|
2018-09-18 21:20:34 +02:00
|
|
|
"time"
|
2018-08-23 00:28:31 +02:00
|
|
|
|
2018-02-13 18:47:05 +01:00
|
|
|
"github.com/aws/aws-sdk-go/aws/credentials"
|
|
|
|
"github.com/aws/aws-sdk-go/service/s3"
|
|
|
|
"github.com/aws/aws-sdk-go/service/s3/s3manager"
|
2019-01-11 02:34:34 +01:00
|
|
|
"github.com/lbryio/lbry.go/extras/errors"
|
|
|
|
"github.com/lbryio/lbry.go/extras/jsonrpc"
|
2018-10-08 22:19:17 +02:00
|
|
|
"github.com/lbryio/ytsync/namer"
|
2018-02-13 18:47:05 +01:00
|
|
|
|
|
|
|
"github.com/aws/aws-sdk-go/aws"
|
|
|
|
"github.com/aws/aws-sdk-go/aws/session"
|
|
|
|
log "github.com/sirupsen/logrus"
|
|
|
|
)
|
|
|
|
|
|
|
|
type ucbVideo struct {
|
2018-08-23 00:28:31 +02:00
|
|
|
id string
|
|
|
|
title string
|
|
|
|
channel string
|
|
|
|
description string
|
|
|
|
publishedAt time.Time
|
|
|
|
dir string
|
|
|
|
claimNames map[string]bool
|
|
|
|
syncedVideosMux *sync.RWMutex
|
2018-02-13 18:47:05 +01:00
|
|
|
}
|
|
|
|
|
2018-08-17 16:05:54 +02:00
|
|
|
func NewUCBVideo(id, title, channel, description, publishedAt, dir string) *ucbVideo {
|
2018-02-13 18:47:05 +01:00
|
|
|
p, _ := time.Parse(time.RFC3339Nano, publishedAt) // ignore parse errors
|
2018-08-17 16:05:54 +02:00
|
|
|
return &ucbVideo{
|
2018-02-13 18:47:05 +01:00
|
|
|
id: id,
|
|
|
|
title: title,
|
|
|
|
description: description,
|
|
|
|
channel: channel,
|
|
|
|
dir: dir,
|
|
|
|
publishedAt: p,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-08-17 16:05:54 +02:00
|
|
|
func (v *ucbVideo) ID() string {
|
2018-02-13 18:47:05 +01:00
|
|
|
return v.id
|
|
|
|
}
|
|
|
|
|
2018-08-17 16:05:54 +02:00
|
|
|
func (v *ucbVideo) PlaylistPosition() int {
|
2018-04-25 20:56:26 +02:00
|
|
|
return 0
|
|
|
|
}
|
|
|
|
|
2018-08-17 16:05:54 +02:00
|
|
|
func (v *ucbVideo) IDAndNum() string {
|
2018-02-13 18:47:05 +01:00
|
|
|
return v.ID() + " (?)"
|
|
|
|
}
|
|
|
|
|
2018-08-17 16:05:54 +02:00
|
|
|
func (v *ucbVideo) PublishedAt() time.Time {
|
2018-02-13 18:47:05 +01:00
|
|
|
return v.publishedAt
|
|
|
|
//r := regexp.MustCompile(`(\d\d\d\d)-(\d\d)-(\d\d)`)
|
|
|
|
//matches := r.FindStringSubmatch(v.title)
|
|
|
|
//if len(matches) > 0 {
|
|
|
|
// year, _ := strconv.Atoi(matches[1])
|
|
|
|
// month, _ := strconv.Atoi(matches[2])
|
|
|
|
// day, _ := strconv.Atoi(matches[3])
|
|
|
|
// return time.Date(year, time.Month(month), day, 0, 0, 0, 0, time.UTC)
|
|
|
|
//}
|
|
|
|
//return time.Now()
|
|
|
|
}
|
|
|
|
|
2018-08-17 16:05:54 +02:00
|
|
|
func (v *ucbVideo) getFilename() string {
|
2018-02-13 18:47:05 +01:00
|
|
|
return v.dir + "/" + v.id + ".mp4"
|
|
|
|
}
|
|
|
|
|
2018-08-17 16:05:54 +02:00
|
|
|
func (v *ucbVideo) getClaimName(attempt int) string {
|
2018-02-13 18:47:05 +01:00
|
|
|
reg := regexp.MustCompile(`[^a-zA-Z0-9]+`)
|
|
|
|
suffix := ""
|
|
|
|
if attempt > 1 {
|
|
|
|
suffix = "-" + strconv.Itoa(attempt)
|
|
|
|
}
|
|
|
|
maxLen := 40 - len(suffix)
|
|
|
|
|
|
|
|
chunks := strings.Split(strings.ToLower(strings.Trim(reg.ReplaceAllString(v.title, "-"), "-")), "-")
|
|
|
|
|
|
|
|
name := chunks[0]
|
|
|
|
if len(name) > maxLen {
|
|
|
|
return name[:maxLen]
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, chunk := range chunks[1:] {
|
|
|
|
tmpName := name + "-" + chunk
|
|
|
|
if len(tmpName) > maxLen {
|
|
|
|
if len(name) < 20 {
|
|
|
|
name = tmpName[:maxLen]
|
|
|
|
}
|
|
|
|
break
|
|
|
|
}
|
|
|
|
name = tmpName
|
|
|
|
}
|
|
|
|
|
|
|
|
return name + suffix
|
|
|
|
}
|
|
|
|
|
2018-08-17 16:05:54 +02:00
|
|
|
func (v *ucbVideo) getAbbrevDescription() string {
|
2018-02-13 18:47:05 +01:00
|
|
|
maxLines := 10
|
|
|
|
description := strings.TrimSpace(v.description)
|
|
|
|
if strings.Count(description, "\n") < maxLines {
|
|
|
|
return description
|
|
|
|
}
|
|
|
|
return strings.Join(strings.Split(description, "\n")[:maxLines], "\n") + "\n..."
|
|
|
|
}
|
|
|
|
|
2018-08-17 16:05:54 +02:00
|
|
|
func (v *ucbVideo) download() error {
|
2018-02-13 18:47:05 +01:00
|
|
|
videoPath := v.getFilename()
|
|
|
|
|
|
|
|
_, err := os.Stat(videoPath)
|
|
|
|
if err != nil && !os.IsNotExist(err) {
|
|
|
|
return err
|
|
|
|
} else if err == nil {
|
|
|
|
log.Debugln(v.id + " already exists at " + videoPath)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
creds := credentials.NewStaticCredentials("ID-GOES-HERE", "SECRET-GOES-HERE", "")
|
|
|
|
s, err := session.NewSession(&aws.Config{Region: aws.String("us-east-2"), Credentials: creds})
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
downloader := s3manager.NewDownloader(s)
|
|
|
|
|
|
|
|
out, err := os.Create(videoPath)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
defer out.Close()
|
|
|
|
|
|
|
|
log.Println("lbry-niko2/videos/" + v.channel + "/" + v.id)
|
|
|
|
|
|
|
|
bytesWritten, err := downloader.Download(out, &s3.GetObjectInput{
|
|
|
|
Bucket: aws.String("lbry-niko2"),
|
|
|
|
Key: aws.String("/videos/" + v.channel + "/" + v.id + ".mp4"),
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
} else if bytesWritten == 0 {
|
2018-03-09 17:47:38 +01:00
|
|
|
return errors.Err("zero bytes written")
|
2018-02-13 18:47:05 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2018-08-17 16:05:54 +02:00
|
|
|
func (v *ucbVideo) saveThumbnail() error {
|
2018-02-13 18:47:05 +01:00
|
|
|
resp, err := http.Get("https://s3.us-east-2.amazonaws.com/lbry-niko2/thumbnails/" + v.id)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
defer resp.Body.Close()
|
|
|
|
|
|
|
|
creds := credentials.NewStaticCredentials("ID-GOES-HERE", "SECRET-GOES-HERE", "")
|
|
|
|
s, err := session.NewSession(&aws.Config{Region: aws.String("us-east-2"), Credentials: creds})
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
uploader := s3manager.NewUploader(s)
|
|
|
|
|
|
|
|
_, err = uploader.Upload(&s3manager.UploadInput{
|
|
|
|
Bucket: aws.String("berk.ninja"),
|
|
|
|
Key: aws.String("thumbnails/" + v.id),
|
|
|
|
ContentType: aws.String("image/jpeg"),
|
|
|
|
Body: resp.Body,
|
|
|
|
})
|
|
|
|
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2018-09-18 22:57:25 +02:00
|
|
|
func (v *ucbVideo) publish(daemon *jsonrpc.Client, claimAddress string, amount float64, channelID string, namer *namer.Namer) (*SyncSummary, error) {
|
2018-02-13 18:47:05 +01:00
|
|
|
options := jsonrpc.PublishOptions{
|
2019-01-30 13:42:23 +01:00
|
|
|
Metadata: &jsonrpc.Metadata{
|
|
|
|
Title: v.title,
|
|
|
|
Description: v.getAbbrevDescription(),
|
|
|
|
Author: "UC Berkeley",
|
|
|
|
Language: "en",
|
|
|
|
License: "see description",
|
|
|
|
Thumbnail: strPtr("https://berk.ninja/thumbnails/" + v.id),
|
|
|
|
NSFW: false,
|
|
|
|
},
|
2018-08-03 23:19:36 +02:00
|
|
|
ChannelID: &channelID,
|
2019-01-30 13:42:23 +01:00
|
|
|
ClaimAddress: &claimAddress,
|
2018-08-03 23:19:36 +02:00
|
|
|
ChangeAddress: &claimAddress,
|
2018-02-13 18:47:05 +01:00
|
|
|
}
|
2018-09-18 21:20:34 +02:00
|
|
|
return publishAndRetryExistingNames(daemon, v.title, v.getFilename(), amount, options, namer)
|
2018-02-13 18:47:05 +01:00
|
|
|
}
|
|
|
|
|
2018-08-17 16:05:54 +02:00
|
|
|
func (v *ucbVideo) Size() *int64 {
|
2018-08-14 17:09:23 +02:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-01-03 19:55:27 +01:00
|
|
|
func (v *ucbVideo) Sync(daemon *jsonrpc.Client, claimAddress string, amount float64, channelID string, maxVideoSize int, namer *namer.Namer, maxVideoLength float64) (*SyncSummary, error) {
|
2018-02-13 18:47:05 +01:00
|
|
|
//download and thumbnail can be done in parallel
|
|
|
|
err := v.download()
|
|
|
|
if err != nil {
|
2018-07-21 01:56:36 +02:00
|
|
|
return nil, errors.Prefix("download error", err)
|
2018-02-13 18:47:05 +01:00
|
|
|
}
|
|
|
|
log.Debugln("Downloaded " + v.id)
|
|
|
|
|
|
|
|
//err = v.SaveThumbnail()
|
|
|
|
//if err != nil {
|
|
|
|
// return errors.WrapPrefix(err, "thumbnail error", 0)
|
|
|
|
//}
|
|
|
|
//log.Debugln("Created thumbnail for " + v.id)
|
|
|
|
|
2018-09-18 22:57:25 +02:00
|
|
|
summary, err := v.publish(daemon, claimAddress, amount, channelID, namer)
|
2018-02-13 18:47:05 +01:00
|
|
|
if err != nil {
|
2018-07-21 01:56:36 +02:00
|
|
|
return nil, errors.Prefix("publish error", err)
|
2018-02-13 18:47:05 +01:00
|
|
|
}
|
|
|
|
|
2018-07-21 01:56:36 +02:00
|
|
|
return summary, nil
|
2018-02-13 18:47:05 +01:00
|
|
|
}
|