2020-10-01 23:55:38 -07:00
|
|
|
package yp
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
2021-11-20 14:42:50 +08:00
|
|
|
"encoding/json"
|
|
|
|
"io"
|
2020-10-01 23:55:38 -07:00
|
|
|
"net/http"
|
2020-11-05 09:20:33 -08:00
|
|
|
"net/url"
|
2020-10-01 23:55:38 -07:00
|
|
|
"time"
|
|
|
|
|
2020-10-06 01:07:09 +08:00
|
|
|
"github.com/owncast/owncast/config"
|
|
|
|
"github.com/owncast/owncast/models"
|
2024-11-15 19:20:58 -08:00
|
|
|
"github.com/owncast/owncast/persistence/configrepository"
|
2020-10-01 23:55:38 -07:00
|
|
|
|
|
|
|
log "github.com/sirupsen/logrus"
|
|
|
|
)
|
|
|
|
|
|
|
|
const pingInterval = 4 * time.Minute
|
|
|
|
|
2023-10-08 14:22:28 -07:00
|
|
|
var (
|
|
|
|
getStatus func() models.Status
|
|
|
|
_inErrorState = false
|
|
|
|
)
|
2020-10-01 23:55:38 -07:00
|
|
|
|
2021-10-25 00:31:45 -07:00
|
|
|
// YP is a service for handling listing in the Owncast directory.
|
2020-10-01 23:55:38 -07:00
|
|
|
type YP struct {
|
|
|
|
timer *time.Ticker
|
|
|
|
}
|
|
|
|
|
|
|
|
type ypPingResponse struct {
|
|
|
|
Key string `json:"key"`
|
|
|
|
Error string `json:"error"`
|
|
|
|
ErrorCode int `json:"errorCode"`
|
2023-05-30 10:31:43 -07:00
|
|
|
Success bool `json:"success"`
|
2020-10-01 23:55:38 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
type ypPingRequest struct {
|
|
|
|
Key string `json:"key"`
|
|
|
|
URL string `json:"url"`
|
|
|
|
}
|
|
|
|
|
2020-11-13 00:14:59 +01:00
|
|
|
// NewYP creates a new instance of the YP service handler.
|
2020-10-01 23:55:38 -07:00
|
|
|
func NewYP(getStatusFunc func() models.Status) *YP {
|
|
|
|
getStatus = getStatusFunc
|
|
|
|
return &YP{}
|
|
|
|
}
|
|
|
|
|
2020-11-13 00:14:59 +01:00
|
|
|
// Start is run when a live stream begins to start pinging YP.
|
2020-10-01 23:55:38 -07:00
|
|
|
func (yp *YP) Start() {
|
|
|
|
yp.timer = time.NewTicker(pingInterval)
|
2020-11-14 18:39:53 -08:00
|
|
|
for range yp.timer.C {
|
|
|
|
yp.ping()
|
|
|
|
}
|
2020-10-01 23:55:38 -07:00
|
|
|
|
|
|
|
yp.ping()
|
|
|
|
}
|
|
|
|
|
2020-11-13 00:14:59 +01:00
|
|
|
// Stop stops the pinging of YP.
|
2020-10-01 23:55:38 -07:00
|
|
|
func (yp *YP) Stop() {
|
|
|
|
yp.timer.Stop()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (yp *YP) ping() {
|
2024-11-15 19:20:58 -08:00
|
|
|
configRepository := configrepository.Get()
|
|
|
|
|
|
|
|
if !configRepository.GetDirectoryEnabled() {
|
2021-03-04 01:48:10 -08:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2021-03-22 20:31:25 -07:00
|
|
|
// Hack: Don't allow ping'ing when offline.
|
|
|
|
// It shouldn't even be trying to, but on some instances the ping timer isn't stopping.
|
|
|
|
if !getStatus().Online {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2024-11-15 19:20:58 -08:00
|
|
|
myInstanceURL := configRepository.GetServerURL()
|
2021-02-18 23:05:52 -08:00
|
|
|
if myInstanceURL == "" {
|
|
|
|
log.Warnln("Server URL not set in the configuration. Directory access is disabled until this is set.")
|
|
|
|
return
|
|
|
|
}
|
2021-09-12 00:18:15 -07:00
|
|
|
isValidInstanceURL := isURL(myInstanceURL)
|
2020-11-05 09:20:33 -08:00
|
|
|
if myInstanceURL == "" || !isValidInstanceURL {
|
|
|
|
if !_inErrorState {
|
|
|
|
log.Warnln("YP Error: unable to use", myInstanceURL, "as a public instance URL. Fix this value in your configuration.")
|
|
|
|
}
|
|
|
|
_inErrorState = true
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2024-11-15 19:20:58 -08:00
|
|
|
key := configRepository.GetDirectoryRegistrationKey()
|
2020-10-01 23:55:38 -07:00
|
|
|
|
2024-11-15 19:20:58 -08:00
|
|
|
log.Traceln("Pinging YP as: ", configRepository.GetServerName(), "with key", key)
|
2020-10-01 23:55:38 -07:00
|
|
|
|
|
|
|
request := ypPingRequest{
|
|
|
|
Key: key,
|
|
|
|
URL: myInstanceURL,
|
|
|
|
}
|
|
|
|
|
|
|
|
req, err := json.Marshal(request)
|
|
|
|
if err != nil {
|
|
|
|
log.Errorln(err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2021-03-16 13:57:28 -07:00
|
|
|
pingURL := config.GetDefaults().YPServer + "/api/ping"
|
2020-11-14 18:39:53 -08:00
|
|
|
resp, err := http.Post(pingURL, "application/json", bytes.NewBuffer(req)) //nolint
|
2020-10-01 23:55:38 -07:00
|
|
|
if err != nil {
|
|
|
|
log.Errorln(err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
defer resp.Body.Close()
|
2021-11-20 14:42:50 +08:00
|
|
|
body, err := io.ReadAll(resp.Body)
|
2020-10-01 23:55:38 -07:00
|
|
|
if err != nil {
|
|
|
|
log.Errorln(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
pingResponse := ypPingResponse{}
|
2021-07-09 20:16:44 +02:00
|
|
|
if err := json.Unmarshal(body, &pingResponse); err != nil {
|
2020-11-14 18:39:53 -08:00
|
|
|
log.Errorln(err)
|
|
|
|
}
|
2020-10-01 23:55:38 -07:00
|
|
|
|
|
|
|
if !pingResponse.Success {
|
2020-11-05 09:14:47 -08:00
|
|
|
if !_inErrorState {
|
|
|
|
log.Warnln("YP Ping error returned from service:", pingResponse.Error)
|
|
|
|
}
|
|
|
|
_inErrorState = true
|
2020-10-01 23:55:38 -07:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2020-11-05 09:14:47 -08:00
|
|
|
_inErrorState = false
|
|
|
|
|
2020-10-01 23:55:38 -07:00
|
|
|
if pingResponse.Key != key {
|
2024-11-15 19:20:58 -08:00
|
|
|
if err := configRepository.SetDirectoryRegistrationKey(key); err != nil {
|
2021-07-19 23:37:06 -07:00
|
|
|
log.Errorln("unable to save directory key:", err)
|
|
|
|
}
|
2020-10-01 23:55:38 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-09-12 00:18:15 -07:00
|
|
|
func isURL(str string) bool {
|
2020-11-05 09:20:33 -08:00
|
|
|
u, err := url.Parse(str)
|
|
|
|
return err == nil && u.Scheme != "" && u.Host != ""
|
|
|
|
}
|