service.go 8.9 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14
// Copyright 2017 fatedier, fatedier@gmail.com
//
// 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.

F
fatedier 已提交
15 16 17 18
package server

import (
	"fmt"
F
fatedier 已提交
19 20
	"net"
	"net/http"
F
fatedier 已提交
21 22 23
	"time"

	"github.com/fatedier/frp/assets"
F
fatedier 已提交
24
	"github.com/fatedier/frp/g"
F
fatedier 已提交
25 26
	"github.com/fatedier/frp/models/msg"
	"github.com/fatedier/frp/utils/log"
27
	frpNet "github.com/fatedier/frp/utils/net"
F
fatedier 已提交
28 29 30
	"github.com/fatedier/frp/utils/util"
	"github.com/fatedier/frp/utils/version"
	"github.com/fatedier/frp/utils/vhost"
31

F
fatedier 已提交
32
	fmux "github.com/hashicorp/yamux"
F
fatedier 已提交
33 34
)

F
fatedier 已提交
35 36 37 38
const (
	connReadTimeout time.Duration = 10 * time.Second
)

39 40
var ServerService *Service

F
fatedier 已提交
41 42 43
// Server service.
type Service struct {
	// Accept connections from client.
44
	listener frpNet.Listener
F
fatedier 已提交
45

F
fatedier 已提交
46 47 48
	// Accept connections using kcp.
	kcpListener frpNet.Listener

F
fatedier 已提交
49 50 51
	// For https proxies, route requests to different clients by hostname and other infomation.
	VhostHttpsMuxer *vhost.HttpsMuxer

F
fatedier 已提交
52 53
	httpReverseProxy *vhost.HttpReverseProxy

F
fatedier 已提交
54 55 56 57 58
	// Manage all controllers.
	ctlManager *ControlManager

	// Manage all proxies.
	pxyManager *ProxyManager
F
fatedier 已提交
59

F
fatedier 已提交
60 61
	// Manage all visitor listeners.
	visitorManager *VisitorManager
F
fatedier 已提交
62

63 64 65 66 67 68
	// Manage all tcp ports.
	tcpPortManager *PortManager

	// Manage all udp ports.
	udpPortManager *PortManager

F
fatedier 已提交
69 70
	// Controller for nat hole connections.
	natHoleController *NatHoleController
F
fatedier 已提交
71 72 73
}

