mirror of
https://github.com/MHSanaei/3x-ui.git
synced 2025-03-01 01:20:49 +03:00
102 lines
1.7 KiB
Go
102 lines
1.7 KiB
Go
package global
|
|
|
|
import (
|
|
"crypto/md5"
|
|
"encoding/hex"
|
|
"regexp"
|
|
"sync"
|
|
"time"
|
|
"x-ui/util/common"
|
|
)
|
|
|
|
type HashEntry struct {
|
|
Hash string
|
|
Value string
|
|
Timestamp time.Time
|
|
}
|
|
|
|
type HashStorage struct {
|
|
sync.RWMutex
|
|
Data map[string]HashEntry
|
|
Expiration time.Duration
|
|
ForceSave bool
|
|
}
|
|
|
|
func NewHashStorage(expiration time.Duration, forceSave bool) *HashStorage {
|
|
return &HashStorage{
|
|
Data: make(map[string]HashEntry),
|
|
Expiration: expiration,
|
|
ForceSave: forceSave,
|
|
}
|
|
}
|
|
|
|
func (h *HashStorage) AddHash(query string) string {
|
|
if h.ForceSave {
|
|
return h.saveValue(query)
|
|
}
|
|
|
|
// we only need to hash for more than 64 chars by default
|
|
if len(query) <= 64 {
|
|
return query
|
|
}
|
|
|
|
return h.saveValue(query)
|
|
}
|
|
|
|
func (h *HashStorage) saveValue(query string) string {
|
|
h.Lock()
|
|
defer h.Unlock()
|
|
|
|
md5Hash := md5.Sum([]byte(query))
|
|
md5HashString := hex.EncodeToString(md5Hash[:])
|
|
|
|
entry := HashEntry{
|
|
Hash: md5HashString,
|
|
Value: query,
|
|
Timestamp: time.Now(),
|
|
}
|
|
|
|
h.Data[md5HashString] = entry
|
|
|
|
return md5HashString
|
|
}
|
|
|
|
func (h *HashStorage) GetValue(hash string) (string, error) {
|
|
h.RLock()
|
|
defer h.RUnlock()
|
|
|
|
entry, exists := h.Data[hash]
|
|
if !exists {
|
|
if h.isMD5(hash) {
|
|
return "", common.NewError("hash not found in storage!")
|
|
}
|
|
return hash, nil
|
|
}
|
|
return entry.Value, nil
|
|
}
|
|
|
|
func (h *HashStorage) isMD5(hash string) bool {
|
|
match, _ := regexp.MatchString("^[a-f0-9]{32}$", hash)
|
|
return match
|
|
}
|
|
|
|
func (h *HashStorage) RemoveExpiredHashes() {
|
|
h.Lock()
|
|
defer h.Unlock()
|
|
|
|
now := time.Now()
|
|
|
|
for hash, entry := range h.Data {
|
|
if now.Sub(entry.Timestamp) > h.Expiration {
|
|
delete(h.Data, hash)
|
|
}
|
|
}
|
|
}
|
|
|
|
func (h *HashStorage) Reset() {
|
|
h.Lock()
|
|
defer h.Unlock()
|
|
|
|
h.Data = make(map[string]HashEntry)
|
|
}
|