forked from mindjiver/gopherstack
-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathwait.go
More file actions
130 lines (114 loc) · 2.83 KB
/
wait.go
File metadata and controls
130 lines (114 loc) · 2.83 KB
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
package gopherstack
import (
"fmt"
"log"
"time"
)
// waitForAsyncJob simply blocks until the the asynchronous job has
// executed or has timed out.
func (c CloudstackClient) WaitForAsyncJob(jobId string, timeout time.Duration) error {
done := make(chan struct{})
defer close(done)
result := make(chan error, 1)
go func() {
attempts := 0
for {
attempts += 1
log.Printf("Checking async job status... (attempt: %d)", attempts)
response, err := c.QueryAsyncJobResult(jobId)
if err != nil {
result <- err
return
}
// Check status of the job we issued.
// 0 - pending / in progress
// 1 - succedded
// 2 - failed
// 3 - cancelled
status := response.Queryasyncjobresultresponse.Jobstatus
switch status {
case 0:
continue
case 1:
result <- nil
return
case 2:
err := fmt.Errorf("WaitForAsyncJob failed")
result <- err
return
case 3:
err := fmt.Errorf("WaitForAsyncJob was cancelled")
result <- err
return
}
// Wait 3 seconds in between
time.Sleep(3 * time.Second)
// Verify we shouldn't exit
select {
case <- done:
// We finished, so just exit the goroutine
return
default:
// Keep going
}
}
}()
log.Printf("Waiting for up to %d seconds for async job %s", timeout, jobId)
select {
case err := <- result:
return err
case <- time.After(timeout):
err := fmt.Errorf("Timeout while waiting to for async job to finish")
return err
}
}
// WaitForVirtualMachineState simply blocks until the virtual machine
// is in the specified state.
func (c CloudstackClient) WaitForVirtualMachineState(vmid string, wantedState string, timeout time.Duration) error {
done := make(chan struct{})
defer close(done)
result := make(chan error, 1)
go func() {
attempts := 0
for {
attempts += 1
log.Printf("Checking virtual machine state... (attempt: %d)", attempts)
response, err := c.ListVirtualMachines(vmid)
if err != nil {
result <- err
return
}
count := response.Listvirtualmachinesresponse.Count
if count != 1 {
result <- err
return
}
currentState := response.Listvirtualmachinesresponse.Virtualmachine[0].State
// check what the real state will be.
log.Printf("current state: %s", currentState)
log.Printf("wanted state: %s", wantedState)
if currentState == wantedState {
result <- nil
return
}
// Wait 3 seconds in between
time.Sleep(3 * time.Second)
// Verify we shouldn't exit
select {
case <- done:
// We finished, so just exit the goroutine
return
default:
// Keep going
}
}
}()
log.Printf("Waiting for up to %d seconds for Virtual Machine state to converge", timeout)
select {
case err := <- result:
return err
case <- time.After(timeout):
err := fmt.Errorf("Timeout while waiting to for Virtual Machine to converge")
return err
}
}