2022-01-10 17:11:30 +00:00
|
|
|
package net
|
2022-01-04 16:47:18 +00:00
|
|
|
|
|
|
|
import (
|
2022-05-08 05:39:35 +00:00
|
|
|
"errors"
|
2022-05-08 07:55:13 +00:00
|
|
|
"io"
|
2022-01-04 16:47:18 +00:00
|
|
|
"net"
|
2022-03-04 23:46:47 +00:00
|
|
|
"time"
|
2022-01-11 15:05:16 +00:00
|
|
|
|
|
|
|
log "github.com/sirupsen/logrus"
|
2022-01-11 16:59:55 +00:00
|
|
|
"github.com/xvzc/SpoofDPI/doh"
|
|
|
|
"github.com/xvzc/SpoofDPI/packet"
|
2022-01-04 16:47:18 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
const BUF_SIZE = 1024
|
|
|
|
|
2022-01-10 17:11:30 +00:00
|
|
|
type Conn struct {
|
2022-01-11 17:15:45 +00:00
|
|
|
conn net.Conn
|
2022-01-10 17:11:30 +00:00
|
|
|
}
|
|
|
|
|
2022-01-13 16:06:14 +00:00
|
|
|
func (c *Conn) CloseWrite() {
|
|
|
|
c.conn.(*net.TCPConn).CloseWrite()
|
|
|
|
}
|
|
|
|
|
2022-01-11 17:15:45 +00:00
|
|
|
func (c *Conn) Close() {
|
|
|
|
c.conn.Close()
|
2022-01-10 17:11:30 +00:00
|
|
|
}
|
|
|
|
|
2022-01-11 17:15:45 +00:00
|
|
|
func (c *Conn) RemoteAddr() net.Addr {
|
|
|
|
return c.conn.RemoteAddr()
|
2022-01-11 15:05:16 +00:00
|
|
|
}
|
|
|
|
|
2022-01-11 17:15:45 +00:00
|
|
|
func (c *Conn) LocalAddr() net.Addr {
|
|
|
|
return c.conn.LocalAddr()
|
2022-01-11 15:05:16 +00:00
|
|
|
}
|
|
|
|
|
2022-01-11 17:15:45 +00:00
|
|
|
func (c *Conn) Read(b []byte) (n int, err error) {
|
|
|
|
return c.conn.Read(b)
|
2022-01-10 17:11:30 +00:00
|
|
|
}
|
|
|
|
|
2022-01-11 17:15:45 +00:00
|
|
|
func (c *Conn) Write(b []byte) (n int, err error) {
|
|
|
|
return c.conn.Write(b)
|
2022-01-10 17:11:30 +00:00
|
|
|
}
|
|
|
|
|
2022-05-01 15:31:22 +00:00
|
|
|
func (c *Conn) SetDeadLine(t time.Time) (error) {
|
|
|
|
c.conn.SetDeadline(t)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Conn) SetKeepAlive(b bool) (error) {
|
|
|
|
c.conn.(*net.TCPConn).SetKeepAlive(b)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2022-01-10 17:11:30 +00:00
|
|
|
func (conn *Conn) WriteChunks(c [][]byte) (n int, err error) {
|
|
|
|
total := 0
|
|
|
|
for i := 0; i < len(c); i++ {
|
|
|
|
b, err := conn.Write(c[i])
|
|
|
|
if err != nil {
|
|
|
|
return 0, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
b += total
|
|
|
|
}
|
|
|
|
|
|
|
|
return total, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (conn *Conn) ReadBytes() ([]byte, error) {
|
2022-01-04 16:47:18 +00:00
|
|
|
ret := make([]byte, 0)
|
|
|
|
buf := make([]byte, BUF_SIZE)
|
|
|
|
|
2022-03-06 08:44:11 +00:00
|
|
|
conn.conn.SetReadDeadline(time.Now().Add(5 * time.Second))
|
2022-03-04 23:46:47 +00:00
|
|
|
|
2022-01-04 16:47:18 +00:00
|
|
|
for {
|
|
|
|
n, err := conn.Read(buf)
|
2022-03-05 00:56:53 +00:00
|
|
|
if err != nil {
|
|
|
|
switch err.(type) {
|
|
|
|
case *net.OpError:
|
|
|
|
return nil, errors.New("timed out")
|
|
|
|
default:
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
2022-01-04 16:47:18 +00:00
|
|
|
ret = append(ret, buf[:n]...)
|
|
|
|
|
|
|
|
if n < BUF_SIZE {
|
2022-05-08 05:39:35 +00:00
|
|
|
break
|
2022-01-04 16:47:18 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-05-08 05:39:35 +00:00
|
|
|
return ret, nil
|
2022-01-04 16:47:18 +00:00
|
|
|
}
|
|
|
|
|
2022-01-11 16:59:55 +00:00
|
|
|
func (lConn *Conn) HandleHttp(p packet.HttpPacket) {
|
2022-05-08 07:55:13 +00:00
|
|
|
defer lConn.Close()
|
2022-01-15 16:44:00 +00:00
|
|
|
p.Tidy()
|
|
|
|
|
|
|
|
log.Debug("[HTTP] request: \n\n" + string(p.Raw()))
|
|
|
|
|
2022-01-11 17:15:45 +00:00
|
|
|
ip, err := doh.Lookup(p.Domain())
|
2022-01-11 16:59:55 +00:00
|
|
|
if err != nil {
|
2022-05-01 15:31:22 +00:00
|
|
|
// log.Error("[HTTP DOH] Error looking up for domain with ", p.Domain() , " ", err)
|
|
|
|
log.Error(lConn.RemoteAddr().String())
|
|
|
|
// log.Error(string(p.Raw()))
|
|
|
|
lConn.Write([]byte(p.Version() + " 502 Bad Gateway\r\n\r\n"))
|
|
|
|
lConn.Close()
|
2022-03-05 00:56:53 +00:00
|
|
|
return
|
2022-01-11 16:59:55 +00:00
|
|
|
}
|
2022-03-05 00:56:53 +00:00
|
|
|
|
|
|
|
log.Debug("[DOH] Found ", ip, " with ", p.Domain())
|
2022-01-11 16:59:55 +00:00
|
|
|
|
2022-01-14 08:23:41 +00:00
|
|
|
// Create connection to server
|
2022-03-05 02:51:17 +00:00
|
|
|
var port = ":80"
|
|
|
|
if p.Port() != "" {
|
|
|
|
port = p.Port()
|
|
|
|
}
|
|
|
|
|
|
|
|
rConn, err := Dial("tcp", ip + port)
|
2022-01-11 16:59:55 +00:00
|
|
|
if err != nil {
|
2022-01-14 08:23:41 +00:00
|
|
|
log.Debug("[HTTPS] ", err)
|
2022-05-01 15:31:22 +00:00
|
|
|
lConn.Close()
|
2022-01-13 16:06:14 +00:00
|
|
|
return
|
|
|
|
}
|
2022-05-08 07:55:13 +00:00
|
|
|
defer rConn.Close()
|
2022-01-11 16:59:55 +00:00
|
|
|
|
2022-03-05 00:56:53 +00:00
|
|
|
log.Debug("[HTTP] Connected to ", p.Domain())
|
2022-01-11 16:59:55 +00:00
|
|
|
|
2022-05-08 07:55:13 +00:00
|
|
|
// go lConn.Serve(rConn, "[HTTP]", "localhost", p.Domain())
|
|
|
|
// go rConn.Serve(lConn, "[HTTP]", p.Domain(), "localhost")
|
|
|
|
go io.Copy(lConn, rConn)
|
2022-03-05 02:51:17 +00:00
|
|
|
|
2022-01-14 08:23:41 +00:00
|
|
|
_, err = rConn.Write(p.Raw())
|
|
|
|
if err != nil {
|
2022-03-05 00:56:53 +00:00
|
|
|
log.Debug("[HTTP] Error sending request to ", p.Domain(), err)
|
2022-05-01 15:31:22 +00:00
|
|
|
lConn.Close()
|
|
|
|
rConn.Close()
|
2022-01-13 16:06:14 +00:00
|
|
|
return
|
|
|
|
}
|
2022-03-04 15:46:33 +00:00
|
|
|
|
2022-03-05 00:56:53 +00:00
|
|
|
log.Debug("[HTTP] Sent a request to ", p.Domain())
|
2022-05-08 07:55:13 +00:00
|
|
|
|
|
|
|
io.Copy(rConn, lConn)
|
|
|
|
|
|
|
|
log.Debug("[HTTP] Closing Connection..", p.Domain())
|
|
|
|
|
2022-01-11 16:59:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (lConn *Conn) HandleHttps(p packet.HttpPacket) {
|
2022-05-08 07:55:13 +00:00
|
|
|
defer lConn.Close()
|
2022-01-15 16:44:00 +00:00
|
|
|
log.Debug("[HTTPS] request: \n\n" + string(p.Raw()))
|
|
|
|
|
2022-01-11 17:15:45 +00:00
|
|
|
ip, err := doh.Lookup(p.Domain())
|
2022-01-11 16:59:55 +00:00
|
|
|
if err != nil {
|
2022-05-01 15:31:22 +00:00
|
|
|
log.Error("[HTTPS DOH] Error looking up for domain: ", p.Domain(), " ", err)
|
|
|
|
lConn.Write([]byte(p.Version() + " 502 Bad Gateway\r\n\r\n"))
|
|
|
|
lConn.Close()
|
2022-03-05 00:56:53 +00:00
|
|
|
return
|
2022-01-11 16:59:55 +00:00
|
|
|
}
|
2022-03-05 00:56:53 +00:00
|
|
|
|
|
|
|
log.Debug("[DOH] Found ", ip, " with ", p.Domain())
|
2022-01-11 16:59:55 +00:00
|
|
|
|
|
|
|
// Create a connection to the requested server
|
2022-03-05 02:51:17 +00:00
|
|
|
var port = ":443"
|
|
|
|
if p.Port() != "" {
|
|
|
|
port = p.Port()
|
|
|
|
}
|
|
|
|
|
|
|
|
rConn, err := Dial("tcp", ip + port)
|
2022-01-11 16:59:55 +00:00
|
|
|
if err != nil {
|
2022-01-11 18:06:14 +00:00
|
|
|
log.Debug("[HTTPS] ", err)
|
2022-05-01 15:31:22 +00:00
|
|
|
lConn.Close()
|
2022-01-11 16:59:55 +00:00
|
|
|
return
|
|
|
|
}
|
2022-05-08 07:55:13 +00:00
|
|
|
defer rConn.Close()
|
2022-01-11 16:59:55 +00:00
|
|
|
|
2022-03-05 00:56:53 +00:00
|
|
|
log.Debug("[HTTPS] Connected to ", p.Domain())
|
2022-01-11 16:59:55 +00:00
|
|
|
|
2022-03-05 02:51:17 +00:00
|
|
|
_, err = lConn.Write([]byte(p.Version() + " 200 Connection Established\r\n\r\n"))
|
2022-01-11 16:59:55 +00:00
|
|
|
if err != nil {
|
2022-03-05 00:56:53 +00:00
|
|
|
log.Debug("[HTTPS] Error sending 200 Connection Established to the client", err)
|
2022-05-01 15:31:22 +00:00
|
|
|
lConn.Close()
|
|
|
|
rConn.Close()
|
2022-03-05 00:56:53 +00:00
|
|
|
return
|
2022-01-11 16:59:55 +00:00
|
|
|
}
|
2022-03-05 00:56:53 +00:00
|
|
|
log.Debug("[HTTPS] Sent 200 Connection Estabalished to the client")
|
2022-01-11 16:59:55 +00:00
|
|
|
|
|
|
|
// Read client hello
|
|
|
|
clientHello, err := lConn.ReadBytes()
|
|
|
|
if err != nil {
|
2022-03-05 00:56:53 +00:00
|
|
|
log.Debug("[HTTPS] Error reading client hello from the client", err)
|
|
|
|
log.Debug("[HTTPS] Closing local connection..")
|
2022-05-01 15:31:22 +00:00
|
|
|
lConn.Close()
|
|
|
|
rConn.Close()
|
2022-03-05 00:56:53 +00:00
|
|
|
return
|
2022-01-11 16:59:55 +00:00
|
|
|
}
|
|
|
|
|
2022-03-05 00:56:53 +00:00
|
|
|
log.Debug("[HTTPS] Client sent hello ", len(clientHello), "bytes")
|
2022-01-11 16:59:55 +00:00
|
|
|
|
2022-03-05 02:51:17 +00:00
|
|
|
// Generate a go routine that reads from the server
|
2022-05-08 07:55:13 +00:00
|
|
|
go io.Copy(lConn, rConn)
|
2022-03-05 02:51:17 +00:00
|
|
|
|
2022-01-11 16:59:55 +00:00
|
|
|
pkt := packet.NewHttpsPacket(clientHello)
|
|
|
|
|
|
|
|
chunks := pkt.SplitInChunks()
|
|
|
|
|
|
|
|
if _, err := rConn.WriteChunks(chunks); err != nil {
|
2022-03-05 00:56:53 +00:00
|
|
|
log.Debug("[HTTPS] Error writing client hello to ", p.Domain(), err)
|
2022-05-01 15:31:22 +00:00
|
|
|
lConn.Close()
|
|
|
|
rConn.Close()
|
2022-01-11 16:59:55 +00:00
|
|
|
return
|
|
|
|
}
|
2022-03-04 15:46:33 +00:00
|
|
|
|
2022-05-08 07:55:13 +00:00
|
|
|
io.Copy(rConn, lConn)
|
|
|
|
|
|
|
|
log.Debug("[HTTPS] Closing Connection..", p.Domain())
|
|
|
|
// go lConn.Serve(rConn, "[HTTPS]", "localhost", p.Domain())
|
|
|
|
// go rConn.Serve(lConn, "[HTTPS]", p.Domain(), "localhost")
|
|
|
|
|
2022-01-11 16:59:55 +00:00
|
|
|
}
|
|
|
|
|
2022-03-05 00:56:53 +00:00
|
|
|
func (from *Conn) Serve(to *Conn, proto string, fd string, td string) {
|
2022-01-14 08:23:41 +00:00
|
|
|
defer from.Close()
|
2022-05-01 15:31:22 +00:00
|
|
|
defer to.Close()
|
2022-01-14 08:23:41 +00:00
|
|
|
|
|
|
|
proto += " "
|
|
|
|
|
2022-01-04 16:47:18 +00:00
|
|
|
for {
|
2022-01-10 17:11:30 +00:00
|
|
|
buf, err := from.ReadBytes()
|
2022-01-04 16:47:18 +00:00
|
|
|
if err != nil {
|
2022-03-05 00:56:53 +00:00
|
|
|
log.Debug(proto, "Error reading from ", fd, " ", err)
|
2022-01-04 16:47:18 +00:00
|
|
|
break
|
|
|
|
}
|
2022-01-14 08:23:41 +00:00
|
|
|
|
2022-03-05 00:56:53 +00:00
|
|
|
// log.Debug(proto, fd, " sent data: ", len(buf), "bytes")
|
2022-01-04 16:47:18 +00:00
|
|
|
|
2022-01-11 15:05:16 +00:00
|
|
|
if _, err := to.Write(buf); err != nil {
|
2022-03-05 00:56:53 +00:00
|
|
|
log.Debug(proto, "Error Writing to ", td)
|
2022-01-04 16:47:18 +00:00
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|