gf/net/gtcp/gtcp_server.go

191 lines
5.3 KiB
Go
Raw Normal View History

2021-01-17 21:46:25 +08:00
// Copyright GoFrame Author(https://goframe.org). All Rights Reserved.
2017-12-29 16:03:30 +08:00
//
// This Source Code Form is subject to the terms of the MIT License.
// If a copy of the MIT was not distributed with this file,
// You can obtain one at https://github.com/gogf/gf.
2017-12-29 16:03:30 +08:00
2017-11-23 10:21:28 +08:00
package gtcp
import (
2019-05-20 08:59:16 +08:00
"crypto/tls"
"fmt"
2019-05-20 08:59:16 +08:00
"net"
"sync"
2019-07-29 21:01:19 +08:00
"github.com/gogf/gf/v2/container/gmap"
2021-11-13 23:23:55 +08:00
"github.com/gogf/gf/v2/errors/gcode"
"github.com/gogf/gf/v2/errors/gerror"
"github.com/gogf/gf/v2/text/gstr"
2021-10-11 21:41:56 +08:00
"github.com/gogf/gf/v2/util/gconv"
2017-11-23 10:21:28 +08:00
)
2018-07-20 11:41:23 +08:00
const (
// FreePortAddress marks the server listens using random free port.
FreePortAddress = ":0"
)
const (
defaultServer = "default"
2018-07-20 11:41:23 +08:00
)
// Server is a TCP server.
2018-07-20 11:41:23 +08:00
type Server struct {
mu sync.Mutex // Used for Server.listen concurrent safety. -- The golang test with data race checks this.
listen net.Listener // TCP address listener.
address string // Server listening address.
handler func(*Conn) // Connection handler.
tlsConfig *tls.Config // TLS configuration.
2018-07-20 11:41:23 +08:00
}
2019-05-20 08:59:16 +08:00
// Map for name to server, for singleton purpose.
var serverMapping = gmap.NewStrAnyMap(true)
2018-07-20 11:41:23 +08:00
// GetServer returns the TCP server with specified `name`,
// or it returns a new normal TCP server named `name` if it does not exist.
// The parameter `name` is used to specify the TCP server
2019-06-19 09:06:52 +08:00
func GetServer(name ...interface{}) *Server {
serverName := defaultServer
2019-07-28 17:37:13 +08:00
if len(name) > 0 && name[0] != "" {
2019-06-19 09:06:52 +08:00
serverName = gconv.String(name[0])
}
return serverMapping.GetOrSetFuncLock(serverName, func() interface{} {
return NewServer("", nil)
}).(*Server)
2018-07-20 11:41:23 +08:00
}
2019-05-20 08:59:16 +08:00
// NewServer creates and returns a new normal TCP server.
// The parameter `name` is optional, which is used to specify the instance name of the server.
2019-06-19 09:06:52 +08:00
func NewServer(address string, handler func(*Conn), name ...string) *Server {
s := &Server{
address: address,
handler: handler,
}
if len(name) > 0 && name[0] != "" {
2019-06-19 09:06:52 +08:00
serverMapping.Set(name[0], s)
}
return s
2018-07-20 11:41:23 +08:00
}
2019-06-04 23:33:46 +08:00
// NewServerTLS creates and returns a new TCP server with TLS support.
// The parameter `name` is optional, which is used to specify the instance name of the server.
2019-06-19 09:06:52 +08:00
func NewServerTLS(address string, tlsConfig *tls.Config, handler func(*Conn), name ...string) *Server {
2019-05-20 08:59:16 +08:00
s := NewServer(address, handler, name...)
2019-06-04 23:33:46 +08:00
s.SetTLSConfig(tlsConfig)
return s
}
// NewServerKeyCrt creates and returns a new TCP server with TLS support.
// The parameter `name` is optional, which is used to specify the instance name of the server.
func NewServerKeyCrt(address, crtFile, keyFile string, handler func(*Conn), name ...string) (*Server, error) {
2019-06-04 23:33:46 +08:00
s := NewServer(address, handler, name...)
if err := s.SetTLSKeyCrt(crtFile, keyFile); err != nil {
return nil, err
2019-06-04 23:33:46 +08:00
}
return s, nil
2019-05-20 08:59:16 +08:00
}
2019-06-04 18:26:32 +08:00
// SetAddress sets the listening address for server.
2019-06-04 23:33:46 +08:00
func (s *Server) SetAddress(address string) {
2019-06-19 09:06:52 +08:00
s.address = address
2017-12-30 17:09:00 +08:00
}
// GetAddress get the listening address for server.
func (s *Server) GetAddress() string {
return s.address
}
2019-06-04 18:26:32 +08:00
// SetHandler sets the connection handler for server.
2019-06-19 09:06:52 +08:00
func (s *Server) SetHandler(handler func(*Conn)) {
s.handler = handler
2017-12-30 17:09:00 +08:00
}
// SetTLSKeyCrt sets the certificate and key file for TLS configuration of server.
2019-06-04 23:33:46 +08:00
func (s *Server) SetTLSKeyCrt(crtFile, keyFile string) error {
tlsConfig, err := LoadKeyCrt(crtFile, keyFile)
2019-05-20 08:59:16 +08:00
if err != nil {
return err
}
2019-06-04 23:33:46 +08:00
s.tlsConfig = tlsConfig
2019-05-20 08:59:16 +08:00
return nil
}
// SetTLSConfig sets the TLS configuration of server.
2019-05-20 08:59:16 +08:00
func (s *Server) SetTLSConfig(tlsConfig *tls.Config) {
s.tlsConfig = tlsConfig
}
// Close closes the listener and shutdowns the server.
func (s *Server) Close() error {
s.mu.Lock()
defer s.mu.Unlock()
if s.listen == nil {
return nil
}
return s.listen.Close()
}
2019-05-20 08:59:16 +08:00
// Run starts running the TCP Server.
func (s *Server) Run() (err error) {
2019-06-19 09:06:52 +08:00
if s.handler == nil {
err = gerror.NewCode(gcode.CodeMissingConfiguration, "start running failed: socket handler not defined")
2019-06-19 09:06:52 +08:00
return
}
if s.tlsConfig != nil {
// TLS Server
s.mu.Lock()
2019-06-19 09:06:52 +08:00
s.listen, err = tls.Listen("tcp", s.address, s.tlsConfig)
s.mu.Unlock()
2019-06-19 09:06:52 +08:00
if err != nil {
err = gerror.Wrapf(err, `tls.Listen failed for address "%s"`, s.address)
2019-06-19 09:06:52 +08:00
return
}
} else {
// Normal Server
var tcpAddr *net.TCPAddr
if tcpAddr, err = net.ResolveTCPAddr("tcp", s.address); err != nil {
err = gerror.Wrapf(err, `net.ResolveTCPAddr failed for address "%s"`, s.address)
2019-06-19 09:06:52 +08:00
return err
}
s.mu.Lock()
s.listen, err = net.ListenTCP("tcp", tcpAddr)
s.mu.Unlock()
2019-06-19 09:06:52 +08:00
if err != nil {
err = gerror.Wrapf(err, `net.ListenTCP failed for address "%s"`, s.address)
2019-06-19 09:06:52 +08:00
return err
}
}
2019-12-11 14:50:25 +08:00
// Listening loop.
2019-06-19 09:06:52 +08:00
for {
var conn net.Conn
if conn, err = s.listen.Accept(); err != nil {
err = gerror.Wrapf(err, `Listener.Accept failed`)
2019-06-19 09:06:52 +08:00
return err
} else if conn != nil {
go s.handler(NewConnByNetConn(conn))
}
}
2017-11-23 10:21:28 +08:00
}
// GetListenedAddress retrieves and returns the address string which are listened by current server.
func (s *Server) GetListenedAddress() string {
if !gstr.Contains(s.address, FreePortAddress) {
return s.address
}
var (
address = s.address
listenedPort = s.GetListenedPort()
)
address = gstr.Replace(address, FreePortAddress, fmt.Sprintf(`:%d`, listenedPort))
return address
}
// GetListenedPort retrieves and returns one port which is listened to by current server.
func (s *Server) GetListenedPort() int {
s.mu.Lock()
defer s.mu.Unlock()
if ln := s.listen; ln != nil {
return ln.Addr().(*net.TCPAddr).Port
}
return -1
}