mirror of https://github.com/synctv-org/synctv
				
				
				
			
			You cannot select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
	
	
		
			295 lines
		
	
	
		
			6.6 KiB
		
	
	
	
		
			Go
		
	
			
		
		
	
	
			295 lines
		
	
	
		
			6.6 KiB
		
	
	
	
		
			Go
		
	
| package op
 | |
| 
 | |
| import (
 | |
| 	"errors"
 | |
| 	"fmt"
 | |
| 	"hash/crc32"
 | |
| 	"net/url"
 | |
| 	"sync/atomic"
 | |
| 	"time"
 | |
| 
 | |
| 	"github.com/go-resty/resty/v2"
 | |
| 	"github.com/synctv-org/synctv/internal/cache"
 | |
| 	"github.com/synctv-org/synctv/internal/conf"
 | |
| 	"github.com/synctv-org/synctv/internal/model"
 | |
| 	"github.com/synctv-org/synctv/internal/settings"
 | |
| 	"github.com/synctv-org/synctv/utils"
 | |
| 	"github.com/zijiren233/livelib/av"
 | |
| 	"github.com/zijiren233/livelib/container/flv"
 | |
| 	"github.com/zijiren233/livelib/protocol/hls"
 | |
| 	rtmpProto "github.com/zijiren233/livelib/protocol/rtmp"
 | |
| 	"github.com/zijiren233/livelib/protocol/rtmp/core"
 | |
| 	rtmps "github.com/zijiren233/livelib/server"
 | |
| )
 | |
| 
 | |
| type Movie struct {
 | |
| 	*model.Movie
 | |
| 	channel       atomic.Pointer[rtmps.Channel]
 | |
| 	alistCache    atomic.Pointer[cache.AlistMovieCache]
 | |
| 	bilibiliCache atomic.Pointer[cache.BilibiliMovieCache]
 | |
| 	embyCache     atomic.Pointer[cache.EmbyMovieCache]
 | |
| }
 | |
| 
 | |
