2020-05-22 17:52:24 +02:00
|
|
|
package client
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
2020-05-23 12:34:44 +02:00
|
|
|
"hash/fnv"
|
2020-05-22 17:52:24 +02:00
|
|
|
"time"
|
|
|
|
|
2020-05-23 12:34:44 +02:00
|
|
|
cache "github.com/patrickmn/go-cache"
|
2022-10-20 13:00:50 +02:00
|
|
|
|
2024-06-04 22:40:43 +02:00
|
|
|
"go-micro.dev/v5/metadata"
|
|
|
|
"go-micro.dev/v5/transport/headers"
|
2020-05-22 17:52:24 +02:00
|
|
|
)
|
|
|
|
|
2022-09-30 16:27:07 +02:00
|
|
|
// NewCache returns an initialized cache.
|
2020-05-22 17:52:24 +02:00
|
|
|
func NewCache() *Cache {
|
|
|
|
return &Cache{
|
2020-05-23 12:34:44 +02:00
|
|
|
cache: cache.New(cache.NoExpiration, 30*time.Second),
|
2020-05-22 17:52:24 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-09-30 16:27:07 +02:00
|
|
|
// Cache for responses.
|
2020-05-22 17:52:24 +02:00
|
|
|
type Cache struct {
|
2020-05-23 12:34:44 +02:00
|
|
|
cache *cache.Cache
|
2020-05-22 17:52:24 +02:00
|
|
|
}
|
|
|
|
|
2022-09-30 16:27:07 +02:00
|
|
|
// Get a response from the cache.
|
2020-05-23 12:34:44 +02:00
|
|
|
func (c *Cache) Get(ctx context.Context, req *Request) (interface{}, bool) {
|
|
|
|
return c.cache.Get(key(ctx, req))
|
2020-05-22 17:52:24 +02:00
|
|
|
}
|
|
|
|
|
2022-09-30 16:27:07 +02:00
|
|
|
// Set a response in the cache.
|
2020-05-22 17:52:24 +02:00
|
|
|
func (c *Cache) Set(ctx context.Context, req *Request, rsp interface{}, expiry time.Duration) {
|
2020-05-23 12:34:44 +02:00
|
|
|
c.cache.Set(key(ctx, req), rsp, expiry)
|
2020-05-22 17:52:24 +02:00
|
|
|
}
|
|
|
|
|
2022-09-30 16:27:07 +02:00
|
|
|
// List the key value pairs in the cache.
|
2020-05-24 19:45:57 +02:00
|
|
|
func (c *Cache) List() map[string]string {
|
|
|
|
items := c.cache.Items()
|
|
|
|
|
|
|
|
rsp := make(map[string]string, len(items))
|
2022-10-20 13:00:50 +02:00
|
|
|
|
2020-05-24 19:45:57 +02:00
|
|
|
for k, v := range items {
|
|
|
|
bytes, _ := json.Marshal(v.Object)
|
|
|
|
rsp[k] = string(bytes)
|
|
|
|
}
|
|
|
|
|
|
|
|
return rsp
|
|
|
|
}
|
|
|
|
|
2022-09-30 16:27:07 +02:00
|
|
|
// key returns a hash for the context and request.
|
2020-05-23 12:34:44 +02:00
|
|
|
func key(ctx context.Context, req *Request) string {
|
2022-10-20 13:00:50 +02:00
|
|
|
ns, _ := metadata.Get(ctx, headers.Namespace)
|
2020-05-22 17:52:24 +02:00
|
|
|
|
2020-05-23 12:34:44 +02:00
|
|
|
bytes, _ := json.Marshal(map[string]interface{}{
|
2020-05-27 10:07:59 +02:00
|
|
|
"namespace": ns,
|
2020-05-23 12:34:44 +02:00
|
|
|
"request": map[string]interface{}{
|
|
|
|
"service": (*req).Service(),
|
|
|
|
"endpoint": (*req).Endpoint(),
|
|
|
|
"method": (*req).Method(),
|
|
|
|
"body": (*req).Body(),
|
|
|
|
},
|
|
|
|
})
|
|
|
|
|
|
|
|
h := fnv.New64()
|
2020-05-22 17:52:24 +02:00
|
|
|
h.Write(bytes)
|
2022-10-20 13:00:50 +02:00
|
|
|
|
2020-05-22 17:52:24 +02:00
|
|
|
return fmt.Sprintf("%x", h.Sum(nil))
|
|
|
|
}
|