func NewService() (svr *Service, err error) {
F
fatedier 已提交
74
	cfg := &g.GlbServerCfg.ServerCommonConf
F
fatedier 已提交
75
	svr = &Service{
F
fatedier 已提交
76 77 78
		ctlManager:     NewControlManager(),
		pxyManager:     NewProxyManager(),
		visitorManager: NewVisitorManager(),
F
fatedier 已提交
79 80
		tcpPortManager: NewPortManager("tcp", cfg.ProxyBindAddr, cfg.AllowPorts),
		udpPortManager: NewPortManager("udp", cfg.ProxyBindAddr, cfg.AllowPorts),
F
fatedier 已提交
81 82 83
	}

	// Init assets.
F
fatedier 已提交
84
	err = assets.Load(cfg.AssetsDir)
F
fatedier 已提交
85 86 87 88 89 90
	if err != nil {
		err = fmt.Errorf("Load assets error: %v", err)
		return
	}

	// Listen for accepting connections from client.
F
fatedier 已提交
91
	svr.listener, err = frpNet.ListenTcp(cfg.BindAddr, cfg.BindPort)
F
fatedier 已提交
92 93 94 95
	if err != nil {
		err = fmt.Errorf("Create server listener error, %v", err)
		return
	}
F
fatedier 已提交
96
	log.Info("frps tcp listen on %s:%d", cfg.BindAddr, cfg.BindPort)
F
fatedier 已提交
97 98

	// Listen for accepting connections from client using kcp protocol.
F
fatedier 已提交
99 100
	if cfg.KcpBindPort > 0 {
		svr.kcpListener, err = frpNet.ListenKcp(cfg.BindAddr, cfg.KcpBindPort)
F
fatedier 已提交
101
		if err != nil {
F
fatedier 已提交
102
			err = fmt.Errorf("Listen on kcp address udp [%s:%d] error: %v", cfg.BindAddr, cfg.KcpBindPort, err)
F
fatedier 已提交
103 104
			return
		}
F
fatedier 已提交
105
		log.Info("frps kcp listen on udp %s:%d", cfg.BindAddr, cfg.KcpBindPort)
F
fatedier 已提交
106
	}
F
fatedier 已提交
107 108

	// Create http vhost muxer.
F
fatedier 已提交
109
	if cfg.VhostHttpPort > 0 {
F
fatedier 已提交
110 111 112 113 114 115 116
		rp := vhost.NewHttpReverseProxy()
		svr.httpReverseProxy = rp

		address := fmt.Sprintf("%s:%d", cfg.ProxyBindAddr, cfg.VhostHttpPort)
		server := &http.Server{
			Addr:    address,
			Handler: rp,
F
fatedier 已提交
117
		}
F
fatedier 已提交
118 119
		var l net.Listener
		l, err = net.Listen("tcp", address)
F
fatedier 已提交
120
		if err != nil {
F
fatedier 已提交
121
			err = fmt.Errorf("Create vhost http listener error, %v", err)
F
fatedier 已提交
122 123
			return
		}
F
fatedier 已提交
124
		go server.Serve(l)
F
fatedier 已提交
125
		log.Info("http service listen on %s:%d", cfg.ProxyBindAddr, cfg.VhostHttpPort)
F
fatedier 已提交
126 127 128
	}

	// Create https vhost muxer.
F
fatedier 已提交
129
	if cfg.VhostHttpsPort > 0 {
130
		var l frpNet.Listener
F
fatedier 已提交
131
		l, err = frpNet.ListenTcp(cfg.ProxyBindAddr, cfg.VhostHttpsPort)
F
fatedier 已提交
132 133 134 135 136 137 138 139 140
		if err != nil {
			err = fmt.Errorf("Create vhost https listener error, %v", err)
			return
		}
		svr.VhostHttpsMuxer, err = vhost.NewHttpsMuxer(l, 30*time.Second)
		if err != nil {
			err = fmt.Errorf("Create vhost httpsMuxer error, %v", err)
			return
		}
F
fatedier 已提交
141 142 143 144 145 146 147 148 149 150 151 152 153 154
		log.Info("https service listen on %s:%d", cfg.ProxyBindAddr, cfg.VhostHttpsPort)
	}

	// Create nat hole controller.
	if cfg.BindUdpPort > 0 {
		var nc *NatHoleController
		addr := fmt.Sprintf("%s:%d", cfg.BindAddr, cfg.BindUdpPort)
		nc, err = NewNatHoleController(addr)
		if err != nil {
			err = fmt.Errorf("Create nat hole controller error, %v", err)
			return
		}
		svr.natHoleController = nc
		log.Info("nat hole udp service listen on %s:%d", cfg.BindAddr, cfg.BindUdpPort)
F
fatedier 已提交
155 156 157
	}

	// Create dashboard web server.
F
fatedier 已提交
158
	if cfg.DashboardPort > 0 {
T
timerever 已提交
159
		err = RunDashboardServer(cfg.DashboardAddr, cfg.DashboardPort)
F
fatedier 已提交
160 161 162 163
		if err != nil {
			err = fmt.Errorf("Create dashboard web server error, %v", err)
			return
		}
T
timerever 已提交
164
		log.Info("Dashboard listen on %s:%d", cfg.DashboardAddr, cfg.DashboardPort)
F
fatedier 已提交
165 166 167 168 169
	}
	return
}

func (svr *Service) Run() {
F
fatedier 已提交
170 171 172
	if svr.natHoleController != nil {
		go svr.natHoleController.Run()
	}
F
fatedier 已提交
173
	if g.GlbServerCfg.KcpBindPort > 0 {
F
fatedier 已提交
174 175 176 177 178 179 180
		go svr.HandleListener(svr.kcpListener)
	}
	svr.HandleListener(svr.listener)

}

func (svr *Service) HandleListener(l frpNet.Listener) {
F
fatedier 已提交
181 182
	// Listen for incoming connections from client.
	for {
F
fatedier 已提交
183
		c, err := l.Accept()
F
fatedier 已提交
184 185 186 187 188 189
		if err != nil {
			log.Warn("Listener for incoming connections from client closed")
			return
		}

		// Start a new goroutine for dealing connections.
190 191 192 193 194
		go func(frpConn frpNet.Conn) {
			dealFn := func(conn frpNet.Conn) {
				var rawMsg msg.Message
				conn.SetReadDeadline(time.Now().Add(connReadTimeout))
				if rawMsg, err = msg.ReadMsg(conn); err != nil {
F
fatedier 已提交
195
					log.Trace("Failed to read message: %v", err)
196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215
					conn.Close()
					return
				}
				conn.SetReadDeadline(time.Time{})

				switch m := rawMsg.(type) {
				case *msg.Login:
					err = svr.RegisterControl(conn, m)
					// If login failed, send error message there.
					// Otherwise send success message in control's work goroutine.
					if err != nil {
						conn.Warn("%v", err)
						msg.WriteMsg(conn, &msg.LoginResp{
							Version: version.Full(),
							Error:   err.Error(),
						})
						conn.Close()
					}
				case *msg.NewWorkConn:
					svr.RegisterWorkConn(conn, m)
F
fatedier 已提交
216 217
				case *msg.NewVisitorConn:
					if err = svr.RegisterVisitorConn(conn, m); err != nil {
F
fatedier 已提交
218
						conn.Warn("%v", err)
F
fatedier 已提交
219
						msg.WriteMsg(conn, &msg.NewVisitorConnResp{
F
fatedier 已提交
220 221 222 223 224
							ProxyName: m.ProxyName,
							Error:     err.Error(),
						})
						conn.Close()
					} else {
F
fatedier 已提交
225
						msg.WriteMsg(conn, &msg.NewVisitorConnResp{
F
fatedier 已提交
226 227 228 229
							ProxyName: m.ProxyName,
							Error:     "",
						})
					}
230 231 232 233
				default:
					log.Warn("Error message type for the new connection [%s]", conn.RemoteAddr().String())
					conn.Close()
				}
F
fatedier 已提交
234 235
			}

F
fatedier 已提交
236
			if g.GlbServerCfg.TcpMux {
F
fatedier 已提交
237
				session, err := fmux.Server(frpConn, nil)
F
fatedier 已提交
238
				if err != nil {
239
					log.Warn("Failed to create mux connection: %v", err)
F
fatedier 已提交
240
					frpConn.Close()
241 242 243 244 245 246 247
					return
				}

				for {
					stream, err := session.AcceptStream()
					if err != nil {
						log.Warn("Accept new mux stream error: %v", err)
F
fatedier 已提交
248
						session.Close()
249 250 251 252
						return
					}
					wrapConn := frpNet.WrapConn(stream)
					go dealFn(wrapConn)
F
fatedier 已提交
253
				}
254 255
			} else {
				dealFn(frpConn)
F
fatedier 已提交
256 257 258 259 260
			}
		}(c)
	}
}

