-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmain.go
343 lines (285 loc) · 9.68 KB
/
main.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
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
package main
import (
"context"
"fmt"
"log"
"math"
"math/rand"
"os"
"path/filepath"
"strings"
"sync"
"time"
"github.com/eoscanada/eos-go"
"github.com/eoscanada/eos-go/ecc"
"github.com/spf13/cobra"
)
// voter info struct from the ABI of eosio contract
type VoterInfo struct {
Owner eos.AccountName `json:"owner"`
Proxy eos.AccountName `json:"proxy"`
Producers []eos.AccountName `json:"producers"`
Staked eos.Int64 `json:"staked"`
UnpaidVoteshare eos.Float64 `json:"unpaid_voteshare"`
UnpaidVoteshareLastUpdated eos.TimePoint `json:"unpaid_voteshare_last_updated"`
UnpaidVoteshareChangeRate eos.Float64 `json:"unpaid_voteshare_change_rate"`
LastClaimTime eos.TimePoint `json:"last_claim_time"`
LastVoteWeight eos.Float64 `json:"last_vote_weight"`
ProxiedVoteWeight eos.Float64 `json:"proxied_vote_weight"`
IsProxy eos.Bool `json:"is_proxy"`
Flags1 uint32 `json:"flags1"`
Reserved2 uint32 `json:"reserved2"`
Reserved3 any `json:"reserved3"`
}
// voteproducer action input
type Voteproducer struct {
Voter eos.AccountName `json:"voter"`
Proxy eos.AccountName `json:"proxy"`
Producers []eos.AccountName `json:"producers"`
}
// claimgbmvote action input
type Claimgbmvote struct {
Owner eos.AccountName `json:"owner"`
}
// stakeclaim Account config struct
type Account struct {
Address eos.AccountName
Permission eos.PermissionName
PrivateKey ecc.PrivateKey
Proxy eos.AccountName
}
type VoterCacheItem struct {
UnpaidVoteshareLastUpdated eos.TimePoint
LastClaimTime eos.TimePoint
}
// list of WAX endpoints
var endpoints = []string{
"api-wax-mainnet.wecan.dev",
"api.wax.alohaeos.com",
"api.wax.bountyblok.io",
"api.wax.greeneosio.com",
"api.waxsweden.org",
"apiwax.3dkrender.com",
"wax.blacklusion.io",
"wax.cryptolions.io",
"wax.dapplica.io",
"wax.defibox.xyz",
"wax.eosphere.io",
"wax.eosusa.io",
"wax.eu.eosamsterdam.net",
"wax.greymass.com",
"wax.pink.gg",
}
// map to keep the voter info cached, to avoid unneccessary api calls
var voterInfoCache = map[eos.AccountName]VoterCacheItem{}
// parseConfig parses the config.txt file and returns a list of Account structs
func parseConfig(cfgFile string) []Account {
// get the full path of cfgFile
fullPath, err := filepath.Abs(cfgFile)
if err != nil {
log.Fatalf("Unable to determine full path of %s %v", cfgFile, err)
}
// check if the config file exists
_, err = os.Stat(fullPath)
if err != nil {
log.Fatalf("Config file %v does not exist", cfgFile)
}
// read the file contents as a string
data, err := os.ReadFile(fullPath)
if err != nil {
log.Fatalf("Unable to read config file %s %v", fullPath, err)
}
content := string(data)
lines := strings.Split(content, "\n")
if len(lines) == 0 {
log.Fatalf("%s is empty", fullPath)
}
accounts := make([]Account, 0)
for i, rawline := range lines {
// trim spaces from each line
line := strings.TrimSpace(rawline)
// ignore comments
if strings.HasPrefix(line, "#") {
continue
}
// ignore empty lines
if len(line) == 0 {
continue
}
// split the line into parts
parts := strings.Split(line, ":")
if len(parts) != 4 {
log.Fatalf("Unable to parse config line %d: %v", i, line)
}
address, permission, key, proxy := parts[0], parts[1], parts[2], parts[3]
eccKey, err := ecc.NewPrivateKey(key)
if err != nil {
log.Fatalf("Invalid private key on line %d: %v", i, err)
}
accounts = append(accounts, Account{eos.AccountName(address), eos.PermissionName(permission), *eccKey, eos.AccountName(proxy)})
}
return accounts
}
// fetchLastClaim fetches the last claim time and unpaid voteshare from the voters table in eosio contract
func fetchLastClaim(account eos.AccountName) (*eos.TimePoint, *eos.TimePoint, error) {
// check if account is cached
if val, ok := voterInfoCache[account]; ok {
return &val.LastClaimTime, &val.UnpaidVoteshareLastUpdated, nil
}
// pick a random endpoint
endpoint := endpoints[rand.Intn(len(endpoints))]
api := eos.New(fmt.Sprintf("https://%s", endpoint))
log.Printf("Fetching voter info for account %v using %v\n", account, endpoint)
results, err := api.GetTableRows(context.Background(), eos.GetTableRowsRequest{
Code: "eosio",
Scope: "eosio",
Table: "voters",
LowerBound: string(account),
UpperBound: string(account),
Limit: 1,
JSON: true,
})
if err != nil {
log.Printf("Error fetching voter info for account: %v using %v\n", account, endpoint)
return nil, nil, err
}
var rows []VoterInfo
err = results.JSONToStructs(&rows)
if err != nil {
log.Printf("Error decoding voter info for account: %v %v\n", account, err)
return nil, nil, err
}
if len(rows) == 0 {
log.Printf("Account %v has not voted yet\n", account)
var lct eos.TimePoint = eos.TimePoint(0)
var lvsu eos.TimePoint = eos.TimePoint(0)
return &lct, &lvsu, nil
}
// save the row in the cache
voterInfoCache[account] = VoterCacheItem{
LastClaimTime: rows[0].LastClaimTime,
UnpaidVoteshareLastUpdated: rows[0].UnpaidVoteshareLastUpdated,
}
return &rows[0].LastClaimTime, &rows[0].UnpaidVoteshareLastUpdated, nil
}
// run runs the script, check if the account is ready to vote & claim or wait until it is
func run(account Account) {
lastClaimTime, lastVoteshareUpdated, err := fetchLastClaim(account.Address)
if err != nil {
return
}
var timeDiff time.Duration
if lastClaimTime != nil {
// calculate the time difference between the last claim and now
timeDiff = time.Since(time.UnixMicro(int64(*lastClaimTime)))
}
// if it's less than 24 hours, sleep
if timeDiff < 24*time.Hour {
// calculate remaining time until next claim
remaining := time.Until(time.UnixMicro(int64(*lastClaimTime)).Add(24 * time.Hour))
// pick the shortest duration between `remaining` and 1 hour
// and make sure it is not negative
shortest := math.Min(remaining.Seconds(), 3600)
if shortest < 0 {
shortest = 0
}
// calculate the sleep time
sleepTime := (time.Duration(shortest) * time.Second)
// sleep a max of 60 minutes or until the next claim time
log.Printf("Account %v Sleeping %v", account.Address, sleepTime)
// add an extra 5 seconds to fix a bug (?)
// where the time.Sleep wakes up a few ms too soon
time.Sleep(sleepTime + (5 * time.Second))
go run(account)
return
}
// construct the actions
actions := make([]*eos.Action, 0)
// add the voting action
actions = append(actions, &eos.Action{
Account: "eosio",
Name: "voteproducer",
Authorization: []eos.PermissionLevel{{Actor: account.Address, Permission: account.Permission}},
ActionData: eos.NewActionData(Voteproducer{
Voter: account.Address,
Producers: []eos.AccountName{},
Proxy: account.Proxy,
}),
})
// if the user has voted before, add the claim action
if *lastVoteshareUpdated > 0 {
actions = append([]*eos.Action{{
Account: "eosio",
Name: "claimgbmvote",
Authorization: []eos.PermissionLevel{{Actor: account.Address, Permission: account.Permission}},
ActionData: eos.NewActionData(Claimgbmvote{
Owner: account.Address,
}),
}}, actions...)
}
log.Printf("Sending transaction for account %v", account.Address)
// send the transaction
transact(account, actions)
// sleep for 30 seconds after the transaction, then re-run again
// in case the tx failed in the first time
// or it will just sleep if it had succeeded
time.Sleep(30 * time.Second)
run(account)
}
// prepare and submit a transaction to the blockchain
func transact(account Account, actions []*eos.Action) {
// pick a random endpoint
endpoint := endpoints[rand.Intn(len(endpoints))]
api := eos.New(fmt.Sprintf("https://%s", endpoint))
keyBag := &eos.KeyBag{}
keyBag.ImportPrivateKey(context.Background(), account.PrivateKey.String())
api.SetSigner(keyBag)
api.SetCustomGetRequiredKeys(func(ctx context.Context, tx *eos.Transaction) ([]ecc.PublicKey, error) {
return keyBag.AvailableKeys(ctx)
})
txOpts := &eos.TxOptions{}
if err := txOpts.FillFromChain(context.Background(), api); err != nil {
log.Printf("Error filling tx opts: %v", err)
return
}
tx := eos.NewTransaction(actions, txOpts)
_, packedTx, err := api.SignTransaction(context.Background(), tx, txOpts.ChainID, eos.CompressionNone)
if err != nil {
log.Printf("Error signing transaction: %v", err)
return
}
response, err := api.SendTransaction(context.Background(), packedTx)
if err != nil {
log.Printf("Error sending transaction for account %v using %v %v", account.Address, endpoint, err)
return
}
// delete the cache for this account
delete(voterInfoCache, account.Address)
log.Printf("Transaction success for account %v: %v", account.Address, response.TransactionID)
}
func main() {
var cfgFile string
var rootCmd = &cobra.Command{
Use: "stakeclaim",
Short: "A simple utility to claim & refresh vote strength on WAX",
Long: `A simple utility to claim vote rewards and refresh vote strength daily
Made by Benjie (https://github.com/benjiewheeler)`,
Run: func(cmd *cobra.Command, args []string) {
// log the pid of this process
log.Printf("Stakeclaim running: %v", os.Getpid())
// parse the config.txt file
accounts := parseConfig(cfgFile)
var wg sync.WaitGroup
for _, account := range accounts {
wg.Add(1)
go run(account)
}
wg.Wait()
},
}
rootCmd.PersistentFlags().StringVar(&cfgFile, "config-file", "./config.txt", "config file to read the account keys from")
if err := rootCmd.Execute(); err != nil {
fmt.Println(err)
os.Exit(1)
}
}