Files
cloudpods/pkg/util/hashcache/cache.go
Qiu Jian 77c4a53059 feature: 1. ak/sk auth support 2. a full feature s3gateway works with
Cyberduck

注意:以下两个文件的修改还未被官方合并,所以下次make
mod的时候会被覆盖,需要注意checkout恢复:

vendor/github.com/Azure/azure-sdk-for-go/storage/blockblob.go
vendor/github.com/tencentyun/cos-go-sdk-v5/object_part.go
2019-08-30 00:12:39 +08:00

156 lines
3.2 KiB
Go

// Copyright 2019 Yunion
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package hashcache
import (
"crypto/md5"
"crypto/sha1"
"crypto/sha256"
"crypto/sha512"
"sync"
"time"
)
var initTime = time.Time{}
type cacheNode struct {
key string
expire time.Time
value interface{}
}
func (node *cacheNode) reset() {
if !node.expire.IsZero() {
node.key = ""
node.expire = initTime
node.value = nil
}
}
type Cache struct {
table []cacheNode
lock *sync.Mutex
size uint32
defaultTtl time.Duration
}
func NewCache(size uint32, defaultTTL time.Duration) *Cache {
ca := &Cache{table: make([]cacheNode, size), lock: &sync.Mutex{}, size: size, defaultTtl: defaultTTL}
return ca
}
const (
HASH_ALG_MD5 int = iota
HASH_ALG_SHA1
HASH_ALG_SHA256
HASH_ALG_SHA512
)
func bytes2int(b []byte) uint32 {
return ((uint32(b[0]) << 24) | (uint32(b[1]) << 16) | (uint32(b[2]) << 8) | (uint32(b[3])))
}
func checksum(alg int, key string) uint32 {
var hash uint32 = 0
switch alg {
case HASH_ALG_MD5:
v := md5.Sum([]byte(key))
hash = bytes2int(v[0:4])
case HASH_ALG_SHA1:
v := sha1.Sum([]byte(key))
hash = bytes2int(v[0:4])
case HASH_ALG_SHA256:
v := sha256.Sum224([]byte(key))
hash = bytes2int(v[0:4])
case HASH_ALG_SHA512:
v := sha512.Sum512([]byte(key))
hash = bytes2int(v[0:4])
}
return hash
}
func (c *Cache) find(key string) (bool, uint32) {
var idx uint32
now := time.Now()
for _, alg := range []int{HASH_ALG_MD5, HASH_ALG_SHA1, HASH_ALG_SHA256, HASH_ALG_SHA512} {
idx = checksum(alg, key) % c.size
if c.table[idx].key == key {
if c.table[idx].expire.IsZero() || c.table[idx].expire.After(now) {
return true, idx
}
break
}
}
return false, idx
}
func (c *Cache) Get(key string) interface{} {
find, idx := c.find(key)
if find {
return c.table[idx].value
}
return nil
}
func (c *Cache) AtomicGet(key string) interface{} {
c.lock.Lock()
defer c.lock.Unlock()
return c.Get(key)
}
func (c *Cache) Set(key string, val interface{}, expire ...time.Time) {
find, idx := c.find(key)
if !find {
c.table[idx].key = key
}
c.table[idx].value = val
if len(expire) > 0 && !expire[0].IsZero() {
c.table[idx].expire = expire[0]
} else if c.defaultTtl > time.Millisecond {
c.table[idx].expire = time.Now().Add(c.defaultTtl)
} else {
c.table[idx].expire = time.Time{}
}
}
func (c *Cache) AtomicSet(key string, val interface{}) {
c.lock.Lock()
defer c.lock.Unlock()
c.Set(key, val)
}
func (c *Cache) Remove(key string) {
find, idx := c.find(key)
if !find {
return
}
c.table[idx].reset()
}
func (c *Cache) AtomicRemove(key string) {
c.lock.Lock()
defer c.lock.Unlock()
c.Remove(key)
}
func (c *Cache) Invalidate() {
c.lock.Lock()
defer c.lock.Unlock()
for i := range c.table {
c.table[i].reset()
}
}