Merge pull request #363 from technosophos/feat/rest-refactor

fix(manager): refactor REST API to make it modular
pull/302/head
Adam Reese 9 years ago
commit c00a711b0c

@ -56,7 +56,7 @@ container: all
.PHONY: test-unit .PHONY: test-unit
test-unit: test-unit:
@echo Running tests... @echo Running tests...
go test -v $(GO_PKGS) go test -race -v $(GO_PKGS)
.PHONY: test-flake8 .PHONY: test-flake8
test-flake8: test-flake8:

@ -18,7 +18,7 @@ package main
import ( import (
"encoding/json" "encoding/json"
"flag" "errors"
"fmt" "fmt"
"io" "io"
"io/ioutil" "io/ioutil"
@ -32,21 +32,20 @@ import (
"github.com/ghodss/yaml" "github.com/ghodss/yaml"
"github.com/gorilla/mux" "github.com/gorilla/mux"
"github.com/kubernetes/deployment-manager/cmd/manager/manager" "github.com/kubernetes/deployment-manager/cmd/manager/manager"
"github.com/kubernetes/deployment-manager/cmd/manager/repository" "github.com/kubernetes/deployment-manager/cmd/manager/repository"
"github.com/kubernetes/deployment-manager/cmd/manager/repository/persistent" "github.com/kubernetes/deployment-manager/cmd/manager/repository/persistent"
"github.com/kubernetes/deployment-manager/cmd/manager/repository/transient" "github.com/kubernetes/deployment-manager/cmd/manager/repository/transient"
"github.com/kubernetes/deployment-manager/cmd/manager/router"
"github.com/kubernetes/deployment-manager/pkg/common" "github.com/kubernetes/deployment-manager/pkg/common"
"github.com/kubernetes/deployment-manager/pkg/httputil"
"github.com/kubernetes/deployment-manager/pkg/registry" "github.com/kubernetes/deployment-manager/pkg/registry"
"github.com/kubernetes/deployment-manager/pkg/util" "github.com/kubernetes/deployment-manager/pkg/util"
) )
var deployments = []Route{ var deployments = []Route{
{"ListDeployments", "/deployments", "GET", listDeploymentsHandlerFunc, ""},
{"GetDeployment", "/deployments/{deployment}", "GET", getDeploymentHandlerFunc, ""},
{"CreateDeployment", "/deployments", "POST", createDeploymentHandlerFunc, "JSON"}, {"CreateDeployment", "/deployments", "POST", createDeploymentHandlerFunc, "JSON"},
{"DeleteDeployment", "/deployments/{deployment}", "DELETE", deleteDeploymentHandlerFunc, ""}, {"DeleteDeplyment", "/deployments/{deployment}", "DELETE", deleteDeploymentHandlerFunc, ""},
{"PutDeployment", "/deployments/{deployment}", "PUT", putDeploymentHandlerFunc, "JSON"}, {"PutDeployment", "/deployments/{deployment}", "PUT", putDeploymentHandlerFunc, "JSON"},
{"ListManifests", "/deployments/{deployment}/manifests", "GET", listManifestsHandlerFunc, ""}, {"ListManifests", "/deployments/{deployment}/manifests", "GET", listManifestsHandlerFunc, ""},
{"GetManifest", "/deployments/{deployment}/manifests/{manifest}", "GET", getManifestHandlerFunc, ""}, {"GetManifest", "/deployments/{deployment}/manifests/{manifest}", "GET", getManifestHandlerFunc, ""},
@ -65,57 +64,83 @@ var deployments = []Route{
{"GetCredential", "/credentials/{credential}", "GET", getCredentialHandlerFunc, ""}, {"GetCredential", "/credentials/{credential}", "GET", getCredentialHandlerFunc, ""},
} }
var ( // Deprecated. Use Context.Manager instead.
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 var backend manager.Manager
func init() { // Route defines a routing table entry to be registered with gorilla/mux.
if !flag.Parsed() { //
flag.Parse() // Route is deprecated. Use router.Routes instead.
type Route struct {
Name string
Path string
Methods string
HandlerFunc http.HandlerFunc
Type string
}
func registerRoutes(c *router.Context, h *router.Handler) {
re := regexp.MustCompile("{[a-z]+}")
h.Add("GET /healthz", healthz)
h.Add("GET /deployments", listDeploymentsHandlerFunc)
h.Add("GET /deployments/*", getDeploymentHandlerFunc)
// TODO: Replace these routes with updated ones.
for _, d := range deployments {
path := fmt.Sprintf("%s %s", d.Methods, re.ReplaceAllString(d.Path, "*"))
fmt.Printf("\t%s\n", path)
h.Add(path, func(w http.ResponseWriter, r *http.Request, c *router.Context) error {
d.HandlerFunc(w, r)
return nil
})
} }
}
func healthz(w http.ResponseWriter, r *http.Request, c *router.Context) error {
log.Println("manager: healthz checkpoint")
// TODO: This should check the availability of the repository, and fail if it
// cannot connect.
fmt.Fprintln(w, "OK")
return nil
}
routes = append(routes, deployments...) func setupDependencies(c *router.Context) error {
var credentialProvider common.CredentialProvider var credentialProvider common.CredentialProvider
if *credentialFile != "" { if c.Config.CredentialFile != "" {
if *credentialSecrets { if c.Config.CredentialSecrets {
panic(fmt.Errorf("Both credentialFile and credentialSecrets are set")) return errors.New("Both credentialFile and credentialSecrets are set")
} }
var err error var err error
credentialProvider, err = registry.NewFilebasedCredentialProvider(*credentialFile) credentialProvider, err = registry.NewFilebasedCredentialProvider(c.Config.CredentialFile)
if err != nil { if err != nil {
panic(fmt.Errorf("cannot create credential provider %s: %s", *credentialFile, err)) return fmt.Errorf("cannot create credential provider %s: %s", c.Config.CredentialFile, err)
} }
} else if *credentialSecrets { } else if *credentialSecrets {
credentialProvider = registry.NewSecretsCredentialProvider() credentialProvider = registry.NewSecretsCredentialProvider()
} else { } else {
credentialProvider = registry.NewInmemCredentialProvider() credentialProvider = registry.NewInmemCredentialProvider()
} }
backend = newManager(credentialProvider) c.CredentialProvider = credentialProvider
c.Manager = newManager(c)
// FIXME: As soon as we can, we need to get rid of this.
backend = c.Manager
return nil
} }
const expanderPort = "8080" const expanderPort = "8080"
const deployerPort = "8080" const deployerPort = "8080"
func newManager(cp common.CredentialProvider) manager.Manager { func newManager(c *router.Context) manager.Manager {
cfg := c.Config
service := registry.NewInmemRegistryService() service := registry.NewInmemRegistryService()
registryProvider := registry.NewDefaultRegistryProvider(cp, service) registryProvider := registry.NewDefaultRegistryProvider(c.CredentialProvider, service)
resolver := manager.NewTypeResolver(registryProvider, util.DefaultHTTPClient()) resolver := manager.NewTypeResolver(registryProvider, util.DefaultHTTPClient())
expander := manager.NewExpander(getServiceURL(*expanderURL, *expanderName, expanderPort), resolver) expander := manager.NewExpander(getServiceURL(cfg.ExpanderURL, cfg.ExpanderName, expanderPort), resolver)
deployer := manager.NewDeployer(getServiceURL(*deployerURL, *deployerName, deployerPort)) deployer := manager.NewDeployer(getServiceURL(cfg.DeployerURL, cfg.DeployerName, deployerPort))
address := strings.TrimPrefix(getServiceURL(*mongoAddress, *mongoName, *mongoPort), "http://") address := strings.TrimPrefix(getServiceURL(cfg.MongoAddress, cfg.MongoName, cfg.MongoPort), "http://")
repository := createRepository(address) repository := createRepository(address)
return manager.NewManager(expander, deployer, repository, registryProvider, service, cp) return manager.NewManager(expander, deployer, repository, registryProvider, service, c.CredentialProvider)
} }
func createRepository(address string) repository.Repository { func createRepository(address string) repository.Repository {
@ -162,13 +187,13 @@ func makeEnvVariableName(str string) string {
return strings.ToUpper(strings.Replace(str, "-", "_", -1)) return strings.ToUpper(strings.Replace(str, "-", "_", -1))
} }
func listDeploymentsHandlerFunc(w http.ResponseWriter, r *http.Request) { func listDeploymentsHandlerFunc(w http.ResponseWriter, r *http.Request, c *router.Context) error {
handler := "manager: list deployments" handler := "manager: list deployments"
util.LogHandlerEntry(handler, r) util.LogHandlerEntry(handler, r)
l, err := backend.ListDeployments() l, err := backend.ListDeployments()
if err != nil { if err != nil {
util.LogAndReturnError(handler, http.StatusInternalServerError, err, w) util.LogAndReturnError(handler, http.StatusInternalServerError, err, w)
return return nil
} }
var names []string var names []string
for _, d := range l { for _, d := range l {
@ -176,23 +201,25 @@ func listDeploymentsHandlerFunc(w http.ResponseWriter, r *http.Request) {
} }
util.LogHandlerExitWithJSON(handler, w, names, http.StatusOK) util.LogHandlerExitWithJSON(handler, w, names, http.StatusOK)
return nil
} }
func getDeploymentHandlerFunc(w http.ResponseWriter, r *http.Request) { func getDeploymentHandlerFunc(w http.ResponseWriter, r *http.Request, c *router.Context) error {
handler := "manager: get deployment" handler := "manager: get deployment"
util.LogHandlerEntry(handler, r) util.LogHandlerEntry(handler, r)
name, err := getPathVariable(w, r, "deployment", handler) name, err := getPathVariable(w, r, "deployment", handler)
if err != nil { if err != nil {
return return nil
} }
d, err := backend.GetDeployment(name) d, err := backend.GetDeployment(name)
if err != nil { if err != nil {
util.LogAndReturnError(handler, http.StatusBadRequest, err, w) util.LogAndReturnError(handler, http.StatusBadRequest, err, w)
return return nil
} }
util.LogHandlerExitWithJSON(handler, w, d, http.StatusOK) util.LogHandlerExitWithJSON(handler, w, d, http.StatusOK)
return nil
} }
func createDeploymentHandlerFunc(w http.ResponseWriter, r *http.Request) { func createDeploymentHandlerFunc(w http.ResponseWriter, r *http.Request) {
@ -216,7 +243,7 @@ func deleteDeploymentHandlerFunc(w http.ResponseWriter, r *http.Request) {
handler := "manager: delete deployment" handler := "manager: delete deployment"
util.LogHandlerEntry(handler, r) util.LogHandlerEntry(handler, r)
defer r.Body.Close() defer r.Body.Close()
name, err := getPathVariable(w, r, "deployment", handler) name, err := pos(w, r, 2) //getPathVariable(w, r, "deployment", handler)
if err != nil { if err != nil {
return return
} }
@ -234,7 +261,7 @@ func putDeploymentHandlerFunc(w http.ResponseWriter, r *http.Request) {
handler := "manager: update deployment" handler := "manager: update deployment"
util.LogHandlerEntry(handler, r) util.LogHandlerEntry(handler, r)
defer r.Body.Close() defer r.Body.Close()
name, err := getPathVariable(w, r, "deployment", handler) name, err := pos(w, r, 2) //getPathVariable(w, r, "deployment", handler)
if err != nil { if err != nil {
return return
} }
@ -251,6 +278,15 @@ func putDeploymentHandlerFunc(w http.ResponseWriter, r *http.Request) {
} }
} }
func pos(w http.ResponseWriter, r *http.Request, i int) (string, error) {
parts := strings.Split(r.URL.Path, "/")
if len(parts) < i-1 {
httputil.BadRequest(w, r)
return "", fmt.Errorf("No index for %d", i)
}
return parts[i], nil
}
func getPathVariable(w http.ResponseWriter, r *http.Request, variable, handler string) (string, error) { func getPathVariable(w http.ResponseWriter, r *http.Request, variable, handler string) (string, error) {
vars := mux.Vars(r) vars := mux.Vars(r)
escaped, ok := vars[variable] escaped, ok := vars[variable]
@ -291,7 +327,7 @@ func getTemplate(w http.ResponseWriter, r *http.Request, handler string) *common
func listManifestsHandlerFunc(w http.ResponseWriter, r *http.Request) { func listManifestsHandlerFunc(w http.ResponseWriter, r *http.Request) {
handler := "manager: list manifests" handler := "manager: list manifests"
util.LogHandlerEntry(handler, r) util.LogHandlerEntry(handler, r)
deploymentName, err := getPathVariable(w, r, "deployment", handler) deploymentName, err := pos(w, r, 2)
if err != nil { if err != nil {
return return
} }
@ -313,12 +349,12 @@ func listManifestsHandlerFunc(w http.ResponseWriter, r *http.Request) {
func getManifestHandlerFunc(w http.ResponseWriter, r *http.Request) { func getManifestHandlerFunc(w http.ResponseWriter, r *http.Request) {
handler := "manager: get manifest" handler := "manager: get manifest"
util.LogHandlerEntry(handler, r) util.LogHandlerEntry(handler, r)
deploymentName, err := getPathVariable(w, r, "deployment", handler) deploymentName, err := pos(w, r, 2)
if err != nil { if err != nil {
return return
} }
manifestName, err := getPathVariable(w, r, "manifest", handler) manifestName, err := pos(w, r, 4)
if err != nil { if err != nil {
return return
} }
@ -366,7 +402,7 @@ func listTypesHandlerFunc(w http.ResponseWriter, r *http.Request) {
func listTypeInstancesHandlerFunc(w http.ResponseWriter, r *http.Request) { func listTypeInstancesHandlerFunc(w http.ResponseWriter, r *http.Request) {
handler := "manager: list instances" handler := "manager: list instances"
util.LogHandlerEntry(handler, r) util.LogHandlerEntry(handler, r)
typeName, err := getPathVariable(w, r, "type", handler) typeName, err := pos(w, r, 2)
if err != nil { if err != nil {
return return
} }
@ -383,7 +419,7 @@ func listTypeInstancesHandlerFunc(w http.ResponseWriter, r *http.Request) {
func getRegistryForTypeHandlerFunc(w http.ResponseWriter, r *http.Request) { func getRegistryForTypeHandlerFunc(w http.ResponseWriter, r *http.Request) {
handler := "manager: get type registry" handler := "manager: get type registry"
util.LogHandlerEntry(handler, r) util.LogHandlerEntry(handler, r)
typeName, err := getPathVariable(w, r, "type", handler) typeName, err := pos(w, r, 2)
if err != nil { if err != nil {
return return
} }
@ -400,7 +436,7 @@ func getRegistryForTypeHandlerFunc(w http.ResponseWriter, r *http.Request) {
func getMetadataForTypeHandlerFunc(w http.ResponseWriter, r *http.Request) { func getMetadataForTypeHandlerFunc(w http.ResponseWriter, r *http.Request) {
handler := "manager: get type metadata" handler := "manager: get type metadata"
util.LogHandlerEntry(handler, r) util.LogHandlerEntry(handler, r)
typeName, err := getPathVariable(w, r, "type", handler) typeName, err := pos(w, r, 2)
if err != nil { if err != nil {
return return
} }
@ -430,7 +466,7 @@ func listRegistriesHandlerFunc(w http.ResponseWriter, r *http.Request) {
func getRegistryHandlerFunc(w http.ResponseWriter, r *http.Request) { func getRegistryHandlerFunc(w http.ResponseWriter, r *http.Request) {
handler := "manager: get registry" handler := "manager: get registry"
util.LogHandlerEntry(handler, r) util.LogHandlerEntry(handler, r)
registryName, err := getPathVariable(w, r, "registry", handler) registryName, err := pos(w, r, 2)
if err != nil { if err != nil {
return return
} }
@ -465,7 +501,7 @@ func createRegistryHandlerFunc(w http.ResponseWriter, r *http.Request) {
handler := "manager: create registry" handler := "manager: create registry"
util.LogHandlerEntry(handler, r) util.LogHandlerEntry(handler, r)
defer r.Body.Close() defer r.Body.Close()
registryName, err := getPathVariable(w, r, "registry", handler) registryName, err := pos(w, r, 2)
if err != nil { if err != nil {
return return
} }
@ -490,7 +526,7 @@ func createRegistryHandlerFunc(w http.ResponseWriter, r *http.Request) {
func listRegistryTypesHandlerFunc(w http.ResponseWriter, r *http.Request) { func listRegistryTypesHandlerFunc(w http.ResponseWriter, r *http.Request) {
handler := "manager: list registry types" handler := "manager: list registry types"
util.LogHandlerEntry(handler, r) util.LogHandlerEntry(handler, r)
registryName, err := getPathVariable(w, r, "registry", handler) registryName, err := pos(w, r, 2)
if err != nil { if err != nil {
return return
} }
@ -523,12 +559,12 @@ func listRegistryTypesHandlerFunc(w http.ResponseWriter, r *http.Request) {
func getDownloadURLsHandlerFunc(w http.ResponseWriter, r *http.Request) { func getDownloadURLsHandlerFunc(w http.ResponseWriter, r *http.Request) {
handler := "manager: get download URLs" handler := "manager: get download URLs"
util.LogHandlerEntry(handler, r) util.LogHandlerEntry(handler, r)
registryName, err := getPathVariable(w, r, "registry", handler) registryName, err := pos(w, r, 2)
if err != nil { if err != nil {
return return
} }
typeName, err := getPathVariable(w, r, "type", handler) typeName, err := pos(w, r, 4)
if err != nil { if err != nil {
return return
} }
@ -555,7 +591,7 @@ func getDownloadURLsHandlerFunc(w http.ResponseWriter, r *http.Request) {
func getFileHandlerFunc(w http.ResponseWriter, r *http.Request) { func getFileHandlerFunc(w http.ResponseWriter, r *http.Request) {
handler := "manager: get file" handler := "manager: get file"
util.LogHandlerEntry(handler, r) util.LogHandlerEntry(handler, r)
registryName, err := getPathVariable(w, r, "registry", handler) registryName, err := pos(w, r, 2)
if err != nil { if err != nil {
return return
} }
@ -595,7 +631,7 @@ func createCredentialHandlerFunc(w http.ResponseWriter, r *http.Request) {
handler := "manager: create credential" handler := "manager: create credential"
util.LogHandlerEntry(handler, r) util.LogHandlerEntry(handler, r)
defer r.Body.Close() defer r.Body.Close()
credentialName, err := getPathVariable(w, r, "credential", handler) credentialName, err := pos(w, r, 2)
if err != nil { if err != nil {
return return
} }
@ -615,7 +651,7 @@ func createCredentialHandlerFunc(w http.ResponseWriter, r *http.Request) {
func getCredentialHandlerFunc(w http.ResponseWriter, r *http.Request) { func getCredentialHandlerFunc(w http.ResponseWriter, r *http.Request) {
handler := "manager: get credential" handler := "manager: get credential"
util.LogHandlerEntry(handler, r) util.LogHandlerEntry(handler, r)
credentialName, err := getPathVariable(w, r, "credential", handler) credentialName, err := pos(w, r, 2)
if err != nil { if err != nil {
return return
} }

@ -0,0 +1,39 @@
package main
import (
"net/http"
"net/http/httptest"
"testing"
"github.com/kubernetes/deployment-manager/cmd/manager/router"
)
func TestHealthz(t *testing.T) {
c := mockContext()
s := httpHarness(c, "GET /", healthz)
defer s.Close()
res, err := http.Get(s.URL)
if err != nil {
t.Fatalf("Failed to GET healthz: %s", err)
} else if res.StatusCode != 200 {
t.Fatalf("Unexpected status: %d", res.StatusCode)
}
// TODO: Get the body and check on the content type and the body.
}
// httpHarness is a simple test server fixture.
// Simple fixture for standing up a test server with a single route.
//
// You must Close() the returned server.
func httpHarness(c *router.Context, route string, fn router.HandlerFunc) *httptest.Server {
h := router.NewHandler(c)
h.Add(route, fn)
return httptest.NewServer(h)
}
func mockContext() *router.Context {
// TODO: We need mocks for credentials and manager.
return &router.Context{}
}

@ -17,7 +17,7 @@ limitations under the License.
package main package main
import ( import (
"github.com/kubernetes/deployment-manager/pkg/util" "github.com/kubernetes/deployment-manager/cmd/manager/router"
"github.com/kubernetes/deployment-manager/pkg/version" "github.com/kubernetes/deployment-manager/pkg/version"
"flag" "flag"
@ -25,67 +25,58 @@ import (
"log" "log"
"net/http" "net/http"
"os" "os"
"github.com/gorilla/handlers"
"github.com/gorilla/mux"
) )
// Route defines a routing table entry to be registered with gorilla/mux. var (
type Route struct { port = flag.Int("port", 8080, "The port to listen on")
Name string maxLength = flag.Int64("maxLength", 1024, "The maximum length (KB) of a template.")
Path string expanderName = flag.String("expander", "expandybird-service", "The DNS name of the expander service.")
Methods string expanderURL = flag.String("expanderURL", "", "The URL for the expander service.")
HandlerFunc http.HandlerFunc deployerName = flag.String("deployer", "resourcifier-service", "The DNS name of the deployer service.")
Type string 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.")
var routes = []Route{ mongoName = flag.String("mongoName", "mongodb", "The DNS name of the mongodb service.")
{"HealthCheck", "/healthz", "GET", healthCheckHandlerFunc, ""}, mongoPort = flag.String("mongoPort", "27017", "The port of the mongodb service.")
} mongoAddress = flag.String("mongoAddress", "mongodb:27017", "The address of the mongodb service.")
)
// port to listen on
var port = flag.Int("port", 8080, "The port to listen on")
func init() {
if !flag.Parsed() {
flag.Parse()
}
}
func main() { func main() {
if !flag.Parsed() { // Set up dependencies
flag.Parse() c := &router.Context{
Config: parseFlags(),
} }
router := mux.NewRouter() if err := setupDependencies(c); err != nil {
router.StrictSlash(true) fmt.Fprintln(os.Stderr, err)
for _, route := range routes { os.Exit(1)
handler := http.Handler(http.HandlerFunc(route.HandlerFunc))
switch route.Type {
case "JSON":
handler = handlers.ContentTypeHandler(handler, "application/json")
case "":
break
default:
log.Fatalf("invalid route type: %v", route.Type)
} }
r := router.NewRoute() // Set up routes
r.Name(route.Name). handler := router.NewHandler(c)
Path(route.Path). registerRoutes(c, handler)
Methods(route.Methods).
Handler(handler)
}
address := fmt.Sprintf(":%d", *port) // Now create a server.
handler := handlers.CombinedLoggingHandler(os.Stderr, router) log.Printf("Starting Manager %s on %s", version.Version, c.Config.Address)
log.Printf("Version: %s", version.Version) if err := http.ListenAndServe(c.Config.Address, handler); err != nil {
log.Printf("Listening on port %d...", *port) log.Printf("Server exited with error %s", err)
log.Fatal(http.ListenAndServe(address, handler)) os.Exit(1)
}
} }
func healthCheckHandlerFunc(w http.ResponseWriter, r *http.Request) { func parseFlags() *router.Config {
handler := "manager: get health" flag.Parse()
util.LogHandlerEntry(handler, r) return &router.Config{
util.LogHandlerExitWithText(handler, w, "OK", http.StatusOK) Address: fmt.Sprintf(":%d", *port),
MaxTemplateLength: *maxLength,
ExpanderName: *expanderName,
ExpanderURL: *expanderURL,
DeployerName: *deployerName,
DeployerURL: *deployerURL,
CredentialFile: *credentialFile,
CredentialSecrets: *credentialSecrets,
MongoName: *mongoName,
MongoPort: *mongoPort,
MongoAddress: *mongoAddress,
}
} }

@ -0,0 +1,51 @@
package router
import (
"github.com/kubernetes/deployment-manager/cmd/manager/manager"
"github.com/kubernetes/deployment-manager/pkg/common"
helmhttp "github.com/kubernetes/deployment-manager/pkg/httputil"
)
// Config holds the global configuration parameters passed into the router.
//
// Config is used concurrently. Once a config is created, it should be treated
// as immutable.
type Config struct {
// Address is the host and port (:8080)
Address string
// MaxTemplateLength is the maximum length of a template.
MaxTemplateLength int64
// ExpanderName is the DNS name of the expansion service.
ExpanderName string
// ExpanderURL is the expander service's URL.
ExpanderURL string
// DeployerName is the deployer's DNS name
DeployerName string
// DeployerURL is the deployer's URL
DeployerURL string
// CredentialFile is the file to the credentials.
CredentialFile string
// CredentialSecrets tells the service to use a secrets file instead.
CredentialSecrets bool
// MongoName is the DNS name of the mongo server.
MongoName string
// MongoPort is the port for the MongoDB protocol on the mongo server.
// It is a string for historical reasons.
MongoPort string
// MongoAddress is the name and port.
MongoAddress string
}
// Context contains dependencies that are passed to each handler function.
//
// Context carries typed information, often scoped to interfaces, so that the
// caller's contract with the service is known at compile time.
//
// Members of the context must be concurrency safe.
type Context struct {
Config *Config
// Manager is a deployment-manager/manager/manager.Manager
Manager manager.Manager
Encoder helmhttp.Encoder
CredentialProvider common.CredentialProvider
}

@ -0,0 +1,93 @@
/*
Copyright 2016 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 router is an HTTP router.
This router provides appropriate dependency injection/encapsulation for the
HTTP routing layer. This removes the requirement to set global variables for
resources like database handles.
This library does not replace the default HTTP mux because there is no need.
Instead, it implements an HTTP handler.
It then defines a handler function that is given a context as well as a
request and response.
*/
package router
import (
"log"
"net/http"
"github.com/Masterminds/httputil"
helmhttp "github.com/kubernetes/deployment-manager/pkg/httputil"
)
// HandlerFunc responds to an individual HTTP request.
//
// Returned errors will be captured, logged, and returned as HTTP 500 errors.
type HandlerFunc func(w http.ResponseWriter, r *http.Request, c *Context) error
// Handler implements an http.Handler.
//
// This is the top level route handler.
type Handler struct {
c *Context
resolver *httputil.Resolver
routes map[string]HandlerFunc
paths []string
}
// NewHandler creates a new Handler.
//
// Routes cannot be modified after construction. The order that the route
// names are returned by Routes.Paths() determines the lookup order.
func NewHandler(c *Context) *Handler {
return &Handler{
c: c,
resolver: httputil.NewResolver([]string{}),
routes: map[string]HandlerFunc{},
paths: []string{},
}
}
// Add a route to a handler.
//
// The route name is "VERB /ENPOINT/PATH", e.g. "GET /foo".
func (h *Handler) Add(route string, fn HandlerFunc) {
h.routes[route] = fn
h.paths = append(h.paths, route)
h.resolver = httputil.NewResolver(h.paths)
}
// ServeHTTP serves an HTTP request.
func (h *Handler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
log.Printf(helmhttp.LogAccess, r.Method, r.URL)
route, err := h.resolver.Resolve(r)
if err != nil {
helmhttp.NotFound(w, r)
return
}
fn, ok := h.routes[route]
if !ok {
helmhttp.Fatal(w, r, "route %s missing", route)
}
if err := fn(w, r, h.c); err != nil {
helmhttp.Fatal(w, r, err.Error())
}
}

@ -0,0 +1,56 @@
/*
Copyright 2016 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 router
import (
"fmt"
"io/ioutil"
"net/http"
"net/http/httptest"
"testing"
)
func TestHandler(t *testing.T) {
c := &Context{}
h := NewHandler(c)
h.Add("GET /", func(w http.ResponseWriter, r *http.Request, c *Context) error {
fmt.Fprintln(w, "hello")
return nil
})
h.Add("POST /", func(w http.ResponseWriter, r *http.Request, c *Context) error {
fmt.Fprintln(w, "goodbye")
return nil
})
s := httptest.NewServer(h)
defer s.Close()
res, err := http.Get(s.URL)
if err != nil {
t.Fatal(err)
}
data, err := ioutil.ReadAll(res.Body)
res.Body.Close()
if err != nil {
t.Fatal(err)
}
if "hello\n" != string(data) {
t.Errorf("Expected 'hello', got %q", data)
}
}

8
glide.lock generated

@ -1,5 +1,5 @@
hash: d9beab9a799ac8dd0d76c4f7a3a32753d44833dd3527b3caa8e786865ea26816 hash: f34e830b237fcba5202f84c55b21764226b7e8c201f7622e9639c7a628e33b0b
updated: 2016-03-04T09:54:13.155442463-07:00 updated: 2016-03-14T10:53:18.091793732-06:00
imports: imports:
- name: github.com/aokoli/goutils - name: github.com/aokoli/goutils
version: 9c37978a95bd5c709a15883b6242714ea6709e64 version: 9c37978a95bd5c709a15883b6242714ea6709e64
@ -25,10 +25,12 @@ imports:
version: 8f2758070a82adb7a3ad6b223a0b91878f32d400 version: 8f2758070a82adb7a3ad6b223a0b91878f32d400
- name: github.com/gorilla/mux - name: github.com/gorilla/mux
version: 26a6070f849969ba72b72256e9f14cf519751690 version: 26a6070f849969ba72b72256e9f14cf519751690
- name: github.com/Masterminds/httputil
version: e9b977e9cf16f9d339573e18f0f1f7ce5d3f419a
- name: github.com/Masterminds/semver - name: github.com/Masterminds/semver
version: c4f7ef0702f269161a60489ccbbc9f1241ad1265 version: c4f7ef0702f269161a60489ccbbc9f1241ad1265
- name: github.com/Masterminds/sprig - name: github.com/Masterminds/sprig
version: fd057ca403105755181f84645696d705a58852dd version: 0199893f008a87287bf2b4e3e390e66bb074c659
- name: golang.org/x/net - name: golang.org/x/net
version: 04b9de9b512f58addf28c9853d50ebef61c3953e version: 04b9de9b512f58addf28c9853d50ebef61c3953e
subpackages: subpackages:

@ -14,3 +14,4 @@ import:
- package: github.com/gorilla/mux - package: github.com/gorilla/mux
- package: gopkg.in/yaml.v2 - package: gopkg.in/yaml.v2
- package: github.com/Masterminds/sprig - package: github.com/Masterminds/sprig
- package: github.com/Masterminds/httputil

@ -0,0 +1,21 @@
/*
Copyright 2016 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 httputil provides common HTTP tools.
This package provides tools for working with HTTP requests and responses.
*/
package httputil

@ -0,0 +1,132 @@
/*
Copyright 2016 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 httputil
import (
"encoding/json"
"errors"
"fmt"
"mime"
"net/http"
"reflect"
"strings"
"github.com/ghodss/yaml"
)
// Encoder takes input and translate it to an expected encoded output.
//
// Implementations of encoders may use details of the HTTP request and response
// to correctly encode an object for return to the client.
//
// Encoders are expected to produce output, even if that output is an error
// message.
type Encoder interface {
// Encode encoders a given response
//
// When an encoder fails, it logs any necessary data and then responds to
// the client.
Encode(http.ResponseWriter, *http.Request, interface{})
}
// AcceptEncoder uses the accept headers on a request to determine the response type.
//
// It supports the following encodings:
// - application/json: passed to encoding/json.Marshal
// - text/yaml: passed to gopkg.in/yaml.v2.Marshal
// - text/plain: passed to fmt.Sprintf("%V")
//
// It treats `application/x-yaml` as `text/yaml`.
type AcceptEncoder struct {
DefaultEncoding string
}
// Encode encodeds the given interface to the first available type in the Accept header.
func (e *AcceptEncoder) Encode(w http.ResponseWriter, r *http.Request, out interface{}) {
a := r.Header.Get("accept")
fn := encoders[e.DefaultEncoding]
mt := e.DefaultEncoding
if a != "" {
mt, fn = e.parseAccept(a)
}
data, err := fn(out)
if err != nil {
Fatal(w, r, "Could not marshal data: %s", err)
return
}
w.Header().Add("content-type", mt)
w.Write(data)
}
// parseAccept parses the value of an Accept: header and returns the best match.
//
// This returns the matched MIME type and the Marshal function.
func (e *AcceptEncoder) parseAccept(h string) (string, Marshaler) {
keys := strings.Split(h, ",")
for _, k := range keys {
mt, _, err := mime.ParseMediaType(k)
if err != nil {
continue
}
if enc, ok := encoders[mt]; ok {
return mt, enc
}
}
return e.DefaultEncoding, encoders[e.DefaultEncoding]
}
// Marshaler marshals an interface{} into a []byte.
type Marshaler func(interface{}) ([]byte, error)
var encoders = map[string]Marshaler{
"application/json": json.Marshal,
"text/yaml": yaml.Marshal,
"application/x-yaml": yaml.Marshal,
"text/plain": textMarshal,
}
// ErrUnsupportedKind indicates that the marshal cannot marshal a particular Go Kind (e.g. struct or chan).
var ErrUnsupportedKind = errors.New("unsupported kind")
// textMarshal marshals v into a text representation ONLY IN NARROW CASES.
//
// An error will have its Error() method called.
// A fmt.Stringer will have its String() method called.
// Scalar types will be marshaled with fmt.Sprintf("%v").
//
// This will only marshal scalar types for securoty reasons (namely, we don't
// want the possibility of forcing exposure of non-exported data or ptr
// addresses, etc.)
func textMarshal(v interface{}) ([]byte, error) {
switch s := v.(type) {
case error:
return []byte(s.Error()), nil
case fmt.Stringer:
return []byte(s.String()), nil
}
// Error on kinds we don't support.
val := reflect.Indirect(reflect.ValueOf(v))
switch val.Kind() {
case reflect.Invalid, reflect.Array, reflect.Chan, reflect.Func, reflect.Interface,
reflect.Map, reflect.Ptr, reflect.Slice, reflect.Struct, reflect.UnsafePointer:
return []byte{}, ErrUnsupportedKind
}
return []byte(fmt.Sprintf("%v", v)), nil
}

@ -0,0 +1,110 @@
/*
Copyright 2016 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 httputil
import (
"encoding/json"
"errors"
"io/ioutil"
"net/http"
"net/http/httptest"
"testing"
)
var _ Encoder = &AcceptEncoder{}
func TestParseAccept(t *testing.T) {
e := &AcceptEncoder{
DefaultEncoding: "application/json",
}
tests := map[string]string{
"": e.DefaultEncoding,
"*/*": e.DefaultEncoding,
// To stay true to spec, this _should_ be an error. But our thought
// on this case is that we'd rather send a default format.
"audio/*; q=0.2, audio/basic": e.DefaultEncoding,
"text/html; q=0.8, text/yaml,application/json": "text/yaml",
"application/x-yaml; foo=bar": "application/x-yaml",
"text/monkey, TEXT/YAML ; zoom=zoom ": "text/yaml",
}
for in, expects := range tests {
mt, enc := e.parseAccept(in)
if mt != expects {
t.Errorf("Expected %q, got %q", expects, mt)
continue
}
_, err := enc([]string{"hello", "world"})
if err != nil {
t.Fatalf("Failed to marshal: %s", err)
}
}
}
func TestTextMarshal(t *testing.T) {
tests := map[string]interface{}{
"foo": "foo",
"5": 5,
"stinky cheese": errors.New("stinky cheese"),
}
for expect, in := range tests {
if o, err := textMarshal(in); err != nil || string(o) != expect {
t.Errorf("Expected %q, got %q", expect, o)
}
}
if _, err := textMarshal(struct{ foo int }{5}); err != ErrUnsupportedKind {
t.Fatalf("Expected unsupported kind, got %v", err)
}
}
func TestAcceptEncoder(t *testing.T) {
enc := &AcceptEncoder{
DefaultEncoding: "application/json",
}
fn := func(w http.ResponseWriter, r *http.Request) {
enc.Encode(w, r, []string{"hello", "world"})
}
s := httptest.NewServer(http.HandlerFunc(fn))
defer s.Close()
res, err := http.Get(s.URL)
if err != nil {
t.Fatal(err)
}
if res.StatusCode != 200 {
t.Fatalf("Unexpected response code %d", res.StatusCode)
}
if mt := res.Header.Get("content-type"); mt != "application/json" {
t.Errorf("Unexpected content type: %q", mt)
}
data, err := ioutil.ReadAll(res.Body)
res.Body.Close()
if err != nil {
t.Fatalf("Failed to read response body: %s", err)
}
out := []string{}
if err := json.Unmarshal(data, &out); err != nil {
t.Fatalf("Failed to unmarshal JSON: %s", err)
}
if out[0] != "hello" {
t.Fatalf("Unexpected JSON data in slot 0: %s", out[0])
}
}

@ -0,0 +1,56 @@
/*
Copyright 2016 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 httputil
import (
"fmt"
"log"
"net/http"
)
const (
// LogAccess is for logging access messages. Form: Access r.Method, r.URL
LogAccess = "Access: %s %s"
// LogNotFound is for logging 404 errors. Form: Not Found r.Method, r.URL
LogNotFound = "Not Found: %s %s"
// LogFatal is for logging 500 errors. Form: Internal Server Error r.Method r.URL message
LogFatal = "Internal Server Error: %s %s %s"
// LogBadRequest logs 400 errors.
LogBadRequest = "Bad Request: %s %s"
)
// NotFound writes a 404 error to the client and logs an error.
func NotFound(w http.ResponseWriter, r *http.Request) {
log.Printf(LogNotFound, r.Method, r.URL)
w.WriteHeader(http.StatusNotFound)
fmt.Fprintln(w, "File Not Found")
}
func BadRequest(w http.ResponseWriter, r *http.Request) {
log.Printf(LogNotFound, r.Method, r.URL)
w.WriteHeader(http.StatusBadRequest)
fmt.Fprintln(w, "Bad Request")
}
// Fatal writes a 500 response to the client and logs the message.
//
// Additional arguments are past into the the formatter as params to msg.
func Fatal(w http.ResponseWriter, r *http.Request, msg string, v ...interface{}) {
log.Printf(LogFatal, r.Method, r.URL, fmt.Sprintf(msg, v...))
w.WriteHeader(http.StatusInternalServerError)
fmt.Fprintln(w, "Internal Server Error")
}

@ -0,0 +1,50 @@
/*
Copyright 2016 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 httputil
import (
"net/http"
"net/http/httptest"
"testing"
)
func TestNotFound(t *testing.T) {
fn := func(w http.ResponseWriter, r *http.Request) {
NotFound(w, r)
}
testStatusCode(http.HandlerFunc(fn), 404, t)
}
func TestFatal(t *testing.T) {
fn := func(w http.ResponseWriter, r *http.Request) {
Fatal(w, r, "fatal %s", "foo")
}
testStatusCode(http.HandlerFunc(fn), 500, t)
}
func testStatusCode(fn http.HandlerFunc, expect int, t *testing.T) {
s := httptest.NewServer(fn)
defer s.Close()
res, err := http.Get(s.URL)
if err != nil {
t.Fatal(err)
}
if res.StatusCode != expect {
t.Errorf("Expected %d, got %d", expect, res.StatusCode)
}
}
Loading…
Cancel
Save