package twitch import ( "bytes" "encoding/json" "fmt" "io/ioutil" "net/http" "net/url" "regexp" "strings" "sync" "text/template" "time" "github.com/go-chi/chi/v5" "github.com/rs/zerolog/log" "github.com/velour/catbase/bot" "github.com/velour/catbase/bot/msg" "github.com/velour/catbase/config" ) const ( isStreamingTplFallback = "{{.Name}} is streaming {{.Game}} at {{.URL}}" notStreamingTplFallback = "{{.Name}} is not streaming" stoppedStreamingTplFallback = "{{.Name}} just stopped streaming" ) type Twitch struct { b bot.Bot c *config.Config twitchList map[string]*Twitcher tbl bot.HandlerTable ircConnected bool irc *IRC ircLock sync.Mutex bridgeMap map[string]string } type Twitcher struct { name string gameID string isStreaming bool } func (t Twitcher) URL() string { u, _ := url.Parse("https://twitch.tv/") u2, _ := url.Parse(t.name) return u.ResolveReference(u2).String() } type stream struct { Data []struct { ID string `json:"id"` UserID string `json:"user_id"` GameID string `json:"game_id"` CommunityIds []string `json:"community_ids"` Type string `json:"type"` Title string `json:"title"` ViewerCount int `json:"viewer_count"` StartedAt time.Time `json:"started_at"` Language string `json:"language"` ThumbnailURL string `json:"thumbnail_url"` } `json:"data"` Pagination struct { Cursor string `json:"cursor"` } `json:"pagination"` } func New(b bot.Bot) *Twitch { p := &Twitch{ b: b, c: b.Config(), twitchList: map[string]*Twitcher{}, bridgeMap: map[string]string{}, } for _, ch := range p.c.GetArray("Twitch.Channels", []string{}) { for _, twitcherName := range p.c.GetArray("Twitch."+ch+".Users", []string{}) { twitcherName = strings.ToLower(twitcherName) if _, ok := p.twitchList[twitcherName]; !ok { p.twitchList[twitcherName] = &Twitcher{ name: twitcherName, gameID: "", } } } go p.twitchChannelLoop(b.DefaultConnector(), ch) } go p.twitchAuthLoop(b.DefaultConnector()) p.register() p.registerWeb() return p } func (p *Twitch) registerWeb() { r := chi.NewRouter() r.HandleFunc("/{user}", p.serveStreaming) p.b.RegisterWeb(r, "/isstreaming") } func (p *Twitch) serveStreaming(w http.ResponseWriter, r *http.Request) { userName := strings.ToLower(chi.URLParam(r, "user")) if userName == "" { fmt.Fprint(w, "User not found.") return } twitcher := p.twitchList[userName] if twitcher == nil { fmt.Fprint(w, "User not found.") return } status := "NO." if twitcher.gameID != "" { status = "YES." } context := map[string]any{"Name": twitcher.name, "Status": status} t, err := template.New("streaming").Parse(page) if err != nil { log.Error().Err(err).Msg("Could not parse template!") return } err = t.Execute(w, context) if err != nil { log.Error().Err(err).Msg("Could not execute template!") } } func (p *Twitch) register() { p.tbl = bot.HandlerTable{ { Kind: bot.Message, IsCmd: true, Regex: regexp.MustCompile(`(?i)^twitch status$`), HelpText: "Get status of all twitchers", Handler: p.twitchStatus, }, { Kind: bot.Message, IsCmd: false, Regex: regexp.MustCompile(`(?i)^is (?P.+) streaming\??$`), HelpText: "Check if a specific twitcher is streaming", Handler: p.twitchUserStatus, }, { Kind: bot.Message, IsCmd: true, Regex: regexp.MustCompile(`(?i)^reset twitch$`), HelpText: "Reset the twitch templates", Handler: p.resetTwitch, }, { Kind: bot.Message, IsCmd: true, Regex: regexp.MustCompile(`(?i)^track (?P.+)$`), HelpText: "Bridge to this channel", Handler: p.mkBridge, }, { Kind: bot.Message, IsCmd: true, Regex: regexp.MustCompile(`(?i)^untrack$`), HelpText: "Disconnect a bridge (only in bridged channel)", Handler: p.rmBridge, }, { Kind: bot.Message, IsCmd: false, Regex: regexp.MustCompile(`.*`), Handler: p.bridgeMsg, }, } p.b.Register(p, bot.Help, p.help) p.b.RegisterTable(p, p.tbl) } func (p *Twitch) twitchStatus(r bot.Request) bool { channel := r.Msg.Channel if users := p.c.GetArray("Twitch."+channel+".Users", []string{}); len(users) > 0 { for _, twitcherName := range users { twitcherName = strings.ToLower(twitcherName) // we could re-add them here instead of needing to restart the bot. if t, ok := p.twitchList[twitcherName]; ok { err := p.checkTwitch(r.Conn, channel, t, true) if err != nil { log.Error().Err(err).Msgf("error in checking twitch") } } } } return true } func (p *Twitch) twitchUserStatus(r bot.Request) bool { who := strings.ToLower(r.Values["who"]) if t, ok := p.twitchList[who]; ok { err := p.checkTwitch(r.Conn, r.Msg.Channel, t, true) if err != nil { log.Error().Err(err).Msgf("error in checking twitch") p.b.Send(r.Conn, bot.Message, r.Msg.Channel, "I had trouble with that.") } } else { p.b.Send(r.Conn, bot.Message, r.Msg.Channel, "I don't know who that is.") } return true } func (p *Twitch) resetTwitch(r bot.Request) bool { p.c.Set("twitch.istpl", isStreamingTplFallback) p.c.Set("twitch.nottpl", notStreamingTplFallback) p.c.Set("twitch.stoppedtpl", stoppedStreamingTplFallback) p.b.Send(r.Conn, bot.Message, r.Msg.Channel, "The Twitch templates have been reset.") return true } func (p *Twitch) help(c bot.Connector, kind bot.Kind, message msg.Message, args ...any) bool { msg := "You can set the templates for streams with\n" msg += fmt.Sprintf("twitch.istpl (default: %s)\n", isStreamingTplFallback) msg += fmt.Sprintf("twitch.nottpl (default: %s)\n", notStreamingTplFallback) msg += fmt.Sprintf("twitch.stoppedtpl (default: %s)\n", stoppedStreamingTplFallback) msg += "You can reset all messages with `!reset twitch`" msg += "And you can ask who is streaming with `!twitch status`" p.b.Send(c, bot.Message, message.Channel, msg) return true } func (p *Twitch) twitchAuthLoop(c bot.Connector) { frequency := p.c.GetInt("Twitch.AuthFreq", 60*60) cid := p.c.Get("twitch.clientid", "") secret := p.c.Get("twitch.secret", "") if cid == "" || secret == "" { log.Info().Msgf("Disabling twitch autoauth.") return } log.Info().Msgf("Checking auth every %d seconds", frequency) if err := p.validateCredentials(); err != nil { log.Error().Err(err).Msgf("error checking twitch validity") } for { select { case <-time.After(time.Duration(frequency) * time.Second): if err := p.validateCredentials(); err != nil { log.Error().Err(err).Msgf("error checking twitch validity") } } } } func (p *Twitch) twitchChannelLoop(c bot.Connector, channel string) { frequency := p.c.GetInt("Twitch.Freq", 60) if p.c.Get("twitch.clientid", "") == "" || p.c.Get("twitch.secret", "") == "" { log.Info().Msgf("Disabling twitch autochecking.") return } log.Info().Msgf("Checking channels every %d seconds", frequency) for { time.Sleep(time.Duration(frequency) * time.Second) for _, twitcherName := range p.c.GetArray("Twitch."+channel+".Users", []string{}) { log.Debug().Msgf("checking %s on twiwch", twitcherName) twitcherName = strings.ToLower(twitcherName) if err := p.checkTwitch(c, channel, p.twitchList[twitcherName], false); err != nil { log.Error().Err(err).Msgf("error in twitch loop") } } } } func getRequest(url, clientID, token string) ([]byte, int, bool) { bearer := fmt.Sprintf("Bearer %s", token) var body []byte var resp *http.Response client := &http.Client{} req, err := http.NewRequest("GET", url, nil) if err != nil { goto errCase } req.Header.Add("Client-ID", clientID) req.Header.Add("Authorization", bearer) resp, err = client.Do(req) if err != nil { goto errCase } body, err = ioutil.ReadAll(resp.Body) if err != nil { goto errCase } return body, resp.StatusCode, true errCase: log.Error().Err(err) return []byte{}, resp.StatusCode, false } type twitchInfo struct { Name string Game string URL string } func (p *Twitch) checkTwitch(c bot.Connector, channel string, twitcher *Twitcher, alwaysPrintStatus bool) error { baseURL, err := url.Parse("https://api.twitch.tv/helix/streams") if err != nil { err := fmt.Errorf("error parsing twitch stream URL") log.Error().Msg(err.Error()) return err } query := baseURL.Query() query.Add("user_login", twitcher.name) baseURL.RawQuery = query.Encode() cid := p.c.Get("twitch.clientid", "") token := p.c.Get("twitch.token", "") if cid == token && cid == "" { log.Info().Msgf("Twitch plugin not enabled.") return nil } body, status, ok := getRequest(baseURL.String(), cid, token) if !ok { return fmt.Errorf("got status %d: %s", status, string(body)) } var s stream err = json.Unmarshal(body, &s) if err != nil { log.Error().Err(err).Msgf("error reading twitch data") return err } games := s.Data gameID, title := "", "" if len(games) == 0 { p.twitchList[twitcher.name].isStreaming = false } else { p.twitchList[twitcher.name].isStreaming = true gameID = games[0].GameID if gameID == "" { gameID = "unknown" } title = games[0].Title } info := twitchInfo{ twitcher.name, title, twitcher.URL(), } log.Debug().Interface("info", info).Interface("twitcher", twitcher).Msgf("checking twitch") if alwaysPrintStatus && twitcher.isStreaming { p.streaming(c, channel, info) } else if alwaysPrintStatus && !twitcher.isStreaming { p.notStreaming(c, channel, info) } else if twitcher.gameID != "" && !twitcher.isStreaming { twitcher.gameID = "" p.disconnectBridge(c, twitcher) p.stopped(c, channel, info) twitcher.gameID = "" } else if twitcher.gameID != gameID && twitcher.isStreaming { p.streaming(c, channel, info) p.connectBridge(c, channel, info, twitcher) twitcher.gameID = gameID } return nil } func (p *Twitch) connectBridge(c bot.Connector, ch string, info twitchInfo, twitcher *Twitcher) { msg := fmt.Sprintf("This post is tracking #%s\n<%s>", twitcher.name, twitcher.URL()) err := p.startBridgeMsg( fmt.Sprintf("%s-%s-%s", info.Name, info.Game, time.Now().Format("2006-01-02-15:04")), twitcher.name, msg, ) if err != nil { log.Error().Err(err).Msgf("unable to start bridge") p.b.Send(c, bot.Message, ch, fmt.Sprintf("Unable to start bridge: %s", err)) } } func (p *Twitch) disconnectBridge(c bot.Connector, twitcher *Twitcher) { log.Debug().Msgf("Disconnecting bridge: %s -> %+v", twitcher.name, p.bridgeMap) for threadID, ircCh := range p.bridgeMap { if strings.HasSuffix(ircCh, twitcher.name) { delete(p.bridgeMap, threadID) p.b.Send(c, bot.Message, threadID, "Stopped tracking #"+twitcher.name) } } } func (p *Twitch) stopped(c bot.Connector, ch string, info twitchInfo) { notStreamingTpl := p.c.Get("Twitch.StoppedTpl", stoppedStreamingTplFallback) buf := bytes.Buffer{} t, err := template.New("notStreaming").Parse(notStreamingTpl) if err != nil { log.Error().Err(err) p.b.Send(c, bot.Message, ch, err) t = template.Must(template.New("notStreaming").Parse(stoppedStreamingTplFallback)) } t.Execute(&buf, info) p.b.Send(c, bot.Message, ch, buf.String()) } func (p *Twitch) streaming(c bot.Connector, channel string, info twitchInfo) { isStreamingTpl := p.c.Get("Twitch.IsTpl", isStreamingTplFallback) buf := bytes.Buffer{} t, err := template.New("isStreaming").Parse(isStreamingTpl) if err != nil { log.Error().Err(err) p.b.Send(c, bot.Message, channel, err) t = template.Must(template.New("isStreaming").Parse(isStreamingTplFallback)) } t.Execute(&buf, info) p.b.Send(c, bot.Message, channel, buf.String()) } func (p *Twitch) notStreaming(c bot.Connector, ch string, info twitchInfo) { notStreamingTpl := p.c.Get("Twitch.NotTpl", notStreamingTplFallback) buf := bytes.Buffer{} t, err := template.New("notStreaming").Parse(notStreamingTpl) if err != nil { log.Error().Err(err) p.b.Send(c, bot.Message, ch, err) t = template.Must(template.New("notStreaming").Parse(notStreamingTplFallback)) } t.Execute(&buf, info) p.b.Send(c, bot.Message, ch, buf.String()) } func (p *Twitch) validateCredentials() error { cid := p.c.Get("twitch.clientid", "") token := p.c.Get("twitch.token", "") if token == "" { return p.reAuthenticate() } _, status, ok := getRequest("https://id.twitch.tv/oauth2/validate", cid, token) if !ok || status == http.StatusUnauthorized { return p.reAuthenticate() } log.Debug().Msgf("checked credentials and they were valid") return nil } func (p *Twitch) reAuthenticate() error { cid := p.c.Get("twitch.clientid", "") secret := p.c.Get("twitch.secret", "") if cid == "" || secret == "" { return fmt.Errorf("could not request a new token without config values set") } resp, err := http.PostForm("https://id.twitch.tv/oauth2/token", url.Values{ "client_id": {cid}, "client_secret": {secret}, "grant_type": {"client_credentials"}, }) if err != nil { return err } body, err := ioutil.ReadAll(resp.Body) if err != nil { return err } credentials := struct { AccessToken string `json:"access_token"` ExpiresIn int `json:"expires_in"` TokenType string `json:"token_type"` }{} err = json.Unmarshal(body, &credentials) log.Debug().Int("expires", credentials.ExpiresIn).Msgf("setting new twitch token") return p.c.RegisterSecret("twitch.token", credentials.AccessToken) }