| func (m *Movie) ExpireId() uint64 {
 | |
| 	switch {
 | |
| 	case m.Movie.Base.VendorInfo.Vendor == model.VendorAlist:
 | |
| 		amcd, _ := m.AlistCache().Raw()
 | |
| 		if amcd != nil && amcd.Ali != nil {
 | |
| 			return uint64(m.AlistCache().Last())
 | |
| 		}
 | |
| 		fallthrough
 | |
| 	default:
 | |
| 		return uint64(crc32.ChecksumIEEE([]byte(m.Movie.ID)))
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func (m *Movie) CheckExpired(expireId uint64) bool {
 | |
| 	switch {
 | |
| 	case m.Movie.Base.VendorInfo.Vendor == model.VendorAlist:
 | |
| 		amcd, _ := m.AlistCache().Raw()
 | |
| 		if amcd != nil && amcd.Ali != nil {
 | |
| 			return time.Now().UnixNano()-int64(expireId) > m.AlistCache().MaxAge()
 | |
| 		}
 | |
| 		fallthrough
 | |
| 	default:
 | |
| 		return expireId != m.ExpireId()
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func (m *Movie) ClearCache() {
 | |
| 	m.alistCache.Store(nil)
 | |
| 
 | |
| 	bmc := m.bilibiliCache.Swap(nil)
 | |
| 	if bmc != nil {
 | |
| 		bmc.NoSharedMovie.Clear()
 | |
| 	}
 | |
| 
 | |
| 	m.embyCache.Store(nil)
 | |
| }
 | |
| 
 | |
| func (m *Movie) AlistCache() *cache.AlistMovieCache {
 | |
| 	c := m.alistCache.Load()
 | |
| 	if c == nil {
 | |
| 		c = cache.NewAlistMovieCache(m.Movie)
 | |
| 		if !m.alistCache.CompareAndSwap(nil, c) {
 | |
| 			return m.AlistCache()
 | |
| 		}
 | |
| 	}
 | |
| 	return c
 | |
| }
 | |
| 
 | |
| func (m *Movie) BilibiliCache() *cache.BilibiliMovieCache {
 | |
| 	c := m.bilibiliCache.Load()
 | |
| 	if c == nil {
 | |
| 		c = cache.NewBilibiliMovieCache(m.Movie)
 | |
| 		if !m.bilibiliCache.CompareAndSwap(nil, c) {
 | |
| 			return m.BilibiliCache()
 | |
| 		}
 | |
| 	}
 | |
| 	return c
 | |
| }
 | |
| 
 | |
| func (m *Movie) EmbyCache() *cache.EmbyMovieCache {
 | |
| 	c := m.embyCache.Load()
 | |
| 	if c == nil {
 | |
| 		c = cache.NewEmbyMovieCache(m.Movie)
 | |
| 		if !m.embyCache.CompareAndSwap(nil, c) {
 | |
| 			return m.EmbyCache()
 | |
| 		}
 | |
| 	}
 | |
| 	return c
 | |
| }
 | |
| 
 | |
| func (m *Movie) Channel() (*rtmps.Channel, error) {
 | |
| 	err := m.initChannel()
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 	return m.channel.Load(), nil
 | |
| }
 | |
| 
 | |
| func genTsName() string {
 | |
| 	return utils.SortUUID()
 | |
| }
 | |
| 
 | |
| func (m *Movie) compareAndSwapInitChannel() *rtmps.Channel {
 | |
| 	c := m.channel.Load()
 | |
| 	if c == nil {
 | |
| 		c = rtmps.NewChannel()
 | |
| 		if !m.channel.CompareAndSwap(nil, c) {
 | |
| 			return m.compareAndSwapInitChannel()
 | |
| 		}
 | |
| 		c.InitHlsPlayer(hls.WithGenTsNameFunc(genTsName))
 | |
| 	}
 | |
| 	return c
 | |
| }
 | |
| 
 | |
| func (m *Movie) initChannel() error {
 | |
| 	switch {
 | |
| 	case m.Movie.Base.Live && m.Movie.Base.RtmpSource:
 | |
| 		m.compareAndSwapInitChannel()
 | |
| 	case m.Movie.Base.Live && m.Movie.Base.Proxy:
 | |
| 		u, err := url.Parse(m.Movie.Base.Url)
 | |
| 		if err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 		switch u.Scheme {
 | |
| 		case "rtmp":
 | |
| 			c := m.compareAndSwapInitChannel()
 | |
| 			err = c.InitHlsPlayer(hls.WithGenTsNameFunc(genTsName))
 | |
| 			if err != nil {
 | |
| 				return err
 | |
| 			}
 | |
| 			go func() {
 | |
| 				for {
 | |
| 					if c.Closed() {
 | |
| 						return
 | |
| 					}
 | |
| 					cli := core.NewConnClient()
 | |
| 					if err = cli.Start(m.Movie.Base.Url, av.PLAY); err != nil {
 | |
| 						cli.Close()
 | |
| 						time.Sleep(time.Second)
 | |
| 						continue
 | |
| 					}
 | |
| 					if err := c.PushStart(rtmpProto.NewReader(cli)); err != nil {
 | |
| 						cli.Close()
 | |
| 						time.Sleep(time.Second)
 | |
| 					}
 | |
| 				}
 | |
| 			}()
 | |
| 		case "http", "https":
 | |
| 			c := m.compareAndSwapInitChannel()
 | |
| 			err := c.InitHlsPlayer(hls.WithGenTsNameFunc(genTsName))
 | |
| 			if err != nil {
 | |
| 				return err
 | |
| 			}
 | |
| 			go func() {
 | |
| 				for {
 | |
| 					if c.Closed() {
 | |
| 						return
 | |
| 					}
 | |
| 					r := resty.New().R()
 | |
| 					for k, v := range m.Movie.Base.Headers {
 | |
| 						r.SetHeader(k, v)
 | |
| 					}
 | |
| 					// r.SetHeader("User-Agent", UserAgent)
 | |
| 					resp, err := r.Get(m.Movie.Base.Url)
 | |
| 					if err != nil {
 | |
| 						time.Sleep(time.Second)
 | |
| 						continue
 | |
| 					}
 | |
| 					if err := c.PushStart(flv.NewReader(resp.RawBody())); err != nil {
 | |
| 						time.Sleep(time.Second)
 | |
| 					}
 | |
| 					resp.RawBody().Close()
 | |
| 				}
 | |
| 			}()
 | |
| 		default:
 | |
| 			return errors.New("unsupported scheme")
 | |
| 		}
 | |
| 	default:
 | |
| 		return errors.New("this movie not support channel")
 | |
| 	}
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func (movie *Movie) Validate() error {
 | |
| 	m := movie.Movie.Base
 | |
| 	if m.VendorInfo.Vendor != "" {
 | |
| 		err := movie.validateVendorMovie()
 | |
| 		if err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 	}
 | |
| 	switch {
 | |
| 	case m.RtmpSource && m.Proxy:
 | |
| 		return errors.New("rtmp source and proxy can't be true at the same time")
 | |
| 	case m.Live && m.RtmpSource:
 | |
| 		if !conf.Conf.Server.Rtmp.Enable {
 | |
| 			return errors.New("rtmp is not enabled")
 | |
| 		}
 | |
| 	case m.Live && m.Proxy:
 | |
| 		if !settings.LiveProxy.Get() {
 | |
| 			return errors.New("live proxy is not enabled")
 | |
| 		}
 | |
| 		u, err := url.Parse(m.Url)
 | |
| 		if err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 		if !settings.AllowProxyToLocal.Get() && utils.IsLocalIP(u.Host) {
 | |
| 			return errors.New("local ip is not allowed")
 | |
| 		}
 | |
| 		switch u.Scheme {
 | |
| 		case "rtmp":
 | |
| 		case "http", "https":
 | |
| 		default:
 | |
| 			return errors.New("unsupported scheme")
 | |
| 		}
 | |
| 	case !m.Live && m.RtmpSource:
 | |
| 		return errors.New("rtmp source can't be true when movie is not live")
 | |
| 	case !m.Live && m.Proxy:
 | |
| 		if !settings.MovieProxy.Get() {
 | |
| 			return errors.New("movie proxy is not enabled")
 | |
| 		}
 | |
| 		if m.VendorInfo.Vendor != "" {
 | |
| 			return nil
 | |
| 		}
 | |
| 		u, err := url.Parse(m.Url)
 | |
| 		if err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 		if !settings.AllowProxyToLocal.Get() && utils.IsLocalIP(u.Host) {
 | |
| 			return errors.New("local ip is not allowed")
 | |
| 		}
 | |
| 		if u.Scheme != "http" && u.Scheme != "https" {
 | |
| 			return errors.New("unsupported scheme")
 | |
| 		}
 | |
| 	case !m.Live && !m.Proxy, m.Live && !m.Proxy && !m.RtmpSource:
 | |
| 		if m.VendorInfo.Vendor == "" {
 | |
| 			u, err := url.Parse(m.Url)
 | |
| 			if err != nil {
 | |
| 				return err
 | |
| 			}
 | |
| 			if u.Scheme != "http" && u.Scheme != "https" {
 | |
| 				return errors.New("unsupported scheme")
 | |
| 			}
 | |
| 		}
 | |
| 	default:
 | |
| 		return errors.New("unknown error")
 | |
| 	}
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func (movie *Movie) validateVendorMovie() error {
 | |
| 	switch movie.Movie.Base.VendorInfo.Vendor {
 | |
| 	case model.VendorBilibili:
 | |
| 		return movie.Movie.Base.VendorInfo.Bilibili.Validate()
 | |
| 
 | |
| 	case model.VendorAlist:
 | |
| 		return movie.Movie.Base.VendorInfo.Alist.Validate()
 | |
| 
 | |
| 	case model.VendorEmby:
 | |
| 		return movie.Movie.Base.VendorInfo.Emby.Validate()
 | |
| 
 | |
| 	default:
 | |
| 		return fmt.Errorf("vendor not implement validate")
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func (m *Movie) Terminate() error {
 | |
| 	c := m.channel.Swap(nil)
 | |
| 	if c != nil {
 | |
| 		err := c.Close()
 | |
| 		if err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 	}
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func (m *Movie) Update(movie *model.BaseMovie) error {
 | |
| 	m.Movie.Base = *movie
 | |
| 	m.ClearCache()
 | |
| 	return m.Terminate()
 | |
| }
 |