2018-06-26 16:15:48 +08:00
|
|
|
package mixed
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"net"
|
|
|
|
"net/url"
|
|
|
|
|
2020-10-01 22:49:14 +08:00
|
|
|
"github.com/nadoo/glider/log"
|
2018-06-26 16:15:48 +08:00
|
|
|
"github.com/nadoo/glider/proxy"
|
|
|
|
"github.com/nadoo/glider/proxy/http"
|
|
|
|
"github.com/nadoo/glider/proxy/socks5"
|
|
|
|
)
|
|
|
|
|
2019-09-18 22:08:48 +08:00
|
|
|
// Mixed struct.
|
|
|
|
type Mixed struct {
|
2019-09-18 19:40:14 +08:00
|
|
|
proxy proxy.Proxy
|
|
|
|
addr string
|
2018-08-12 12:37:25 +08:00
|
|
|
|
2019-10-19 22:16:15 +08:00
|
|
|
httpServer *http.HTTP
|
|
|
|
socks5Server *socks5.Socks5
|
2018-08-12 12:37:25 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
func init() {
|
2019-09-18 22:08:48 +08:00
|
|
|
proxy.RegisterServer("mixed", NewMixedServer)
|
2018-06-26 16:15:48 +08:00
|
|
|
}
|
|
|
|
|
2019-09-18 22:08:48 +08:00
|
|
|
// NewMixed returns a mixed proxy.
|
|
|
|
func NewMixed(s string, p proxy.Proxy) (*Mixed, error) {
|
2018-06-26 16:15:48 +08:00
|
|
|
u, err := url.Parse(s)
|
|
|
|
if err != nil {
|
|
|
|
log.F("parse err: %s", err)
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2019-09-18 22:08:48 +08:00
|
|
|
m := &Mixed{
|
2019-09-18 19:40:14 +08:00
|
|
|
proxy: p,
|
|
|
|
addr: u.Host,
|
2018-06-26 16:15:48 +08:00
|
|
|
}
|
|
|
|
|
2019-10-19 22:16:15 +08:00
|
|
|
m.httpServer, err = http.NewHTTP(s, nil, p)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
m.socks5Server, err = socks5.NewSocks5(s, nil, p)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2018-06-26 16:15:48 +08:00
|
|
|
|
2019-09-18 19:40:14 +08:00
|
|
|
return m, nil
|
2018-06-26 16:15:48 +08:00
|
|
|
}
|
|
|
|
|
2019-09-18 22:08:48 +08:00
|
|
|
// NewMixedServer returns a mixed server.
|
|
|
|
func NewMixedServer(s string, p proxy.Proxy) (proxy.Server, error) {
|
|
|
|
return NewMixed(s, p)
|
2018-06-26 16:15:48 +08:00
|
|
|
}
|
|
|
|
|
2019-09-18 22:08:48 +08:00
|
|
|
// ListenAndServe listens on server's addr and serves connections.
|
|
|
|
func (m *Mixed) ListenAndServe() {
|
2019-10-19 22:16:15 +08:00
|
|
|
go m.socks5Server.ListenAndServeUDP()
|
2018-06-26 16:15:48 +08:00
|
|
|
|
2019-09-18 22:08:48 +08:00
|
|
|
l, err := net.Listen("tcp", m.addr)
|
2018-11-25 13:18:15 +08:00
|
|
|
if err != nil {
|
2019-09-18 22:08:48 +08:00
|
|
|
log.F("[mixed] failed to listen on %s: %v", m.addr, err)
|
2018-11-25 13:18:15 +08:00
|
|
|
return
|
|
|
|
}
|
2018-06-26 16:15:48 +08:00
|
|
|
|
2019-09-18 22:08:48 +08:00
|
|
|
log.F("[mixed] listening TCP on %s", m.addr)
|
2018-11-25 13:18:15 +08:00
|
|
|
|
|
|
|
for {
|
|
|
|
c, err := l.Accept()
|
2018-06-26 16:15:48 +08:00
|
|
|
if err != nil {
|
2018-11-25 13:18:15 +08:00
|
|
|
log.F("[mixed] failed to accept: %v", err)
|
|
|
|
continue
|
2018-06-26 16:15:48 +08:00
|
|
|
}
|
|
|
|
|
2019-09-18 22:08:48 +08:00
|
|
|
go m.Serve(c)
|
2018-06-26 16:15:48 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-18 22:08:48 +08:00
|
|
|
// Serve serves connections.
|
|
|
|
func (m *Mixed) Serve(c net.Conn) {
|
2018-06-26 16:15:48 +08:00
|
|
|
defer c.Close()
|
|
|
|
|
|
|
|
if c, ok := c.(*net.TCPConn); ok {
|
|
|
|
c.SetKeepAlive(true)
|
|
|
|
}
|
|
|
|
|
2020-10-01 22:38:34 +08:00
|
|
|
cc := proxy.NewConn(c)
|
2019-10-19 22:16:15 +08:00
|
|
|
head, err := cc.Peek(1)
|
|
|
|
if err != nil {
|
|
|
|
// log.F("[mixed] socks5 peek error: %s", err)
|
|
|
|
return
|
|
|
|
}
|
2018-06-26 16:15:48 +08:00
|
|
|
|
2019-10-19 22:16:15 +08:00
|
|
|
// check socks5, client send socksversion: 5 as the first byte
|
|
|
|
if head[0] == socks5.Version {
|
|
|
|
m.socks5Server.Serve(cc)
|
|
|
|
return
|
2018-06-26 16:15:48 +08:00
|
|
|
}
|
|
|
|
|
2019-10-19 22:16:15 +08:00
|
|
|
head, err = cc.Peek(8)
|
|
|
|
if err != nil {
|
|
|
|
log.F("[mixed] http peek error: %s", err)
|
|
|
|
return
|
|
|
|
}
|
2018-06-26 16:15:48 +08:00
|
|
|
|
2019-10-19 22:16:15 +08:00
|
|
|
for _, method := range http.Methods {
|
|
|
|
if bytes.HasPrefix(head, method) {
|
|
|
|
m.httpServer.Serve(cc)
|
|
|
|
return
|
2018-06-26 16:15:48 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-10-21 14:58:18 +08:00
|
|
|
log.F("[mixed] unknown request from %s, ignored", c.RemoteAddr())
|
2018-06-26 16:15:48 +08:00
|
|
|
}
|