forked from odise/etcd-backup
-
Notifications
You must be signed in to change notification settings - Fork 0
/
restore.go
125 lines (104 loc) · 3.41 KB
/
restore.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
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
package main
import (
"encoding/json"
"errors"
"fmt"
"os"
"sync"
"sync/atomic"
"time"
"github.com/coreos/go-etcd/etcd"
)
func LoadDataSet(dumpFilePath string) *[]BackupKey {
file, error := os.Open(dumpFilePath)
defer file.Close()
if error != nil {
config.LogFatal("Error when trying to open the file `"+dumpFilePath+"`. Error: ", error)
}
jsonDataSet := &[]BackupKey{}
jsonParser := json.NewDecoder(file)
if err := jsonParser.Decode(jsonDataSet); err != nil {
config.LogFatal("Error when trying to load data set into json. Error: ", err)
}
return jsonDataSet
}
func RestoreDataSet(backupKeys []BackupKey, config *Config, etcdClient EtcdClient) {
statistics := NewRestoreStatistics(backupKeys)
throttle := make(chan int, config.ConcurrentRequests)
var wg sync.WaitGroup
for index := range backupKeys {
backupKey := backupKeys[index]
RestoreKeyWithThrottle(&backupKey, statistics, &wg, throttle, etcdClient)
}
wg.Wait()
printStatistics(statistics)
}
func NewRestoreStatistics(backupKeys []BackupKey) map[string]*int32 {
DataSetSize := int32(len(backupKeys))
KeysInserted := int32(0)
DirectoriesInserted := int32(0)
return map[string]*int32{
"DataSetSize": &DataSetSize,
"KeysInserted": &KeysInserted,
"EmptyDirectories": &DirectoriesInserted,
}
}
func printStatistics(statistics map[string]*int32) {
config.LogPrintln("Backup restored succesfully! Results:")
for keyName, value := range statistics {
config.LogPrintln(keyName + ": " + fmt.Sprintf("%#v", *value))
}
}
func RestoreKeyWithThrottle(backupKey *BackupKey, statistics map[string]*int32, wg *sync.WaitGroup, throttle chan int, etcdClient EtcdClient) {
if backupKey.MatchBackupStrategy(config.BackupStrategy) {
wg.Add(1)
throttle <- 1
go RestoreKey(backupKey, statistics, wg, throttle, etcdClient)
}
}
func RestoreKey(backupKey *BackupKey, statistics map[string]*int32, wg *sync.WaitGroup, throttle chan int, etcdClient EtcdClient) {
defer wg.Done()
if !backupKey.IsExpired() {
if backupKey.IsDirectory() {
restoreKeyWithRetries(setDirectory, 0, backupKey, etcdClient)
atomic.AddInt32(statistics["EmptyDirectories"], 1)
} else {
restoreKeyWithRetries(setKey, 0, backupKey, etcdClient)
}
atomic.AddInt32(statistics["KeysInserted"], 1)
}
<-throttle
}
func restoreKeyWithRetries(
request func(*BackupKey, EtcdClient) (*etcd.Response, error),
retries int, backupKey *BackupKey, etcdClient EtcdClient,
) {
_, err := request(backupKey, etcdClient)
if err != nil {
if retries > config.Retries {
config.LogFatal(err)
} else {
retries += 1
time.Sleep(time.Duration(retries * 1000))
restoreKeyWithRetries(request, retries, backupKey, etcdClient)
}
}
}
func setKey(backupKey *BackupKey, etcdClient EtcdClient) (*etcd.Response, error) {
response, err := etcdClient.Set(backupKey.Key, *backupKey.Value, uint64(backupKey.TTL))
if err != nil {
err = errors.New("Error when trying to set the following key: " + backupKey.Key + ". Error: " + err.Error())
}
return response, err
}
func setDirectory(backupKey *BackupKey, etcdClient EtcdClient) (*etcd.Response, error) {
response, err := etcdClient.SetDir(backupKey.Key, uint64(backupKey.TTL))
if err != nil {
if err.(*etcd.EtcdError) != nil && err.(*etcd.EtcdError).ErrorCode != 102 {
err = errors.New("Error when trying to set the following directory : " + backupKey.Key + ". Error: " + err.Error())
} else {
err = nil
}
}
return response, err
}