2017-08-14 21:06:22 +03:00
|
|
|
|
package db
|
2017-06-30 18:40:25 +03:00
|
|
|
|
|
|
|
|
|
import (
|
2017-08-10 03:14:28 +03:00
|
|
|
|
"bytes"
|
|
|
|
|
"compress/zlib"
|
2017-06-30 18:40:25 +03:00
|
|
|
|
"encoding/json"
|
2017-08-14 21:06:22 +03:00
|
|
|
|
"fmt"
|
2017-08-10 03:14:28 +03:00
|
|
|
|
"io"
|
2017-06-30 18:40:25 +03:00
|
|
|
|
"log"
|
2017-08-10 03:14:28 +03:00
|
|
|
|
"strings"
|
2017-08-15 01:03:02 +03:00
|
|
|
|
"time"
|
2017-06-30 18:40:25 +03:00
|
|
|
|
|
2017-08-10 03:14:28 +03:00
|
|
|
|
"github.com/mediocregopher/radix.v2/pool"
|
2017-06-30 18:40:25 +03:00
|
|
|
|
"github.com/mediocregopher/radix.v2/redis"
|
|
|
|
|
"github.com/mediocregopher/radix.v2/util"
|
2017-08-10 03:14:28 +03:00
|
|
|
|
|
|
|
|
|
"elyby/minecraft-skinsystem/model"
|
2017-08-18 00:50:23 +03:00
|
|
|
|
"elyby/minecraft-skinsystem/interfaces"
|
2017-06-30 18:40:25 +03:00
|
|
|
|
)
|
|
|
|
|
|
2017-08-14 21:06:22 +03:00
|
|
|
|
type RedisFactory struct {
|
|
|
|
|
Host string
|
|
|
|
|
Port int
|
|
|
|
|
PoolSize int
|
|
|
|
|
connection util.Cmder
|
2017-08-10 03:14:28 +03:00
|
|
|
|
}
|
|
|
|
|
|
2017-08-18 00:50:23 +03:00
|
|
|
|
func (f RedisFactory) CreateSkinsRepository() (interfaces.SkinsRepository, error) {
|
2017-08-14 21:06:22 +03:00
|
|
|
|
connection, err := f.getConnection()
|
2017-08-10 03:14:28 +03:00
|
|
|
|
if err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
|
2017-08-14 21:06:22 +03:00
|
|
|
|
return &redisDb{connection}, nil
|
|
|
|
|
}
|
|
|
|
|
|
2017-08-18 00:50:23 +03:00
|
|
|
|
func (f RedisFactory) CreateCapesRepository() (interfaces.CapesRepository, error) {
|
2017-08-14 21:06:22 +03:00
|
|
|
|
panic("capes repository not supported for this storage type")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (f RedisFactory) getConnection() (util.Cmder, error) {
|
|
|
|
|
if f.connection == nil {
|
|
|
|
|
if f.Host == "" {
|
|
|
|
|
return nil, &ParamRequired{"host"}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if f.Port == 0 {
|
|
|
|
|
return nil, &ParamRequired{"port"}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
addr := fmt.Sprintf("%s:%d", f.Host, f.Port)
|
2017-08-15 01:03:02 +03:00
|
|
|
|
conn, err := createConnection(addr, f.PoolSize)
|
2017-08-14 21:06:22 +03:00
|
|
|
|
if err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
f.connection = conn
|
2017-08-15 01:03:02 +03:00
|
|
|
|
|
|
|
|
|
go func() {
|
|
|
|
|
period := 5
|
|
|
|
|
for {
|
|
|
|
|
time.Sleep(time.Duration(period) * time.Second)
|
|
|
|
|
resp := f.connection.Cmd("PING")
|
|
|
|
|
if resp.Err == nil {
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
log.Println("Redis not pinged. Try to reconnect")
|
|
|
|
|
conn, err := createConnection(addr, f.PoolSize)
|
|
|
|
|
if err != nil {
|
|
|
|
|
log.Printf("Cannot reconnect to redis: %v\n", err)
|
|
|
|
|
log.Printf("Waiting %d seconds to retry\n", period)
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
f.connection = conn
|
|
|
|
|
log.Println("Reconnected")
|
|
|
|
|
}
|
|
|
|
|
}()
|
2017-08-14 21:06:22 +03:00
|
|
|
|
}
|
2017-08-10 03:14:28 +03:00
|
|
|
|
|
2017-08-14 21:06:22 +03:00
|
|
|
|
return f.connection, nil
|
2017-08-10 03:14:28 +03:00
|
|
|
|
}
|
|
|
|
|
|
2017-08-15 01:03:02 +03:00
|
|
|
|
func createConnection(addr string, poolSize int) (util.Cmder, error) {
|
|
|
|
|
if poolSize > 1 {
|
|
|
|
|
return pool.New("tcp", addr, poolSize)
|
|
|
|
|
} else {
|
|
|
|
|
return redis.Dial("tcp", addr)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2017-06-30 18:40:25 +03:00
|
|
|
|
type redisDb struct {
|
|
|
|
|
conn util.Cmder
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const accountIdToUsernameKey string = "hash:username-to-account-id"
|
|
|
|
|
|
2017-08-17 02:47:35 +03:00
|
|
|
|
func (db *redisDb) FindByUsername(username string) (*model.Skin, error) {
|
2017-08-09 19:11:53 +03:00
|
|
|
|
if username == "" {
|
2017-08-17 02:47:35 +03:00
|
|
|
|
return nil, &SkinNotFoundError{username}
|
2017-08-09 19:11:53 +03:00
|
|
|
|
}
|
|
|
|
|
|
2017-06-30 18:40:25 +03:00
|
|
|
|
redisKey := buildKey(username)
|
|
|
|
|
response := db.conn.Cmd("GET", redisKey)
|
|
|
|
|
if response.IsType(redis.Nil) {
|
2017-08-17 02:47:35 +03:00
|
|
|
|
return nil, &SkinNotFoundError{username}
|
2017-06-30 18:40:25 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
encodedResult, err := response.Bytes()
|
2017-08-17 02:47:35 +03:00
|
|
|
|
if err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
2017-06-30 18:40:25 +03:00
|
|
|
|
|
2017-08-17 02:47:35 +03:00
|
|
|
|
result, err := zlibDecode(encodedResult)
|
|
|
|
|
if err != nil {
|
|
|
|
|
log.Println("Cannot uncompress zlib for key " + redisKey) // TODO: replace with valid error
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
2017-06-30 18:40:25 +03:00
|
|
|
|
|
2017-08-17 02:47:35 +03:00
|
|
|
|
var skin *model.Skin
|
|
|
|
|
err = json.Unmarshal(result, &skin)
|
|
|
|
|
if err != nil {
|
|
|
|
|
log.Println("Cannot decode record data for key" + redisKey) // TODO: replace with valid error
|
|
|
|
|
return nil, nil
|
2017-06-30 18:40:25 +03:00
|
|
|
|
}
|
|
|
|
|
|
2017-08-17 02:47:35 +03:00
|
|
|
|
skin.OldUsername = skin.Username
|
|
|
|
|
|
|
|
|
|
return skin, nil
|
2017-06-30 18:40:25 +03:00
|
|
|
|
}
|
|
|
|
|
|
2017-08-17 02:47:35 +03:00
|
|
|
|
func (db *redisDb) FindByUserId(id int) (*model.Skin, error) {
|
2017-06-30 18:40:25 +03:00
|
|
|
|
response := db.conn.Cmd("HGET", accountIdToUsernameKey, id)
|
|
|
|
|
if response.IsType(redis.Nil) {
|
2017-08-17 02:47:35 +03:00
|
|
|
|
return nil, SkinNotFoundError{"unknown"}
|
2017-06-30 18:40:25 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
username, _ := response.Str()
|
|
|
|
|
|
|
|
|
|
return db.FindByUsername(username)
|
|
|
|
|
}
|
2017-08-10 03:14:28 +03:00
|
|
|
|
|
2017-08-17 02:47:35 +03:00
|
|
|
|
func (db *redisDb) Save(skin *model.Skin) error {
|
|
|
|
|
conn := db.conn
|
|
|
|
|
if poolConn, isPool := conn.(*pool.Pool); isPool {
|
|
|
|
|
conn, _ = poolConn.Get()
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
conn.Cmd("MULTI")
|
|
|
|
|
|
|
|
|
|
// Если пользователь сменил ник, то мы должны удать его ключ
|
|
|
|
|
if skin.OldUsername != "" && skin.OldUsername != skin.Username {
|
|
|
|
|
conn.Cmd("DEL", buildKey(skin.OldUsername))
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Если это новая запись или если пользователь сменил ник, то обновляем значение в хэш-таблице
|
|
|
|
|
if skin.OldUsername != "" || skin.OldUsername != skin.Username {
|
|
|
|
|
conn.Cmd("HSET", accountIdToUsernameKey, skin.UserId, skin.Username)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
str, _ := json.Marshal(skin)
|
|
|
|
|
conn.Cmd("SET", buildKey(skin.Username), zlibEncode(str))
|
|
|
|
|
|
|
|
|
|
conn.Cmd("EXEC")
|
|
|
|
|
|
|
|
|
|
skin.OldUsername = skin.Username
|
|
|
|
|
|
|
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
|
2017-08-10 03:14:28 +03:00
|
|
|
|
func buildKey(username string) string {
|
|
|
|
|
return "username:" + strings.ToLower(username)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//noinspection GoUnusedFunction
|
|
|
|
|
func zlibEncode(str []byte) []byte {
|
|
|
|
|
var buff bytes.Buffer
|
|
|
|
|
writer := zlib.NewWriter(&buff)
|
|
|
|
|
writer.Write(str)
|
|
|
|
|
writer.Close()
|
|
|
|
|
|
|
|
|
|
return buff.Bytes()
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func zlibDecode(bts []byte) ([]byte, error) {
|
|
|
|
|
buff := bytes.NewReader(bts)
|
|
|
|
|
reader, readError := zlib.NewReader(buff)
|
|
|
|
|
if readError != nil {
|
|
|
|
|
return nil, readError
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
resultBuffer := new(bytes.Buffer)
|
|
|
|
|
io.Copy(resultBuffer, reader)
|
|
|
|
|
reader.Close()
|
|
|
|
|
|
|
|
|
|
return resultBuffer.Bytes(), nil
|
|
|
|
|
}
|