2017-08-03 14:13:53 +02:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"os"
|
2018-05-05 13:56:53 +02:00
|
|
|
"strconv"
|
|
|
|
"strings"
|
2017-08-03 14:13:53 +02:00
|
|
|
|
2020-01-24 16:15:17 +01:00
|
|
|
"github.com/gofrs/uuid/v3"
|
2018-05-05 13:56:53 +02:00
|
|
|
log "github.com/sirupsen/logrus"
|
2017-08-03 14:13:53 +02:00
|
|
|
"github.com/xuyu/goredis"
|
|
|
|
)
|
|
|
|
|
|
|
|
type storageRedis struct {
|
|
|
|
conn *goredis.Redis
|
|
|
|
}
|
|
|
|
|
|
|
|
func newStorageRedis() (storage, error) {
|
|
|
|
if os.Getenv("REDIS_URL") == "" {
|
|
|
|
return nil, fmt.Errorf("REDIS_URL environment variable not set")
|
|
|
|
}
|
|
|
|
|
|
|
|
c, err := goredis.DialURL(os.Getenv("REDIS_URL"))
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2018-05-05 13:56:53 +02:00
|
|
|
s := &storageRedis{
|
2017-08-03 14:13:53 +02:00
|
|
|
conn: c,
|
2018-05-05 13:56:53 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if err := s.migrate(); err != nil { // Move from the old to the new storage format
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return s, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s storageRedis) migrate() error {
|
|
|
|
t, err := s.conn.Type(s.redisKey())
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
log.Printf("Key %q type: %s", s.redisKey(), t)
|
|
|
|
|
|
|
|
if t == "hash" {
|
|
|
|
hashs, err := s.conn.HGetAll(s.redisKey())
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
for k, v := range hashs {
|
|
|
|
if err := s.writeKey(k, v); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if _, err = s.conn.Del(s.redisKey()); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s storageRedis) redisExpiry() int {
|
2020-01-24 16:33:14 +01:00
|
|
|
var expStr string
|
|
|
|
for _, eVar := range []string{"SECRET_EXPIRY", "REDIS_EXPIRY"} {
|
|
|
|
if v := os.Getenv(eVar); v != "" {
|
|
|
|
expStr = v
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if expStr == "" {
|
2018-05-05 13:56:53 +02:00
|
|
|
return 0
|
|
|
|
}
|
|
|
|
|
2020-01-24 16:33:14 +01:00
|
|
|
e, err := strconv.ParseInt(expStr, 10, 64)
|
2018-05-05 13:56:53 +02:00
|
|
|
if err != nil {
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
|
|
|
|
return int(e)
|
2017-08-03 14:13:53 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (s storageRedis) redisKey() string {
|
|
|
|
key := os.Getenv("REDIS_KEY")
|
|
|
|
if key == "" {
|
|
|
|
key = "io.luzifer.ots"
|
|
|
|
}
|
|
|
|
|
|
|
|
return key
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s storageRedis) Create(secret string) (string, error) {
|
2018-10-06 20:08:21 +02:00
|
|
|
id := uuid.Must(uuid.NewV4()).String()
|
2018-05-05 13:56:53 +02:00
|
|
|
err := s.writeKey(id, secret)
|
2017-08-03 14:13:53 +02:00
|
|
|
|
|
|
|
return id, err
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s storageRedis) ReadAndDestroy(id string) (string, error) {
|
2018-05-05 13:56:53 +02:00
|
|
|
secret, err := s.conn.Get(strings.Join([]string{s.redisKey(), id}, ":"))
|
2017-08-03 14:13:53 +02:00
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
2017-08-03 18:18:05 +02:00
|
|
|
if secret == nil {
|
|
|
|
return "", errSecretNotFound
|
|
|
|
}
|
|
|
|
|
2018-05-05 13:56:53 +02:00
|
|
|
_, err = s.conn.Del(strings.Join([]string{s.redisKey(), id}, ":"))
|
2017-08-03 14:13:53 +02:00
|
|
|
return string(secret), err
|
|
|
|
}
|
2018-05-05 13:56:53 +02:00
|
|
|
|
|
|
|
func (s storageRedis) writeKey(id, value string) error {
|
|
|
|
return s.conn.Set(
|
|
|
|
strings.Join([]string{s.redisKey(), id}, ":"), // Key
|
|
|
|
value, // Secret
|
|
|
|
s.redisExpiry(), // Expiry in seconds
|
|
|
|
0, // Expiry milliseconds
|
|
|
|
false, // MustExist
|
|
|
|
true, // MustNotExist
|
|
|
|
)
|
|
|
|
}
|