David K. Bainbridge | f0da873 | 2016-06-01 16:15:37 -0700 | [diff] [blame] | 1 | package main |
| 2 | |
| 3 | import ( |
| 4 | "bufio" |
| 5 | "encoding/json" |
| 6 | "github.com/gorilla/mux" |
David K. Bainbridge | 97ee805 | 2016-06-14 00:52:07 -0700 | [diff] [blame] | 7 | "log" |
David K. Bainbridge | f0da873 | 2016-06-01 16:15:37 -0700 | [diff] [blame] | 8 | "net/http" |
| 9 | "strings" |
| 10 | ) |
| 11 | |
| 12 | type RequestInfo struct { |
David K. Bainbridge | 97ee805 | 2016-06-14 00:52:07 -0700 | [diff] [blame] | 13 | Id string `json:"id"` |
| 14 | Name string `json:"name"` |
| 15 | Ip string `json:"ip"` |
| 16 | Mac string `json:"mac"` |
| 17 | RoleSelector string `json:"role_selector"` |
| 18 | Role string `json:"role"` |
| 19 | Script string `json:"script"` |
David K. Bainbridge | f0da873 | 2016-06-01 16:15:37 -0700 | [diff] [blame] | 20 | } |
| 21 | |
| 22 | func (c *Context) GetRole(info *RequestInfo) (string, error) { |
David K. Bainbridge | 97ee805 | 2016-06-14 00:52:07 -0700 | [diff] [blame] | 23 | if info.Role != "" { |
| 24 | return info.Role, nil |
| 25 | } else if c.config.RoleSelectorURL == "" && info.RoleSelector == "" { |
David K. Bainbridge | f0da873 | 2016-06-01 16:15:37 -0700 | [diff] [blame] | 26 | return c.config.DefaultRole, nil |
| 27 | } |
David K. Bainbridge | 97ee805 | 2016-06-14 00:52:07 -0700 | [diff] [blame] | 28 | selector := c.config.RoleSelectorURL |
| 29 | if info.RoleSelector != "" { |
| 30 | selector = info.RoleSelector |
| 31 | } |
| 32 | |
| 33 | r, err := http.Get(selector) |
David K. Bainbridge | f0da873 | 2016-06-01 16:15:37 -0700 | [diff] [blame] | 34 | if err != nil { |
| 35 | return "", err |
| 36 | } |
| 37 | |
| 38 | s := bufio.NewScanner(r.Body) |
| 39 | defer r.Body.Close() |
| 40 | role := strings.TrimSpace(s.Text()) |
| 41 | if role == "" { |
| 42 | return c.config.DefaultRole, nil |
| 43 | } |
| 44 | return role, nil |
| 45 | } |
| 46 | |
| 47 | func (c *Context) validateData(info *RequestInfo) bool { |
| 48 | if strings.TrimSpace(info.Id) == "" || |
| 49 | strings.TrimSpace(info.Name) == "" || |
| 50 | strings.TrimSpace(info.Ip) == "" || |
| 51 | strings.TrimSpace(info.Mac) == "" { |
| 52 | return false |
| 53 | } |
| 54 | return true |
| 55 | } |
| 56 | |
| 57 | func (c *Context) ProvisionRequestHandler(w http.ResponseWriter, r *http.Request) { |
| 58 | var info RequestInfo |
| 59 | decoder := json.NewDecoder(r.Body) |
| 60 | defer r.Body.Close() |
David K. Bainbridge | d86d96d | 2016-06-01 17:28:46 -0700 | [diff] [blame] | 61 | if err := decoder.Decode(&info); err != nil { |
David K. Bainbridge | 3ee7641 | 2016-06-15 18:56:08 -0700 | [diff] [blame] | 62 | log.Printf("[error] Unable to decode request to provision : %s", err) |
David K. Bainbridge | d86d96d | 2016-06-01 17:28:46 -0700 | [diff] [blame] | 63 | http.Error(w, err.Error(), http.StatusBadRequest) |
| 64 | return |
| 65 | } |
David K. Bainbridge | 97ee805 | 2016-06-14 00:52:07 -0700 | [diff] [blame] | 66 | |
David K. Bainbridge | d86d96d | 2016-06-01 17:28:46 -0700 | [diff] [blame] | 67 | if !c.validateData(&info) { |
David K. Bainbridge | 3ee7641 | 2016-06-15 18:56:08 -0700 | [diff] [blame] | 68 | log.Printf("[errpr] Provisioning request not valid for '%s'", info.Name) |
David K. Bainbridge | f0da873 | 2016-06-01 16:15:37 -0700 | [diff] [blame] | 69 | w.WriteHeader(http.StatusBadRequest) |
| 70 | return |
| 71 | } |
| 72 | |
David K. Bainbridge | f0da873 | 2016-06-01 16:15:37 -0700 | [diff] [blame] | 73 | role, err := c.GetRole(&info) |
| 74 | if err != nil { |
David K. Bainbridge | 3ee7641 | 2016-06-15 18:56:08 -0700 | [diff] [blame] | 75 | log.Printf("[error] unable to get provisioning role for node '%s' : %s", info.Name, err) |
David K. Bainbridge | f0da873 | 2016-06-01 16:15:37 -0700 | [diff] [blame] | 76 | http.Error(w, err.Error(), http.StatusInternalServerError) |
| 77 | return |
| 78 | } |
David K. Bainbridge | 3850158 | 2016-06-01 18:15:45 -0700 | [diff] [blame] | 79 | err = c.dispatcher.Dispatch(&info, role, c.config.Script) |
David K. Bainbridge | f0da873 | 2016-06-01 16:15:37 -0700 | [diff] [blame] | 80 | if err != nil { |
David K. Bainbridge | 3ee7641 | 2016-06-15 18:56:08 -0700 | [diff] [blame] | 81 | log.Printf("[errpr] unable to dispatch provisioning request for node '%s' : %s", info.Name, err) |
David K. Bainbridge | f0da873 | 2016-06-01 16:15:37 -0700 | [diff] [blame] | 82 | http.Error(w, err.Error(), http.StatusInternalServerError) |
| 83 | return |
| 84 | } |
| 85 | |
| 86 | w.WriteHeader(http.StatusAccepted) |
| 87 | } |
| 88 | |
| 89 | func (c *Context) ListRequestsHandler(w http.ResponseWriter, r *http.Request) { |
| 90 | list, err := c.storage.List() |
| 91 | bytes, err := json.Marshal(list) |
| 92 | if err != nil { |
| 93 | http.Error(w, err.Error(), http.StatusInternalServerError) |
| 94 | return |
| 95 | } |
| 96 | w.Write(bytes) |
| 97 | } |
| 98 | |
| 99 | func (c *Context) QueryStatusHandler(w http.ResponseWriter, r *http.Request) { |
| 100 | vars := mux.Vars(r) |
| 101 | id, ok := vars["nodeid"] |
| 102 | if !ok || strings.TrimSpace(id) == "" { |
| 103 | w.WriteHeader(http.StatusBadRequest) |
| 104 | return |
| 105 | } |
| 106 | s, err := c.storage.Get(id) |
| 107 | if err != nil { |
David K. Bainbridge | 97ee805 | 2016-06-14 00:52:07 -0700 | [diff] [blame] | 108 | log.Printf("[warn] Error while retrieving status for '%s' from strorage : %s", id, err) |
David K. Bainbridge | f0da873 | 2016-06-01 16:15:37 -0700 | [diff] [blame] | 109 | http.Error(w, err.Error(), http.StatusInternalServerError) |
| 110 | return |
| 111 | } |
| 112 | if s == nil { |
| 113 | w.WriteHeader(http.StatusNotFound) |
| 114 | return |
| 115 | } |
| 116 | bytes, err := json.Marshal(s) |
| 117 | if err != nil { |
David K. Bainbridge | 97ee805 | 2016-06-14 00:52:07 -0700 | [diff] [blame] | 118 | log.Printf("[error] Error while attempting to marshal status for '%s' from storage : %s", id, err) |
David K. Bainbridge | f0da873 | 2016-06-01 16:15:37 -0700 | [diff] [blame] | 119 | http.Error(w, err.Error(), http.StatusInternalServerError) |
| 120 | return |
| 121 | } |
David K. Bainbridge | 8352c59 | 2016-06-02 12:48:37 -0700 | [diff] [blame] | 122 | |
| 123 | switch s.Status { |
| 124 | case Pending, Running: |
| 125 | w.WriteHeader(http.StatusAccepted) |
David K. Bainbridge | 97ee805 | 2016-06-14 00:52:07 -0700 | [diff] [blame] | 126 | case Failed, Complete: |
David K. Bainbridge | 8352c59 | 2016-06-02 12:48:37 -0700 | [diff] [blame] | 127 | w.WriteHeader(http.StatusOK) |
| 128 | default: |
| 129 | w.WriteHeader(http.StatusInternalServerError) |
| 130 | } |
| 131 | |
David K. Bainbridge | f0da873 | 2016-06-01 16:15:37 -0700 | [diff] [blame] | 132 | w.Write(bytes) |
| 133 | } |