rogueserver/api/daily/common.go

177 lines
3.8 KiB
Go
Raw Normal View History

2024-04-29 17:26:46 -04:00
/*
Copyright (C) 2024 Pagefault Games
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU Affero General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU Affero General Public License for more details.
You should have received a copy of the GNU Affero General Public License
along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
2024-04-29 15:32:58 -04:00
package daily
2024-03-16 21:51:13 -04:00
import (
"bytes"
"context"
2024-03-18 19:55:02 -04:00
"crypto/md5"
"crypto/rand"
"encoding/base64"
"encoding/binary"
"encoding/json"
2024-03-17 13:18:51 -04:00
"fmt"
2024-03-16 21:51:13 -04:00
"log"
2024-03-18 19:55:02 -04:00
"os"
"time"
2024-03-16 21:51:13 -04:00
"github.com/aws/aws-sdk-go-v2/aws"
"github.com/aws/aws-sdk-go-v2/config"
"github.com/aws/aws-sdk-go-v2/service/s3"
2024-04-29 15:22:27 -04:00
"github.com/pagefaultgames/rogueserver/db"
2024-04-16 21:01:24 -04:00
"github.com/robfig/cron/v3"
2024-03-16 21:51:13 -04:00
)
2024-03-18 19:55:02 -04:00
const secondsPerDay = 60 * 60 * 24
2024-03-16 21:51:13 -04:00
var (
2024-10-22 16:23:54 -04:00
scheduler = cron.New(cron.WithLocation(time.UTC))
secret []byte
2024-03-16 21:51:13 -04:00
)
func Init() error {
2024-04-14 20:13:45 -04:00
var err error
secret, err = os.ReadFile("secret.key")
2024-03-16 21:51:13 -04:00
if err != nil {
2024-03-18 19:55:02 -04:00
if !os.IsNotExist(err) {
2024-04-08 18:15:09 -04:00
return fmt.Errorf("failed to read daily seed secret: %s", err)
2024-03-18 19:55:02 -04:00
}
2024-03-18 19:55:02 -04:00
newSecret := make([]byte, 32)
_, err := rand.Read(newSecret)
if err != nil {
2024-04-08 18:15:09 -04:00
return fmt.Errorf("failed to generate daily seed secret: %s", err)
}
2024-03-18 19:55:02 -04:00
err = os.WriteFile("secret.key", newSecret, 0400)
if err != nil {
2024-04-08 18:15:09 -04:00
return fmt.Errorf("failed to write daily seed secret: %s", err)
2024-03-18 19:55:02 -04:00
}
secret = newSecret
}
2024-06-07 21:10:57 -04:00
seed, err := db.TryAddDailyRun(Seed())
2024-03-18 19:55:02 -04:00
if err != nil {
2024-04-01 22:54:55 -04:00
log.Print(err)
2024-03-16 21:51:13 -04:00
}
2024-04-01 22:54:55 -04:00
log.Printf("Daily Run Seed: %s", seed)
_, err = scheduler.AddFunc("@daily", func() {
time.Sleep(time.Second)
2024-06-07 21:10:57 -04:00
seed, err = db.TryAddDailyRun(Seed())
2024-04-16 21:01:24 -04:00
if err != nil {
log.Printf("error while recording new daily: %s", err)
} else {
log.Printf("Daily Run Seed: %s", seed)
2024-04-16 21:01:24 -04:00
}
})
if err != nil {
return err
}
2024-04-16 21:01:24 -04:00
scheduler.Start()
2024-04-08 18:15:09 -04:00
2024-10-22 16:12:16 -04:00
if os.Getenv("AWS_ENDPOINT_URL_S3") != "" {
2024-10-22 16:23:54 -04:00
go func() {
for {
err = S3SaveMigration()
if err != nil {
return
}
}
}()
}
2024-04-08 18:15:09 -04:00
return nil
2024-03-16 21:51:13 -04:00
}
func Seed() string {
2024-04-14 20:13:45 -04:00
return base64.StdEncoding.EncodeToString(deriveSeed(time.Now().UTC()))
}
2024-04-14 20:13:45 -04:00
func deriveSeed(seedTime time.Time) []byte {
2024-03-18 19:55:02 -04:00
day := make([]byte, 8)
binary.BigEndian.PutUint64(day, uint64(seedTime.Unix()/secondsPerDay))
2024-04-14 20:13:45 -04:00
hashedSeed := md5.Sum(append(day, secret...))
2024-03-18 19:55:02 -04:00
return hashedSeed[:]
}
2024-10-22 16:23:54 -04:00
func S3SaveMigration() error {
cfg, _ := config.LoadDefaultConfig(context.TODO())
svc := s3.NewFromConfig(cfg, func(o *s3.Options) {
o.BaseEndpoint = aws.String(os.Getenv("AWS_ENDPOINT_URL_S3"))
})
2024-10-22 15:50:01 -04:00
_, err := svc.CreateBucket(context.Background(), &s3.CreateBucketInput{
Bucket: aws.String("pokerogue-system"),
})
if err != nil {
2024-10-22 16:23:54 -04:00
log.Printf("error while creating bucket (already exists?): %s", err)
}
2024-10-23 09:23:24 -04:00
// retrieve accounts from db
2024-10-22 15:50:01 -04:00
accounts, err := db.RetrieveOldAccounts()
if err != nil {
2024-10-22 16:23:54 -04:00
return fmt.Errorf("failed to retrieve old accounts: %s", err)
2024-10-22 15:50:01 -04:00
}
for _, user := range accounts {
2024-10-22 15:50:01 -04:00
data, err := db.ReadSystemSaveData(user)
if err != nil {
continue
}
username, err := db.FetchUsernameFromUUID(user)
if err != nil {
continue
}
json, err := json.Marshal(data)
if err != nil {
continue
}
_, err = svc.PutObject(context.Background(), &s3.PutObjectInput{
Bucket: aws.String("pokerogue-system"),
Key: aws.String(username),
Body: bytes.NewReader(json),
})
if err != nil {
2024-10-22 16:12:16 -04:00
log.Printf("error while saving data in S3 for user %s: %s", username, err)
continue
}
err = db.UpdateSystemSaveLocation(user)
2024-10-22 15:50:01 -04:00
if err != nil {
2024-10-22 16:23:54 -04:00
log.Printf("failed to update location for user %s: %s", username, err)
2024-10-22 15:50:01 -04:00
continue
}
2024-10-22 16:23:54 -04:00
log.Printf("saved data in S3 for user %s", username)
}
2024-10-22 16:23:54 -04:00
return nil
}