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
|
package applesilicon
import (
"time"
"github.com/scaleway/scaleway-sdk-go/errors"
"github.com/scaleway/scaleway-sdk-go/internal/async"
"github.com/scaleway/scaleway-sdk-go/scw"
)
const (
defaultRetryInterval = 15 * time.Second
defaultTimeout = 60 * time.Minute
)
// WaitForInstanceRequest is used by WaitForServer method.
type WaitForServerRequest struct {
ServerID string
Zone scw.Zone
Timeout *time.Duration
RetryInterval *time.Duration
}
// WaitForServer waits for the instance to be in a "terminal state" before returning.
// This function can be used to wait for an instance to be ready for example.
func (s *API) WaitForServer(req *WaitForServerRequest, opts ...scw.RequestOption) (*Server, error) {
timeout := defaultTimeout
if req.Timeout != nil {
timeout = *req.Timeout
}
retryInterval := defaultRetryInterval
if req.RetryInterval != nil {
retryInterval = *req.RetryInterval
}
terminalStatus := map[ServerStatus]struct{}{
ServerStatusReady: {},
ServerStatusError: {},
}
server, err := async.WaitSync(&async.WaitSyncConfig{
Get: func() (interface{}, bool, error) {
res, err := s.GetServer(&GetServerRequest{
ServerID: req.ServerID,
Zone: req.Zone,
}, opts...)
if err != nil {
return nil, false, err
}
_, isTerminal := terminalStatus[res.Status]
return res, isTerminal, nil
},
Timeout: timeout,
IntervalStrategy: async.LinearIntervalStrategy(retryInterval),
})
if err != nil {
return nil, errors.Wrap(err, "waiting for server failed")
}
return server.(*Server), nil
}
func (s *API) WaitForPossibleDeletion(req *WaitForServerRequest, opts ...scw.RequestOption) (*Server, error) {
server, err := s.WaitForServer(&WaitForServerRequest{
ServerID: req.ServerID,
Zone: req.Zone,
Timeout: scw.TimeDurationPtr(defaultTimeout),
RetryInterval: scw.TimeDurationPtr(defaultRetryInterval),
}, opts...,
)
if err != nil {
return nil, errors.Wrap(err, "waiting for server failed")
}
timeToDelete := *server.DeletableAt
time.Sleep(time.Until(timeToDelete))
return server, nil
}
func (s *PrivateNetworkAPI) WaitForServerPrivateNetworks(req *WaitForServerRequest, opts ...scw.RequestOption) ([]*ServerPrivateNetwork, error) {
timeout := defaultTimeout
if req.Timeout != nil {
timeout = *req.Timeout
}
retryInterval := defaultRetryInterval
if req.RetryInterval != nil {
retryInterval = *req.RetryInterval
}
terminalStatus := map[ServerPrivateNetworkServerStatus]struct{}{
ServerPrivateNetworkServerStatusAttached: {},
ServerPrivateNetworkServerStatusError: {},
ServerPrivateNetworkServerStatusUnknownStatus: {},
ServerPrivateNetworkServerStatusLocked: {},
}
serverPrivateNetworks, err := async.WaitSync(&async.WaitSyncConfig{
Get: func() (interface{}, bool, error) {
res, err := s.ListServerPrivateNetworks(&PrivateNetworkAPIListServerPrivateNetworksRequest{
Zone: req.Zone,
ServerID: &req.ServerID,
}, opts...)
if err != nil {
return nil, false, err
}
for _, serverPrivateNetwork := range res.ServerPrivateNetworks {
_, isTerminal := terminalStatus[serverPrivateNetwork.Status]
if !isTerminal {
return res.ServerPrivateNetworks, isTerminal, err
}
}
return res.ServerPrivateNetworks, true, err
},
Timeout: timeout,
IntervalStrategy: async.LinearIntervalStrategy(retryInterval),
})
if err != nil {
return nil, errors.Wrap(err, "waiting for server private network failed")
}
return serverPrivateNetworks.([]*ServerPrivateNetwork), nil
}
|