-
Notifications
You must be signed in to change notification settings - Fork 56
/
ws_reverseporxy_option.go
140 lines (116 loc) · 3.58 KB
/
ws_reverseporxy_option.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
package proxy
import (
"errors"
"net/http"
"net/url"
"github.com/fasthttp/websocket"
"github.com/valyala/fasthttp"
)
// OptionWS to define all options to reverse web socket proxy.
type OptionWS interface {
apply(o *buildOptionWS)
}
type funcBuildOptionWS struct {
f func(o *buildOptionWS)
}
func newFuncBuildOptionWS(f func(o *buildOptionWS)) funcBuildOptionWS { return funcBuildOptionWS{f: f} }
func (fb funcBuildOptionWS) apply(o *buildOptionWS) { fb.f(o) }
type forwardHeaderHandler func(ctx *fasthttp.RequestCtx) (forwardHeader http.Header)
// buildOptionWS is Option for WS reverse-proxy
type buildOptionWS struct {
// logger is used to log messages.
logger __Logger
// debug is used to enable debug mode.
debug bool
// target indicates which backend server to proxy.
target *url.URL
// fn is forwardHeaderHandler which allows users customize themselves' forward headers
// to be proxied to backend server.
fn forwardHeaderHandler
// dialer contains options for connecting to the backend WebSocket server.
// If nil, DefaultDialer is used.
dialer *websocket.Dialer
// upgrader specifies the parameters for upgrading a incoming HTTP
// connection to a WebSocket connection. If nil, DefaultUpgrader is used.
upgrader *websocket.FastHTTPUpgrader
// dynamicPathFeature specifies if developer would be able to override path for base URL of`target`dynamically
// Default value is false
dynamicPathFeature *dynamicPathFeatureOp
}
type dynamicPathFeatureOp struct {
enable bool
headerValue string
}
func (o *buildOptionWS) validate() error {
if o == nil {
return errors.New("option is nil")
}
if o.target == nil {
return errors.New("target is nil")
}
return nil
}
func defaultBuildOptionWS() *buildOptionWS {
return &buildOptionWS{
logger: &nopLogger{},
debug: false,
target: nil,
fn: nil,
dialer: nil,
upgrader: nil,
dynamicPathFeature: nil,
}
}
// WithURL_OptionWS specify the url to backend websocket server.
// WithURL_OptionWS("ws://YOUR_WEBSOCKET_HOST:PORT/AND/PATH")
func WithURL_OptionWS(u string) OptionWS {
return newFuncBuildOptionWS(func(o *buildOptionWS) {
URL, err := url.Parse(u)
if err != nil {
panic(err)
}
o.target = URL
})
}
// WithDebug_OptionWS is used to enable debug mode.
func WithDebug_OptionWS() OptionWS {
return newFuncBuildOptionWS(func(o *buildOptionWS) {
o.debug = true
})
}
// WithDialer_OptionWS use specified dialer
func WithDialer_OptionWS(dialer *websocket.Dialer) OptionWS {
return newFuncBuildOptionWS(func(o *buildOptionWS) {
o.dialer = dialer
})
}
// WithUpgrader_OptionWS use specified upgrader.
func WithUpgrader_OptionWS(upgrader *websocket.FastHTTPUpgrader) OptionWS {
return newFuncBuildOptionWS(func(o *buildOptionWS) {
o.upgrader = upgrader
})
}
// WithForwardHeadersHandlers_OptionWS allows users to customize forward headers.
func WithForwardHeadersHandlers_OptionWS(handler forwardHeaderHandler) OptionWS {
return newFuncBuildOptionWS(func(o *buildOptionWS) {
o.fn = handler
})
}
// WithDynamicPath_OptionWS enable/disable dynamic path overriding explicitly
// WithDynamicPath_OptionWS(true)
func WithDynamicPath_OptionWS(t bool, header ...string) OptionWS {
return newFuncBuildOptionWS(func(o *buildOptionWS) {
if o.dynamicPathFeature == nil {
o.dynamicPathFeature = &dynamicPathFeatureOp{}
}
o.dynamicPathFeature.enable = t
if !t {
return
}
if len(header) > 0 {
o.dynamicPathFeature.headerValue = header[0]
} else {
o.dynamicPathFeature.headerValue = DefaultOverrideHeader
}
})
}