261
func (svr *Service) RegisterControl(ctlConn frpNet.Conn, loginMsg *msg.Login) (err error) {
F
fatedier 已提交
262 263 264 265 266 267 268 269 270 271 272
	ctlConn.Info("client login info: ip [%s] version [%s] hostname [%s] os [%s] arch [%s]",
		ctlConn.RemoteAddr().String(), loginMsg.Version, loginMsg.Hostname, loginMsg.Os, loginMsg.Arch)

	// Check client version.
	if ok, msg := version.Compat(loginMsg.Version); !ok {
		err = fmt.Errorf("%s", msg)
		return
	}

	// Check auth.
	nowTime := time.Now().Unix()
F
fatedier 已提交
273
	if g.GlbServerCfg.AuthTimeout != 0 && nowTime-loginMsg.Timestamp > g.GlbServerCfg.AuthTimeout {
F
fatedier 已提交
274 275 276
		err = fmt.Errorf("authorization timeout")
		return
	}
F
fatedier 已提交
277
	if util.GetAuthKey(g.GlbServerCfg.Token, loginMsg.Timestamp) != loginMsg.PrivilegeKey {
F
fatedier 已提交
278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293
		err = fmt.Errorf("authorization failed")
		return
	}

	// If client's RunId is empty, it's a new client, we just create a new controller.
	// Otherwise, we check if there is one controller has the same run id. If so, we release previous controller and start new one.
	if loginMsg.RunId == "" {
		loginMsg.RunId, err = util.RandId()
		if err != nil {
			return
		}
	}

	ctl := NewControl(svr, ctlConn, loginMsg)

	if oldCtl := svr.ctlManager.Add(loginMsg.RunId, ctl); oldCtl != nil {
F
fatedier 已提交
294
		oldCtl.allShutdown.WaitDone()
F
fatedier 已提交
295 296 297 298
	}

	ctlConn.AddLogPrefix(loginMsg.RunId)
	ctl.Start()
299 300 301

	// for statistics
	StatsNewClient()
F
fatedier 已提交
302 303 304 305
	return
}

// RegisterWorkConn register a new work connection to control and proxies need it.
306
func (svr *Service) RegisterWorkConn(workConn frpNet.Conn, newMsg *msg.NewWorkConn) {
F
fatedier 已提交
307 308 309 310 311 312 313 314 315
	ctl, exist := svr.ctlManager.GetById(newMsg.RunId)
	if !exist {
		workConn.Warn("No client control found for run id [%s]", newMsg.RunId)
		return
	}
	ctl.RegisterWorkConn(workConn)
	return
}

F
fatedier 已提交
316 317
func (svr *Service) RegisterVisitorConn(visitorConn frpNet.Conn, newMsg *msg.NewVisitorConn) error {
	return svr.visitorManager.NewConn(newMsg.ProxyName, visitorConn, newMsg.Timestamp, newMsg.SignKey,
F
fatedier 已提交
318 319 320
		newMsg.UseEncryption, newMsg.UseCompression)
}

F
fatedier 已提交
321
func (svr *Service) RegisterProxy(name string, pxy Proxy) error {
F
fatedier 已提交
322
	return svr.pxyManager.Add(name, pxy)
F
fatedier 已提交
323 324 325 326 327
}

func (svr *Service) DelProxy(name string) {
	svr.pxyManager.Del(name)
}