2020-04-20 22:18:27 +03:00
|
|
|
package redis
|
2017-06-30 18:40:25 +03:00
|
|
|
|
|
|
|
import (
|
2023-12-14 02:15:59 +01:00
|
|
|
"context"
|
2021-03-03 01:32:38 +01:00
|
|
|
"fmt"
|
2017-08-10 03:14:28 +03:00
|
|
|
"strings"
|
2017-06-30 18:40:25 +03:00
|
|
|
|
2023-12-14 02:15:59 +01:00
|
|
|
"github.com/mediocregopher/radix/v4"
|
2017-08-10 03:14:28 +03:00
|
|
|
|
2024-02-01 08:12:34 +01:00
|
|
|
"ely.by/chrly/internal/db"
|
2017-06-30 18:40:25 +03:00
|
|
|
)
|
|
|
|
|
2024-01-30 09:05:04 +01:00
|
|
|
const usernameToProfileKey = "hash:username-to-profile"
|
|
|
|
const userUuidToUsernameKey = "hash:uuid-to-username"
|
2020-04-21 02:20:45 +03:00
|
|
|
|
2024-01-30 09:05:04 +01:00
|
|
|
type Redis struct {
|
|
|
|
client radix.Client
|
|
|
|
serializer db.ProfileSerializer
|
|
|
|
}
|
|
|
|
|
|
|
|
func New(ctx context.Context, profileSerializer db.ProfileSerializer, addr string, poolSize int) (*Redis, error) {
|
2023-12-14 02:15:59 +01:00
|
|
|
client, err := (radix.PoolConfig{Size: poolSize}).New(ctx, "tcp", addr)
|
2017-08-10 03:14:28 +03:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2020-04-20 22:18:27 +03:00
|
|
|
return &Redis{
|
2024-01-30 09:05:04 +01:00
|
|
|
client: client,
|
|
|
|
serializer: profileSerializer,
|
2020-04-20 22:18:27 +03:00
|
|
|
}, nil
|
2017-08-14 21:06:22 +03:00
|
|
|
}
|
|
|
|
|
2024-02-13 02:08:42 +01:00
|
|
|
func (r *Redis) FindProfileByUsername(ctx context.Context, username string) (*db.Profile, error) {
|
2024-01-30 09:05:04 +01:00
|
|
|
var profile *db.Profile
|
2024-02-13 02:08:42 +01:00
|
|
|
err := r.client.Do(ctx, radix.WithConn("", func(ctx context.Context, conn radix.Conn) error {
|
2023-12-14 02:15:59 +01:00
|
|
|
var err error
|
2024-01-30 09:05:04 +01:00
|
|
|
profile, err = r.findProfileByUsername(ctx, conn, username)
|
2023-12-14 02:15:59 +01:00
|
|
|
|
|
|
|
return err
|
|
|
|
}))
|
2019-05-01 02:10:11 +03:00
|
|
|
|
2024-01-30 09:05:04 +01:00
|
|
|
return profile, err
|
2018-01-23 18:43:37 +03:00
|
|
|
}
|
|
|
|
|
2024-01-30 09:05:04 +01:00
|
|
|
func (r *Redis) findProfileByUsername(ctx context.Context, conn radix.Conn, username string) (*db.Profile, error) {
|
2023-12-14 02:15:59 +01:00
|
|
|
var encodedResult []byte
|
2024-01-30 09:05:04 +01:00
|
|
|
err := conn.Do(ctx, radix.Cmd(&encodedResult, "HGET", usernameToProfileKey, usernameHashKey(username)))
|
2023-12-14 02:15:59 +01:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(encodedResult) == 0 {
|
2020-04-20 22:18:27 +03:00
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
2024-01-30 09:05:04 +01:00
|
|
|
return r.serializer.Deserialize(encodedResult)
|
2018-01-23 18:43:37 +03:00
|
|
|
}
|
|
|
|
|
2024-01-30 09:05:04 +01:00
|
|
|
func (r *Redis) findUsernameHashKeyByUuid(ctx context.Context, conn radix.Conn, uuid string) (string, error) {
|
|
|
|
var username string
|
|
|
|
return username, conn.Do(ctx, radix.FlatCmd(&username, "HGET", userUuidToUsernameKey, normalizeUuid(uuid)))
|
|
|
|
}
|
|
|
|
|
2024-02-13 02:08:42 +01:00
|
|
|
func (r *Redis) SaveProfile(ctx context.Context, profile *db.Profile) error {
|
|
|
|
return r.client.Do(ctx, radix.WithConn("", func(ctx context.Context, conn radix.Conn) error {
|
2024-01-30 09:05:04 +01:00
|
|
|
return r.saveProfile(ctx, conn, profile)
|
2023-12-14 02:15:59 +01:00
|
|
|
}))
|
|
|
|
}
|
|
|
|
|
2024-01-30 09:05:04 +01:00
|
|
|
func (r *Redis) saveProfile(ctx context.Context, conn radix.Conn, profile *db.Profile) error {
|
|
|
|
newUsernameHashKey := usernameHashKey(profile.Username)
|
|
|
|
existsUsernameHashKey, err := r.findUsernameHashKeyByUuid(ctx, conn, profile.Uuid)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
err = conn.Do(ctx, radix.Cmd(nil, "MULTI"))
|
2019-05-06 17:12:37 +03:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2017-06-30 18:40:25 +03:00
|
|
|
|
2020-04-20 22:18:27 +03:00
|
|
|
// If user has changed username, then we must delete his old username record
|
2024-01-30 09:05:04 +01:00
|
|
|
if existsUsernameHashKey != "" && existsUsernameHashKey != newUsernameHashKey {
|
|
|
|
err = conn.Do(ctx, radix.Cmd(nil, "HDEL", usernameToProfileKey, existsUsernameHashKey))
|
2023-12-14 02:15:59 +01:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2017-08-17 02:47:35 +03:00
|
|
|
}
|
2017-06-30 18:40:25 +03:00
|
|
|
|
2024-01-30 09:05:04 +01:00
|
|
|
err = conn.Do(ctx, radix.FlatCmd(nil, "HSET", userUuidToUsernameKey, normalizeUuid(profile.Uuid), newUsernameHashKey))
|
2023-12-14 02:15:59 +01:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2020-04-20 22:18:27 +03:00
|
|
|
|
2024-01-30 09:05:04 +01:00
|
|
|
serializedProfile, err := r.serializer.Serialize(profile)
|
2023-12-14 02:15:59 +01:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2017-06-30 18:40:25 +03:00
|
|
|
|
2024-01-30 09:05:04 +01:00
|
|
|
err = conn.Do(ctx, radix.FlatCmd(nil, "HSET", usernameToProfileKey, newUsernameHashKey, serializedProfile))
|
2020-04-20 15:16:15 +03:00
|
|
|
if err != nil {
|
2020-04-20 22:18:27 +03:00
|
|
|
return err
|
2020-04-20 15:16:15 +03:00
|
|
|
}
|
2017-06-30 18:40:25 +03:00
|
|
|
|
2024-01-30 09:05:04 +01:00
|
|
|
err = conn.Do(ctx, radix.Cmd(nil, "EXEC"))
|
2023-12-14 02:15:59 +01:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2017-08-17 02:47:35 +03:00
|
|
|
|
2024-01-30 09:05:04 +01:00
|
|
|
return nil
|
2018-01-23 18:43:37 +03:00
|
|
|
}
|
|
|
|
|
2024-02-13 02:08:42 +01:00
|
|
|
func (r *Redis) RemoveProfileByUuid(ctx context.Context, uuid string) error {
|
|
|
|
return r.client.Do(ctx, radix.WithConn("", func(ctx context.Context, conn radix.Conn) error {
|
2024-01-30 09:05:04 +01:00
|
|
|
return r.removeProfileByUuid(ctx, conn, uuid)
|
2023-12-14 02:15:59 +01:00
|
|
|
}))
|
2020-04-20 22:18:27 +03:00
|
|
|
}
|
|
|
|
|
2024-01-30 09:05:04 +01:00
|
|
|
func (r *Redis) removeProfileByUuid(ctx context.Context, conn radix.Conn, uuid string) error {
|
|
|
|
username, err := r.findUsernameHashKeyByUuid(ctx, conn, uuid)
|
2018-01-23 18:43:37 +03:00
|
|
|
if err != nil {
|
2019-05-06 17:12:37 +03:00
|
|
|
return err
|
2018-01-23 18:43:37 +03:00
|
|
|
}
|
|
|
|
|
2023-12-14 02:15:59 +01:00
|
|
|
err = conn.Do(ctx, radix.Cmd(nil, "MULTI"))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2018-01-23 18:43:37 +03:00
|
|
|
|
2024-01-30 09:05:04 +01:00
|
|
|
err = conn.Do(ctx, radix.FlatCmd(nil, "HDEL", userUuidToUsernameKey, normalizeUuid(uuid)))
|
2023-12-14 02:15:59 +01:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2018-01-23 18:43:37 +03:00
|
|
|
|
2024-01-30 09:05:04 +01:00
|
|
|
if username != "" {
|
|
|
|
err = conn.Do(ctx, radix.Cmd(nil, "HDEL", usernameToProfileKey, usernameHashKey(username)))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2023-12-14 02:15:59 +01:00
|
|
|
}
|
2018-01-23 18:43:37 +03:00
|
|
|
|
2023-12-14 02:15:59 +01:00
|
|
|
return conn.Do(ctx, radix.Cmd(nil, "EXEC"))
|
2018-01-23 18:43:37 +03:00
|
|
|
}
|
|
|
|
|
2024-02-13 02:08:42 +01:00
|
|
|
func (r *Redis) GetUuidForMojangUsername(ctx context.Context, username string) (string, string, error) {
|
2023-12-14 02:15:59 +01:00
|
|
|
var uuid string
|
2024-01-10 01:42:10 +01:00
|
|
|
foundUsername := username
|
2024-02-13 02:08:42 +01:00
|
|
|
err := r.client.Do(ctx, radix.WithConn("", func(ctx context.Context, conn radix.Conn) error {
|
2023-12-14 02:15:59 +01:00
|
|
|
var err error
|
2024-01-10 01:42:10 +01:00
|
|
|
uuid, foundUsername, err = findMojangUuidByUsername(ctx, conn, username)
|
2017-08-17 02:47:35 +03:00
|
|
|
|
2023-12-14 02:15:59 +01:00
|
|
|
return err
|
|
|
|
}))
|
|
|
|
|
2024-01-10 01:42:10 +01:00
|
|
|
return uuid, foundUsername, err
|
2017-08-17 02:47:35 +03:00
|
|
|
}
|
|
|
|
|
2024-01-10 01:42:10 +01:00
|
|
|
func findMojangUuidByUsername(ctx context.Context, conn radix.Conn, username string) (string, string, error) {
|
2024-01-30 09:05:04 +01:00
|
|
|
key := buildMojangUsernameKey(username)
|
2023-12-14 02:15:59 +01:00
|
|
|
var result string
|
2024-01-30 09:05:04 +01:00
|
|
|
err := conn.Do(ctx, radix.Cmd(&result, "GET", key))
|
2023-12-14 02:15:59 +01:00
|
|
|
if err != nil {
|
2024-01-10 01:42:10 +01:00
|
|
|
return "", "", err
|
2023-12-14 02:15:59 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if result == "" {
|
2024-01-10 01:42:10 +01:00
|
|
|
return "", "", nil
|
2019-04-25 02:23:10 +03:00
|
|
|
}
|
|
|
|
|
2023-12-14 02:15:59 +01:00
|
|
|
parts := strings.Split(result, ":")
|
|
|
|
|
2024-01-30 09:05:04 +01:00
|
|
|
return parts[1], parts[0], nil
|
2019-04-25 02:23:10 +03:00
|
|
|
}
|
|
|
|
|
2024-02-13 02:08:42 +01:00
|
|
|
func (r *Redis) StoreMojangUuid(ctx context.Context, username string, uuid string) error {
|
|
|
|
return r.client.Do(ctx, radix.WithConn("", func(ctx context.Context, conn radix.Conn) error {
|
2023-12-14 02:15:59 +01:00
|
|
|
return storeMojangUuid(ctx, conn, username, uuid)
|
|
|
|
}))
|
2020-04-20 22:18:27 +03:00
|
|
|
}
|
|
|
|
|
2023-12-14 02:15:59 +01:00
|
|
|
func storeMojangUuid(ctx context.Context, conn radix.Conn, username string, uuid string) error {
|
2024-01-30 09:05:04 +01:00
|
|
|
value := fmt.Sprintf("%s:%s", username, uuid)
|
|
|
|
err := conn.Do(ctx, radix.FlatCmd(nil, "SET", buildMojangUsernameKey(username), value, "EX", 60*60*24*30))
|
2023-12-14 02:15:59 +01:00
|
|
|
if err != nil {
|
|
|
|
return err
|
2019-05-06 17:12:37 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
2019-04-25 02:23:10 +03:00
|
|
|
}
|
|
|
|
|
2024-02-07 17:34:57 +01:00
|
|
|
func (r *Redis) Ping(ctx context.Context) error {
|
|
|
|
return r.client.Do(ctx, radix.Cmd(nil, "PING"))
|
2020-05-01 02:46:12 +03:00
|
|
|
}
|
|
|
|
|
2024-01-30 09:05:04 +01:00
|
|
|
func normalizeUuid(uuid string) string {
|
|
|
|
return strings.ToLower(strings.ReplaceAll(uuid, "-", ""))
|
2017-08-10 03:14:28 +03:00
|
|
|
}
|
|
|
|
|
2024-01-30 09:05:04 +01:00
|
|
|
func usernameHashKey(username string) string {
|
|
|
|
return strings.ToLower(username)
|
2017-08-10 03:14:28 +03:00
|
|
|
}
|
|
|
|
|
2024-01-30 09:05:04 +01:00
|
|
|
func buildMojangUsernameKey(username string) string {
|
|
|
|
return fmt.Sprintf("mojang:uuid:%s", usernameHashKey(username))
|
2017-08-10 03:14:28 +03:00
|
|
|
}
|