| giolekva | d5a58c4 | 2020-05-07 22:18:35 +0400 | [diff] [blame] | 1 | package main |
| 2 | |
| 3 | import ( |
| giolekva | 6d46459 | 2020-05-13 20:12:18 +0400 | [diff] [blame] | 4 | "bytes" |
| giolekva | ebea582 | 2020-05-12 20:52:19 +0400 | [diff] [blame] | 5 | "context" |
| giolekva | 6d46459 | 2020-05-13 20:12:18 +0400 | [diff] [blame] | 6 | "encoding/json" |
| giolekva | a4a153b | 2020-05-12 11:49:53 +0400 | [diff] [blame] | 7 | "flag" |
| 8 | "fmt" |
| 9 | "io" |
| giolekva | b1f19ee | 2020-05-16 11:31:20 +0400 | [diff] [blame] | 10 | "io/ioutil" |
| giolekva | a4a153b | 2020-05-12 11:49:53 +0400 | [diff] [blame] | 11 | "log" |
| 12 | "net/http" |
| 13 | "os" |
| giolekva | ebea582 | 2020-05-12 20:52:19 +0400 | [diff] [blame] | 14 | "syscall" |
| giolekva | d5a58c4 | 2020-05-07 22:18:35 +0400 | [diff] [blame] | 15 | |
| giolekva | fe0765f | 2020-05-12 14:09:09 +0400 | [diff] [blame] | 16 | "github.com/golang/glog" |
| giolekva | ebea582 | 2020-05-12 20:52:19 +0400 | [diff] [blame] | 17 | "github.com/google/uuid" |
| 18 | apiv1 "k8s.io/api/core/v1" |
| 19 | metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" |
| 20 | "k8s.io/client-go/kubernetes" |
| 21 | corev1 "k8s.io/client-go/kubernetes/typed/core/v1" |
| 22 | "k8s.io/client-go/rest" |
| 23 | "k8s.io/client-go/tools/clientcmd" |
| giolekva | a4a153b | 2020-05-12 11:49:53 +0400 | [diff] [blame] | 24 | |
| 25 | app "github.com/giolekva/pcloud/appmanager" |
| giolekva | d5a58c4 | 2020-05-07 22:18:35 +0400 | [diff] [blame] | 26 | ) |
| 27 | |
| giolekva | ebea582 | 2020-05-12 20:52:19 +0400 | [diff] [blame] | 28 | var kubeconfig = flag.String("kubeconfig", "", "Absolute path to the kubeconfig file.") |
| giolekva | b22550e | 2020-05-12 22:09:03 +0400 | [diff] [blame] | 29 | var helmBin = flag.String("helm_bin", "/usr/local/bin/helm", "Path to the Helm binary.") |
| giolekva | a4a153b | 2020-05-12 11:49:53 +0400 | [diff] [blame] | 30 | var port = flag.Int("port", 1234, "Port to listen on.") |
| giolekva | fe0765f | 2020-05-12 14:09:09 +0400 | [diff] [blame] | 31 | var apiAddr = flag.String("api_addr", "", "PCloud API service address.") |
| giolekva | 6d46459 | 2020-05-13 20:12:18 +0400 | [diff] [blame] | 32 | var managerStoreFile = flag.String("manager_store_file", "", "Persistent file containing installed application information.") |
| giolekva | a4a153b | 2020-05-12 11:49:53 +0400 | [diff] [blame] | 33 | |
| 34 | var helmUploadPage = ` |
| 35 | <html> |
| 36 | <head> |
| 37 | <title>Upload Helm chart</title> |
| 38 | </head> |
| 39 | <body> |
| giolekva | b22550e | 2020-05-12 22:09:03 +0400 | [diff] [blame] | 40 | <form enctype="multipart/form-data" method="post"> |
| giolekva | a4a153b | 2020-05-12 11:49:53 +0400 | [diff] [blame] | 41 | <input type="file" name="chartfile" /> |
| 42 | <input type="submit" value="upload" /> |
| 43 | </form> |
| 44 | </body> |
| 45 | </html> |
| 46 | ` |
| 47 | |
| giolekva | ebea582 | 2020-05-12 20:52:19 +0400 | [diff] [blame] | 48 | type handler struct { |
| giolekva | b1f19ee | 2020-05-16 11:31:20 +0400 | [diff] [blame] | 49 | client *kubernetes.Clientset |
| giolekva | 6d46459 | 2020-05-13 20:12:18 +0400 | [diff] [blame] | 50 | manager *app.Manager |
| giolekva | b1f19ee | 2020-05-16 11:31:20 +0400 | [diff] [blame] | 51 | launcher app.Launcher |
| giolekva | ebea582 | 2020-05-12 20:52:19 +0400 | [diff] [blame] | 52 | } |
| 53 | |
| giolekva | 6d46459 | 2020-05-13 20:12:18 +0400 | [diff] [blame] | 54 | func (hn *handler) handleInstall(w http.ResponseWriter, r *http.Request) { |
| giolekva | a4a153b | 2020-05-12 11:49:53 +0400 | [diff] [blame] | 55 | if r.Method == "GET" { |
| 56 | _, err := io.WriteString(w, helmUploadPage) |
| 57 | if err != nil { |
| 58 | http.Error(w, err.Error(), http.StatusInternalServerError) |
| 59 | } |
| 60 | } else if r.Method == "POST" { |
| 61 | r.ParseMultipartForm(1000000) |
| 62 | file, handler, err := r.FormFile("chartfile") |
| 63 | if err != nil { |
| 64 | http.Error(w, err.Error(), http.StatusInternalServerError) |
| 65 | return |
| 66 | } |
| 67 | defer file.Close() |
| giolekva | ebea582 | 2020-05-12 20:52:19 +0400 | [diff] [blame] | 68 | tmp := uuid.New().String() |
| 69 | if tmp == "" { |
| 70 | http.Error(w, "Could not generate temp dir", http.StatusInternalServerError) |
| 71 | return |
| 72 | } |
| 73 | p := "/tmp/" + tmp |
| 74 | // TODO(giolekva): defer rmdir |
| 75 | if err := syscall.Mkdir(p, 0777); err != nil { |
| 76 | http.Error(w, "Could not create temp dir", http.StatusInternalServerError) |
| 77 | return |
| 78 | } |
| 79 | p += "/" + handler.Filename |
| giolekva | a4a153b | 2020-05-12 11:49:53 +0400 | [diff] [blame] | 80 | f, err := os.OpenFile(p, os.O_WRONLY|os.O_CREATE, 0666) |
| 81 | if err != nil { |
| 82 | fmt.Println(err) |
| 83 | return |
| 84 | } |
| 85 | defer f.Close() |
| 86 | _, err = io.Copy(f, file) |
| 87 | if err != nil { |
| 88 | http.Error(w, err.Error(), http.StatusInternalServerError) |
| 89 | return |
| 90 | } |
| giolekva | 6d46459 | 2020-05-13 20:12:18 +0400 | [diff] [blame] | 91 | if err = hn.installHelmChart(p); err != nil { |
| giolekva | a4a153b | 2020-05-12 11:49:53 +0400 | [diff] [blame] | 92 | http.Error(w, err.Error(), http.StatusInternalServerError) |
| 93 | return |
| 94 | } |
| 95 | w.Write([]byte("Installed")) |
| giolekva | d5a58c4 | 2020-05-07 22:18:35 +0400 | [diff] [blame] | 96 | } |
| giolekva | a4a153b | 2020-05-12 11:49:53 +0400 | [diff] [blame] | 97 | } |
| 98 | |
| giolekva | 6d46459 | 2020-05-13 20:12:18 +0400 | [diff] [blame] | 99 | type trigger struct { |
| giolekva | d6cbd8f | 2020-05-16 13:52:01 +0400 | [diff] [blame] | 100 | App string `json:"app"` |
| 101 | Action string `json:"action"` |
| giolekva | 6d46459 | 2020-05-13 20:12:18 +0400 | [diff] [blame] | 102 | } |
| 103 | |
| 104 | func (hn *handler) handleTriggers(w http.ResponseWriter, r *http.Request) { |
| 105 | if r.Method != "GET" { |
| 106 | http.Error(w, "Only GET method is supported on /triggers", http.StatusBadRequest) |
| 107 | return |
| 108 | } |
| 109 | if err := r.ParseForm(); err != nil { |
| 110 | http.Error(w, err.Error(), http.StatusBadRequest) |
| 111 | return |
| 112 | } |
| 113 | // TODO(giolekva): check if exists |
| 114 | triggerOnType := r.Form["trigger_on_type"][0] |
| 115 | triggerOnEvent := r.Form["trigger_on_event"][0] |
| 116 | var triggers []trigger |
| 117 | for _, a := range hn.manager.Apps { |
| giolekva | 6d46459 | 2020-05-13 20:12:18 +0400 | [diff] [blame] | 118 | for _, t := range a.Triggers.Triggers { |
| 119 | if t.TriggerOn.Type == triggerOnType && t.TriggerOn.Event == triggerOnEvent { |
| giolekva | d6cbd8f | 2020-05-16 13:52:01 +0400 | [diff] [blame] | 120 | triggers = append(triggers, trigger{a.Name, t.Action}) |
| giolekva | 6d46459 | 2020-05-13 20:12:18 +0400 | [diff] [blame] | 121 | } |
| 122 | } |
| 123 | } |
| 124 | respBody, err := json.Marshal(triggers) |
| 125 | if err != nil { |
| 126 | http.Error(w, err.Error(), http.StatusInternalServerError) |
| 127 | return |
| 128 | } |
| 129 | if _, err := io.Copy(w, bytes.NewReader(respBody)); err != nil { |
| 130 | http.Error(w, err.Error(), http.StatusInternalServerError) |
| 131 | return |
| 132 | } |
| 133 | } |
| 134 | |
| giolekva | b1f19ee | 2020-05-16 11:31:20 +0400 | [diff] [blame] | 135 | type actionReq struct { |
| 136 | App string `json:"app"` |
| 137 | Action string `json:"action"` |
| 138 | Args map[string]interface{} `json:"args"` |
| 139 | } |
| 140 | |
| 141 | func (hn *handler) handleLaunchAction(w http.ResponseWriter, r *http.Request) { |
| 142 | actionStr, err := ioutil.ReadAll(r.Body) |
| 143 | if err != nil { |
| 144 | http.Error(w, err.Error(), http.StatusInternalServerError) |
| 145 | return |
| 146 | } |
| 147 | var req actionReq |
| 148 | if err := json.Unmarshal(actionStr, &req); err != nil { |
| 149 | http.Error(w, err.Error(), http.StatusBadRequest) |
| 150 | return |
| 151 | } |
| giolekva | 1c0372c | 2020-05-16 21:18:59 +0400 | [diff] [blame] | 152 | if err := hn.launchAction(req); err != nil { |
| 153 | http.Error(w, err.Error(), http.StatusBadRequest) |
| 154 | return |
| 155 | } |
| 156 | } |
| 157 | |
| 158 | func (hn *handler) launchAction(req actionReq) error { |
| giolekva | b1f19ee | 2020-05-16 11:31:20 +0400 | [diff] [blame] | 159 | for _, a := range hn.manager.Apps { |
| 160 | if a.Name != req.App { |
| 161 | continue |
| 162 | } |
| 163 | for _, action := range a.Actions.Actions { |
| giolekva | 1c0372c | 2020-05-16 21:18:59 +0400 | [diff] [blame] | 164 | if action.Name == req.Action { |
| 165 | return hn.launcher.Launch(a.Namespace, action.Template, req.Args) |
| giolekva | b1f19ee | 2020-05-16 11:31:20 +0400 | [diff] [blame] | 166 | } |
| giolekva | b1f19ee | 2020-05-16 11:31:20 +0400 | [diff] [blame] | 167 | } |
| 168 | } |
| giolekva | 1c0372c | 2020-05-16 21:18:59 +0400 | [diff] [blame] | 169 | return fmt.Errorf("Action not found: %s %s", req.App, req.Action) |
| giolekva | b1f19ee | 2020-05-16 11:31:20 +0400 | [diff] [blame] | 170 | } |
| 171 | |
| giolekva | 6d46459 | 2020-05-13 20:12:18 +0400 | [diff] [blame] | 172 | func (hn *handler) installHelmChart(path string) error { |
| giolekva | ebea582 | 2020-05-12 20:52:19 +0400 | [diff] [blame] | 173 | h, err := app.HelmChartFromTar(path) |
| giolekva | a4a153b | 2020-05-12 11:49:53 +0400 | [diff] [blame] | 174 | if err != nil { |
| 175 | return err |
| 176 | } |
| giolekva | daa8559 | 2020-05-16 23:32:02 +0400 | [diff] [blame] | 177 | if err := h.Render( |
| 178 | *helmBin, |
| 179 | map[string]string{}); err != nil { |
| 180 | return err |
| 181 | } |
| 182 | glog.Info("Rendered templates") |
| giolekva | ebea582 | 2020-05-12 20:52:19 +0400 | [diff] [blame] | 183 | if err = app.InstallSchema(h.Schema, *apiAddr); err != nil { |
| giolekva | fe0765f | 2020-05-12 14:09:09 +0400 | [diff] [blame] | 184 | return err |
| 185 | } |
| 186 | glog.Infof("Installed schema: %s", h.Schema) |
| giolekva | daa8559 | 2020-05-16 23:32:02 +0400 | [diff] [blame] | 187 | err = createNamespace(hn.client.CoreV1().Namespaces(), h.Namespace) |
| giolekva | b1f19ee | 2020-05-16 11:31:20 +0400 | [diff] [blame] | 188 | if err != nil { |
| giolekva | ebea582 | 2020-05-12 20:52:19 +0400 | [diff] [blame] | 189 | return err |
| 190 | } |
| giolekva | daa8559 | 2020-05-16 23:32:02 +0400 | [diff] [blame] | 191 | glog.Infof("Created namespaces: %s", h.Namespace) |
| giolekva | 30036e7 | 2020-05-13 22:00:40 +0400 | [diff] [blame] | 192 | if h.Type == "application" { |
| giolekva | daa8559 | 2020-05-16 23:32:02 +0400 | [diff] [blame] | 193 | if err := h.Install(*helmBin); err != nil { |
| giolekva | 30036e7 | 2020-05-13 22:00:40 +0400 | [diff] [blame] | 194 | return err |
| 195 | } |
| 196 | glog.Info("Deployed") |
| 197 | } else { |
| 198 | glog.Info("Skipping deployment as we got library chart.") |
| giolekva | ebea582 | 2020-05-12 20:52:19 +0400 | [diff] [blame] | 199 | } |
| giolekva | daa8559 | 2020-05-16 23:32:02 +0400 | [diff] [blame] | 200 | hn.manager.Apps[h.Name] = app.App{h.Name, h.Namespace, h.Triggers, h.Actions} |
| giolekva | 6d46459 | 2020-05-13 20:12:18 +0400 | [diff] [blame] | 201 | app.StoreManagerStateToFile(hn.manager, *managerStoreFile) |
| giolekva | 1c0372c | 2020-05-16 21:18:59 +0400 | [diff] [blame] | 202 | for _, a := range h.Init.PostInstall.CallAction { |
| 203 | if err := hn.launchAction(actionReq{a.App, a.Action, a.Args}); err != nil { |
| 204 | return err |
| 205 | } |
| 206 | } |
| giolekva | ebea582 | 2020-05-12 20:52:19 +0400 | [diff] [blame] | 207 | glog.Info("Installed") |
| 208 | return nil |
| 209 | } |
| 210 | |
| 211 | func createNamespace(nsClient corev1.NamespaceInterface, name string) error { |
| 212 | _, err := nsClient.Create( |
| 213 | context.TODO(), |
| 214 | &apiv1.Namespace{ |
| 215 | ObjectMeta: metav1.ObjectMeta{ |
| 216 | Name: name}}, |
| 217 | metav1.CreateOptions{}) |
| giolekva | a4a153b | 2020-05-12 11:49:53 +0400 | [diff] [blame] | 218 | return err |
| 219 | } |
| 220 | |
| giolekva | ebea582 | 2020-05-12 20:52:19 +0400 | [diff] [blame] | 221 | func getKubeConfig() (*rest.Config, error) { |
| 222 | if *kubeconfig != "" { |
| 223 | return clientcmd.BuildConfigFromFlags("", *kubeconfig) |
| 224 | } else { |
| 225 | return rest.InClusterConfig() |
| 226 | } |
| 227 | } |
| 228 | |
| giolekva | a4a153b | 2020-05-12 11:49:53 +0400 | [diff] [blame] | 229 | func main() { |
| 230 | flag.Parse() |
| giolekva | ebea582 | 2020-05-12 20:52:19 +0400 | [diff] [blame] | 231 | config, err := getKubeConfig() |
| 232 | if err != nil { |
| 233 | glog.Fatalf("Could not initialize Kubeconfig: %v", err) |
| 234 | } |
| 235 | clientset, err := kubernetes.NewForConfig(config) |
| 236 | if err != nil { |
| 237 | glog.Fatalf("Could not create Kubernetes API client: %v", err) |
| 238 | } |
| giolekva | 6d46459 | 2020-05-13 20:12:18 +0400 | [diff] [blame] | 239 | manager, err := app.LoadManagerStateFromFile(*managerStoreFile) |
| 240 | if err != nil { |
| 241 | glog.Fatalf("Could ot initialize manager: %v", err) |
| 242 | } |
| giolekva | d6cbd8f | 2020-05-16 13:52:01 +0400 | [diff] [blame] | 243 | glog.Info(manager) |
| giolekva | b1f19ee | 2020-05-16 11:31:20 +0400 | [diff] [blame] | 244 | h := handler{clientset, manager, app.NewK8sLauncher(clientset)} |
| giolekva | 6d46459 | 2020-05-13 20:12:18 +0400 | [diff] [blame] | 245 | http.HandleFunc("/triggers", h.handleTriggers) |
| giolekva | b1f19ee | 2020-05-16 11:31:20 +0400 | [diff] [blame] | 246 | http.HandleFunc("/launch_action", h.handleLaunchAction) |
| giolekva | 6d46459 | 2020-05-13 20:12:18 +0400 | [diff] [blame] | 247 | http.HandleFunc("/", h.handleInstall) |
| giolekva | a4a153b | 2020-05-12 11:49:53 +0400 | [diff] [blame] | 248 | log.Fatal(http.ListenAndServe(fmt.Sprintf(":%d", *port), nil)) |
| 249 | |
| giolekva | d5a58c4 | 2020-05-07 22:18:35 +0400 | [diff] [blame] | 250 | } |