2016-01-28 16:08:32 +00:00
|
|
|
package hub
|
2016-01-27 21:11:31 +00:00
|
|
|
|
|
|
|
import (
|
2016-05-02 22:16:07 +00:00
|
|
|
"crypto/tls"
|
2016-03-09 10:34:27 +00:00
|
|
|
"errors"
|
2016-01-27 21:11:31 +00:00
|
|
|
"net"
|
2016-05-29 20:33:04 +00:00
|
|
|
"sync"
|
2016-01-27 21:11:31 +00:00
|
|
|
|
|
|
|
"github.com/v2ray/v2ray-core/common/log"
|
|
|
|
v2net "github.com/v2ray/v2ray-core/common/net"
|
|
|
|
)
|
|
|
|
|
2016-03-09 10:34:27 +00:00
|
|
|
var (
|
|
|
|
ErrorClosedConnection = errors.New("Connection already closed.")
|
|
|
|
)
|
|
|
|
|
2016-01-28 19:47:00 +00:00
|
|
|
type TCPHub struct {
|
2016-05-29 20:33:04 +00:00
|
|
|
sync.Mutex
|
2016-05-02 22:16:07 +00:00
|
|
|
listener net.Listener
|
2016-04-27 21:01:31 +00:00
|
|
|
connCallback ConnectionHandler
|
2016-01-27 21:11:31 +00:00
|
|
|
accepting bool
|
|
|
|
}
|
|
|
|
|
2016-05-29 14:37:52 +00:00
|
|
|
func ListenTCP(address v2net.Address, port v2net.Port, callback ConnectionHandler, tlsConfig *tls.Config) (*TCPHub, error) {
|
2016-01-27 21:11:31 +00:00
|
|
|
listener, err := net.ListenTCP("tcp", &net.TCPAddr{
|
2016-05-29 14:37:52 +00:00
|
|
|
IP: address.IP(),
|
2016-01-27 21:11:31 +00:00
|
|
|
Port: int(port),
|
|
|
|
Zone: "",
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2016-05-02 22:16:07 +00:00
|
|
|
var hub *TCPHub
|
|
|
|
if tlsConfig != nil {
|
|
|
|
tlsListener := tls.NewListener(listener, tlsConfig)
|
|
|
|
hub = &TCPHub{
|
|
|
|
listener: tlsListener,
|
|
|
|
connCallback: callback,
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
hub = &TCPHub{
|
|
|
|
listener: listener,
|
|
|
|
connCallback: callback,
|
|
|
|
}
|
2016-01-27 21:11:31 +00:00
|
|
|
}
|
2016-05-02 22:16:07 +00:00
|
|
|
|
|
|
|
go hub.start()
|
|
|
|
return hub, nil
|
2016-01-27 21:11:31 +00:00
|
|
|
}
|
|
|
|
|
2016-01-28 19:47:00 +00:00
|
|
|
func (this *TCPHub) Close() {
|
2016-01-27 21:11:31 +00:00
|
|
|
this.accepting = false
|
|
|
|
this.listener.Close()
|
|
|
|
}
|
|
|
|
|
2016-01-28 19:47:00 +00:00
|
|
|
func (this *TCPHub) start() {
|
2016-01-27 21:11:31 +00:00
|
|
|
this.accepting = true
|
|
|
|
for this.accepting {
|
2016-05-02 22:16:07 +00:00
|
|
|
conn, err := this.listener.Accept()
|
2016-05-29 20:33:04 +00:00
|
|
|
|
2016-01-27 21:11:31 +00:00
|
|
|
if err != nil {
|
2016-02-18 10:36:21 +00:00
|
|
|
if this.accepting {
|
|
|
|
log.Warning("Listener: Failed to accept new TCP connection: ", err)
|
|
|
|
}
|
2016-01-27 21:11:31 +00:00
|
|
|
continue
|
|
|
|
}
|
2016-05-02 21:53:16 +00:00
|
|
|
go this.connCallback(&Connection{
|
2016-05-30 22:21:41 +00:00
|
|
|
dest: conn.RemoteAddr().String(),
|
2016-01-27 21:11:31 +00:00
|
|
|
conn: conn,
|
|
|
|
listener: this,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-05-29 20:33:04 +00:00
|
|
|
// @Private
|
2016-05-30 22:21:41 +00:00
|
|
|
func (this *TCPHub) Recycle(dest string, conn net.Conn) {
|
2016-05-29 20:33:04 +00:00
|
|
|
if this.accepting {
|
|
|
|
go this.connCallback(&Connection{
|
2016-05-30 22:21:41 +00:00
|
|
|
dest: dest,
|
2016-05-29 20:33:04 +00:00
|
|
|
conn: conn,
|
|
|
|
listener: this,
|
|
|
|
})
|
|
|
|
}
|
2016-01-27 21:11:31 +00:00
|
|
|
}
|