blob: 1ee00a470df46c408e03be66a2fd120a362e2408 [file] [log] [blame]
Jamie Hannafordb6927c12014-11-03 10:31:26 +01001package nodes
Jamie Hannaford3cfa00a2014-11-03 11:16:35 +01002
3import (
Jamie Hannaford0a9a6be2014-11-03 12:55:38 +01004 "errors"
Jamie Hannaford3cfa00a2014-11-03 11:16:35 +01005 "fmt"
6
Jamie Hannaforded8b89a2014-11-03 12:24:19 +01007 "github.com/racker/perigee"
Jamie Hannaford3cfa00a2014-11-03 11:16:35 +01008 "github.com/rackspace/gophercloud"
9 "github.com/rackspace/gophercloud/pagination"
10)
11
Jamie Hannaford8cdaa802014-11-03 15:11:39 +010012// List is the operation responsible for returning a paginated collection of
13// load balancer nodes. It requires the node ID, its parent load balancer ID,
14// and optional limit integer (passed in either as a pointer or a nil poitner).
Jamie Hannaford3cfa00a2014-11-03 11:16:35 +010015func List(client *gophercloud.ServiceClient, loadBalancerID int, limit *int) pagination.Pager {
16 url := rootURL(client, loadBalancerID)
Jamie Hannaford3cfa00a2014-11-03 11:16:35 +010017 if limit != nil {
18 url += fmt.Sprintf("?limit=%d", limit)
19 }
20
Jamie Hannaforded8b89a2014-11-03 12:24:19 +010021 return pagination.NewPager(client, url, func(r pagination.PageResult) pagination.Page {
Jamie Hannaford3cfa00a2014-11-03 11:16:35 +010022 return NodePage{pagination.SinglePageBase(r)}
Jamie Hannaforded8b89a2014-11-03 12:24:19 +010023 })
24}
25
Jamie Hannaford8cdaa802014-11-03 15:11:39 +010026// CreateOptsBuilder is the interface responsible for generating the JSON
27// for a Create operation.
Jamie Hannaforded8b89a2014-11-03 12:24:19 +010028type CreateOptsBuilder interface {
29 ToNodeCreateMap() (map[string]interface{}, error)
30}
31
Jamie Hannaford8cdaa802014-11-03 15:11:39 +010032// CreateOpts is a slice of CreateOpt structs, that allow the user to create
33// multiple nodes in a single operation (one node per CreateOpt).
Jamie Hannaforded8b89a2014-11-03 12:24:19 +010034type CreateOpts []CreateOpt
35
Jamie Hannaford8cdaa802014-11-03 15:11:39 +010036// CreateOpt represents the options to create a single node.
Jamie Hannaforded8b89a2014-11-03 12:24:19 +010037type CreateOpt struct {
Jamie Hannaford8cdaa802014-11-03 15:11:39 +010038 // Required - the IP address or CIDR for this back-end node. It can either be
39 // a private IP (ServiceNet) or a public IP.
40 Address string
41
42 // Optional - the port on which traffic is sent and received.
43 Port int
44
45 // Optional - the condition of the node. See the consts in Results.go.
Jamie Hannaforded8b89a2014-11-03 12:24:19 +010046 Condition Condition
Jamie Hannaford8cdaa802014-11-03 15:11:39 +010047
48 // Optional - the type of the node. See the consts in Results.go.
49 Type Type
50
51 // Optional - a pointer to an integer between 0 and 100.
52 Weight *int
Jamie Hannaford00222d72014-11-03 13:58:52 +010053}
54
55func validateWeight(weight *int) error {
56 if weight != nil && (*weight > 100 || *weight < 0) {
57 return errors.New("Weight must be a valid int between 0 and 100")
58 }
59 return nil
Jamie Hannaforded8b89a2014-11-03 12:24:19 +010060}
61
Jamie Hannaford8cdaa802014-11-03 15:11:39 +010062// ToNodeCreateMap converts a slice of options into a map that can be used for
63// the JSON.
Jamie Hannaforded8b89a2014-11-03 12:24:19 +010064func (opts CreateOpts) ToNodeCreateMap() (map[string]interface{}, error) {
65 type nodeMap map[string]interface{}
66 nodes := []nodeMap{}
67
68 for k, v := range opts {
69 if v.Address == "" {
70 return nodeMap{}, fmt.Errorf("ID is a required attribute, none provided for %d CreateOpt element", k)
71 }
Jamie Hannaford00222d72014-11-03 13:58:52 +010072 if weightErr := validateWeight(v.Weight); weightErr != nil {
73 return nodeMap{}, weightErr
74 }
Jamie Hannaforded8b89a2014-11-03 12:24:19 +010075
76 node := make(map[string]interface{})
77 node["address"] = v.Address
78
79 if v.Port > 0 {
80 node["port"] = v.Port
81 }
82 if v.Condition != "" {
83 node["condition"] = v.Condition
84 }
85 if v.Type != "" {
86 node["type"] = v.Type
87 }
Jamie Hannaford00222d72014-11-03 13:58:52 +010088 if v.Weight != nil {
89 node["weight"] = &v.Weight
90 }
Jamie Hannaforded8b89a2014-11-03 12:24:19 +010091
92 nodes = append(nodes, node)
Jamie Hannaford3cfa00a2014-11-03 11:16:35 +010093 }
Jamie Hannaforded8b89a2014-11-03 12:24:19 +010094
95 return nodeMap{"nodes": nodes}, nil
96}
97
Jamie Hannaford8cdaa802014-11-03 15:11:39 +010098// Create is the operation responsible for creating a new node on a load
99// balancer. Since every load balancer exists in both ServiceNet and the public
100// Internet, both private and public IP addresses can be used for nodes.
101//
102// If nodes need time to boot up services before they become operational, you
103// can temporarily prevent traffic from being sent to that node by setting the
104// Condition field to DRAINING. Health checks will still be performed; but once
105// your node is ready, you can update its condition to ENABLED and have it
106// handle traffic.
Jamie Hannaforded8b89a2014-11-03 12:24:19 +0100107func Create(client *gophercloud.ServiceClient, loadBalancerID int, opts CreateOptsBuilder) CreateResult {
108 var res CreateResult
109
110 reqBody, err := opts.ToNodeCreateMap()
111 if err != nil {
112 res.Err = err
113 return res
114 }
115
116 resp, err := perigee.Request("POST", rootURL(client, loadBalancerID), perigee.Options{
117 MoreHeaders: client.AuthenticatedHeaders(),
118 ReqBody: &reqBody,
119 Results: &res.Body,
Jamie Hannaford703527e2014-11-05 12:38:15 +0100120 OkCodes: []int{202},
Jamie Hannaforded8b89a2014-11-03 12:24:19 +0100121 })
122 if err != nil {
123 res.Err = err
124 return res
125 }
126
Ash Wilson9716ec32015-02-12 15:10:48 -0500127 pr, err := pagination.PageResultFrom(&resp.HttpResponse)
Jamie Hannaforded8b89a2014-11-03 12:24:19 +0100128 if err != nil {
129 res.Err = err
130 return res
131 }
132
133 return CreateResult{pagination.SinglePageBase(pr)}
Jamie Hannaford3cfa00a2014-11-03 11:16:35 +0100134}
Jamie Hannaford16bebfc2014-11-03 12:52:30 +0100135
Jamie Hannaford8cdaa802014-11-03 15:11:39 +0100136// BulkDelete is the operation responsible for batch deleting multiple nodes in
137// a single operation. It accepts a slice of integer IDs and will remove them
138// from the load balancer. The maximum limit is 10 node removals at once.
Jamie Hannaford16bebfc2014-11-03 12:52:30 +0100139func BulkDelete(c *gophercloud.ServiceClient, loadBalancerID int, nodeIDs []int) DeleteResult {
140 var res DeleteResult
141
Jamie Hannaford0a9a6be2014-11-03 12:55:38 +0100142 if len(nodeIDs) > 10 || len(nodeIDs) == 0 {
143 res.Err = errors.New("You must provide a minimum of 1 and a maximum of 10 node IDs")
144 return res
145 }
146
Jamie Hannaford16bebfc2014-11-03 12:52:30 +0100147 url := rootURL(c, loadBalancerID)
Jamie Hannaford950561c2014-11-12 11:12:20 +0100148 url += gophercloud.IDSliceToQueryString("id", nodeIDs)
Jamie Hannaford16bebfc2014-11-03 12:52:30 +0100149
150 _, res.Err = perigee.Request("DELETE", url, perigee.Options{
151 MoreHeaders: c.AuthenticatedHeaders(),
152 OkCodes: []int{202},
153 })
154
155 return res
156}
Jamie Hannaford51175a02014-11-03 13:29:44 +0100157
Jamie Hannaford8cdaa802014-11-03 15:11:39 +0100158// Get is the operation responsible for showing details for a single node.
Jamie Hannaford51175a02014-11-03 13:29:44 +0100159func Get(c *gophercloud.ServiceClient, lbID, nodeID int) GetResult {
160 var res GetResult
161
162 _, res.Err = perigee.Request("GET", resourceURL(c, lbID, nodeID), perigee.Options{
163 MoreHeaders: c.AuthenticatedHeaders(),
164 Results: &res.Body,
165 OkCodes: []int{200},
166 })
167
168 return res
169}
Jamie Hannaford00222d72014-11-03 13:58:52 +0100170
Jamie Hannaford8cdaa802014-11-03 15:11:39 +0100171// UpdateOptsBuilder represents a type that can be converted into a JSON-like
172// map structure.
Jamie Hannaford00222d72014-11-03 13:58:52 +0100173type UpdateOptsBuilder interface {
174 ToNodeUpdateMap() (map[string]interface{}, error)
175}
176
Jamie Hannaford8cdaa802014-11-03 15:11:39 +0100177// UpdateOpts represent the options for updating an existing node.
Jamie Hannaford00222d72014-11-03 13:58:52 +0100178type UpdateOpts struct {
Jamie Hannaford8cdaa802014-11-03 15:11:39 +0100179 // Optional - the condition of the node. See the consts in Results.go.
Jamie Hannaford00222d72014-11-03 13:58:52 +0100180 Condition Condition
Jamie Hannaford8cdaa802014-11-03 15:11:39 +0100181
182 // Optional - the type of the node. See the consts in Results.go.
183 Type Type
184
185 // Optional - a pointer to an integer between 0 and 100.
186 Weight *int
Jamie Hannaford00222d72014-11-03 13:58:52 +0100187}
188
Jamie Hannaford8cdaa802014-11-03 15:11:39 +0100189// ToNodeUpdateMap converts an options struct into a JSON-like map.
Jamie Hannaford00222d72014-11-03 13:58:52 +0100190func (opts UpdateOpts) ToNodeUpdateMap() (map[string]interface{}, error) {
191 node := make(map[string]interface{})
192
Jamie Hannaford00222d72014-11-03 13:58:52 +0100193 if opts.Condition != "" {
194 node["condition"] = opts.Condition
195 }
196 if opts.Weight != nil {
197 if weightErr := validateWeight(opts.Weight); weightErr != nil {
198 return node, weightErr
199 }
200 node["weight"] = &opts.Weight
201 }
202 if opts.Type != "" {
203 node["type"] = opts.Type
204 }
205
206 return map[string]interface{}{"node": node}, nil
207}
208
Jamie Hannaford8cdaa802014-11-03 15:11:39 +0100209// Update is the operation responsible for updating an existing node. A node's
210// IP, port, and status are immutable attributes and cannot be modified.
Jamie Hannaford00222d72014-11-03 13:58:52 +0100211func Update(c *gophercloud.ServiceClient, lbID, nodeID int, opts UpdateOptsBuilder) UpdateResult {
212 var res UpdateResult
213
214 reqBody, err := opts.ToNodeUpdateMap()
215 if err != nil {
216 res.Err = err
217 return res
218 }
219
220 _, res.Err = perigee.Request("PUT", resourceURL(c, lbID, nodeID), perigee.Options{
221 MoreHeaders: c.AuthenticatedHeaders(),
222 ReqBody: &reqBody,
223 OkCodes: []int{202},
224 })
225
226 return res
227}
Jamie Hannaford9f4870f2014-11-03 14:03:16 +0100228
Jamie Hannaford8cdaa802014-11-03 15:11:39 +0100229// Delete is the operation responsible for permanently deleting a node.
Jamie Hannaford9f4870f2014-11-03 14:03:16 +0100230func Delete(c *gophercloud.ServiceClient, lbID, nodeID int) DeleteResult {
231 var res DeleteResult
Jamie Hannaford9f4870f2014-11-03 14:03:16 +0100232 _, res.Err = perigee.Request("DELETE", resourceURL(c, lbID, nodeID), perigee.Options{
233 MoreHeaders: c.AuthenticatedHeaders(),
Jamie Hannaford703527e2014-11-05 12:38:15 +0100234 OkCodes: []int{202},
Jamie Hannaford9f4870f2014-11-03 14:03:16 +0100235 })
Jamie Hannaford9f4870f2014-11-03 14:03:16 +0100236 return res
237}
Jamie Hannaford1fac9dd2014-11-03 14:22:40 +0100238
Jamie Hannaford8cdaa802014-11-03 15:11:39 +0100239// ListEventsOptsBuilder allows extensions to add additional parameters to the
240// List request.
241type ListEventsOptsBuilder interface {
242 ToEventsListQuery() (string, error)
243}
244
245// ListEventsOpts allows the filtering and sorting of paginated collections through
246// the API.
247type ListEventsOpts struct {
248 Marker string `q:"marker"`
249 Limit int `q:"limit"`
250}
251
252// ToEventsListQuery formats a ListOpts into a query string.
253func (opts ListEventsOpts) ToEventsListQuery() (string, error) {
254 q, err := gophercloud.BuildQueryString(opts)
255 if err != nil {
256 return "", err
257 }
258 return q.String(), nil
259}
260
261// ListEvents is the operation responsible for listing all the events
262// associated with the activity between the node and the load balancer. The
263// events report errors found with the node. The detailedMessage provides the
264// detailed reason for the error.
Jamie Hannaford703527e2014-11-05 12:38:15 +0100265func ListEvents(client *gophercloud.ServiceClient, loadBalancerID int, opts ListEventsOptsBuilder) pagination.Pager {
266 url := eventsURL(client, loadBalancerID)
Jamie Hannaford8cdaa802014-11-03 15:11:39 +0100267
268 if opts != nil {
269 query, err := opts.ToEventsListQuery()
270 if err != nil {
271 return pagination.Pager{Err: err}
272 }
273 url += query
274 }
275
Jamie Hannaford1fac9dd2014-11-03 14:22:40 +0100276 return pagination.NewPager(client, url, func(r pagination.PageResult) pagination.Page {
277 return NodeEventPage{pagination.SinglePageBase(r)}
278 })
279}