-
Notifications
You must be signed in to change notification settings - Fork 39
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Showing
4 changed files
with
355 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,82 @@ | ||
package cache | ||
|
||
import ( | ||
"time" | ||
|
||
"github.com/ethereum/go-ethereum/log" | ||
|
||
"github.com/dgraph-io/ristretto" | ||
) | ||
|
||
const ( | ||
numCounters = 1e7 // number of keys to track frequency of (10M). | ||
maxCost = 1 << 30 // maximum cost of cache (1GB). | ||
bufferItems = 64 // number of keys per Get buffer. | ||
defaultConst = 1 // default cost of cache. | ||
) | ||
|
||
type RistrettoCache struct { | ||
cache *ristretto.Cache | ||
quit chan struct{} | ||
} | ||
|
||
// NewRistrettoCache returns a new RistrettoCache. | ||
func NewRistrettoCache(logger log.Logger) (*RistrettoCache, error) { | ||
cache, err := ristretto.NewCache(&ristretto.Config{ | ||
NumCounters: numCounters, | ||
MaxCost: maxCost, | ||
BufferItems: bufferItems, | ||
Metrics: true, | ||
}) | ||
if err != nil { | ||
return nil, err | ||
} | ||
|
||
c := &RistrettoCache{ | ||
cache: cache, | ||
quit: make(chan struct{}), | ||
} | ||
|
||
// Start the metrics logging | ||
go c.startMetricsLogging(logger) | ||
|
||
return c, nil | ||
} | ||
|
||
// Set adds the key and value to the cache. | ||
func (c *RistrettoCache) Set(key string, value map[string]interface{}, ttl time.Duration) bool { | ||
return c.cache.SetWithTTL(key, value, defaultConst, ttl) | ||
} | ||
|
||
// Get returns the value for the given key if it exists. | ||
func (c *RistrettoCache) Get(key string) (value map[string]interface{}, ok bool) { | ||
item, found := c.cache.Get(key) | ||
if !found { | ||
return nil, false | ||
} | ||
|
||
// Assuming the item is stored as a map[string]interface{}, otherwise you need to type assert to the correct type. | ||
value, ok = item.(map[string]interface{}) | ||
if !ok { | ||
// The item isn't of type map[string]interface{} | ||
return nil, false | ||
} | ||
|
||
return value, true | ||
} | ||
|
||
// startMetricsLogging starts logging cache metrics every hour. | ||
func (c *RistrettoCache) startMetricsLogging(logger log.Logger) { | ||
ticker := time.NewTicker(1 * time.Hour) | ||
for { | ||
select { | ||
case <-ticker.C: | ||
metrics := c.cache.Metrics | ||
logger.Info("Cache metrics: Hits: %d, Misses: %d, Cost Added: %d\n", | ||
metrics.Hits(), metrics.Misses(), metrics.CostAdded()) | ||
case <-c.quit: | ||
ticker.Stop() | ||
return | ||
} | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,98 @@ | ||
package cache | ||
|
||
import ( | ||
"crypto/sha256" | ||
"encoding/json" | ||
"fmt" | ||
"time" | ||
|
||
"github.com/ethereum/go-ethereum/log" | ||
|
||
"github.com/ten-protocol/go-ten/tools/walletextension/common" | ||
) | ||
|
||
const ( | ||
longCacheTTL = 5 * time.Hour | ||
shortCacheTTL = 1 * time.Second | ||
) | ||
|
||
// CacheableRPCMethods is a map of Ethereum JSON-RPC methods that can be cached and their TTL | ||
var cacheableRPCMethods = map[string]time.Duration{ | ||
// Ethereum JSON-RPC methods that can be cached long time | ||
"eth_getBlockByNumber": longCacheTTL, | ||
"eth_getBlockByHash": longCacheTTL, | ||
"eth_getTransactionByHash": longCacheTTL, | ||
"eth_chainId": longCacheTTL, | ||
|
||
// Ethereum JSON-RPC methods that can be cached short time | ||
"eth_blockNumber": shortCacheTTL, | ||
"eth_getCode": shortCacheTTL, | ||
"eth_getBalance": shortCacheTTL, | ||
"eth_getTransactionReceipt": shortCacheTTL, | ||
"eth_call": shortCacheTTL, | ||
"eth_gasPrice": shortCacheTTL, | ||
"eth_getTransactionCount": shortCacheTTL, | ||
"eth_estimateGas": shortCacheTTL, | ||
"eth_feeHistory": shortCacheTTL, | ||
} | ||
|
||
type Cache interface { | ||
Set(key string, value map[string]interface{}, ttl time.Duration) bool | ||
Get(key string) (value map[string]interface{}, ok bool) | ||
} | ||
|
||
func NewCache(logger log.Logger) (Cache, error) { | ||
return NewRistrettoCache(logger) | ||
} | ||
|
||
// IsCacheable checks if the given RPC request is cacheable and returns the cache key and TTL | ||
func IsCacheable(key *common.RPCRequest) (bool, string, time.Duration) { | ||
if key == nil || key.Method == "" { | ||
return false, "", 0 | ||
} | ||
|
||
// Check if the method is cacheable | ||
ttl, isCacheable := cacheableRPCMethods[key.Method] | ||
|
||
if isCacheable { | ||
// method is cacheable - select cache key | ||
switch key.Method { | ||
case "eth_getCode", "eth_getBalance", "eth_getTransactionCount", "eth_estimateGas", "eth_call": | ||
if len(key.Params) == 1 || len(key.Params) == 2 && (key.Params[1] == "latest" || key.Params[1] == "pending") { | ||
return true, GenerateCacheKey(key.Method, key.Params...), ttl | ||
} | ||
// in this case, we have a fixed block number, and we can cache the result for a long time | ||
return true, GenerateCacheKey(key.Method, key.Params...), longCacheTTL | ||
case "eth_feeHistory": | ||
if len(key.Params) == 2 || len(key.Params) == 3 && (key.Params[2] == "latest" || key.Params[2] == "pending") { | ||
return true, GenerateCacheKey(key.Method, key.Params...), ttl | ||
} | ||
// in this case, we have a fixed block number, and we can cache the result for a long time | ||
return true, GenerateCacheKey(key.Method, key.Params...), longCacheTTL | ||
default: | ||
return true, GenerateCacheKey(key.Method, key.Params...), ttl | ||
} | ||
} | ||
|
||
// method is not cacheable | ||
return false, "", 0 | ||
} | ||
|
||
// GenerateCacheKey generates a cache key for the given method and parameters | ||
func GenerateCacheKey(method string, params ...interface{}) string { | ||
// Serialize parameters | ||
paramBytes, err := json.Marshal(params) | ||
if err != nil { | ||
return "" | ||
} | ||
|
||
// Concatenate method name and parameters | ||
rawKey := method + string(paramBytes) | ||
|
||
// Optional: Apply hashing | ||
hasher := sha256.New() | ||
hasher.Write([]byte(rawKey)) | ||
hashedKey := fmt.Sprintf("%x", hasher.Sum(nil)) | ||
|
||
return hashedKey | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,147 @@ | ||
package cache | ||
|
||
import ( | ||
"reflect" | ||
"testing" | ||
"time" | ||
|
||
"github.com/ethereum/go-ethereum/log" | ||
|
||
"github.com/ten-protocol/go-ten/tools/walletextension/common" | ||
) | ||
|
||
var tests = map[string]func(t *testing.T){ | ||
"testCacheableMethods": testCacheableMethods, | ||
"testNonCacheableMethods": testNonCacheableMethods, | ||
"testMethodsWithLatestOrPendingParameter": testMethodsWithLatestOrPendingParameter, | ||
} | ||
|
||
var cacheTests = map[string]func(cache Cache, t *testing.T){ | ||
"testResultsAreCached": testResultsAreCached, | ||
"testCacheTTL": testCacheTTL, | ||
} | ||
|
||
var nonCacheableMethods = []string{"eth_sendrawtransaction", "eth_sendtransaction", "join", "authenticate"} | ||
|
||
func TestGatewayCaching(t *testing.T) { | ||
for name, test := range tests { | ||
t.Run(name, func(t *testing.T) { | ||
test(t) | ||
}) | ||
} | ||
|
||
// cache tests | ||
for name, test := range cacheTests { | ||
t.Run(name, func(t *testing.T) { | ||
logger := log.New() | ||
cache, err := NewCache(logger) | ||
if err != nil { | ||
t.Errorf("failed to create cache: %v", err) | ||
} | ||
test(cache, t) | ||
}) | ||
} | ||
} | ||
|
||
// testCacheableMethods tests if the cacheable methods are cacheable | ||
func testCacheableMethods(t *testing.T) { | ||
for method := range cacheableRPCMethods { | ||
key := &common.RPCRequest{Method: method} | ||
isCacheable, _, _ := IsCacheable(key) | ||
if isCacheable != true { | ||
t.Errorf("method %s should be cacheable", method) | ||
} | ||
} | ||
} | ||
|
||
// testNonCacheableMethods tests if the non-cacheable methods are not cacheable | ||
func testNonCacheableMethods(t *testing.T) { | ||
for _, method := range nonCacheableMethods { | ||
key := &common.RPCRequest{Method: method} | ||
isCacheable, _, _ := IsCacheable(key) | ||
if isCacheable == true { | ||
t.Errorf("method %s should not be cacheable", method) | ||
} | ||
} | ||
} | ||
|
||
// testMethodsWithLatestOrPendingParameter tests if the methods with latest or pending parameter are cacheable | ||
func testMethodsWithLatestOrPendingParameter(t *testing.T) { | ||
methods := []string{"eth_getCode", "eth_getBalance", "eth_getTransactionCount", "eth_estimateGas", "eth_call"} | ||
for _, method := range methods { | ||
key := &common.RPCRequest{Method: method, Params: []interface{}{"0x123", "latest"}} | ||
_, _, ttl := IsCacheable(key) | ||
if ttl != shortCacheTTL { | ||
t.Errorf("method %s with latest parameter should have TTL of %s, but %s received", method, shortCacheTTL, ttl) | ||
} | ||
|
||
key = &common.RPCRequest{Method: method, Params: []interface{}{"0x123", "pending"}} | ||
_, _, ttl = IsCacheable(key) | ||
if ttl != shortCacheTTL { | ||
t.Errorf("method %s with pending parameter should have TTL of %s, but %s received", method, shortCacheTTL, ttl) | ||
} | ||
} | ||
} | ||
|
||
// testResultsAreCached tests if the results are cached as expected | ||
func testResultsAreCached(cache Cache, t *testing.T) { | ||
// prepare a cacheable request and imaginary response | ||
req := &common.RPCRequest{Method: "eth_getBlockByNumber", Params: []interface{}{"0x123"}} | ||
res := map[string]interface{}{"result": "block"} | ||
isCacheable, key, ttl := IsCacheable(req) | ||
if !isCacheable { | ||
t.Errorf("method %s should be cacheable", req.Method) | ||
} | ||
// set the response in the cache with a TTL | ||
if !cache.Set(key, res, ttl) { | ||
t.Errorf("failed to set value in cache for %s", req) | ||
} | ||
|
||
time.Sleep(50 * time.Millisecond) // wait for the cache to be set | ||
value, ok := cache.Get(key) | ||
if !ok { | ||
t.Errorf("failed to get cached value for %s", req) | ||
} | ||
|
||
if !reflect.DeepEqual(value, res) { | ||
t.Errorf("expected %v, got %v", res, value) | ||
} | ||
} | ||
|
||
// testCacheTTL tests if the cache TTL is working as expected | ||
func testCacheTTL(cache Cache, t *testing.T) { | ||
req := &common.RPCRequest{Method: "eth_getBalance", Params: []interface{}{"0x123"}} | ||
res := map[string]interface{}{"result": "100"} | ||
isCacheable, key, ttl := IsCacheable(req) | ||
|
||
if !isCacheable { | ||
t.Errorf("method %s should be cacheable", req.Method) | ||
} | ||
|
||
if ttl != shortCacheTTL { | ||
t.Errorf("method %s should have TTL of %s, but %s received", req.Method, shortCacheTTL, ttl) | ||
} | ||
|
||
// set the response in the cache with a TTL | ||
if !cache.Set(key, res, ttl) { | ||
t.Errorf("failed to set value in cache for %s", req) | ||
} | ||
time.Sleep(50 * time.Millisecond) // wait for the cache to be set | ||
|
||
// check if the value is in the cache | ||
value, ok := cache.Get(key) | ||
if !ok { | ||
t.Errorf("failed to get cached value for %s", req) | ||
} | ||
|
||
if !reflect.DeepEqual(value, res) { | ||
t.Errorf("expected %v, got %v", res, value) | ||
} | ||
|
||
// sleep for the TTL to expire | ||
time.Sleep(shortCacheTTL + 100*time.Millisecond) | ||
_, ok = cache.Get(key) | ||
if ok { | ||
t.Errorf("value should not be in the cache after TTL") | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters