mirror of
https://github.com/Luzifer/twitch-bot.git
synced 2025-01-04 10:46:02 +00:00
346 lines
11 KiB
Go
346 lines
11 KiB
Go
// Package access contains a service to manage Twitch tokens and scopes
|
|
package access
|
|
|
|
import (
|
|
"context"
|
|
"strings"
|
|
|
|
"github.com/pkg/errors"
|
|
"gorm.io/gorm"
|
|
"gorm.io/gorm/clause"
|
|
|
|
"github.com/Luzifer/go_helpers/v2/backoff"
|
|
"github.com/Luzifer/go_helpers/v2/str"
|
|
"github.com/Luzifer/twitch-bot/v3/internal/helpers"
|
|
"github.com/Luzifer/twitch-bot/v3/pkg/database"
|
|
"github.com/Luzifer/twitch-bot/v3/pkg/twitch"
|
|
)
|
|
|
|
const (
|
|
coreMetaKeyBotToken = "bot_access_token"
|
|
coreMetaKeyBotUsername = "bot_username"
|
|
coreMetaKeyBotRefreshToken = "bot_refresh_token" //#nosec:G101 // That's a key, not a credential
|
|
)
|
|
|
|
type (
|
|
// ClientConfig contains a configuration to derive new Twitch clients
|
|
// from
|
|
ClientConfig struct {
|
|
TwitchClient string
|
|
TwitchClientSecret string
|
|
|
|
FallbackToken string // DEPRECATED
|
|
|
|
TokenUpdateHook func()
|
|
}
|
|
|
|
extendedPermission struct {
|
|
Channel string `gorm:"primaryKey"`
|
|
AccessToken string
|
|
RefreshToken string
|
|
Scopes string
|
|
}
|
|
|
|
// Service manages the permission database
|
|
Service struct{ db database.Connector }
|
|
)
|
|
|
|
// ErrChannelNotAuthorized denotes there is no valid authoriztion for
|
|
// the given channel
|
|
var ErrChannelNotAuthorized = errors.New("channel is not authorized")
|
|
|
|
// New creates a new Service on the given database
|
|
func New(db database.Connector) (*Service, error) {
|
|
return &Service{db}, errors.Wrap(
|
|
db.DB().AutoMigrate(&extendedPermission{}),
|
|
"migrating database schema",
|
|
)
|
|
}
|
|
|
|
// CopyDatabase enables the bot to migrate the access database
|
|
func (*Service) CopyDatabase(src, target *gorm.DB) error {
|
|
return database.CopyObjects(src, target, &extendedPermission{}) //nolint:wrapcheck // Internal helper
|
|
}
|
|
|
|
// GetBotUsername gets the cached bot username
|
|
func (s Service) GetBotUsername() (botUsername string, err error) {
|
|
err = s.db.ReadCoreMeta(coreMetaKeyBotUsername, &botUsername)
|
|
return botUsername, errors.Wrap(err, "reading bot username")
|
|
}
|
|
|
|
// GetChannelPermissions returns the scopes granted for the given channel
|
|
func (s Service) GetChannelPermissions(channel string) ([]string, error) {
|
|
var (
|
|
err error
|
|
perm extendedPermission
|
|
)
|
|
|
|
if err = helpers.Retry(func() error {
|
|
err = s.db.DB().First(&perm, "channel = ?", strings.TrimLeft(channel, "#")).Error
|
|
if errors.Is(err, gorm.ErrRecordNotFound) {
|
|
return nil
|
|
}
|
|
|
|
return errors.Wrap(err, "getting twitch credential from database")
|
|
}); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return strings.Split(perm.Scopes, " "), nil
|
|
}
|
|
|
|
// GetBotTwitchClient returns a twitch.Client configured to act as the
|
|
// bot user
|
|
func (s Service) GetBotTwitchClient(cfg ClientConfig) (*twitch.Client, error) {
|
|
botUsername, err := s.GetBotUsername()
|
|
switch {
|
|
case errors.Is(err, nil):
|
|
// This is fine, we have a username
|
|
return s.GetTwitchClientForChannel(botUsername, cfg)
|
|
|
|
case errors.Is(err, database.ErrCoreMetaNotFound):
|
|
// The bot has no username stored, we try to auto-migrate below
|
|
|
|
default:
|
|
return nil, errors.Wrap(err, "getting bot username from database")
|
|
}
|
|
|
|
// Bot username is not set, either we're running from fallback token
|
|
// or did not yet execute the v3.5.0 migration
|
|
|
|
var botAccessToken, botRefreshToken string
|
|
err = s.db.ReadEncryptedCoreMeta(coreMetaKeyBotToken, &botAccessToken)
|
|
switch {
|
|
case errors.Is(err, nil):
|
|
// This is fine, we do have a pre-v3.5.0 config, lets do the migration
|
|
|
|
case errors.Is(err, database.ErrCoreMetaNotFound):
|
|
// We're don't have a stored pre-v3.5.0 token either, so we're
|
|
// running from the fallback token (which might be empty)
|
|
return twitch.New(cfg.TwitchClient, cfg.TwitchClientSecret, cfg.FallbackToken, ""), nil
|
|
|
|
default:
|
|
return nil, errors.Wrap(err, "getting bot access token from database")
|
|
}
|
|
|
|
if err = s.db.ReadEncryptedCoreMeta(coreMetaKeyBotRefreshToken, &botRefreshToken); err != nil {
|
|
return nil, errors.Wrap(err, "getting bot refresh token from database")
|
|
}
|
|
|
|
// Now we do have (hopefully valid) tokens for the bot and therefore
|
|
// can determine who the bot is. That means we can set the username
|
|
// for later reference and afterwards delete the duplicated tokens.
|
|
|
|
_, botUser, err := twitch.New(cfg.TwitchClient, cfg.TwitchClientSecret, botAccessToken, botRefreshToken).GetAuthorizedUser(context.Background())
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "validating stored access token")
|
|
}
|
|
|
|
if err = s.db.StoreCoreMeta(coreMetaKeyBotUsername, botUser); err != nil {
|
|
return nil, errors.Wrap(err, "setting bot username")
|
|
}
|
|
|
|
if _, err = s.GetTwitchClientForChannel(botUser, cfg); errors.Is(err, gorm.ErrRecordNotFound) {
|
|
// There is no extended permission for that channel, we probably
|
|
// are in a state created by the v2 migrator. Lets just store the
|
|
// token without any permissions as we cannot know the permissions
|
|
// assigned to that token
|
|
if err = s.SetExtendedTwitchCredentials(botUser, botAccessToken, botRefreshToken, nil); err != nil {
|
|
return nil, errors.Wrap(err, "moving bot access token")
|
|
}
|
|
}
|
|
|
|
if err = s.db.DeleteCoreMeta(coreMetaKeyBotToken); err != nil {
|
|
return nil, errors.Wrap(err, "deleting deprecated bot token")
|
|
}
|
|
|
|
if err = s.db.DeleteCoreMeta(coreMetaKeyBotRefreshToken); err != nil {
|
|
return nil, errors.Wrap(err, "deleting deprecated bot refresh-token")
|
|
}
|
|
|
|
return s.GetTwitchClientForChannel(botUser, cfg)
|
|
}
|
|
|
|
// GetTwitchClientForChannel returns a twitch.Client configured to act
|
|
// as the owner of the given channel
|
|
func (s Service) GetTwitchClientForChannel(channel string, cfg ClientConfig) (*twitch.Client, error) {
|
|
var (
|
|
err error
|
|
perm extendedPermission
|
|
)
|
|
|
|
if err = helpers.Retry(func() error {
|
|
err = s.db.DB().First(&perm, "channel = ?", strings.TrimLeft(channel, "#")).Error
|
|
if errors.Is(err, gorm.ErrRecordNotFound) {
|
|
return backoff.NewErrCannotRetry(ErrChannelNotAuthorized) //nolint:wrapcheck // We get our own error
|
|
}
|
|
return errors.Wrap(err, "getting twitch credential from database")
|
|
}); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if perm.AccessToken, err = s.db.DecryptField(perm.AccessToken); err != nil {
|
|
return nil, errors.Wrap(err, "decrypting access token")
|
|
}
|
|
|
|
if perm.RefreshToken, err = s.db.DecryptField(perm.RefreshToken); err != nil {
|
|
return nil, errors.Wrap(err, "decrypting refresh token")
|
|
}
|
|
|
|
if perm.AccessToken == "" && perm.RefreshToken == "" {
|
|
// We have no tokens but an entry in the permission table: Means
|
|
// we still can't do stuff on behalf of that channel so we treat
|
|
// that as an unauthorized channel
|
|
return nil, ErrChannelNotAuthorized
|
|
}
|
|
|
|
scopes := strings.Split(perm.Scopes, " ")
|
|
|
|
tc := twitch.New(cfg.TwitchClient, cfg.TwitchClientSecret, perm.AccessToken, perm.RefreshToken)
|
|
tc.SetTokenUpdateHook(func(at, rt string) error {
|
|
return errors.Wrap(s.SetExtendedTwitchCredentials(channel, at, rt, scopes), "updating extended permissions token")
|
|
})
|
|
|
|
return tc, nil
|
|
}
|
|
|
|
// HasAnyPermissionForChannel checks whether any of the given scopes
|
|
// are granted for the given channel
|
|
func (s Service) HasAnyPermissionForChannel(channel string, scopes ...string) (bool, error) {
|
|
storedScopes, err := s.GetChannelPermissions(channel)
|
|
if err != nil {
|
|
return false, errors.Wrap(err, "getting channel scopes")
|
|
}
|
|
|
|
for _, scope := range scopes {
|
|
if str.StringInSlice(scope, storedScopes) {
|
|
return true, nil
|
|
}
|
|
}
|
|
|
|
return false, nil
|
|
}
|
|
|
|
// HasPermissionsForChannel checks whether all of the given scopes
|
|
// are granted for the given channel
|
|
func (s Service) HasPermissionsForChannel(channel string, scopes ...string) (bool, error) {
|
|
storedScopes, err := s.GetChannelPermissions(channel)
|
|
if err != nil {
|
|
return false, errors.Wrap(err, "getting channel scopes")
|
|
}
|
|
|
|
for _, scope := range scopes {
|
|
if !str.StringInSlice(scope, storedScopes) {
|
|
return false, nil
|
|
}
|
|
}
|
|
|
|
return true, nil
|
|
}
|
|
|
|
// HasTokensForChannel retrieves and decrypts stored access- and
|
|
// refresh-tokens to evaluate whether tokens are available. Those
|
|
// tokens are NOT validated in this request, it's just checked whether
|
|
// they are present
|
|
func (s Service) HasTokensForChannel(channel string) (bool, error) {
|
|
var (
|
|
err error
|
|
perm extendedPermission
|
|
)
|
|
|
|
if err = helpers.Retry(func() error {
|
|
err = s.db.DB().First(&perm, "channel = ?", strings.TrimLeft(channel, "#")).Error
|
|
if errors.Is(err, gorm.ErrRecordNotFound) {
|
|
return backoff.NewErrCannotRetry(ErrChannelNotAuthorized) //nolint:wrapcheck // We'll get our own error
|
|
}
|
|
return errors.Wrap(err, "getting twitch credential from database")
|
|
}); err != nil {
|
|
if errors.Is(err, ErrChannelNotAuthorized) {
|
|
return false, nil
|
|
}
|
|
return false, err
|
|
}
|
|
|
|
if perm.AccessToken, err = s.db.DecryptField(perm.AccessToken); err != nil {
|
|
return false, errors.Wrap(err, "decrypting access token")
|
|
}
|
|
|
|
if perm.RefreshToken, err = s.db.DecryptField(perm.RefreshToken); err != nil {
|
|
return false, errors.Wrap(err, "decrypting refresh token")
|
|
}
|
|
|
|
return perm.AccessToken != "" && perm.RefreshToken != "", nil
|
|
}
|
|
|
|
// ListPermittedChannels returns a list of all channels having a token
|
|
// for the channels owner
|
|
func (s Service) ListPermittedChannels() (out []string, err error) {
|
|
var perms []extendedPermission
|
|
if err = helpers.Retry(func() error {
|
|
return errors.Wrap(s.db.DB().Find(&perms).Error, "listing permissions")
|
|
}); err != nil {
|
|
return nil, err //nolint:wrapcheck // is already wrapped on the inside
|
|
}
|
|
|
|
for _, perm := range perms {
|
|
out = append(out, perm.Channel)
|
|
}
|
|
|
|
return out, nil
|
|
}
|
|
|
|
// RemoveAllExtendedTwitchCredentials wipes the access database
|
|
func (s Service) RemoveAllExtendedTwitchCredentials() error {
|
|
return errors.Wrap(
|
|
helpers.RetryTransaction(s.db.DB(), func(tx *gorm.DB) error {
|
|
return tx.Delete(&extendedPermission{}, "1 = 1").Error
|
|
}),
|
|
"deleting data from table",
|
|
)
|
|
}
|
|
|
|
// RemoveExendedTwitchCredentials wipes the access database for a given
|
|
// channel
|
|
func (s Service) RemoveExendedTwitchCredentials(channel string) error {
|
|
return errors.Wrap(
|
|
helpers.RetryTransaction(s.db.DB(), func(tx *gorm.DB) error {
|
|
return tx.Delete(&extendedPermission{}, "channel = ?", strings.TrimLeft(channel, "#")).Error
|
|
}),
|
|
"deleting data from table",
|
|
)
|
|
}
|
|
|
|
// SetBotUsername stores the username of the bot
|
|
func (s Service) SetBotUsername(channel string) (err error) {
|
|
return errors.Wrap(
|
|
s.db.StoreCoreMeta(coreMetaKeyBotUsername, strings.TrimLeft(channel, "#")),
|
|
"storing bot username",
|
|
)
|
|
}
|
|
|
|
// SetExtendedTwitchCredentials stores tokens and scopes for the given
|
|
// channel into the access database
|
|
func (s Service) SetExtendedTwitchCredentials(channel, accessToken, refreshToken string, scope []string) (err error) {
|
|
if accessToken, err = s.db.EncryptField(accessToken); err != nil {
|
|
return errors.Wrap(err, "encrypting access token")
|
|
}
|
|
|
|
if refreshToken, err = s.db.EncryptField(refreshToken); err != nil {
|
|
return errors.Wrap(err, "encrypting refresh token")
|
|
}
|
|
|
|
return errors.Wrap(
|
|
helpers.RetryTransaction(s.db.DB(), func(tx *gorm.DB) error {
|
|
return tx.Clauses(clause.OnConflict{
|
|
Columns: []clause.Column{{Name: "channel"}},
|
|
DoUpdates: clause.AssignmentColumns([]string{"access_token", "refresh_token", "scopes"}),
|
|
}).Create(extendedPermission{
|
|
Channel: strings.TrimLeft(channel, "#"),
|
|
AccessToken: accessToken,
|
|
RefreshToken: refreshToken,
|
|
Scopes: strings.Join(scope, " "),
|
|
}).Error
|
|
}),
|
|
"inserting data into table",
|
|
)
|
|
}
|