/* Copyright 2015 The Kubernetes Authors All rights reserved. Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License. */ package main import ( "encoding/json" "flag" "fmt" "io" "io/ioutil" "log" "net" "net/http" "net/url" "os" "regexp" "strings" "github.com/ghodss/yaml" "github.com/gorilla/mux" "github.com/kubernetes/deployment-manager/common" "github.com/kubernetes/deployment-manager/manager/manager" "github.com/kubernetes/deployment-manager/manager/repository" "github.com/kubernetes/deployment-manager/manager/repository/persistent" "github.com/kubernetes/deployment-manager/manager/repository/transient" "github.com/kubernetes/deployment-manager/registry" "github.com/kubernetes/deployment-manager/util" ) var deployments = []Route{ {"ListDeployments", "/deployments", "GET", listDeploymentsHandlerFunc, ""}, {"GetDeployment", "/deployments/{deployment}", "GET", getDeploymentHandlerFunc, ""}, {"CreateDeployment", "/deployments", "POST", createDeploymentHandlerFunc, "JSON"}, {"DeleteDeployment", "/deployments/{deployment}", "DELETE", deleteDeploymentHandlerFunc, ""}, {"PutDeployment", "/deployments/{deployment}", "PUT", putDeploymentHandlerFunc, "JSON"}, {"ListManifests", "/deployments/{deployment}/manifests", "GET", listManifestsHandlerFunc, ""}, {"GetManifest", "/deployments/{deployment}/manifests/{manifest}", "GET", getManifestHandlerFunc, ""}, {"Expand", "/expand", "POST", expandHandlerFunc, ""}, {"ListTypes", "/types", "GET", listTypesHandlerFunc, ""}, {"ListTypeInstances", "/types/{type}/instances", "GET", listTypeInstancesHandlerFunc, ""}, {"GetRegistryForType", "/types/{type}/registry", "GET", getRegistryForTypeHandlerFunc, ""}, {"GetMetadataForType", "/types/{type}/metadata", "GET", getMetadataForTypeHandlerFunc, ""}, {"ListRegistries", "/registries", "GET", listRegistriesHandlerFunc, ""}, {"GetRegistry", "/registries/{registry}", "GET", getRegistryHandlerFunc, ""}, {"CreateRegistry", "/registries/{registry}", "POST", createRegistryHandlerFunc, "JSON"}, {"ListRegistryTypes", "/registries/{registry}/types", "GET", listRegistryTypesHandlerFunc, ""}, {"GetDownloadURLs", "/registries/{registry}/types/{type}", "GET", getDownloadURLsHandlerFunc, ""}, {"GetFile", "/registries/{registry}/download", "GET", getFileHandlerFunc, ""}, {"CreateCredential", "/credentials/{credential}", "POST", createCredentialHandlerFunc, "JSON"}, {"GetCredential", "/credentials/{credential}", "GET", getCredentialHandlerFunc, ""}, } var ( maxLength = flag.Int64("maxLength", 1024, "The maximum length (KB) of a template.") expanderName = flag.String("expander", "expandybird-service", "The DNS name of the expander service.") expanderURL = flag.String("expanderURL", "", "The URL for the expander service.") deployerName = flag.String("deployer", "resourcifier-service", "The DNS name of the deployer service.") deployerURL = flag.String("deployerURL", "", "The URL for the deployer service.") credentialFile = flag.String("credentialFile", "", "Local file to use for credentials.") credentialSecrets = flag.Bool("credentialSecrets", true, "Use secrets for credentials.") mongoName = flag.String("mongoName", "mongodb", "The DNS name of the mongodb service.") mongoPort = flag.String("mongoPort", "27017", "The port of the mongodb service.") mongoAddress = flag.String("mongoAddress", "mongodb:27017", "The address of the mongodb service.") ) var backend manager.Manager func init() { if !flag.Parsed() { flag.Parse() } routes = append(routes, deployments...) var credentialProvider common.CredentialProvider if *credentialFile != "" { if *credentialSecrets { panic(fmt.Errorf("Both credentialFile and credentialSecrets are set")) } var err error credentialProvider, err = registry.NewFilebasedCredentialProvider(*credentialFile) if err != nil { panic(fmt.Errorf("cannot create credential provider %s: %s", *credentialFile, err)) } } else if *credentialSecrets { credentialProvider = registry.NewSecretsCredentialProvider() } else { credentialProvider = registry.NewInmemCredentialProvider() } backend = newManager(credentialProvider) } const expanderPort = "8080" const deployerPort = "8080" func newManager(cp common.CredentialProvider) manager.Manager { service := registry.NewInmemRegistryService() registryProvider := registry.NewDefaultRegistryProvider(cp, service) resolver := manager.NewTypeResolver(registryProvider, util.DefaultHTTPClient()) expander := manager.NewExpander(getServiceURL(*expanderURL, *expanderName, expanderPort), resolver) deployer := manager.NewDeployer(getServiceURL(*deployerURL, *deployerName, deployerPort)) address := strings.TrimPrefix(getServiceURL(*mongoAddress, *mongoName, *mongoPort), "http://") repository := createRepository(address) return manager.NewManager(expander, deployer, repository, registryProvider, service, cp) } func createRepository(address string) repository.Repository { r, err := persistent.NewRepository(address) if err != nil { r = transient.NewRepository() } return r } func getServiceURL(serviceURL, serviceName, servicePort string) string { if serviceURL == "" { serviceURL = makeEnvVariableURL(serviceName) if serviceURL == "" { addrs, err := net.LookupHost(serviceName) if err != nil || len(addrs) < 1 { log.Fatalf("cannot resolve service:%v. environment:%v\n", serviceName, os.Environ()) } serviceURL = fmt.Sprintf("http://%s:%s", addrs[0], servicePort) } } return serviceURL } // makeEnvVariableURL takes a service name and returns the value of the // environment variable that identifies its URL, if it exists, or the empty // string, if it doesn't. func makeEnvVariableURL(str string) string { prefix := makeEnvVariableName(str) url := os.Getenv(prefix + "_PORT") return strings.Replace(url, "tcp", "http", 1) } // makeEnvVariableName is copied from the Kubernetes source, // which is referenced by the documentation for service environment variables. func makeEnvVariableName(str string) string { // TODO: If we simplify to "all names are DNS1123Subdomains" this // will need two tweaks: // 1) Handle leading digits // 2) Handle dots return strings.ToUpper(strings.Replace(str, "-", "_", -1)) } func listDeploymentsHandlerFunc(w http.ResponseWriter, r *http.Request) { handler := "manager: list deployments" util.LogHandlerEntry(handler, r) l, err := backend.ListDeployments() if err != nil { util.LogAndReturnError(handler, http.StatusInternalServerError, err, w) return } var names []string for _, d := range l { names = append(names, d.Name) } util.LogHandlerExitWithJSON(handler, w, names, http.StatusOK) } func getDeploymentHandlerFunc(w http.ResponseWriter, r *http.Request) { handler := "manager: get deployment" util.LogHandlerEntry(handler, r) name, err := getPathVariable(w, r, "deployment", handler) if err != nil { return } d, err := backend.GetDeployment(name) if err != nil { util.LogAndReturnError(handler, http.StatusBadRequest, err, w) return } util.LogHandlerExitWithJSON(handler, w, d, http.StatusOK) } func createDeploymentHandlerFunc(w http.ResponseWriter, r *http.Request) { handler := "manager: create deployment" util.LogHandlerEntry(handler, r) defer r.Body.Close() t := getTemplate(w, r, handler) if t != nil { d, err := backend.CreateDeployment(t) if err != nil { util.LogAndReturnError(handler, http.StatusBadRequest, err, w) return } util.LogHandlerExitWithJSON(handler, w, d, http.StatusCreated) return } } func deleteDeploymentHandlerFunc(w http.ResponseWriter, r *http.Request) { handler := "manager: delete deployment" util.LogHandlerEntry(handler, r) defer r.Body.Close() name, err := getPathVariable(w, r, "deployment", handler) if err != nil { return } d, err := backend.DeleteDeployment(name, true) if err != nil { util.LogAndReturnError(handler, http.StatusBadRequest, err, w) return } util.LogHandlerExitWithJSON(handler, w, d, http.StatusOK) } func putDeploymentHandlerFunc(w http.ResponseWriter, r *http.Request) { handler := "manager: update deployment" util.LogHandlerEntry(handler, r) defer r.Body.Close() name, err := getPathVariable(w, r, "deployment", handler) if err != nil { return } t := getTemplate(w, r, handler) if t != nil { d, err := backend.PutDeployment(name, t) if err != nil { util.LogAndReturnError(handler, http.StatusBadRequest, err, w) return } util.LogHandlerExitWithJSON(handler, w, d, http.StatusCreated) } } func getPathVariable(w http.ResponseWriter, r *http.Request, variable, handler string) (string, error) { vars := mux.Vars(r) escaped, ok := vars[variable] if !ok { e := fmt.Errorf("%s name not found in URL", variable) util.LogAndReturnError(handler, http.StatusBadRequest, e, w) return "", e } unescaped, err := url.QueryUnescape(escaped) if err != nil { e := fmt.Errorf("cannot decode name (%v)", variable) util.LogAndReturnError(handler, http.StatusBadRequest, e, w) return "", e } return unescaped, nil } func getTemplate(w http.ResponseWriter, r *http.Request, handler string) *common.Template { util.LogHandlerEntry(handler, r) j, err := getJSONFromRequest(w, r, handler) if err != nil { return nil } t := &common.Template{} if err := json.Unmarshal(j, t); err != nil { e := fmt.Errorf("%v\n%v", err, string(j)) util.LogAndReturnError(handler, http.StatusBadRequest, e, w) return nil } return t } func listManifestsHandlerFunc(w http.ResponseWriter, r *http.Request) { handler := "manager: list manifests" util.LogHandlerEntry(handler, r) deploymentName, err := getPathVariable(w, r, "deployment", handler) if err != nil { return } m, err := backend.ListManifests(deploymentName) if err != nil { util.LogAndReturnError(handler, http.StatusInternalServerError, err, w) return } var manifestNames []string for _, manifest := range m { manifestNames = append(manifestNames, manifest.Name) } util.LogHandlerExitWithJSON(handler, w, manifestNames, http.StatusOK) } func getManifestHandlerFunc(w http.ResponseWriter, r *http.Request) { handler := "manager: get manifest" util.LogHandlerEntry(handler, r) deploymentName, err := getPathVariable(w, r, "deployment", handler) if err != nil { return } manifestName, err := getPathVariable(w, r, "manifest", handler) if err != nil { return } m, err := backend.GetManifest(deploymentName, manifestName) if err != nil { util.LogAndReturnError(handler, http.StatusBadRequest, err, w) return } util.LogHandlerExitWithJSON(handler, w, m, http.StatusOK) } func expandHandlerFunc(w http.ResponseWriter, r *http.Request) { handler := "manager: expand config" util.LogHandlerEntry(handler, r) defer r.Body.Close() t := getTemplate(w, r, handler) if t != nil { c, err := backend.Expand(t) if err != nil { util.LogAndReturnError(handler, http.StatusBadRequest, err, w) return } util.LogHandlerExitWithJSON(handler, w, c, http.StatusCreated) return } } // Putting Type handlers here for now because deployments.go // currently owns its own Manager backend and doesn't like to share. func listTypesHandlerFunc(w http.ResponseWriter, r *http.Request) { handler := "manager: list types" util.LogHandlerEntry(handler, r) types, err := backend.ListTypes() if err != nil { util.LogAndReturnError(handler, http.StatusBadRequest, err, w) return } util.LogHandlerExitWithJSON(handler, w, types, http.StatusOK) } func listTypeInstancesHandlerFunc(w http.ResponseWriter, r *http.Request) { handler := "manager: list instances" util.LogHandlerEntry(handler, r) typeName, err := getPathVariable(w, r, "type", handler) if err != nil { return } instances, err := backend.ListInstances(typeName) if err != nil { util.LogAndReturnError(handler, http.StatusBadRequest, err, w) return } util.LogHandlerExitWithJSON(handler, w, instances, http.StatusOK) } func getRegistryForTypeHandlerFunc(w http.ResponseWriter, r *http.Request) { handler := "manager: get type registry" util.LogHandlerEntry(handler, r) typeName, err := getPathVariable(w, r, "type", handler) if err != nil { return } registry, err := backend.GetRegistryForType(typeName) if err != nil { util.LogAndReturnError(handler, http.StatusBadRequest, err, w) return } util.LogHandlerExitWithJSON(handler, w, registry, http.StatusOK) } func getMetadataForTypeHandlerFunc(w http.ResponseWriter, r *http.Request) { handler := "manager: get type metadata" util.LogHandlerEntry(handler, r) typeName, err := getPathVariable(w, r, "type", handler) if err != nil { return } metadata, err := backend.GetMetadataForType(typeName) if err != nil { util.LogAndReturnError(handler, http.StatusBadRequest, err, w) return } util.LogHandlerExitWithJSON(handler, w, metadata, http.StatusOK) } // Putting Registry handlers here for now because deployments.go // currently owns its own Manager backend and doesn't like to share. func listRegistriesHandlerFunc(w http.ResponseWriter, r *http.Request) { handler := "manager: list registries" util.LogHandlerEntry(handler, r) registries, err := backend.ListRegistries() if err != nil { return } util.LogHandlerExitWithJSON(handler, w, registries, http.StatusOK) } func getRegistryHandlerFunc(w http.ResponseWriter, r *http.Request) { handler := "manager: get registry" util.LogHandlerEntry(handler, r) registryName, err := getPathVariable(w, r, "registry", handler) if err != nil { return } cr, err := backend.GetRegistry(registryName) if err != nil { util.LogAndReturnError(handler, http.StatusBadRequest, err, w) return } util.LogHandlerExitWithJSON(handler, w, cr, http.StatusOK) } func getRegistry(w http.ResponseWriter, r *http.Request, handler string) *common.Registry { util.LogHandlerEntry(handler, r) j, err := getJSONFromRequest(w, r, handler) if err != nil { return nil } t := &common.Registry{} if err := json.Unmarshal(j, t); err != nil { e := fmt.Errorf("%v\n%v", err, string(j)) util.LogAndReturnError(handler, http.StatusBadRequest, e, w) return nil } return t } func createRegistryHandlerFunc(w http.ResponseWriter, r *http.Request) { handler := "manager: create registry" util.LogHandlerEntry(handler, r) defer r.Body.Close() registryName, err := getPathVariable(w, r, "registry", handler) if err != nil { return } reg := getRegistry(w, r, handler) if reg.Name != registryName { e := fmt.Errorf("Registry name does not match %s != %s", reg.Name, registryName) util.LogAndReturnError(handler, http.StatusBadRequest, e, w) return } if reg != nil { err = backend.CreateRegistry(reg) if err != nil { util.LogAndReturnError(handler, http.StatusBadRequest, err, w) return } } util.LogHandlerExitWithJSON(handler, w, reg, http.StatusOK) } func listRegistryTypesHandlerFunc(w http.ResponseWriter, r *http.Request) { handler := "manager: list registry types" util.LogHandlerEntry(handler, r) registryName, err := getPathVariable(w, r, "registry", handler) if err != nil { return } values, err := url.ParseQuery(r.URL.RawQuery) if err != nil { util.LogAndReturnError(handler, http.StatusBadRequest, err, w) return } var regex *regexp.Regexp regexString := values.Get("regex") if regexString != "" { regex, err = regexp.Compile(regexString) if err != nil { util.LogAndReturnError(handler, http.StatusBadRequest, err, w) return } } registryTypes, err := backend.ListRegistryTypes(registryName, regex) if err != nil { util.LogAndReturnError(handler, http.StatusInternalServerError, err, w) return } util.LogHandlerExitWithJSON(handler, w, registryTypes, http.StatusOK) } func getDownloadURLsHandlerFunc(w http.ResponseWriter, r *http.Request) { handler := "manager: get download URLs" util.LogHandlerEntry(handler, r) registryName, err := getPathVariable(w, r, "registry", handler) if err != nil { return } typeName, err := getPathVariable(w, r, "type", handler) if err != nil { return } tt, err := registry.ParseType(typeName) if err != nil { util.LogAndReturnError(handler, http.StatusInternalServerError, err, w) return } c, err := backend.GetDownloadURLs(registryName, tt) if err != nil { util.LogAndReturnError(handler, http.StatusBadRequest, err, w) return } urls := []string{} for _, u := range c { urls = append(urls, u.String()) } util.LogHandlerExitWithJSON(handler, w, urls, http.StatusOK) } func getFileHandlerFunc(w http.ResponseWriter, r *http.Request) { handler := "manager: get file" util.LogHandlerEntry(handler, r) registryName, err := getPathVariable(w, r, "registry", handler) if err != nil { return } file := r.FormValue("file") if file == "" { return } b, err := backend.GetFile(registryName, file) if err != nil { util.LogAndReturnError(handler, http.StatusBadRequest, err, w) return } util.LogHandlerExitWithJSON(handler, w, b, http.StatusOK) } func getCredential(w http.ResponseWriter, r *http.Request, handler string) *common.RegistryCredential { util.LogHandlerEntry(handler, r) j, err := getJSONFromRequest(w, r, handler) if err != nil { return nil } t := &common.RegistryCredential{} if err := json.Unmarshal(j, t); err != nil { e := fmt.Errorf("%v\n%v", err, string(j)) util.LogAndReturnError(handler, http.StatusBadRequest, e, w) return nil } return t } func createCredentialHandlerFunc(w http.ResponseWriter, r *http.Request) { handler := "manager: create credential" util.LogHandlerEntry(handler, r) defer r.Body.Close() credentialName, err := getPathVariable(w, r, "credential", handler) if err != nil { return } c := getCredential(w, r, handler) if c != nil { err = backend.CreateCredential(credentialName, c) if err != nil { util.LogAndReturnError(handler, http.StatusBadRequest, err, w) return } } util.LogHandlerExitWithJSON(handler, w, c, http.StatusOK) } func getCredentialHandlerFunc(w http.ResponseWriter, r *http.Request) { handler := "manager: get credential" util.LogHandlerEntry(handler, r) credentialName, err := getPathVariable(w, r, "credential", handler) if err != nil { return } c, err := backend.GetCredential(credentialName) if err != nil { util.LogAndReturnError(handler, http.StatusBadRequest, err, w) return } util.LogHandlerExitWithJSON(handler, w, c, http.StatusOK) } func getJSONFromRequest(w http.ResponseWriter, r *http.Request, handler string) ([]byte, error) { util.LogHandlerEntry(handler, r) b := io.LimitReader(r.Body, *maxLength*1024) y, err := ioutil.ReadAll(b) if err != nil { util.LogAndReturnError(handler, http.StatusBadRequest, err, w) return []byte{}, err } // Reject the input if it exceeded the length limit, // since we may not have read all of it into the buffer. if _, err = b.Read(make([]byte, 0, 1)); err != io.EOF { e := fmt.Errorf("template exceeds maximum length of %d KB", *maxLength) util.LogAndReturnError(handler, http.StatusBadRequest, e, w) return []byte{}, err } if err := r.Body.Close(); err != nil { util.LogAndReturnError(handler, http.StatusInternalServerError, err, w) return []byte{}, err } return yaml.YAMLToJSON(y) }