mirror of
https://github.com/unmojang/drasl.git
synced 2025-08-03 19:06:04 -04:00
451 lines
12 KiB
Go
451 lines
12 KiB
Go
package main
|
|
|
|
import (
|
|
"database/sql"
|
|
"errors"
|
|
"fmt"
|
|
mapset "github.com/deckarep/golang-set/v2"
|
|
"github.com/samber/mo"
|
|
"gorm.io/driver/sqlite"
|
|
"gorm.io/gorm"
|
|
"gorm.io/gorm/logger"
|
|
"log"
|
|
"os"
|
|
"path"
|
|
"path/filepath"
|
|
"time"
|
|
)
|
|
|
|
const CURRENT_USER_VERSION = 4
|
|
|
|
const PLAYER_NAME_TAKEN_BY_USERNAME_ERROR = "PLAYER_NAME_TAKEN_BY_USERNAME"
|
|
const USERNAME_TAKEN_BY_PLAYER_NAME_ERROR = "USERNAME_TAKEN_BY_PLAYER_NAME"
|
|
|
|
type Error error
|
|
|
|
func IsErrorUniqueFailed(err error) bool {
|
|
if err == nil {
|
|
return false
|
|
}
|
|
// Work around https://stackoverflow.com/questions/75489773/why-do-i-get-second-argument-to-errors-as-should-not-be-error-build-error-in
|
|
e := (errors.New("UNIQUE constraint failed")).(Error)
|
|
return errors.As(err, &e) || IsErrorPlayerNameTakenByUsername(err) || IsErrorUsernameTakenByPlayerName(err)
|
|
}
|
|
|
|
func IsErrorUniqueFailedField(err error, field string) bool {
|
|
if err == nil {
|
|
return false
|
|
}
|
|
|
|
// The Go programming language 😎
|
|
return err.Error() == "UNIQUE constraint failed: "+field
|
|
}
|
|
|
|
func IsErrorUsernameTakenByPlayerName(err error) bool {
|
|
return err.Error() == USERNAME_TAKEN_BY_PLAYER_NAME_ERROR
|
|
}
|
|
|
|
func IsErrorPlayerNameTakenByUsername(err error) bool {
|
|
return err.Error() == PLAYER_NAME_TAKEN_BY_USERNAME_ERROR
|
|
}
|
|
|
|
type BackwardsMigrationError struct {
|
|
Err error
|
|
}
|
|
|
|
func (e BackwardsMigrationError) Error() string {
|
|
return e.Err.Error()
|
|
}
|
|
|
|
type V1User struct {
|
|
IsAdmin bool
|
|
IsLocked bool
|
|
UUID string `gorm:"primaryKey"`
|
|
Username string `gorm:"unique;not null"`
|
|
PasswordSalt []byte `gorm:"not null"`
|
|
PasswordHash []byte `gorm:"not null"`
|
|
Clients []V1Client `gorm:"foreignKey:UserUUID"`
|
|
ServerID sql.NullString
|
|
PlayerName string `gorm:"unique;not null;type:text collate nocase"`
|
|
FallbackPlayer string
|
|
PreferredLanguage string
|
|
BrowserToken sql.NullString `gorm:"index"`
|
|
APIToken string
|
|
SkinHash sql.NullString `gorm:"index"`
|
|
SkinModel string
|
|
CapeHash sql.NullString `gorm:"index"`
|
|
CreatedAt time.Time
|
|
NameLastChangedAt time.Time
|
|
}
|
|
|
|
func (V1User) TableName() string {
|
|
return "users"
|
|
}
|
|
|
|
type V1Client struct {
|
|
ClientToken string `gorm:"primaryKey"`
|
|
Version int
|
|
UserUUID string
|
|
User V3User
|
|
}
|
|
|
|
func (V1Client) TableName() string {
|
|
return "clients"
|
|
}
|
|
|
|
type V2User = V1User
|
|
|
|
type V2Client struct {
|
|
UUID string `gorm:"primaryKey"`
|
|
ClientToken string
|
|
Version int
|
|
UserUUID string
|
|
User V2User
|
|
}
|
|
|
|
func (V2Client) TableName() string {
|
|
return "clients"
|
|
}
|
|
|
|
type V3User struct {
|
|
IsAdmin bool
|
|
IsLocked bool
|
|
UUID string `gorm:"primaryKey"`
|
|
Username string `gorm:"unique;not null"`
|
|
PasswordSalt []byte `gorm:"not null"`
|
|
PasswordHash []byte `gorm:"not null"`
|
|
Clients []V3Client `gorm:"foreignKey:UserUUID"`
|
|
ServerID sql.NullString
|
|
PlayerName string `gorm:"unique;not null;type:text collate nocase"`
|
|
OfflineUUID string `gorm:"not null"`
|
|
FallbackPlayer string
|
|
PreferredLanguage string
|
|
BrowserToken sql.NullString `gorm:"index"`
|
|
APIToken string
|
|
SkinHash sql.NullString `gorm:"index"`
|
|
SkinModel string
|
|
CapeHash sql.NullString `gorm:"index"`
|
|
CreatedAt time.Time
|
|
NameLastChangedAt time.Time
|
|
}
|
|
|
|
func (V3User) TableName() string {
|
|
return "users"
|
|
}
|
|
|
|
type V3Client = V2Client
|
|
|
|
type V4User = User
|
|
type V4Player = Player
|
|
type V4Client = Client
|
|
|
|
func OpenDB(config *Config) (*gorm.DB, error) {
|
|
dbPath := path.Join(config.StateDirectory, "drasl.db")
|
|
_, err := os.Stat(dbPath)
|
|
alreadyExisted := err == nil
|
|
|
|
db := Unwrap(gorm.Open(sqlite.Open(dbPath), &gorm.Config{
|
|
Logger: logger.Default.LogMode(logger.Silent),
|
|
}))
|
|
err = Migrate(config, mo.Some(dbPath), db, alreadyExisted, CURRENT_USER_VERSION)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("Error migrating database: %w", err)
|
|
}
|
|
|
|
return db, nil
|
|
}
|
|
|
|
func setUserVersion(tx *gorm.DB, userVersion uint) error {
|
|
// PRAGMA user_version = ? doesn't work here
|
|
return tx.Exec(fmt.Sprintf("PRAGMA user_version = %d", userVersion)).Error
|
|
}
|
|
|
|
func Migrate(config *Config, dbPath mo.Option[string], db *gorm.DB, alreadyExisted bool, targetUserVersion uint) error {
|
|
var userVersion uint
|
|
|
|
if alreadyExisted {
|
|
if err := db.Raw("PRAGMA user_version;").Scan(&userVersion).Error; err != nil {
|
|
return nil
|
|
}
|
|
} else {
|
|
userVersion = targetUserVersion
|
|
}
|
|
|
|
initialUserVersion := userVersion
|
|
if initialUserVersion > targetUserVersion {
|
|
return BackwardsMigrationError{
|
|
Err: fmt.Errorf("Database is version %d, migration target version is %d, cannot continue. Are you trying to run an older version of %s with a newer database?", userVersion, targetUserVersion, config.ApplicationName),
|
|
}
|
|
}
|
|
|
|
if initialUserVersion < targetUserVersion {
|
|
log.Printf("Started migration of database version %d to %d.", userVersion, targetUserVersion)
|
|
if !config.PreMigrationBackups {
|
|
log.Printf("PreMigrationBackups disabled, skipping backup.")
|
|
} else if p, ok := dbPath.Get(); ok {
|
|
dbDir := filepath.Dir(p)
|
|
datetime := time.Now().UTC().Format("2006-01-02T15-04-05Z")
|
|
backupPath := path.Join(dbDir, fmt.Sprintf("drasl.%d.%s.db", userVersion, datetime))
|
|
log.Printf("Backing up old database to %s", backupPath)
|
|
_, err := CopyPath(p, backupPath)
|
|
if err != nil {
|
|
return fmt.Errorf("Error backing up database: %w", err)
|
|
}
|
|
log.Printf("Database backed up, proceeding.")
|
|
} else {
|
|
log.Printf("Database path not specified, skipping backup.")
|
|
}
|
|
}
|
|
|
|
err := db.Transaction(func(tx *gorm.DB) error {
|
|
if userVersion == 0 && targetUserVersion >= 1 {
|
|
// Version 0 to 1
|
|
// Add User.OfflineUUID
|
|
if err := tx.AutoMigrate(&V1User{}); err != nil {
|
|
return err
|
|
}
|
|
var users []V1User
|
|
if err := tx.Find(&users).Error; err != nil {
|
|
return err
|
|
}
|
|
for _, user := range users {
|
|
offlineUUID, err := OfflineUUID(user.PlayerName)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if err := tx.Model(&user).Update("offline_uuid", offlineUUID).Error; err != nil {
|
|
return err
|
|
}
|
|
}
|
|
userVersion += 1
|
|
}
|
|
if userVersion == 1 && targetUserVersion >= 2 {
|
|
// Version 1 to 2
|
|
// Change Client primaryKey from ClientToken to UUID
|
|
if err := tx.Exec("ALTER TABLE clients RENAME client_token TO uuid").Error; err != nil {
|
|
return err
|
|
}
|
|
if err := tx.Migrator().AddColumn(&V2Client{}, "client_token"); err != nil {
|
|
return err
|
|
}
|
|
if err := tx.Exec("UPDATE clients SET client_token = uuid").Error; err != nil {
|
|
return err
|
|
}
|
|
userVersion += 1
|
|
}
|
|
if userVersion == 2 && targetUserVersion >= 3 {
|
|
// Version 2 to 3
|
|
// Add User.APIToken
|
|
|
|
if err := tx.Migrator().AddColumn(&V3User{}, "api_token"); err != nil {
|
|
return err
|
|
}
|
|
var users []V3User
|
|
if err := tx.Find(&users).Error; err != nil {
|
|
return err
|
|
}
|
|
for _, user := range users {
|
|
apiToken, err := MakeAPIToken()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if err := tx.Model(&user).Update("api_token", apiToken).Error; err != nil {
|
|
return err
|
|
}
|
|
}
|
|
userVersion += 1
|
|
}
|
|
if userVersion == 3 && targetUserVersion >= 4 {
|
|
// Version 3 to 4
|
|
// Split Users and Players
|
|
|
|
var v3Users []V3User
|
|
if err := tx.Preload("Clients").Find(&v3Users).Error; err != nil {
|
|
return err
|
|
}
|
|
|
|
if err := tx.AutoMigrate(&V4User{}); err != nil {
|
|
return err
|
|
}
|
|
if err := tx.AutoMigrate(&V4Player{}); err != nil {
|
|
return err
|
|
}
|
|
if err := tx.AutoMigrate(&V4Client{}); err != nil {
|
|
return err
|
|
}
|
|
|
|
// Drop player_name and offline_uuid, they have non-null
|
|
// constraints and SQLite has no mechanism to remove them
|
|
if err := tx.Migrator().DropColumn(&V4User{}, "player_name"); err != nil {
|
|
return err
|
|
}
|
|
if err := tx.Migrator().DropColumn(&V4User{}, "offline_uuid"); err != nil {
|
|
return err
|
|
}
|
|
|
|
allUsernames := mapset.NewSet[string]()
|
|
for _, v3User := range v3Users {
|
|
allUsernames.Add(v3User.Username)
|
|
}
|
|
|
|
users := make([]V4User, 0, len(v3Users))
|
|
for _, v3User := range v3Users {
|
|
clients := make([]V4Client, 0, len(v3User.Clients))
|
|
for _, v3Client := range v3User.Clients {
|
|
clients = append(clients, V4Client{
|
|
UUID: v3Client.UUID,
|
|
ClientToken: v3Client.ClientToken,
|
|
Version: v3Client.Version,
|
|
UserUUID: v3Client.UserUUID,
|
|
PlayerUUID: MakeNullString(&v3Client.UserUUID),
|
|
})
|
|
}
|
|
// If the player name is in use as someone else's username,
|
|
// reset the player name to its owner's username
|
|
playerName := v3User.PlayerName
|
|
if playerName != v3User.Username && allUsernames.Contains(playerName) {
|
|
playerName = v3User.Username
|
|
}
|
|
player := V4Player{
|
|
UUID: v3User.UUID,
|
|
Name: playerName,
|
|
OfflineUUID: v3User.OfflineUUID,
|
|
CreatedAt: v3User.CreatedAt,
|
|
NameLastChangedAt: v3User.NameLastChangedAt,
|
|
SkinHash: v3User.SkinHash,
|
|
CapeHash: v3User.CapeHash,
|
|
ServerID: v3User.ServerID,
|
|
FallbackPlayer: v3User.FallbackPlayer,
|
|
Clients: clients,
|
|
UserUUID: v3User.UUID,
|
|
}
|
|
user := V4User{
|
|
IsAdmin: v3User.IsAdmin,
|
|
IsLocked: v3User.IsLocked,
|
|
UUID: v3User.UUID,
|
|
Username: v3User.Username,
|
|
PasswordSalt: v3User.PasswordSalt,
|
|
PasswordHash: v3User.PasswordHash,
|
|
BrowserToken: v3User.BrowserToken,
|
|
APIToken: v3User.APIToken,
|
|
PreferredLanguage: v3User.PreferredLanguage,
|
|
Players: []Player{player},
|
|
MaxPlayerCount: Constants.MaxPlayerCountUseDefault,
|
|
}
|
|
user.Players = append(user.Players, player)
|
|
users = append(users, user)
|
|
}
|
|
if len(users) > 0 {
|
|
if err := tx.Session(&gorm.Session{FullSaveAssociations: true}).Save(&users).Error; err != nil {
|
|
return err
|
|
}
|
|
}
|
|
userVersion += 1
|
|
}
|
|
|
|
err := tx.AutoMigrate(&User{})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
err = tx.AutoMigrate(&Player{})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
err = tx.AutoMigrate(&Client{})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
err = tx.AutoMigrate(&Invite{})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
err = tx.Exec(fmt.Sprintf(`
|
|
DROP TRIGGER IF EXISTS v4_insert_unique_username;
|
|
CREATE TRIGGER v4_insert_unique_username
|
|
BEFORE INSERT ON users
|
|
FOR EACH ROW
|
|
BEGIN
|
|
-- We have to reimplement the regular "UNIQUE constraint
|
|
-- failed" errors here too since we want them to take priority
|
|
SELECT RAISE(ABORT, 'UNIQUE constraint failed: users.username')
|
|
WHERE EXISTS(
|
|
SELECT 1 FROM users WHERE username = NEW.username AND uuid != NEW.uuid
|
|
);
|
|
|
|
SELECT RAISE(ABORT, '%[1]s')
|
|
WHERE EXISTS(
|
|
SELECT 1 from players WHERE name == NEW.username AND user_uuid != NEW.uuid
|
|
);
|
|
END;
|
|
|
|
DROP TRIGGER IF EXISTS v4_update_unique_username;
|
|
CREATE TRIGGER v4_update_unique_username
|
|
BEFORE UPDATE ON users
|
|
FOR EACH ROW
|
|
BEGIN
|
|
SELECT RAISE(ABORT, 'UNIQUE constraint failed: users.username')
|
|
WHERE EXISTS(
|
|
SELECT 1 FROM users WHERE username = NEW.username AND uuid != NEW.uuid
|
|
);
|
|
|
|
SELECT RAISE(ABORT, '%[1]s')
|
|
WHERE EXISTS(
|
|
SELECT 1 from players WHERE name == NEW.username AND user_uuid != NEW.uuid
|
|
);
|
|
END;
|
|
|
|
DROP TRIGGER IF EXISTS v4_insert_unique_player_name;
|
|
CREATE TRIGGER v4_insert_unique_player_name
|
|
BEFORE INSERT ON players
|
|
BEGIN
|
|
SELECT RAISE(ABORT, 'UNIQUE constraint failed: players.name')
|
|
WHERE EXISTS(
|
|
SELECT 1 from players WHERE name == NEW.name AND uuid != NEW.uuid
|
|
);
|
|
|
|
SELECT RAISE(ABORT, '%[2]s')
|
|
WHERE EXISTS(
|
|
SELECT 1 from users WHERE username == NEW.name AND uuid != NEW.user_uuid
|
|
);
|
|
END;
|
|
|
|
DROP TRIGGER IF EXISTS v4_update_unique_player_name;
|
|
CREATE TRIGGER v4_update_unique_player_name
|
|
BEFORE UPDATE ON players
|
|
BEGIN
|
|
SELECT RAISE(ABORT, 'UNIQUE constraint failed: players.name')
|
|
WHERE EXISTS(
|
|
SELECT 1 from players WHERE name == NEW.name AND uuid != NEW.uuid
|
|
);
|
|
|
|
SELECT RAISE(ABORT, '%[2]s')
|
|
WHERE EXISTS(
|
|
SELECT 1 from users WHERE username == NEW.name AND uuid != NEW.user_uuid
|
|
);
|
|
END;
|
|
`, USERNAME_TAKEN_BY_PLAYER_NAME_ERROR, PLAYER_NAME_TAKEN_BY_USERNAME_ERROR)).Error
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if err := setUserVersion(tx, userVersion); err != nil {
|
|
return err
|
|
}
|
|
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if initialUserVersion < targetUserVersion {
|
|
log.Printf("Finished migration from version %d to %d", initialUserVersion, userVersion)
|
|
}
|
|
|
|
return nil
|
|
}
|