2023-05-29 21:18:04 -05:00
|
|
|
package twitch
|
2023-05-20 21:18:10 -05:00
|
|
|
|
|
|
|
import (
|
2023-05-31 18:39:07 -05:00
|
|
|
"encoding/base64"
|
2023-05-20 21:18:10 -05:00
|
|
|
"errors"
|
2023-06-02 07:13:45 -05:00
|
|
|
"os"
|
2023-05-29 21:18:04 -05:00
|
|
|
"safetwitch-backend/extractor"
|
2023-05-20 21:18:10 -05:00
|
|
|
"safetwitch-backend/extractor/structs"
|
2023-06-01 14:49:50 -05:00
|
|
|
"strings"
|
2023-05-22 09:28:31 -05:00
|
|
|
"time"
|
2023-05-20 21:18:10 -05:00
|
|
|
|
|
|
|
"github.com/tidwall/gjson"
|
|
|
|
)
|
|
|
|
|
2023-07-03 14:28:20 -05:00
|
|
|
func ParseStreamer(streamerData gjson.Result, isLive bool, socials []structs.Social, stream *structs.Stream, login string, streamerBanner string) (structs.Streamer, error) {
|
2023-06-06 07:31:12 -05:00
|
|
|
// Store streamerColorHex in memory to use as pointer
|
|
|
|
var streamerColorHex *string
|
|
|
|
rawStreamerColorHex := streamerData.Get("user.primaryColorHex").String()
|
|
|
|
if rawStreamerColorHex == "" {
|
|
|
|
streamerColorHex = nil
|
|
|
|
} else {
|
|
|
|
streamerColorHex = &rawStreamerColorHex
|
|
|
|
}
|
|
|
|
|
2023-07-03 14:28:20 -05:00
|
|
|
var bannerUrl *string = &streamerBanner
|
|
|
|
if *bannerUrl == "" {
|
|
|
|
bannerUrl = nil
|
|
|
|
} else {
|
|
|
|
proxied := extractor.ProxyUrl(*bannerUrl)
|
|
|
|
bannerUrl = &proxied
|
|
|
|
}
|
2023-06-19 21:59:34 -05:00
|
|
|
|
2023-06-06 07:31:12 -05:00
|
|
|
parsedStreamer := structs.Streamer{
|
|
|
|
Username: streamerData.Get("user.displayName").String(),
|
2023-06-19 21:59:34 -05:00
|
|
|
Login: login,
|
2023-06-06 07:31:12 -05:00
|
|
|
About: streamerData.Get("user.description").String(),
|
|
|
|
Pfp: extractor.ProxyUrl(streamerData.Get("user.profileImageURL").String()),
|
2023-07-03 14:28:20 -05:00
|
|
|
Banner: bannerUrl,
|
2023-06-06 07:31:12 -05:00
|
|
|
Followers: int(streamerData.Get("user.followers.totalCount").Int()),
|
|
|
|
Socials: socials,
|
|
|
|
IsLive: isLive,
|
|
|
|
IsPartner: streamerData.Get("user.isPartner").Bool(),
|
|
|
|
ColorHex: streamerColorHex,
|
|
|
|
Id: streamerData.Get("user.id").String(),
|
|
|
|
Stream: stream,
|
|
|
|
}
|
|
|
|
|
|
|
|
return parsedStreamer, nil
|
|
|
|
}
|
|
|
|
|
2023-05-20 21:18:10 -05:00
|
|
|
func ParseSocials(data string) ([]structs.Social, error) {
|
|
|
|
var parsedSocials []structs.Social
|
|
|
|
result := gjson.Get(data, "user.channel.socialMedias")
|
|
|
|
for _, social := range result.Array() {
|
|
|
|
parsedSocials = append(parsedSocials, structs.Social{
|
2023-05-29 15:02:10 -05:00
|
|
|
Name: social.Get("title").String(),
|
|
|
|
Type: social.Get("name").String(),
|
|
|
|
Url: social.Get("url").String(),
|
2023-05-20 21:18:10 -05:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
if !result.Exists() {
|
2023-05-22 09:28:31 -05:00
|
|
|
return parsedSocials, errors.New("error while parsing socials, path does not exist")
|
2023-05-20 21:18:10 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
return parsedSocials, nil
|
|
|
|
}
|
2023-05-22 09:28:31 -05:00
|
|
|
|
|
|
|
func ParseStream(data string) (*structs.Stream, error) {
|
|
|
|
// check if live
|
|
|
|
stream := gjson.Get(data, "1.data.user.stream")
|
|
|
|
if !stream.IsObject() {
|
|
|
|
return nil, errors.New("streamer is not live")
|
|
|
|
}
|
|
|
|
|
|
|
|
var tags []string
|
|
|
|
tagArea := gjson.Get(data, "2.data.user.stream.freeformTags").Array()
|
|
|
|
for _, tag := range tagArea {
|
|
|
|
tags = append(tags, tag.Get("name").String())
|
|
|
|
}
|
|
|
|
|
|
|
|
time, err := time.Parse(time.RFC3339, stream.Get("createdAt").String())
|
|
|
|
if err != nil {
|
|
|
|
return &structs.Stream{}, err
|
|
|
|
}
|
|
|
|
|
|
|
|
parsedStream := structs.Stream{
|
|
|
|
Title: gjson.Get(data, "1.data.user.lastBroadcast.title").String(),
|
|
|
|
Topic: stream.Get("game.name").String(),
|
|
|
|
StartedAt: time,
|
|
|
|
Tags: tags,
|
|
|
|
Viewers: int(gjson.Get(data, "4.data.user.stream.viewersCount").Int()),
|
2023-05-29 21:18:04 -05:00
|
|
|
Preview: extractor.ProxyUrl(gjson.Get(data, "3.data.user.stream.previewImageURL").String()),
|
2023-05-22 09:28:31 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
return &parsedStream, nil
|
|
|
|
}
|
2023-05-26 09:17:55 -05:00
|
|
|
|
|
|
|
// discover
|
2023-05-28 15:19:39 -05:00
|
|
|
func ParseCategory(data gjson.Result) (structs.CategoryPreview, error) {
|
2023-05-26 09:17:55 -05:00
|
|
|
tags := data.Get("node.tags").Array()
|
|
|
|
var parsedTags []string
|
|
|
|
|
|
|
|
for _, tag := range tags {
|
|
|
|
parsedTags = append(parsedTags, tag.Get("localizedName").String())
|
|
|
|
}
|
|
|
|
|
2023-05-28 15:19:39 -05:00
|
|
|
return structs.CategoryPreview{
|
2023-05-26 09:17:55 -05:00
|
|
|
Name: data.Get("node.name").String(),
|
|
|
|
DisplayName: data.Get("node.displayName").String(),
|
2023-05-28 15:19:39 -05:00
|
|
|
Viewers: int(data.Get("node.viewersCount").Int()),
|
2023-05-26 09:17:55 -05:00
|
|
|
CreatedAt: data.Get("node.originalReleaseDate").Time(),
|
|
|
|
Tags: parsedTags,
|
2023-07-29 19:57:23 -05:00
|
|
|
Cursor: data.Get("cursor").String(),
|
2023-05-29 21:18:04 -05:00
|
|
|
Image: extractor.ProxyUrl(data.Get("node.avatarURL").String()),
|
2023-05-26 09:17:55 -05:00
|
|
|
}, nil
|
|
|
|
}
|
2023-05-28 15:19:39 -05:00
|
|
|
|
|
|
|
func ParseMinifiedStream(data gjson.Result) (structs.CategoryMinifiedStream, error) {
|
|
|
|
var tags []string
|
2023-05-31 18:39:07 -05:00
|
|
|
tagArea := data.Get("node.freeformTacgs").Array()
|
2023-05-28 15:19:39 -05:00
|
|
|
for _, tag := range tagArea {
|
|
|
|
tags = append(tags, tag.Get("name").String())
|
|
|
|
}
|
|
|
|
|
|
|
|
// Store streamerColorHex in memory to use as pointer
|
|
|
|
var streamerColorHex *string
|
|
|
|
rawStreamerColorHex := data.Get("node.broadcaster.primaryColorHex").String()
|
|
|
|
if rawStreamerColorHex == "" {
|
|
|
|
streamerColorHex = nil
|
|
|
|
} else {
|
|
|
|
streamerColorHex = &rawStreamerColorHex
|
|
|
|
}
|
|
|
|
|
|
|
|
parsedStream := structs.CategoryMinifiedStream{
|
|
|
|
Title: data.Get("node.title").String(),
|
|
|
|
Viewers: int(data.Get("node.viewersCount").Int()),
|
2023-05-29 21:18:04 -05:00
|
|
|
Preview: extractor.ProxyUrl(data.Get("node.previewImageURL").String()),
|
2023-05-28 15:19:39 -05:00
|
|
|
Tags: tags,
|
|
|
|
Streamer: structs.CategoryMinifiedStreamer{
|
|
|
|
Name: data.Get("node.broadcaster.login").String(),
|
2023-05-29 21:18:04 -05:00
|
|
|
Pfp: extractor.ProxyUrl(data.Get("node.broadcaster.profileImageURL").String()),
|
2023-05-28 15:19:39 -05:00
|
|
|
ColorHex: streamerColorHex,
|
|
|
|
},
|
|
|
|
Cursor: data.Get("cursor").String(),
|
|
|
|
}
|
|
|
|
|
|
|
|
return parsedStream, nil
|
|
|
|
}
|
2023-05-31 18:39:07 -05:00
|
|
|
|
|
|
|
func ParseBadges(data gjson.Result) ([]structs.Badge, error) {
|
|
|
|
var formattedBadges = []structs.Badge{}
|
|
|
|
|
|
|
|
for _, badge := range data.Array() {
|
|
|
|
id := badge.Get("id").String()
|
|
|
|
decodedId, err := base64.StdEncoding.DecodeString(id)
|
|
|
|
if err != nil {
|
|
|
|
return []structs.Badge{}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
formattedBadges = append(formattedBadges, structs.Badge{
|
|
|
|
Id: string(decodedId),
|
|
|
|
SetId: badge.Get("setID").String(),
|
|
|
|
Title: badge.Get("title").String(),
|
|
|
|
Version: badge.Get("version").String(),
|
|
|
|
Images: map[string]string{
|
|
|
|
"image1x": extractor.ProxyUrl(badge.Get("image1x").String()),
|
|
|
|
"image2x": extractor.ProxyUrl(badge.Get("image2x").String()),
|
|
|
|
"image4x": extractor.ProxyUrl(badge.Get("image4x").String()),
|
|
|
|
},
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
return formattedBadges, nil
|
|
|
|
}
|
2023-06-01 14:49:50 -05:00
|
|
|
|
2023-07-17 12:16:33 -05:00
|
|
|
func StreamSubProxyUrl(url string, isVOD bool) string {
|
2023-06-02 07:13:45 -05:00
|
|
|
encodedUrl := base64.StdEncoding.EncodeToString([]byte(url))
|
|
|
|
backendUrl := os.Getenv("URL")
|
|
|
|
|
2023-07-17 12:16:33 -05:00
|
|
|
if isVOD {
|
|
|
|
return backendUrl + "/proxy/vod/sub/" + encodedUrl + "/video.m3u8"
|
|
|
|
} else {
|
|
|
|
return backendUrl + "/proxy/stream/sub/" + encodedUrl
|
|
|
|
}
|
2023-06-02 07:13:45 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
func StreamSegmentProxyUrl(url string) string {
|
|
|
|
encodedUrl := base64.StdEncoding.EncodeToString([]byte(url))
|
|
|
|
backendUrl := os.Getenv("URL")
|
|
|
|
|
|
|
|
return backendUrl + "/proxy/stream/segment/" + encodedUrl
|
|
|
|
}
|
|
|
|
|
2023-07-17 12:16:33 -05:00
|
|
|
func ProxyPlaylistFile(playlist string, isSubPlaylist bool, isVod bool) string {
|
2023-06-01 14:49:50 -05:00
|
|
|
// Split the playlist into individual entries
|
|
|
|
entries := strings.Split(playlist, "\n")[1:] // Ignore the first line which contains the M3U header
|
|
|
|
|
|
|
|
// Loop through each entry and replace the URL
|
|
|
|
for i, entry := range entries {
|
|
|
|
if strings.HasPrefix(entry, "http") { // Only modify lines that contain URLs
|
2023-06-02 07:13:45 -05:00
|
|
|
var newURL string
|
|
|
|
if isSubPlaylist {
|
|
|
|
newURL = StreamSegmentProxyUrl(entry)
|
|
|
|
} else {
|
2023-07-17 12:16:33 -05:00
|
|
|
newURL = StreamSubProxyUrl(entry, isVod)
|
2023-06-02 07:13:45 -05:00
|
|
|
}
|
2023-06-01 14:49:50 -05:00
|
|
|
entries[i] = newURL
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-07-17 12:16:33 -05:00
|
|
|
if isVod {
|
|
|
|
// Loop through each entry and replace the URL
|
|
|
|
for i, entry := range entries {
|
|
|
|
if strings.HasSuffix(entry, ".ts") {
|
|
|
|
entries[i] = entry
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-06-01 14:49:50 -05:00
|
|
|
// Join the modified entries back into a single string separated by a newline
|
|
|
|
modifiedPlaylist := "#EXTM3U\n" + strings.Join(entries, "\n")
|
|
|
|
|
|
|
|
return modifiedPlaylist
|
|
|
|
}
|
2023-07-04 19:54:12 -05:00
|
|
|
|
|
|
|
func ParseMinifiedCategory(data gjson.Result) structs.MinifiedCategory {
|
|
|
|
return structs.MinifiedCategory{
|
|
|
|
Image: extractor.ProxyUrl(data.Get("boxArtURL").String()),
|
|
|
|
Name: data.Get("name").String(),
|
|
|
|
DisplayName: data.Get("displayName").String(),
|
|
|
|
Id: data.Get("id").String(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func ParseMinifiedStreamer(data gjson.Result) structs.MinifiedStreamer {
|
|
|
|
return structs.MinifiedStreamer{
|
|
|
|
Name: data.Get("displayName").String(),
|
|
|
|
Login: data.Get("login").String(),
|
|
|
|
Pfp: extractor.ProxyUrl(data.Get("profileImageURL").String()),
|
|
|
|
ColorHex: data.Get("primaryColorHex").String(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-07-17 12:16:33 -05:00
|
|
|
func ParseVideo(data gjson.Result, streamer structs.Streamer) structs.Video {
|
2023-07-04 19:54:12 -05:00
|
|
|
return structs.Video{
|
2023-07-20 13:13:41 -05:00
|
|
|
Type: "vod",
|
2023-07-04 19:54:12 -05:00
|
|
|
Preview: extractor.ProxyUrl(data.Get("previewThumbnailURL").String()),
|
|
|
|
Game: ParseMinifiedCategory(data.Get("game")),
|
|
|
|
Duration: int(data.Get("lengthSeconds").Int()),
|
|
|
|
Title: data.Get("title").String(),
|
|
|
|
PublishedAt: data.Get("createdAt").Time(),
|
|
|
|
Views: int(data.Get("viewCount").Int()),
|
2023-07-17 12:16:33 -05:00
|
|
|
Streamer: streamer,
|
|
|
|
Id: data.Get("id").String(),
|
2023-07-04 19:54:12 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-07-17 12:16:33 -05:00
|
|
|
func ParseClip(data gjson.Result, streamer structs.Streamer) structs.Video {
|
2023-07-04 19:54:12 -05:00
|
|
|
return structs.Video{
|
2023-07-20 13:13:41 -05:00
|
|
|
Type: "clip",
|
2023-07-04 19:54:12 -05:00
|
|
|
Preview: extractor.ProxyUrl(data.Get("thumbnailURL").String()),
|
|
|
|
Game: ParseMinifiedCategory(data.Get("clipGame")),
|
|
|
|
Duration: int(data.Get("lengthSeconds").Int()),
|
|
|
|
Title: data.Get("clipTitle").String(),
|
|
|
|
PublishedAt: data.Get("publishedAt").Time(),
|
|
|
|
Views: int(data.Get("clipViewCount").Int()),
|
2023-07-17 12:16:33 -05:00
|
|
|
Streamer: streamer,
|
2023-08-21 22:28:23 -05:00
|
|
|
Id: data.Get("slug").String(),
|
2023-07-04 19:54:12 -05:00
|
|
|
}
|
|
|
|
}
|
2023-07-17 12:16:33 -05:00
|
|
|
func ParseShelve(data gjson.Result, streamer structs.Streamer) structs.Shelve {
|
2023-07-04 19:54:12 -05:00
|
|
|
rawVideos := data.Get("items").Array()
|
|
|
|
parsedVideos := []structs.Video{}
|
|
|
|
|
|
|
|
isClip := data.Get("type").String() == "TOP_CLIPS"
|
|
|
|
for _, video := range rawVideos {
|
|
|
|
if isClip {
|
2023-07-17 12:16:33 -05:00
|
|
|
parsedVideos = append(parsedVideos, ParseClip(video, streamer))
|
2023-07-04 19:54:12 -05:00
|
|
|
} else {
|
2023-07-17 12:16:33 -05:00
|
|
|
parsedVideos = append(parsedVideos, ParseVideo(video, streamer))
|
2023-07-04 19:54:12 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return structs.Shelve{
|
|
|
|
Title: data.Get("title").String(),
|
|
|
|
Videos: parsedVideos,
|
|
|
|
}
|
|
|
|
}
|
2023-07-19 19:27:36 -05:00
|
|
|
|
|
|
|
func ParseVODMessage(data gjson.Result) structs.VodComment {
|
|
|
|
messager := structs.MinifiedStreamer{
|
|
|
|
Login: data.Get("node.commenter.login").String(),
|
|
|
|
Name: data.Get("node.commenter.displayName").String(),
|
|
|
|
ColorHex: data.Get("node.message.userColor").String(),
|
|
|
|
}
|
|
|
|
|
|
|
|
parsedBadges := []structs.VodCommentBadge{}
|
|
|
|
for _, badge := range data.Get("node.message.userBadges").Array() {
|
|
|
|
setID := badge.Get("setID").String()
|
|
|
|
version := badge.Get("version").String()
|
|
|
|
|
|
|
|
if version != "" && setID != "" {
|
|
|
|
b := structs.VodCommentBadge{
|
|
|
|
SetID: setID,
|
|
|
|
Version: version,
|
|
|
|
}
|
|
|
|
parsedBadges = append(parsedBadges, b)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return structs.VodComment{
|
|
|
|
Message: data.Get("node.message.fragments.0.text").String(),
|
|
|
|
Offset: int(data.Get("node.contentOffsetSeconds").Int()),
|
|
|
|
Cursor: data.Get("cursor").String(),
|
|
|
|
Messager: messager,
|
|
|
|
Badges: parsedBadges,
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|