
Rather than hard-coding the default response format as HTML, allow the default to be overridden by an environment variable. For example, given a REST API endpoint that defaults to responding in JSON, you may wish to configure the error messages to be JSON by default as well.
184 lines
5.4 KiB
Go
184 lines
5.4 KiB
Go
/*
|
|
Copyright 2017 The Kubernetes Authors.
|
|
|
|
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 (
|
|
"fmt"
|
|
"io"
|
|
"log"
|
|
"mime"
|
|
"net/http"
|
|
"os"
|
|
"strconv"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/prometheus/client_golang/prometheus"
|
|
"github.com/prometheus/client_golang/prometheus/promhttp"
|
|
)
|
|
|
|
const (
|
|
// FormatHeader name of the header used to extract the format
|
|
FormatHeader = "X-Format"
|
|
|
|
// CodeHeader name of the header used as source of the HTTP status code to return
|
|
CodeHeader = "X-Code"
|
|
|
|
// ContentType name of the header that defines the format of the reply
|
|
ContentType = "Content-Type"
|
|
|
|
// OriginalURI name of the header with the original URL from NGINX
|
|
OriginalURI = "X-Original-URI"
|
|
|
|
// Namespace name of the header that contains information about the Ingress namespace
|
|
Namespace = "X-Namespace"
|
|
|
|
// IngressName name of the header that contains the matched Ingress
|
|
IngressName = "X-Ingress-Name"
|
|
|
|
// ServiceName name of the header that contains the matched Service in the Ingress
|
|
ServiceName = "X-Service-Name"
|
|
|
|
// ServicePort name of the header that contains the matched Service port in the Ingress
|
|
ServicePort = "X-Service-Port"
|
|
|
|
// RequestId is a unique ID that identifies the request - same as for backend service
|
|
RequestId = "X-Request-ID"
|
|
|
|
// ErrFilesPathVar is the name of the environment variable indicating
|
|
// the location on disk of files served by the handler.
|
|
ErrFilesPathVar = "ERROR_FILES_PATH"
|
|
|
|
// DefaultFormatVar is the name of the environment variable indicating
|
|
// the default error MIME type that should be returned if either the
|
|
// client does not specify an Accept header, or the Accept header provided
|
|
// cannot be mapped to a file extension.
|
|
DefaultFormatVar = "DEFAULT_RESPONSE_FORMAT"
|
|
)
|
|
|
|
func init() {
|
|
prometheus.MustRegister(requestCount)
|
|
prometheus.MustRegister(requestDuration)
|
|
}
|
|
|
|
func main() {
|
|
errFilesPath := "/www"
|
|
if os.Getenv(ErrFilesPathVar) != "" {
|
|
errFilesPath = os.Getenv(ErrFilesPathVar)
|
|
}
|
|
|
|
defaultFormat := "text/html"
|
|
if os.Getenv(DefaultFormatVar) != "" {
|
|
defaultFormat = os.Getenv(DefaultFormatVar)
|
|
}
|
|
|
|
http.HandleFunc("/", errorHandler(errFilesPath, defaultFormat))
|
|
|
|
http.Handle("/metrics", promhttp.Handler())
|
|
|
|
http.HandleFunc("/healthz", func(w http.ResponseWriter, r *http.Request) {
|
|
w.WriteHeader(http.StatusOK)
|
|
})
|
|
|
|
http.ListenAndServe(fmt.Sprintf(":8080"), nil)
|
|
}
|
|
|
|
func errorHandler(path, defaultFormat string) func(http.ResponseWriter, *http.Request) {
|
|
defaultExts, err := mime.ExtensionsByType(defaultFormat)
|
|
if err != nil || len(defaultExts) == 0 {
|
|
panic("couldn't get file extension for default format")
|
|
}
|
|
defaultExt := defaultExts[0]
|
|
|
|
return func(w http.ResponseWriter, r *http.Request) {
|
|
start := time.Now()
|
|
ext := defaultExt
|
|
|
|
if os.Getenv("DEBUG") != "" {
|
|
w.Header().Set(FormatHeader, r.Header.Get(FormatHeader))
|
|
w.Header().Set(CodeHeader, r.Header.Get(CodeHeader))
|
|
w.Header().Set(ContentType, r.Header.Get(ContentType))
|
|
w.Header().Set(OriginalURI, r.Header.Get(OriginalURI))
|
|
w.Header().Set(Namespace, r.Header.Get(Namespace))
|
|
w.Header().Set(IngressName, r.Header.Get(IngressName))
|
|
w.Header().Set(ServiceName, r.Header.Get(ServiceName))
|
|
w.Header().Set(ServicePort, r.Header.Get(ServicePort))
|
|
w.Header().Set(RequestId, r.Header.Get(RequestId))
|
|
}
|
|
|
|
format := r.Header.Get(FormatHeader)
|
|
if format == "" {
|
|
format = defaultFormat
|
|
log.Printf("format not specified. Using %v", format)
|
|
}
|
|
|
|
cext, err := mime.ExtensionsByType(format)
|
|
if err != nil {
|
|
log.Printf("unexpected error reading media type extension: %v. Using %v", err, ext)
|
|
format = defaultFormat
|
|
} else if len(cext) == 0 {
|
|
log.Printf("couldn't get media type extension. Using %v", ext)
|
|
} else {
|
|
ext = cext[0]
|
|
}
|
|
w.Header().Set(ContentType, format)
|
|
|
|
errCode := r.Header.Get(CodeHeader)
|
|
code, err := strconv.Atoi(errCode)
|
|
if err != nil {
|
|
code = 404
|
|
log.Printf("unexpected error reading return code: %v. Using %v", err, code)
|
|
}
|
|
w.WriteHeader(code)
|
|
|
|
if !strings.HasPrefix(ext, ".") {
|
|
ext = "." + ext
|
|
}
|
|
// special case for compatibility
|
|
if ext == ".htm" {
|
|
ext = ".html"
|
|
}
|
|
file := fmt.Sprintf("%v/%v%v", path, code, ext)
|
|
f, err := os.Open(file)
|
|
if err != nil {
|
|
log.Printf("unexpected error opening file: %v", err)
|
|
scode := strconv.Itoa(code)
|
|
file := fmt.Sprintf("%v/%cxx%v", path, scode[0], ext)
|
|
f, err := os.Open(file)
|
|
if err != nil {
|
|
log.Printf("unexpected error opening file: %v", err)
|
|
http.NotFound(w, r)
|
|
return
|
|
}
|
|
defer f.Close()
|
|
log.Printf("serving custom error response for code %v and format %v from file %v", code, format, file)
|
|
io.Copy(w, f)
|
|
return
|
|
}
|
|
defer f.Close()
|
|
log.Printf("serving custom error response for code %v and format %v from file %v", code, format, file)
|
|
io.Copy(w, f)
|
|
|
|
duration := time.Now().Sub(start).Seconds()
|
|
|
|
proto := strconv.Itoa(r.ProtoMajor)
|
|
proto = fmt.Sprintf("%s.%s", proto, strconv.Itoa(r.ProtoMinor))
|
|
|
|
requestCount.WithLabelValues(proto).Inc()
|
|
requestDuration.WithLabelValues(proto).Observe(duration)
|
|
}
|
|
}
|