mirror of
https://github.com/Luzifer/twitch-bot.git
synced 2025-01-10 13:21:49 +00:00
Compare commits
3 commits
a01ce9aa5f
...
19a30d342a
Author | SHA1 | Date | |
---|---|---|---|
19a30d342a | |||
30305600e7 | |||
5dd6a5323c |
8 changed files with 187 additions and 37 deletions
|
@ -1,3 +1,11 @@
|
||||||
|
# 3.31.0 / 2024-05-13
|
||||||
|
|
||||||
|
* Improvements
|
||||||
|
* [core] Add locking to prevent concurrent rule executions
|
||||||
|
|
||||||
|
* Bugfixes
|
||||||
|
* [spotify] Fix: Refresh-Token gets revoked when using two functions
|
||||||
|
|
||||||
# 3.30.0 / 2024-04-26
|
# 3.30.0 / 2024-04-26
|
||||||
|
|
||||||
* New Features
|
* New Features
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
package main
|
package main
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"path"
|
||||||
"sync"
|
"sync"
|
||||||
|
|
||||||
"github.com/pkg/errors"
|
"github.com/pkg/errors"
|
||||||
|
@ -8,6 +9,7 @@ import (
|
||||||
"gopkg.in/irc.v4"
|
"gopkg.in/irc.v4"
|
||||||
|
|
||||||
"github.com/Luzifer/go_helpers/v2/fieldcollection"
|
"github.com/Luzifer/go_helpers/v2/fieldcollection"
|
||||||
|
"github.com/Luzifer/twitch-bot/v3/internal/locker"
|
||||||
"github.com/Luzifer/twitch-bot/v3/plugins"
|
"github.com/Luzifer/twitch-bot/v3/plugins"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -79,6 +81,9 @@ func handleMessage(c *irc.Client, m *irc.Message, event *string, eventData *fiel
|
||||||
}
|
}
|
||||||
|
|
||||||
func handleMessageRuleExecution(c *irc.Client, m *irc.Message, r *plugins.Rule, eventData *fieldcollection.FieldCollection) {
|
func handleMessageRuleExecution(c *irc.Client, m *irc.Message, r *plugins.Rule, eventData *fieldcollection.FieldCollection) {
|
||||||
|
locker.LockByKey(path.Join("rule-execution", r.MatcherID()))
|
||||||
|
defer locker.UnlockByKey(path.Join("rule-execution", r.MatcherID()))
|
||||||
|
|
||||||
var (
|
var (
|
||||||
ruleEventData = fieldcollection.NewFieldCollection()
|
ruleEventData = fieldcollection.NewFieldCollection()
|
||||||
preventCooldown bool
|
preventCooldown bool
|
||||||
|
|
|
@ -467,12 +467,12 @@ Example:
|
||||||
|
|
||||||
```
|
```
|
||||||
# Your int this hour: {{ printf "%.0f" (mulf (seededRandom (list "int" .username (now | date "2006-01-02 15") | join ":")) 100) }}%
|
# Your int this hour: {{ printf "%.0f" (mulf (seededRandom (list "int" .username (now | date "2006-01-02 15") | join ":")) 100) }}%
|
||||||
< Your int this hour: 72%
|
< Your int this hour: 82%
|
||||||
```
|
```
|
||||||
|
|
||||||
### `spotifyCurrentPlaying`
|
### `spotifyCurrentPlaying`
|
||||||
|
|
||||||
Retrieves the current playing track for the given channel
|
Retrieves the current playing track for the given channel (returns an empty string when nothing is playing)
|
||||||
|
|
||||||
Syntax: `spotifyCurrentPlaying <channel>`
|
Syntax: `spotifyCurrentPlaying <channel>`
|
||||||
|
|
||||||
|
@ -487,7 +487,7 @@ Example:
|
||||||
|
|
||||||
### `spotifyLink`
|
### `spotifyLink`
|
||||||
|
|
||||||
Retrieves the link for the playing track for the given channel
|
Retrieves the link for the playing track for the given channel (returns an empty string when nothing is playing)
|
||||||
|
|
||||||
Syntax: `spotifyLink <channel>`
|
Syntax: `spotifyLink <channel>`
|
||||||
|
|
||||||
|
|
101
internal/actors/spotify/auth.go
Normal file
101
internal/actors/spotify/auth.go
Normal file
|
@ -0,0 +1,101 @@
|
||||||
|
package spotify
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"fmt"
|
||||||
|
"net/http"
|
||||||
|
"strings"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/Luzifer/twitch-bot/v3/internal/locker"
|
||||||
|
"github.com/sirupsen/logrus"
|
||||||
|
"golang.org/x/oauth2"
|
||||||
|
)
|
||||||
|
|
||||||
|
const expiryGrace = 10 * time.Second
|
||||||
|
|
||||||
|
func getAuthorizedClient(channel, redirectURL string) (client *http.Client, err error) {
|
||||||
|
// In templating functions are called multiple times at once which
|
||||||
|
// with Spotify replacing the refresh-token on each renew would kill
|
||||||
|
// the stored token when multiple spotify functions are called at
|
||||||
|
// once. Therefore we do have this method locking itself until it
|
||||||
|
// has successfully made one request to the users profile and therefore
|
||||||
|
// renewed the token. The next request then will use the token the
|
||||||
|
// previous request renewed.
|
||||||
|
locker.LockByKey(strings.Join([]string{"spotify", "api-access", channel}, ":"))
|
||||||
|
defer locker.UnlockByKey(strings.Join([]string{"spotify", "api-access", channel}, ":"))
|
||||||
|
|
||||||
|
conf, err := oauthConfig(channel, redirectURL)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("getting oauth config: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
var token *oauth2.Token
|
||||||
|
if err = db.ReadEncryptedCoreMeta(strings.Join([]string{"spotify-auth", channel}, ":"), &token); err != nil {
|
||||||
|
return nil, fmt.Errorf("loading oauth token: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
ts := conf.TokenSource(context.Background(), token)
|
||||||
|
|
||||||
|
if token.Expiry.After(time.Now().Add(expiryGrace)) {
|
||||||
|
// Token is still valid long enough, we spare the resources to do
|
||||||
|
// the profile fetch and directly return the client with the token
|
||||||
|
// as the scenario described here does not apply.
|
||||||
|
return oauth2.NewClient(context.Background(), ts), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
logrus.WithField("channel", channel).Debug("refreshing spotify token")
|
||||||
|
|
||||||
|
ctx, cancel := context.WithTimeout(context.Background(), spotifyRequestTimeout)
|
||||||
|
defer cancel()
|
||||||
|
|
||||||
|
// We do a request to /me once to refresh the token if needed
|
||||||
|
req, err := http.NewRequestWithContext(ctx, http.MethodGet, "https://api.spotify.com/v1/me", nil)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("creating currently-playing request: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
oauthClient := oauth2.NewClient(context.Background(), ts)
|
||||||
|
|
||||||
|
resp, err := oauthClient.Do(req)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("executing request: %w", err)
|
||||||
|
}
|
||||||
|
defer func() {
|
||||||
|
if err := resp.Body.Close(); err != nil {
|
||||||
|
logrus.WithError(err).Error("closing Spotify response body (leaked fd)")
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
|
||||||
|
if resp.StatusCode != http.StatusOK {
|
||||||
|
return nil, fmt.Errorf("requesting user profile: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
updToken, err := ts.Token()
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("getting updated token: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := db.StoreEncryptedCoreMeta(strings.Join([]string{"spotify-auth", channel}, ":"), updToken); err != nil {
|
||||||
|
logrus.WithError(err).Error("storing back Spotify auth token")
|
||||||
|
}
|
||||||
|
|
||||||
|
return oauthClient, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func oauthConfig(channel, redirectURL string) (conf *oauth2.Config, err error) {
|
||||||
|
clientID, err := getModuleConfig(actorName, channel).String("clientId")
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("getting clientId for channel: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return &oauth2.Config{
|
||||||
|
ClientID: clientID,
|
||||||
|
Endpoint: oauth2.Endpoint{
|
||||||
|
AuthURL: "https://accounts.spotify.com/authorize",
|
||||||
|
TokenURL: "https://accounts.spotify.com/api/token",
|
||||||
|
},
|
||||||
|
RedirectURL: redirectURL,
|
||||||
|
Scopes: []string{"user-read-currently-playing"},
|
||||||
|
}, nil
|
||||||
|
}
|
|
@ -3,34 +3,25 @@ package spotify
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
"net/http"
|
"net/http"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
"github.com/sirupsen/logrus"
|
"github.com/sirupsen/logrus"
|
||||||
"golang.org/x/oauth2"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
|
var errNotPlaying = errors.New("nothing playing")
|
||||||
|
|
||||||
func getCurrentTrackForChannel(channel string) (track currentPlayingTrackResponse, err error) {
|
func getCurrentTrackForChannel(channel string) (track currentPlayingTrackResponse, err error) {
|
||||||
channel = strings.TrimLeft(channel, "#")
|
channel = strings.TrimLeft(channel, "#")
|
||||||
|
|
||||||
conf, err := oauthConfig(channel, "")
|
client, err := getAuthorizedClient(channel, "")
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return track, fmt.Errorf("getting oauth config: %w", err)
|
return track, fmt.Errorf("retrieving authorized Spotify client: %w", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
var token *oauth2.Token
|
|
||||||
if err = db.ReadEncryptedCoreMeta(strings.Join([]string{"spotify-auth", channel}, ":"), &token); err != nil {
|
|
||||||
return track, fmt.Errorf("loading oauth token: %w", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
defer func() {
|
|
||||||
if err := db.StoreEncryptedCoreMeta(strings.Join([]string{"spotify-auth", channel}, ":"), token); err != nil {
|
|
||||||
logrus.WithError(err).Error("storing back Spotify auth token")
|
|
||||||
}
|
|
||||||
}()
|
|
||||||
|
|
||||||
ctx, cancel := context.WithTimeout(context.Background(), spotifyRequestTimeout)
|
ctx, cancel := context.WithTimeout(context.Background(), spotifyRequestTimeout)
|
||||||
defer cancel()
|
defer cancel()
|
||||||
|
|
||||||
|
@ -39,7 +30,7 @@ func getCurrentTrackForChannel(channel string) (track currentPlayingTrackRespons
|
||||||
return track, fmt.Errorf("creating currently-playing request: %w", err)
|
return track, fmt.Errorf("creating currently-playing request: %w", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
resp, err := conf.Client(context.Background(), token).Do(req)
|
resp, err := client.Do(req)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return track, fmt.Errorf("executing request: %w", err)
|
return track, fmt.Errorf("executing request: %w", err)
|
||||||
}
|
}
|
||||||
|
@ -58,6 +49,10 @@ func getCurrentTrackForChannel(channel string) (track currentPlayingTrackRespons
|
||||||
case http.StatusOK:
|
case http.StatusOK:
|
||||||
// This is perfect, continue below
|
// This is perfect, continue below
|
||||||
|
|
||||||
|
case http.StatusNoContent:
|
||||||
|
// User is not playing anything
|
||||||
|
return track, errNotPlaying
|
||||||
|
|
||||||
case http.StatusUnauthorized:
|
case http.StatusUnauthorized:
|
||||||
// The token is FUBAR
|
// The token is FUBAR
|
||||||
return track, fmt.Errorf("token expired (HTTP 401 - unauthorized)")
|
return track, fmt.Errorf("token expired (HTTP 401 - unauthorized)")
|
||||||
|
@ -85,6 +80,10 @@ func getCurrentTrackForChannel(channel string) (track currentPlayingTrackRespons
|
||||||
func getCurrentArtistTitleForChannel(channel string) (artistTitle string, err error) {
|
func getCurrentArtistTitleForChannel(channel string) (artistTitle string, err error) {
|
||||||
track, err := getCurrentTrackForChannel(channel)
|
track, err := getCurrentTrackForChannel(channel)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
if errors.Is(err, errNotPlaying) {
|
||||||
|
return "", nil
|
||||||
|
}
|
||||||
|
|
||||||
return "", fmt.Errorf("getting track info: %w", err)
|
return "", fmt.Errorf("getting track info: %w", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -102,6 +101,10 @@ func getCurrentArtistTitleForChannel(channel string) (artistTitle string, err er
|
||||||
func getCurrentLinkForChannel(channel string) (link string, err error) {
|
func getCurrentLinkForChannel(channel string) (link string, err error) {
|
||||||
track, err := getCurrentTrackForChannel(channel)
|
track, err := getCurrentTrackForChannel(channel)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
if errors.Is(err, errNotPlaying) {
|
||||||
|
return "", nil
|
||||||
|
}
|
||||||
|
|
||||||
return "", fmt.Errorf("getting track info: %w", err)
|
return "", fmt.Errorf("getting track info: %w", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -70,20 +70,3 @@ func handleStartAuth(w http.ResponseWriter, r *http.Request) {
|
||||||
|
|
||||||
fmt.Fprintln(w, "Spotify is now authorized for this channel, you can close this page")
|
fmt.Fprintln(w, "Spotify is now authorized for this channel, you can close this page")
|
||||||
}
|
}
|
||||||
|
|
||||||
func oauthConfig(channel, redirectURL string) (conf *oauth2.Config, err error) {
|
|
||||||
clientID, err := getModuleConfig(actorName, channel).String("clientId")
|
|
||||||
if err != nil {
|
|
||||||
return nil, fmt.Errorf("getting clientId for channel: %w", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
return &oauth2.Config{
|
|
||||||
ClientID: clientID,
|
|
||||||
Endpoint: oauth2.Endpoint{
|
|
||||||
AuthURL: "https://accounts.spotify.com/authorize",
|
|
||||||
TokenURL: "https://accounts.spotify.com/api/token",
|
|
||||||
},
|
|
||||||
RedirectURL: redirectURL,
|
|
||||||
Scopes: []string{"user-read-currently-playing"},
|
|
||||||
}, nil
|
|
||||||
}
|
|
||||||
|
|
|
@ -35,7 +35,7 @@ func Register(args plugins.RegistrationArguments) (err error) {
|
||||||
plugins.GenericTemplateFunctionGetter(getCurrentArtistTitleForChannel),
|
plugins.GenericTemplateFunctionGetter(getCurrentArtistTitleForChannel),
|
||||||
plugins.TemplateFuncDocumentation{
|
plugins.TemplateFuncDocumentation{
|
||||||
Name: "spotifyCurrentPlaying",
|
Name: "spotifyCurrentPlaying",
|
||||||
Description: "Retrieves the current playing track for the given channel",
|
Description: "Retrieves the current playing track for the given channel (returns an empty string when nothing is playing)",
|
||||||
Syntax: "spotifyCurrentPlaying <channel>",
|
Syntax: "spotifyCurrentPlaying <channel>",
|
||||||
Example: &plugins.TemplateFuncDocumentationExample{
|
Example: &plugins.TemplateFuncDocumentationExample{
|
||||||
MatchMessage: "^!spotify",
|
MatchMessage: "^!spotify",
|
||||||
|
@ -51,7 +51,7 @@ func Register(args plugins.RegistrationArguments) (err error) {
|
||||||
plugins.GenericTemplateFunctionGetter(getCurrentLinkForChannel),
|
plugins.GenericTemplateFunctionGetter(getCurrentLinkForChannel),
|
||||||
plugins.TemplateFuncDocumentation{
|
plugins.TemplateFuncDocumentation{
|
||||||
Name: "spotifyLink",
|
Name: "spotifyLink",
|
||||||
Description: "Retrieves the link for the playing track for the given channel",
|
Description: "Retrieves the link for the playing track for the given channel (returns an empty string when nothing is playing)",
|
||||||
Syntax: "spotifyLink <channel>",
|
Syntax: "spotifyLink <channel>",
|
||||||
Example: &plugins.TemplateFuncDocumentationExample{
|
Example: &plugins.TemplateFuncDocumentationExample{
|
||||||
MatchMessage: "^!spotifylink",
|
MatchMessage: "^!spotifylink",
|
||||||
|
|
50
internal/locker/locker.go
Normal file
50
internal/locker/locker.go
Normal file
|
@ -0,0 +1,50 @@
|
||||||
|
// Package locker contains a way to interact with arbitrary locks
|
||||||
|
package locker
|
||||||
|
|
||||||
|
import "sync"
|
||||||
|
|
||||||
|
var (
|
||||||
|
locks = map[string]*sync.RWMutex{}
|
||||||
|
locksOLocks sync.RWMutex
|
||||||
|
)
|
||||||
|
|
||||||
|
// LockByKey takes a key to lock and locks the corresponding RWMutex
|
||||||
|
func LockByKey(key string) { getLockByKey(key).Lock() }
|
||||||
|
|
||||||
|
// RLockByKey takes a key to lock and read-locks the corresponding RWMutex
|
||||||
|
func RLockByKey(key string) { getLockByKey(key).RLock() }
|
||||||
|
|
||||||
|
// RUnlockByKey takes a key to lock and read-unlocks the corresponding RWMutex
|
||||||
|
func RUnlockByKey(key string) { getLockByKey(key).RUnlock() }
|
||||||
|
|
||||||
|
// UnlockByKey takes a key to lock and unlocks the corresponding RWMutex
|
||||||
|
func UnlockByKey(key string) { getLockByKey(key).Unlock() }
|
||||||
|
|
||||||
|
// WithLock takes a key to lock and a function to execute during the
|
||||||
|
// lock of this key
|
||||||
|
func WithLock(key string, fn func()) {
|
||||||
|
LockByKey(key)
|
||||||
|
defer UnlockByKey(key)
|
||||||
|
|
||||||
|
fn()
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithRLock takes a key to lock and a function to execute during the
|
||||||
|
// read-lock of this key
|
||||||
|
func WithRLock(key string, fn func()) {
|
||||||
|
RLockByKey(key)
|
||||||
|
defer RUnlockByKey(key)
|
||||||
|
|
||||||
|
fn()
|
||||||
|
}
|
||||||
|
|
||||||
|
func getLockByKey(key string) *sync.RWMutex {
|
||||||
|
locksOLocks.Lock()
|
||||||
|
defer locksOLocks.Unlock()
|
||||||
|
|
||||||
|
if locks[key] == nil {
|
||||||
|
locks[key] = new(sync.RWMutex)
|
||||||
|
}
|
||||||
|
|
||||||
|
return locks[key]
|
||||||
|
}
|
Loading…
Reference in a new issue