-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathrpc-client.go
462 lines (403 loc) · 13.8 KB
/
rpc-client.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
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
package main
import (
"bufio"
"encoding/json"
"fmt"
"io"
"log"
"net"
"sync"
"time"
)
type Client struct {
Name string //Name of the miner
IP string //Ip to the cgminer including port
Conn net.Conn //Connection made with net.Dial
RefreshInterval int //Seconds between fetching information
MinerInfo *MinerInformation //Struct to put the answers for the webserver
ClientRequests chan RpcRequest //Channel for sending rpc request to cgminer
MHSThresLimit float64 //MHSAv should not be below this
LastSumTimestamp int //Last timestamp from summary request
}
//Main function for fetching information from one client
func rpcClient(name, ip string, refInt int, minerInfo *MinerInformation, wg *sync.WaitGroup, threshold float64) {
//Add everything except the connection
c := Client{name, ip, nil, refInt, minerInfo, nil, threshold, int(time.Now().Unix())}
//Save the Client struct in the MinerInfo
c.MinerInfo.Client = &c
clientRequests := make(chan RpcRequest)
c.ClientRequests = clientRequests
//Start the thread the will keep doing summary requests
go SummaryHandler(clientRequests, minerInfo, &c, wg)
//Start another thread the will ask the devs requests
go DevsHandler(clientRequests, minerInfo, &c, wg)
//Wait for new requst to make from the clienReequest channel
for r := range clientRequests {
//Create a new connection
c.Conn = createConnection(c.IP)
//If c.Conn is still nil then we couldn't connect
//So send back an empty slice of bytes
if c.Conn == nil {
log.Printf("[rpcClient] - Could not connect to the client - %s\n",c.Name)
r.ResultChan <- make([]byte, 0)
} else {
//Send the request to the cgminer
b := sendCommand(&c.Conn, r.Request)
/*
* Note:
*
* It seems that cgminer close the tcp connection
* after each call so we need to reset it for i := 0; i < count; i++ {
}
* the next rpc-call
*/
c.Conn.Close()
//And send back the result
r.ResultChan <- b
}
}
}
//Making summary requests to the cgminer and parse the result.
func SummaryHandler(res chan<- RpcRequest, minerInfo *MinerInformation, c *Client, wg *sync.WaitGroup) {
request := RpcRequest{"{\"command\":\"summary\"}", make(chan []byte), ""}
var response []byte
//Creating an empty instance of everything
summary := SummaryResponse{[]StatusObject{StatusObject{}}, []SummaryObject{SummaryObject{}}, 0}
summaryRow := MinerRow{}
summaryRow.Name = c.Name
//Save the default values
//Lock it
minerInfo.SumWrap.Mu.Lock()
//Save the summary
minerInfo.SumWrap.Summary = summary
minerInfo.SumWrap.SummaryRow = summaryRow
//Now unlock
minerInfo.SumWrap.Mu.Unlock()
//Signal that the thread is started
wg.Done()
for {
summary := SummaryResponse{[]StatusObject{StatusObject{}}, []SummaryObject{SummaryObject{}}, 0}
res <- request
response = <-request.ResultChan
//If we got the response back unmarshal it
if len(response) != 0 {
err := json.Unmarshal(response, &summary)
//Check for errors
if err != nil {
fmt.Println(err.Error())
}
//Update the summaryrow
summaryRow = MinerRow{c.Name, summary.Summary[0].Accepted, summary.Summary[0].Rejected, summary.Summary[0].MHSAv, summary.Summary[0].BestShare}
} else {
//No response so wait somee extra before try again
log.Println("Failed to fetch new data from " + c.Name)
summaryRow = MinerRow{c.Name, 0, 0, 0, 0}
}
//Lock it
minerInfo.SumWrap.Mu.Lock()
//Save the summary
minerInfo.SumWrap.Summary = summary
minerInfo.SumWrap.SummaryRow = summaryRow
//Now unlock
minerInfo.SumWrap.Mu.Unlock()
//Check if we should sleep some extra
if(len(response) == 0){
time.Sleep(time.Duration(c.RefreshInterval*2) * time.Second)
}
//Now sleep
time.Sleep(time.Duration(c.RefreshInterval) * time.Second)
}
}
//Making devs request to the cgminer and parse the result
func DevsHandler(res chan<- RpcRequest, minerInfo *MinerInformation, c *Client, wg *sync.WaitGroup) {
//Signal that the thread is started
wg.Done()
//Now do this forever and ever!
for {
//If it return false it has failed to connect
//So wait abit more before next time
if UpdateDevs(c.Name, true) == false {
log.Println("Failed to fetch new data from: " + c.Name)
//No response so wait somee extra before try again
time.Sleep(time.Duration(c.RefreshInterval*2) * time.Second)
}
//Now sleep
time.Sleep(time.Duration(c.RefreshInterval) * time.Second)
}
}
//Update the devs struct
//name = the name of the miner
//checkTresHold = true if it should check if the miner is below the threshold set for the miner
func UpdateDevs(name string, checkTresHold bool) (ok bool) {
ok = false
request := RpcRequest{"{\"command\":\"devs\"}", make(chan []byte), name}
minerInfo := miners[name]
var devs DevsResponse
//Ignore the error at the moment since it not implement in the Send() yet
response, _ := request.Send()
//Parse the data into a DevsResponse
devs.Parse(response)
//If we got data back
if len(response) != 0 {
//Set ok to true
ok = true
//Should we do a treshold check?
if checkTresHold == true {
//Need to sum up the mhs5s to get the current total hashrate for the miner
mhs5s := 0.0
for i := 0; i < len(devs.Devs); i++ {
var dev = &devs.Devs[i]
mhs5s += dev.MHS5s
}
CheckMhsThresHold(mhs5s, devs.Status[0].When, minerInfo.Client)
//If the threshold is checked then do a alive check as well
CheckAliveStatus(devs, name)
}
}
//Lock it
minerInfo.DevsWrap.Mu.Lock()
//Save the summary
minerInfo.DevsWrap.Devs = devs
//Now unlock
minerInfo.DevsWrap.Mu.Unlock()
return
}
//Check every devs to see if someone is sick or dead
//if so restart the miner
func CheckAliveStatus(devs DevsResponse, name string) {
for i := 0; i < len(devs.Devs); i++ {
var dev = &devs.Devs[i]
if dev.Status != "Alive" {
//Send the restart command
log.Printf("Dev #%s on %s got %s status so sending restart command\n", dev.GPU, name, dev.Status)
restartMiner(name)
}
}
}
//Checks the current mhs average value against the threshold.
//The value should have been lower for 10 minutes
//before it restarts the miner
func CheckMhsThresHold(mhs float64, lasttime int, c *Client) {
switch {
//Good - It's abowe the limit
case mhs >= c.MHSThresLimit:
//Save the last timestamp
c.LastSumTimestamp = lasttime
return
//Meeh - Under the limit but it hasn't gone 10 min yey
case mhs < c.MHSThresLimit && (lasttime-c.LastSumTimestamp) < 600:
//Dont to nothing just wait and see if the hashrate
//goes up or if it keeps down
return
//Oh noes - Below the threshold and for longer then 10 min
case mhs < c.MHSThresLimit && (lasttime-c.LastSumTimestamp) >= 600:
//Restart the miner
log.Printf("Hashrate: Below threshold(%v < %v) for %v secs which is over 10 min\n", mhs, c.MHSThresLimit, (lasttime - c.LastSumTimestamp))
log.Printf("Restarting %s miner\n", c.Name)
restartMiner(c.Name)
c.LastSumTimestamp = lasttime
return
}
}
func restartMiner(name string) {
request := RpcRequest{(fmt.Sprintf("{\"command\":\"restart\"}")), make(chan []byte), name}
request.Send()
}
//Enable or disable a gpu.
func enableDisable(status, device int, name string) {
var request RpcRequest
switch status {
case 0:
request = RpcRequest{fmt.Sprintf("{\"command\":\"gpudisable\",\"parameter\":\"%v\"}", device), make(chan []byte), name}
case 1:
request = RpcRequest{fmt.Sprintf("{\"command\":\"gpuenable\",\"parameter\":\"%v\"}", device), make(chan []byte), name}
}
request.Send()
}
//Change the gpu engine clock
func setGPUEngine(clock, device int, name string) {
request := RpcRequest{(fmt.Sprintf("{\"command\":\"gpuengine\",\"parameter\":\"%v,%v\"}", device, clock)), make(chan []byte), name}
request.Send()
}
//Change the gpu memory clock
func setGPUMemory(clock, device int, name string) {
request := RpcRequest{(fmt.Sprintf("{\"command\":\"gpumem\",\"parameter\":\"%v,%v\"}", device, clock)), make(chan []byte), name}
request.Send()
}
//Change the vddc on gpu
func setVDDC(voltage float64, device int, name string) {
request := RpcRequest{(fmt.Sprintf("{\"command\":\"gpuvddc\",\"parameter\":\"%v,%v\"}", device, voltage)), make(chan []byte), name}
request.Send()
}
//Change the intensity on the gpu
func setIntensity(intensity, device int, name string) {
request := RpcRequest{(fmt.Sprintf("{\"command\":\"gpuintensity\",\"parameter\":\"%v,%v\"}", device, intensity)), make(chan []byte), name}
request.Send()
}
//Write down the config file
func writeConfig(name string) {
request := RpcRequest{(fmt.Sprintf("{\"command\":\"save\",\"parameter\":\"\"}")), make(chan []byte), name}
request.Send()
}
// Returns a TCP connection to the ip
func createConnection(ip string) net.Conn {
conn, err := net.DialTimeout("tcp", ip,10*time.Second)
//Check for errors
if err != nil {
log.Printf("createConnection: %s, check if the ip is correct or cgminer's api is enabled or if the miner is offline\n", err)
return nil
}
return conn
}
// Sends a json rpc command to threw the socket and return the answer
func sendCommand(conn *net.Conn, cmd string) []byte {
//Write the command to the socket
fmt.Fprintf(*conn, cmd)
//Read the response
response, err := bufio.NewReader(*conn).ReadString('\n')
//Check for any errors
if err != nil {
//Check for errors
if err == io.EOF {
/*
* Cgminer sends out EOF after each call.
* Catch this error because it's not really
* an error that crash the program.
*/
} else {
//If the error is not EOF then warn about it
log.Println("Sending command error: ", err)
}
}
//Create the byte array
b := []byte(response)
/*
* Check for \x00 to remove
*/
if b[len(b)-1] == '\x00' {
b = b[0 : len(b)-1]
}
//Return the status we got from the server
return b
}
//Used the send action request to a miner
//I.E a restart or change intensity
type RpcRequest struct {
Request string
ResultChan chan []byte
Name string
}
//Sends the request to the miner and return the answer as byte array
func (request RpcRequest) Send() (response []byte, err error) {
miners[request.Name].Client.ClientRequests <- request
response = <-request.ResultChan
/*
* TODO:
* Check the response for the status and if something
* went wront create and error
*/
return response, nil
}
/*
* Bellow here is only structs defined
* for converting json responces to
* structs.
*/
////////////
// Status //
////////////
type StatusObject struct {
Status string `json:"STATUS"`
When int `json:"When"`
Code int `json:"Code"`
Msg string `json:"Msg"`
Description string `json:"Description"`
}
/////////////
// summary //
/////////////
type SummaryResponse struct {
Status []StatusObject `json:"STATUS"`
Summary []SummaryObject `json:"SUMMARY"`
Id int `json:"id"`
}
type SummaryObject struct {
Elapsed int `json:"Elapsed"`
MHSAv float64 `json:"MHS av"`
FoundBlocks int `json:"Found blocks"`
Getworks int `json:"Getworks"`
Accepted int `json:"Accepted"`
Rejected int `json:"Rejected"`
HardwareErrors int `json:"Hardware Errors"`
Utility float64 `json:"Utility"`
Discarded int `json:"Discarded"`
Stale int `json:"Stale"`
GetFailures int `json:"Get Failures"`
LocalWork int `json:"Local Work"`
RemoteFailures int `json:"Remote Failures"`
NetworkBlocks int `json:"Network Blocks"`
TotalMH float64 `json:"TotalMH"`
WorkUtility float64 `json:"Work Utility"`
DifficultyAccepted float64 `json:"Difficulty Accepted"`
DifficultyRejected float64 `json:"Difficulty Rejected"`
DifficultyStale float64 `json:"Difficulty Stale"`
BestShare int `json:"Best Share"`
}
//////////
// devs //
//////////
type DevsResponse struct {
Status []StatusObject `json:"STATUS"`
Devs []DevObject `json:"DEVS"`
Id int `json:"id"`
}
type DevObject struct {
GPU int `json:"GPU"`
Enabled string `json:"Enabled"`
Status string `json:"Status"`
Temperature float64 `json:"Temperature"`
FanSpeed int `json:"Fan Speed"`
FanPercent int `json:"Fan Percent"`
GPUClock int `json:"GPU Clock"`
MemoryClock int `json:"Memory Clock"`
GPUVoltage float64 `json:"GPU Voltage"`
GPUActivity int `json:"GPU Activity"`
Powertune int `json:"Powertune"`
MHSAv float64 `json:"MHS av"`
MHS5s float64 `json:"MHS 5s"`
Accepted int `json:"Accepted"`
Rejected int `json:"Rejected"`
HardwareErrors int `json:"Hardware Errors"`
Utility float64 `json:"Utility"`
Intensity string `json:"Intensity"`
LastSharePool int `json:"Last Share Pool"`
LastShareTime int `json:"Last Share Time"`
TotalMH float64 `json:"Total MH"`
Diff1Work int `json:"Diff1 Work"`
DifficultyAccepted float64 `json:"Difficulty Accepted"`
DifficultyRejected float64 `json:"Difficulty Rejected"`
LastShareDifficulty float64 `json:"Last Share Difficulty"`
LastValidWork int `json:"Last Valid Work"`
OnOff bool //This is an extra boolean used for html template parsing
}
//Parse raw data from a response to a DevsResponse
func (devs *DevsResponse) Parse(response []byte) {
if len(response) != 0 {
err := json.Unmarshal(response, &devs)
//Check for errors
if err != nil {
fmt.Println(err.Error())
}
//Set the onoff boolean for every device
for i := 0; i < len(devs.Devs); i++ {
//Get the variable
var dev = &devs.Devs[i]
//Make the comparison
if dev.Enabled == "Y" {
dev.OnOff = true
} else {
dev.OnOff = false
}
}
}
}