forked from TykTechnologies/tyk
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathrpc_analytics_purger.go
91 lines (75 loc) · 2.22 KB
/
rpc_analytics_purger.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
package main
import (
"encoding/json"
"github.com/Sirupsen/logrus"
"github.com/lonelycode/gorpc"
"gopkg.in/vmihailenco/msgpack.v2"
"time"
)
// Purger is an interface that will define how the in-memory store will be purged
// of analytics data to prevent it growing too large
type Purger interface {
PurgeCache()
StartPurgeLoop(int)
}
// RPCPurger will purge analytics data into a Mongo database, requires that the Mongo DB string is specified
// in the Config object
type RPCPurger struct {
Store *RedisClusterStorageManager
RPCClient *gorpc.Client
Client *gorpc.DispatcherClient
Address string
}
func (r *RPCPurger) ReConnect() {
r.RPCClient.Stop()
r.Connect()
}
// Connect Connects to RPC
func (r *RPCPurger) Connect() {
log.Info("Connecting to RPC Analytics service")
r.RPCClient = gorpc.NewTCPClient(r.Address)
if log.Level != logrus.DebugLevel {
gorpc.SetErrorLogger(gorpc.NilErrorLogger)
}
r.RPCClient.Start()
d := GetDispatcher()
r.Client = d.NewFuncClient(r.RPCClient)
return
}
// StartPurgeLoop starts the loop that will be started as a goroutine and pull data out of the in-memory
// store and into RPC
func (r RPCPurger) StartPurgeLoop(nextCount int) {
time.Sleep(time.Duration(nextCount) * time.Second)
r.PurgeCache()
r.StartPurgeLoop(nextCount)
}
// PurgeCache will pull the data from the in-memory store and drop it into the specified MongoDB collection
func (r *RPCPurger) PurgeCache() {
//var AnalyticsValues []interface{}
AnalyticsValues := r.Store.GetAndDeleteSet(ANALYTICS_KEYNAME)
if len(AnalyticsValues) > 0 {
keys := make([]AnalyticsRecord, len(AnalyticsValues), len(AnalyticsValues))
for i, v := range AnalyticsValues {
decoded := AnalyticsRecord{}
err := msgpack.Unmarshal(v.([]byte), &decoded)
log.Debug("Decoded Record: ", decoded)
if err != nil {
log.Error("Couldn't unmarshal analytics data:")
log.Error(err)
} else {
keys[i] = decoded
}
}
data, dErr := json.Marshal(keys)
if dErr != nil {
log.Error("Failed to marshal analytics data")
return
}
// Send keys to RPC
_, callErr := r.Client.Call("PurgeAnalyticsData", string(data))
if callErr != nil {
log.Error("Failed to call purge (reconnecting): ", callErr)
r.ReConnect()
}
}
}