2017-02-07 22:33:23 +01:00
|
|
|
package cbreaker
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"net/http"
|
|
|
|
"net/url"
|
|
|
|
"strconv"
|
2017-11-22 18:20:03 +01:00
|
|
|
|
2018-01-22 12:16:03 +01:00
|
|
|
log "github.com/sirupsen/logrus"
|
2017-11-22 18:20:03 +01:00
|
|
|
"github.com/vulcand/oxy/utils"
|
2017-02-07 22:33:23 +01:00
|
|
|
)
|
|
|
|
|
2018-07-11 10:08:03 +02:00
|
|
|
// Response response model
|
2017-02-07 22:33:23 +01:00
|
|
|
type Response struct {
|
|
|
|
StatusCode int
|
|
|
|
ContentType string
|
|
|
|
Body []byte
|
|
|
|
}
|
|
|
|
|
2018-07-11 10:08:03 +02:00
|
|
|
// ResponseFallback fallback response handler
|
2017-02-07 22:33:23 +01:00
|
|
|
type ResponseFallback struct {
|
|
|
|
r Response
|
2018-07-11 10:08:03 +02:00
|
|
|
|
|
|
|
log *log.Logger
|
2017-02-07 22:33:23 +01:00
|
|
|
}
|
|
|
|
|
2018-07-11 10:08:03 +02:00
|
|
|
// NewResponseFallbackWithLogger creates a new ResponseFallback
|
|
|
|
func NewResponseFallbackWithLogger(r Response, l *log.Logger) (*ResponseFallback, error) {
|
2017-02-07 22:33:23 +01:00
|
|
|
if r.StatusCode == 0 {
|
|
|
|
return nil, fmt.Errorf("response code should not be 0")
|
|
|
|
}
|
2018-07-11 10:08:03 +02:00
|
|
|
return &ResponseFallback{r: r, log: l}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewResponseFallback creates a new ResponseFallback
|
|
|
|
func NewResponseFallback(r Response) (*ResponseFallback, error) {
|
|
|
|
return NewResponseFallbackWithLogger(r, log.StandardLogger())
|
2017-02-07 22:33:23 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
func (f *ResponseFallback) ServeHTTP(w http.ResponseWriter, req *http.Request) {
|
2018-07-11 10:08:03 +02:00
|
|
|
if f.log.Level >= log.DebugLevel {
|
|
|
|
logEntry := f.log.WithField("Request", utils.DumpHttpRequest(req))
|
2017-11-22 18:20:03 +01:00
|
|
|
logEntry.Debug("vulcand/oxy/fallback/response: begin ServeHttp on request")
|
2018-06-04 14:14:03 +02:00
|
|
|
defer logEntry.Debug("vulcand/oxy/fallback/response: completed ServeHttp on request")
|
2017-11-22 18:20:03 +01:00
|
|
|
}
|
|
|
|
|
2017-02-07 22:33:23 +01:00
|
|
|
if f.r.ContentType != "" {
|
|
|
|
w.Header().Set("Content-Type", f.r.ContentType)
|
|
|
|
}
|
|
|
|
w.Header().Set("Content-Length", strconv.Itoa(len(f.r.Body)))
|
|
|
|
w.WriteHeader(f.r.StatusCode)
|
2017-11-22 18:20:03 +01:00
|
|
|
_, err := w.Write(f.r.Body)
|
|
|
|
if err != nil {
|
2018-09-25 15:06:03 +02:00
|
|
|
f.log.Errorf("vulcand/oxy/fallback/response: failed to write response, err: %v", err)
|
2017-11-22 18:20:03 +01:00
|
|
|
}
|
2017-02-07 22:33:23 +01:00
|
|
|
}
|
|
|
|
|
2018-07-11 10:08:03 +02:00
|
|
|
// Redirect redirect model
|
2017-02-07 22:33:23 +01:00
|
|
|
type Redirect struct {
|
2017-11-22 18:20:03 +01:00
|
|
|
URL string
|
|
|
|
PreservePath bool
|
2017-02-07 22:33:23 +01:00
|
|
|
}
|
|
|
|
|
2018-07-11 10:08:03 +02:00
|
|
|
// RedirectFallback fallback redirect handler
|
2017-02-07 22:33:23 +01:00
|
|
|
type RedirectFallback struct {
|
2017-11-22 18:20:03 +01:00
|
|
|
r Redirect
|
2018-07-11 10:08:03 +02:00
|
|
|
|
|
|
|
u *url.URL
|
|
|
|
|
|
|
|
log *log.Logger
|
2017-02-07 22:33:23 +01:00
|
|
|
}
|
|
|
|
|
2018-07-11 10:08:03 +02:00
|
|
|
// NewRedirectFallbackWithLogger creates a new RedirectFallback
|
|
|
|
func NewRedirectFallbackWithLogger(r Redirect, l *log.Logger) (*RedirectFallback, error) {
|
2017-02-07 22:33:23 +01:00
|
|
|
u, err := url.ParseRequestURI(r.URL)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2018-07-11 10:08:03 +02:00
|
|
|
return &RedirectFallback{r: r, u: u, log: l}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewRedirectFallback creates a new RedirectFallback
|
|
|
|
func NewRedirectFallback(r Redirect) (*RedirectFallback, error) {
|
|
|
|
return NewRedirectFallbackWithLogger(r, log.StandardLogger())
|
2017-02-07 22:33:23 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
func (f *RedirectFallback) ServeHTTP(w http.ResponseWriter, req *http.Request) {
|
2018-07-11 10:08:03 +02:00
|
|
|
if f.log.Level >= log.DebugLevel {
|
|
|
|
logEntry := f.log.WithField("Request", utils.DumpHttpRequest(req))
|
2017-11-22 18:20:03 +01:00
|
|
|
logEntry.Debug("vulcand/oxy/fallback/redirect: begin ServeHttp on request")
|
2018-06-04 14:14:03 +02:00
|
|
|
defer logEntry.Debug("vulcand/oxy/fallback/redirect: completed ServeHttp on request")
|
2017-11-22 18:20:03 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
location := f.u.String()
|
|
|
|
if f.r.PreservePath {
|
|
|
|
location += req.URL.Path
|
|
|
|
}
|
|
|
|
|
|
|
|
w.Header().Set("Location", location)
|
2017-02-07 22:33:23 +01:00
|
|
|
w.WriteHeader(http.StatusFound)
|
2017-11-22 18:20:03 +01:00
|
|
|
_, err := w.Write([]byte(http.StatusText(http.StatusFound)))
|
|
|
|
if err != nil {
|
2018-09-25 15:06:03 +02:00
|
|
|
f.log.Errorf("vulcand/oxy/fallback/redirect: failed to write response, err: %v", err)
|
2017-11-22 18:20:03 +01:00
|
|
|
}
|
2017-02-07 22:33:23 +01:00
|
|
|
}
|