Merge pull request #460 from containous/fix-websocket-hijack

Fix websocket connection Hijack
This commit is contained in:
Vincent Demeester 2016-06-18 15:50:30 +02:00 committed by GitHub
commit 7133a28fdb
10 changed files with 212 additions and 92 deletions

View file

@ -22,4 +22,4 @@ COPY glide.yaml glide.yaml
COPY glide.lock glide.lock COPY glide.lock glide.lock
RUN glide install RUN glide install
COPY . /go/src/github.com/containous/traefik COPY . /go/src/github.com/containous/traefik

View file

@ -206,8 +206,7 @@ type Certificate struct {
// Retry contains request retry config // Retry contains request retry config
type Retry struct { type Retry struct {
Attempts int `description:"Number of attempts"` Attempts int `description:"Number of attempts"`
MaxMem int64 `description:"Maximum request body to be stored in memory in Mo"`
} }
// NewTraefikDefaultPointersConfiguration creates a TraefikConfiguration with pointers default values // NewTraefikDefaultPointersConfiguration creates a TraefikConfiguration with pointers default values
@ -269,7 +268,7 @@ func NewTraefikDefaultPointersConfiguration() *TraefikConfiguration {
//default Kubernetes //default Kubernetes
var defaultKubernetes provider.Kubernetes var defaultKubernetes provider.Kubernetes
defaultKubernetes.Watch = true defaultKubernetes.Watch = true
defaultKubernetes.Endpoint = "127.0.0.1:8080" defaultKubernetes.Endpoint = "http://127.0.0.1:8080"
defaultKubernetes.Constraints = []types.Constraint{} defaultKubernetes.Constraints = []types.Constraint{}
defaultConfiguration := GlobalConfiguration{ defaultConfiguration := GlobalConfiguration{
@ -283,7 +282,7 @@ func NewTraefikDefaultPointersConfiguration() *TraefikConfiguration {
Zookeeper: &defaultZookeeper, Zookeeper: &defaultZookeeper,
Boltdb: &defaultBoltDb, Boltdb: &defaultBoltDb,
Kubernetes: &defaultKubernetes, Kubernetes: &defaultKubernetes,
Retry: &Retry{MaxMem: 2}, Retry: &Retry{},
} }
return &TraefikConfiguration{ return &TraefikConfiguration{
GlobalConfiguration: defaultConfiguration, GlobalConfiguration: defaultConfiguration,

View file

@ -110,13 +110,6 @@
# Default: (number servers in backend) -1 # Default: (number servers in backend) -1
# #
# attempts = 3 # attempts = 3
# Sets the maximum request body to be stored in memory in Mo
#
# Optional
# Default: 2
#
# maxMem = 3
``` ```
## ACME (Let's Encrypt) configuration ## ACME (Let's Encrypt) configuration

View file

@ -1,21 +1,25 @@
consul: version: '2'
image: progrium/consul services:
command: -server -bootstrap -advertise 12.0.0.254 -log-level debug -ui-dir /ui consul:
ports: image: progrium/consul
- "8400:8400" command: -server -bootstrap -advertise 12.0.0.254 -log-level debug -ui-dir /ui
- "8500:8500" ports:
- "8600:53/udp" - "8400:8400"
expose: - "8500:8500"
- "8300" - "8600:53/udp"
- "8301" expose:
- "8301/udp" - "8300"
- "8302" - "8301"
- "8302/udp" - "8301/udp"
- "8302"
- "8302/udp"
registrator: registrator:
image: gliderlabs/registrator:master depends_on:
command: -internal consulkv://consul:8500/traefik - consul
volumes: image: gliderlabs/registrator:master
- /var/run/docker.sock:/tmp/docker.sock command: -internal consul://consul:8500
links: volumes:
- consul - /var/run/docker.sock:/tmp/docker.sock
links:
- consul

57
glide.lock generated
View file

@ -1,8 +1,8 @@
hash: 5a6dbc30a69abd002736bd5113e0f783c448faee20a0791c724ec2c3c1cfb8bb hash: 7330fcb934bc52f47319f969bbf918e325bac2436518e3fad74de596f13bbda2
updated: 2016-06-03T18:11:43.839017153+02:00 updated: 2016-06-18T12:59:25.604052029+02:00
imports: imports:
- name: github.com/boltdb/bolt - name: github.com/boltdb/bolt
version: dfb21201d9270c1082d5fb0f07f500311ff72f18 version: 3f7947a25d970e1e5f512276c14d5dcf731ccd5e
- name: github.com/BurntSushi/toml - name: github.com/BurntSushi/toml
version: f0aeabca5a127c4078abb8c8d64298b147264b55 version: f0aeabca5a127c4078abb8c8d64298b147264b55
- name: github.com/BurntSushi/ty - name: github.com/BurntSushi/ty
@ -10,26 +10,17 @@ imports:
subpackages: subpackages:
- fun - fun
- name: github.com/cenkalti/backoff - name: github.com/cenkalti/backoff
version: a6030178a585d5972d4d33ce61f4a1fa40eaaed0 version: cdf48bbc1eb78d1349cbda326a4a037f7ba565c6
- name: github.com/codahale/hdrhistogram - name: github.com/codahale/hdrhistogram
version: 9208b142303c12d8899bae836fd524ac9338b4fd version: 9208b142303c12d8899bae836fd524ac9338b4fd
- name: github.com/codegangsta/cli - name: github.com/codegangsta/cli
version: bf4a526f48af7badd25d2cb02d587e1b01be3b50 version: bf4a526f48af7badd25d2cb02d587e1b01be3b50
- name: github.com/codegangsta/negroni - name: github.com/codegangsta/negroni
version: feacfc52d357c844f524c794947493483ed881b3 version: dcaac9107a7a6ba4cf5143afc145e2b70a1c12c2
- name: github.com/containous/flaeg - name: github.com/containous/flaeg
version: b98687da5c323650f4513fda6b6203fcbdec9313 version: b98687da5c323650f4513fda6b6203fcbdec9313
- name: github.com/containous/mux - name: github.com/containous/mux
version: a819b77bba13f0c0cbe36e437bc2e948411b3996 version: a819b77bba13f0c0cbe36e437bc2e948411b3996
- name: github.com/containous/oxy
version: 183212964e13e7b8afe01a08b193d04300554a68
subpackages:
- cbreaker
- connlimit
- forward
- roundrobin
- stream
- utils
- name: github.com/containous/staert - name: github.com/containous/staert
version: e2aa88e235a02dd52aa1d5d9de75f9d9139d1602 version: e2aa88e235a02dd52aa1d5d9de75f9d9139d1602
- name: github.com/coreos/etcd - name: github.com/coreos/etcd
@ -45,7 +36,7 @@ imports:
subpackages: subpackages:
- spew - spew
- name: github.com/docker/distribution - name: github.com/docker/distribution
version: feddf6cd4e439577ab270d8e3ba63a5d7c5c0d55 version: edd7cb5249d0a45262b20bb58b838ecf4fb368bd
subpackages: subpackages:
- reference - reference
- digest - digest
@ -71,13 +62,13 @@ imports:
- types/blkiodev - types/blkiodev
- types/strslice - types/strslice
- name: github.com/docker/go-connections - name: github.com/docker/go-connections
version: c7838b258fbfa3fe88eecfb2a0e08ea0dbd6a646 version: 990a1a1a70b0da4c4cb70e117971a4f0babfbf1a
subpackages: subpackages:
- sockets - sockets
- tlsconfig - tlsconfig
- nat - nat
- name: github.com/docker/go-units - name: github.com/docker/go-units
version: 09dda9d4b0d748c57c14048906d3d094a58ec0c9 version: f2d77a61e3c169b43402a0a1e84f06daf29b8190
- name: github.com/docker/libcompose - name: github.com/docker/libcompose
version: 8ee7bcc364f7b8194581a3c6bd9fa019467c7873 version: 8ee7bcc364f7b8194581a3c6bd9fa019467c7873
- name: github.com/docker/libkv - name: github.com/docker/libkv
@ -103,13 +94,13 @@ imports:
- name: github.com/gorilla/context - name: github.com/gorilla/context
version: aed02d124ae4a0e94fea4541c8effd05bf0c8296 version: aed02d124ae4a0e94fea4541c8effd05bf0c8296
- name: github.com/hashicorp/consul - name: github.com/hashicorp/consul
version: 802b29ab948dedb7f7b1b903f535bdf250388c50 version: af30e17dcd1a6869da033bd55ad949973c7c54e9
subpackages: subpackages:
- api - api
- name: github.com/hashicorp/go-cleanhttp - name: github.com/hashicorp/go-cleanhttp
version: 875fb671b3ddc66f8e2f0acc33829c8cb989a38d version: 875fb671b3ddc66f8e2f0acc33829c8cb989a38d
- name: github.com/hashicorp/serf - name: github.com/hashicorp/serf
version: e4ec8cc423bbe20d26584b96efbeb9102e16d05f version: 6c4672d66fc6312ddde18399262943e21175d831
subpackages: subpackages:
- coordinate - coordinate
- serf - serf
@ -121,8 +112,6 @@ imports:
version: 2f35a4607f1abf71f97f77f99b0de8493ef6f4ef version: 2f35a4607f1abf71f97f77f99b0de8493ef6f4ef
- name: github.com/mailgun/manners - name: github.com/mailgun/manners
version: fada45142db3f93097ca917da107aa3fad0ffcb5 version: fada45142db3f93097ca917da107aa3fad0ffcb5
- name: github.com/mailgun/multibuf
version: 565402cd71fbd9c12aa7e295324ea357e970a61e
- name: github.com/mailgun/timetools - name: github.com/mailgun/timetools
version: fd192d755b00c968d312d23f521eb0cdc6f66bd0 version: fd192d755b00c968d312d23f521eb0cdc6f66bd0
- name: github.com/mattn/go-shellwords - name: github.com/mattn/go-shellwords
@ -130,13 +119,13 @@ imports:
- name: github.com/Microsoft/go-winio - name: github.com/Microsoft/go-winio
version: 4f1a71750d95a5a8a46c40a67ffbed8129c2f138 version: 4f1a71750d95a5a8a46c40a67ffbed8129c2f138
- name: github.com/miekg/dns - name: github.com/miekg/dns
version: 48ab6605c66ac797e07f615101c3e9e10e932b66 version: 5d001d020961ae1c184f9f8152fdc73810481677
- name: github.com/moul/http2curl - name: github.com/moul/http2curl
version: b1479103caacaa39319f75e7f57fc545287fca0d version: b1479103caacaa39319f75e7f57fc545287fca0d
- name: github.com/ogier/pflag - name: github.com/ogier/pflag
version: 45c278ab3607870051a2ea9040bb85fcb8557481 version: 45c278ab3607870051a2ea9040bb85fcb8557481
- name: github.com/opencontainers/runc - name: github.com/opencontainers/runc
version: 3211c9f721237f55a16da9c111e3d7e8777e53b5 version: 5dc3f3576efb5262bf582217e93f86c93944374d
subpackages: subpackages:
- libcontainer/user - libcontainer/user
- name: github.com/parnurzeal/gorequest - name: github.com/parnurzeal/gorequest
@ -148,7 +137,7 @@ imports:
- name: github.com/ryanuber/go-glob - name: github.com/ryanuber/go-glob
version: 572520ed46dbddaed19ea3d9541bdd0494163693 version: 572520ed46dbddaed19ea3d9541bdd0494163693
- name: github.com/samuel/go-zookeeper - name: github.com/samuel/go-zookeeper
version: 4b20de542e40ed2b89d65ae195fc20a330919b92 version: e64db453f3512cade908163702045e0f31137843
subpackages: subpackages:
- zk - zk
- name: github.com/Sirupsen/logrus - name: github.com/Sirupsen/logrus
@ -158,7 +147,7 @@ imports:
- name: github.com/stretchr/objx - name: github.com/stretchr/objx
version: cbeaeb16a013161a98496fad62933b1d21786672 version: cbeaeb16a013161a98496fad62933b1d21786672
- name: github.com/stretchr/testify - name: github.com/stretchr/testify
version: 8d64eb7173c7753d6419fd4a9caf057398611364 version: d77da356e56a7428ad25149ca77381849a6a5232
subpackages: subpackages:
- mock - mock
- assert - assert
@ -171,26 +160,34 @@ imports:
- name: github.com/vdemeester/shakers - name: github.com/vdemeester/shakers
version: 24d7f1d6a71aa5d9cbe7390e4afb66b7eef9e1b3 version: 24d7f1d6a71aa5d9cbe7390e4afb66b7eef9e1b3
- name: github.com/vulcand/oxy - name: github.com/vulcand/oxy
version: 11677428db34c4a05354d66d028174d0e3c6e905 version: 1ca0a134a818f7b8ea85de6e6554fe3312f144c2
repo: https://github.com/containous/oxy.git
vcs: git
subpackages: subpackages:
- memmetrics - cbreaker
- connlimit
- forward
- roundrobin
- stream
- utils - utils
- memmetrics
- name: github.com/vulcand/predicate - name: github.com/vulcand/predicate
version: cb0bff91a7ab7cf7571e661ff883fc997bc554a3 version: cb0bff91a7ab7cf7571e661ff883fc997bc554a3
- name: github.com/vulcand/route - name: github.com/vulcand/route
version: cb89d787ddbb1c5849a7ac9f79004c1fd12a4a32 version: cb89d787ddbb1c5849a7ac9f79004c1fd12a4a32
- name: github.com/vulcand/vulcand - name: github.com/vulcand/vulcand
version: 475540bb016702d5b7cc4674e37f48ee3e144a69 version: 42492a3a85e294bdbdd1bcabb8c12769a81ea284
subpackages: subpackages:
- plugin/rewrite - plugin/rewrite
- plugin - plugin
- conntracker
- router - router
- name: github.com/xenolf/lego - name: github.com/xenolf/lego
version: 30a7a8e8821de3532192d1240a45e53c6204f603 version: 30a7a8e8821de3532192d1240a45e53c6204f603
subpackages: subpackages:
- acme - acme
- name: golang.org/x/crypto - name: golang.org/x/crypto
version: 5bcd134fee4dd1475da17714aac19c0aa0142e2f version: f3241ce8505855877cc8a9717bd61a0f7c4ea83c
subpackages: subpackages:
- ocsp - ocsp
- name: golang.org/x/net - name: golang.org/x/net
@ -207,7 +204,7 @@ imports:
- name: gopkg.in/fsnotify.v1 - name: gopkg.in/fsnotify.v1
version: 30411dbcefb7a1da7e84f75530ad3abe4011b4f8 version: 30411dbcefb7a1da7e84f75530ad3abe4011b4f8
- name: gopkg.in/mgo.v2 - name: gopkg.in/mgo.v2
version: b6e2fa371e64216a45e61072a96d4e3859f169da version: 29cc868a5ca65f401ff318143f9408d02f4799cc
subpackages: subpackages:
- bson - bson
- name: gopkg.in/square/go-jose.v1 - name: gopkg.in/square/go-jose.v1

View file

@ -9,7 +9,10 @@ import:
- package: github.com/codegangsta/negroni - package: github.com/codegangsta/negroni
- package: github.com/containous/flaeg - package: github.com/containous/flaeg
version: b98687da5c323650f4513fda6b6203fcbdec9313 version: b98687da5c323650f4513fda6b6203fcbdec9313
- package: github.com/containous/oxy - package: github.com/vulcand/oxy
vcs: git
repo: https://github.com/containous/oxy.git
version: 1ca0a134a818f7b8ea85de6e6554fe3312f144c2
subpackages: subpackages:
- cbreaker - cbreaker
- connlimit - connlimit
@ -57,6 +60,7 @@ import:
subpackages: subpackages:
- plugin/rewrite - plugin/rewrite
- package: github.com/xenolf/lego - package: github.com/xenolf/lego
version: 30a7a8e8821de3532192d1240a45e53c6204f603
subpackages: subpackages:
- acme - acme
- package: golang.org/x/net - package: golang.org/x/net

View file

@ -3,7 +3,7 @@ package middlewares
import ( import (
"net/http" "net/http"
"github.com/containous/oxy/cbreaker" "github.com/vulcand/oxy/cbreaker"
) )
// CircuitBreaker holds the oxy circuit breaker. // CircuitBreaker holds the oxy circuit breaker.

145
middlewares/retry.go Normal file
View file

@ -0,0 +1,145 @@
package middlewares
import (
"bufio"
"bytes"
log "github.com/Sirupsen/logrus"
"github.com/vulcand/oxy/utils"
"net"
"net/http"
)
// Retry is a middleware that retries requests
type Retry struct {
attempts int
next http.Handler
}
// NewRetry returns a new Retry instance
func NewRetry(attempts int, next http.Handler) *Retry {
return &Retry{
attempts: attempts,
next: next,
}
}
func (retry *Retry) ServeHTTP(rw http.ResponseWriter, r *http.Request) {
attempts := 1
for {
recorder := NewRecorder()
recorder.responseWriter = rw
retry.next.ServeHTTP(recorder, r)
if !isNetworkError(recorder.Code) || attempts >= retry.attempts {
utils.CopyHeaders(recorder.Header(), rw.Header())
rw.WriteHeader(recorder.Code)
rw.Write(recorder.Body.Bytes())
break
}
attempts++
log.Debugf("New attempt %d for request: %v", attempts, r.URL)
}
}
func isNetworkError(status int) bool {
return status == http.StatusBadGateway || status == http.StatusGatewayTimeout
}
// ResponseRecorder is an implementation of http.ResponseWriter that
// records its mutations for later inspection in tests.
type ResponseRecorder struct {
Code int // the HTTP response code from WriteHeader
HeaderMap http.Header // the HTTP response headers
Body *bytes.Buffer // if non-nil, the bytes.Buffer to append written data to
Flushed bool
wroteHeader bool
responseWriter http.ResponseWriter
}
// NewRecorder returns an initialized ResponseRecorder.
func NewRecorder() *ResponseRecorder {
return &ResponseRecorder{
HeaderMap: make(http.Header),
Body: new(bytes.Buffer),
Code: 200,
}
}
// Header returns the response headers.
func (rw *ResponseRecorder) Header() http.Header {
m := rw.HeaderMap
if m == nil {
m = make(http.Header)
rw.HeaderMap = m
}
return m
}
// writeHeader writes a header if it was not written yet and
// detects Content-Type if needed.
//
// bytes or str are the beginning of the response body.
// We pass both to avoid unnecessarily generate garbage
// in rw.WriteString which was created for performance reasons.
// Non-nil bytes win.
func (rw *ResponseRecorder) writeHeader(b []byte, str string) {
if rw.wroteHeader {
return
}
if len(str) > 512 {
str = str[:512]
}
_, hasType := rw.HeaderMap["Content-Type"]
hasTE := rw.HeaderMap.Get("Transfer-Encoding") != ""
if !hasType && !hasTE {
if b == nil {
b = []byte(str)
}
if rw.HeaderMap == nil {
rw.HeaderMap = make(http.Header)
}
rw.HeaderMap.Set("Content-Type", http.DetectContentType(b))
}
rw.WriteHeader(200)
}
// Write always succeeds and writes to rw.Body, if not nil.
func (rw *ResponseRecorder) Write(buf []byte) (int, error) {
rw.writeHeader(buf, "")
if rw.Body != nil {
rw.Body.Write(buf)
}
return len(buf), nil
}
// WriteString always succeeds and writes to rw.Body, if not nil.
func (rw *ResponseRecorder) WriteString(str string) (int, error) {
rw.writeHeader(nil, str)
if rw.Body != nil {
rw.Body.WriteString(str)
}
return len(str), nil
}
// WriteHeader sets rw.Code.
func (rw *ResponseRecorder) WriteHeader(code int) {
if !rw.wroteHeader {
rw.Code = code
rw.wroteHeader = true
}
}
// Flush sets rw.Flushed to true.
func (rw *ResponseRecorder) Flush() {
if !rw.wroteHeader {
rw.WriteHeader(200)
}
rw.Flushed = true
}
// Hijack hijacks the connection
func (rw *ResponseRecorder) Hijack() (net.Conn, *bufio.ReadWriter, error) {
return rw.responseWriter.(http.Hijacker).Hijack()
}

View file

@ -14,25 +14,23 @@ import (
"reflect" "reflect"
"regexp" "regexp"
"sort" "sort"
"strconv"
"syscall" "syscall"
"time" "time"
log "github.com/Sirupsen/logrus" log "github.com/Sirupsen/logrus"
"github.com/codegangsta/negroni" "github.com/codegangsta/negroni"
"github.com/containous/mux" "github.com/containous/mux"
"github.com/containous/oxy/cbreaker"
"github.com/containous/oxy/connlimit"
"github.com/containous/oxy/forward"
"github.com/containous/oxy/roundrobin"
"github.com/containous/oxy/stream"
"github.com/containous/oxy/utils"
"github.com/containous/traefik/middlewares" "github.com/containous/traefik/middlewares"
"github.com/containous/traefik/provider" "github.com/containous/traefik/provider"
"github.com/containous/traefik/safe" "github.com/containous/traefik/safe"
"github.com/containous/traefik/types" "github.com/containous/traefik/types"
"github.com/mailgun/manners" "github.com/mailgun/manners"
"github.com/streamrail/concurrent-map" "github.com/streamrail/concurrent-map"
"github.com/vulcand/oxy/cbreaker"
"github.com/vulcand/oxy/connlimit"
"github.com/vulcand/oxy/forward"
"github.com/vulcand/oxy/roundrobin"
"github.com/vulcand/oxy/utils"
) )
var oxyLogger = &OxyLogger{} var oxyLogger = &OxyLogger{}
@ -469,21 +467,8 @@ func (server *Server) loadConfig(configurations configs, globalConfiguration Glo
if globalConfiguration.Retry.Attempts > 0 { if globalConfiguration.Retry.Attempts > 0 {
retries = globalConfiguration.Retry.Attempts retries = globalConfiguration.Retry.Attempts
} }
maxMem := int64(2 * 1024 * 1024) lb = middlewares.NewRetry(retries, lb)
if globalConfiguration.Retry.MaxMem > 0 {
maxMem = globalConfiguration.Retry.MaxMem
}
lb, err = stream.New(lb,
stream.Logger(oxyLogger),
stream.Retry("IsNetworkError() && Attempts() < "+strconv.Itoa(retries)),
stream.MemRequestBodyBytes(maxMem),
stream.MaxRequestBodyBytes(maxMem),
stream.MemResponseBodyBytes(maxMem),
stream.MaxResponseBodyBytes(maxMem))
log.Debugf("Creating retries max attempts %d", retries) log.Debugf("Creating retries max attempts %d", retries)
if err != nil {
return nil, err
}
} }
var negroni = negroni.New() var negroni = negroni.New()

View file

@ -157,13 +157,6 @@
# #
# attempts = 3 # attempts = 3
# Sets the maximum request body to be stored in memory in Mo
#
# Optional
# Default: 2
#
# maxMem = 3
################################################################ ################################################################
# Web configuration backend # Web configuration backend
################################################################ ################################################################