2020-10-02 00:12:47 -07:00
|
|
|
package rtmp
|
|
|
|
|
|
|
|
import (
|
2022-12-23 21:26:08 -08:00
|
|
|
"crypto/subtle"
|
2020-10-02 00:12:47 -07:00
|
|
|
"encoding/json"
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"regexp"
|
2020-11-16 22:14:22 -06:00
|
|
|
"strings"
|
2020-10-02 00:12:47 -07:00
|
|
|
|
|
|
|
"github.com/nareix/joy5/format/flv/flvio"
|
2020-11-14 18:39:53 -08:00
|
|
|
"github.com/owncast/owncast/models"
|
2021-06-05 05:09:43 +02:00
|
|
|
log "github.com/sirupsen/logrus"
|
2020-10-02 00:12:47 -07:00
|
|
|
)
|
|
|
|
|
2020-12-16 23:01:09 -08:00
|
|
|
const unknownString = "Unknown"
|
|
|
|
|
2021-10-29 20:29:01 +01:00
|
|
|
var _getInboundDetailsFromMetadataRE = regexp.MustCompile(`\{(.*?)\}`)
|
|
|
|
|
2020-10-02 00:12:47 -07:00
|
|
|
func getInboundDetailsFromMetadata(metadata []interface{}) (models.RTMPStreamMetadata, error) {
|
|
|
|
metadataComponentsString := fmt.Sprintf("%+v", metadata)
|
2020-12-05 18:33:17 -08:00
|
|
|
if !strings.Contains(metadataComponentsString, "onMetaData") {
|
|
|
|
return models.RTMPStreamMetadata{}, errors.New("Not a onMetaData message")
|
2020-11-16 22:14:22 -06:00
|
|
|
}
|
2021-10-29 20:29:01 +01:00
|
|
|
|
|
|
|
submatchall := _getInboundDetailsFromMetadataRE.FindAllString(metadataComponentsString, 1)
|
2020-10-02 00:12:47 -07:00
|
|
|
|
|
|
|
if len(submatchall) == 0 {
|
|
|
|
return models.RTMPStreamMetadata{}, errors.New("unable to parse inbound metadata")
|
|
|
|
}
|
|
|
|
|
|
|
|
metadataJSONString := submatchall[0]
|
|
|
|
var details models.RTMPStreamMetadata
|
2020-11-14 18:39:53 -08:00
|
|
|
err := json.Unmarshal([]byte(metadataJSONString), &details)
|
|
|
|
return details, err
|
2020-10-02 00:12:47 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
func getAudioCodec(codec interface{}) string {
|
2020-12-02 00:19:55 -08:00
|
|
|
if codec == nil {
|
|
|
|
return "No audio"
|
|
|
|
}
|
|
|
|
|
2020-10-02 00:12:47 -07:00
|
|
|
var codecID float64
|
|
|
|
if assertedCodecID, ok := codec.(float64); ok {
|
|
|
|
codecID = assertedCodecID
|
|
|
|
} else {
|
|
|
|
return codec.(string)
|
|
|
|
}
|
|
|
|
|
|
|
|
switch codecID {
|
|
|
|
case flvio.SOUND_MP3:
|
|
|
|
return "MP3"
|
|
|
|
case flvio.SOUND_AAC:
|
|
|
|
return "AAC"
|
|
|
|
case flvio.SOUND_SPEEX:
|
|
|
|
return "Speex"
|
|
|
|
}
|
|
|
|
|
2020-12-16 23:01:09 -08:00
|
|
|
return unknownString
|
2020-10-02 00:12:47 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
func getVideoCodec(codec interface{}) string {
|
2020-12-06 14:28:00 -08:00
|
|
|
if codec == nil {
|
2020-12-16 23:01:09 -08:00
|
|
|
return unknownString
|
2020-12-06 14:28:00 -08:00
|
|
|
}
|
|
|
|
|
2020-10-02 00:12:47 -07:00
|
|
|
var codecID float64
|
|
|
|
if assertedCodecID, ok := codec.(float64); ok {
|
|
|
|
codecID = assertedCodecID
|
|
|
|
} else {
|
|
|
|
return codec.(string)
|
|
|
|
}
|
|
|
|
|
|
|
|
switch codecID {
|
|
|
|
case flvio.VIDEO_H264:
|
|
|
|
return "H.264"
|
|
|
|
case flvio.VIDEO_H265:
|
|
|
|
return "H.265"
|
|
|
|
}
|
|
|
|
|
2020-12-16 23:01:09 -08:00
|
|
|
return unknownString
|
2020-10-02 00:12:47 -07:00
|
|
|
}
|
2021-06-05 05:09:43 +02:00
|
|
|
|
|
|
|
func secretMatch(configStreamKey string, path string) bool {
|
|
|
|
prefix := "/live/"
|
|
|
|
|
|
|
|
if !strings.HasPrefix(path, prefix) {
|
|
|
|
log.Debug("RTMP path does not start with " + prefix)
|
|
|
|
return false // We need the path to begin with $prefix
|
|
|
|
}
|
|
|
|
|
|
|
|
streamingKey := path[len(prefix):] // Remove $prefix
|
2022-12-23 21:26:08 -08:00
|
|
|
|
|
|
|
matches := subtle.ConstantTimeCompare([]byte(streamingKey), []byte(configStreamKey)) == 1
|
|
|
|
return matches
|
2021-06-05 05:09:43 +02:00
|
|
|
}
|