2020-11-20 21:51:10 +00:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
|
|
|
"io/ioutil"
|
|
|
|
"net/http"
|
2020-12-30 15:40:23 +00:00
|
|
|
"net/url"
|
2020-11-20 21:51:10 +00:00
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/pkg/errors"
|
|
|
|
log "github.com/sirupsen/logrus"
|
|
|
|
)
|
|
|
|
|
|
|
|
func updateStats() error {
|
|
|
|
log.Debug("Updating statistics from API")
|
|
|
|
for _, fn := range []func() error{
|
|
|
|
updateFollowers,
|
2020-12-30 15:40:23 +00:00
|
|
|
updateSubscriberCount,
|
2021-05-09 16:39:30 +00:00
|
|
|
func() error { return subscriptions.SendAllSockets(msgTypeStore, store, false, false) },
|
2020-11-20 21:51:10 +00:00
|
|
|
} {
|
|
|
|
if err := fn(); err != nil {
|
|
|
|
return errors.Wrap(err, "update statistics module")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func updateFollowers() error {
|
|
|
|
log.Debug("Updating followers from API")
|
|
|
|
ctx, cancel := context.WithTimeout(context.Background(), twitchRequestTimeout)
|
|
|
|
defer cancel()
|
|
|
|
|
|
|
|
req, err := http.NewRequestWithContext(ctx, http.MethodGet, fmt.Sprintf("https://api.twitch.tv/helix/users/follows?to_id=%s", cfg.TwitchID), nil)
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "assemble follower count request")
|
|
|
|
}
|
|
|
|
req.Header.Set("Client-Id", cfg.TwitchClient)
|
|
|
|
req.Header.Set("Authorization", "Bearer "+cfg.TwitchToken)
|
|
|
|
|
|
|
|
resp, err := http.DefaultClient.Do(req)
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "requesting subscribe")
|
|
|
|
}
|
|
|
|
defer resp.Body.Close()
|
|
|
|
|
|
|
|
if resp.StatusCode != http.StatusOK {
|
|
|
|
body, err := ioutil.ReadAll(resp.Body)
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrapf(err, "unexpected status %d, unable to read body", resp.StatusCode)
|
|
|
|
}
|
|
|
|
return errors.Errorf("unexpected status %d: %s", resp.StatusCode, body)
|
|
|
|
}
|
|
|
|
|
|
|
|
payload := struct {
|
|
|
|
Total int64 `json:"total"`
|
|
|
|
Data []struct {
|
|
|
|
FromName string `json:"from_name"`
|
|
|
|
FollowedAt time.Time `json:"followed_at"`
|
|
|
|
} `json:"data"`
|
|
|
|
// Contains more but I don't care.
|
|
|
|
}{}
|
|
|
|
|
|
|
|
if err = json.NewDecoder(resp.Body).Decode(&payload); err != nil {
|
|
|
|
return errors.Wrap(err, "decode json response")
|
|
|
|
}
|
|
|
|
|
|
|
|
var seen []string
|
|
|
|
for _, f := range payload.Data {
|
|
|
|
seen = append(seen, f.FromName)
|
|
|
|
}
|
|
|
|
|
2020-12-30 15:40:23 +00:00
|
|
|
store.WithModLock(func() error {
|
|
|
|
store.Followers.Count = payload.Total
|
|
|
|
store.Followers.Seen = seen
|
2020-11-20 21:51:10 +00:00
|
|
|
|
2020-12-30 15:40:23 +00:00
|
|
|
return nil
|
|
|
|
})
|
|
|
|
|
|
|
|
return errors.Wrap(store.Save(cfg.StoreFile), "save store")
|
|
|
|
}
|
|
|
|
|
|
|
|
func updateSubscriberCount() error {
|
|
|
|
log.Debug("Updating subscriber count from API")
|
2020-11-20 21:51:10 +00:00
|
|
|
|
2020-12-30 15:40:23 +00:00
|
|
|
var (
|
|
|
|
params = url.Values{"broadcaster_id": []string{cfg.TwitchID}}
|
|
|
|
subCount int64
|
2020-11-20 21:51:10 +00:00
|
|
|
)
|
2020-12-30 15:40:23 +00:00
|
|
|
|
|
|
|
for {
|
|
|
|
ctx, cancel := context.WithTimeout(context.Background(), twitchRequestTimeout)
|
|
|
|
defer cancel()
|
|
|
|
|
|
|
|
req, err := http.NewRequestWithContext(ctx, http.MethodGet, fmt.Sprintf("https://api.twitch.tv/helix/subscriptions?%s", params.Encode()), nil)
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "assemble subscriber request")
|
|
|
|
}
|
|
|
|
req.Header.Set("Client-Id", cfg.TwitchClient)
|
|
|
|
req.Header.Set("Authorization", "Bearer "+cfg.TwitchToken)
|
|
|
|
|
|
|
|
resp, err := http.DefaultClient.Do(req)
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "requesting subscribe")
|
|
|
|
}
|
|
|
|
defer resp.Body.Close()
|
|
|
|
|
|
|
|
if resp.StatusCode != http.StatusOK {
|
|
|
|
body, err := ioutil.ReadAll(resp.Body)
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrapf(err, "unexpected status %d, unable to read body", resp.StatusCode)
|
|
|
|
}
|
|
|
|
return errors.Errorf("unexpected status %d: %s", resp.StatusCode, body)
|
|
|
|
}
|
|
|
|
|
|
|
|
payload := struct {
|
|
|
|
Pagination struct {
|
|
|
|
Cursor string `json:"cursor"`
|
|
|
|
} `json:"pagination"`
|
|
|
|
Data []struct {
|
|
|
|
BroadcasterID string `json:"broadcaster_id"`
|
|
|
|
Tier string `json:"tier"`
|
|
|
|
UserID string `json:"user_id"`
|
|
|
|
} `json:"data"`
|
|
|
|
// Contains more but I don't care.
|
|
|
|
}{}
|
|
|
|
|
|
|
|
if err = json.NewDecoder(resp.Body).Decode(&payload); err != nil {
|
|
|
|
return errors.Wrap(err, "decode json response")
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(payload.Data) == 0 {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, sub := range payload.Data {
|
|
|
|
if sub.UserID == sub.BroadcasterID {
|
|
|
|
// Don't count self
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
subCount++
|
|
|
|
}
|
|
|
|
|
|
|
|
params.Set("after", payload.Pagination.Cursor)
|
|
|
|
}
|
|
|
|
|
|
|
|
store.WithModLock(func() error {
|
|
|
|
store.Subs.Count = subCount
|
|
|
|
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
|
|
|
|
return errors.Wrap(store.Save(cfg.StoreFile), "save store")
|
2020-11-20 21:51:10 +00:00
|
|
|
}
|