mirror of
https://github.com/SagerNet/sing.git
synced 2025-04-03 20:07:38 +03:00
Add http connect client
This commit is contained in:
parent
368e41b67b
commit
677c52f01a
1 changed files with 93 additions and 0 deletions
93
protocol/http/client.go
Normal file
93
protocol/http/client.go
Normal file
|
@ -0,0 +1,93 @@
|
|||
package http
|
||||
|
||||
import (
|
||||
std_bufio "bufio"
|
||||
"context"
|
||||
"encoding/base64"
|
||||
"net"
|
||||
"net/http"
|
||||
"net/url"
|
||||
"os"
|
||||
"strings"
|
||||
|
||||
"github.com/sagernet/sing/common/buf"
|
||||
"github.com/sagernet/sing/common/bufio"
|
||||
E "github.com/sagernet/sing/common/exceptions"
|
||||
M "github.com/sagernet/sing/common/metadata"
|
||||
N "github.com/sagernet/sing/common/network"
|
||||
)
|
||||
|
||||
var _ N.Dialer = (*Client)(nil)
|
||||
|
||||
type Client struct {
|
||||
dialer N.Dialer
|
||||
serverAddr M.Socksaddr
|
||||
username string
|
||||
password string
|
||||
}
|
||||
|
||||
func NewClient(dialer N.Dialer, serverAddr M.Socksaddr, username string, password string) *Client {
|
||||
return &Client{
|
||||
dialer,
|
||||
serverAddr,
|
||||
username,
|
||||
password,
|
||||
}
|
||||
}
|
||||
|
||||
func (c *Client) DialContext(ctx context.Context, network string, destination M.Socksaddr) (net.Conn, error) {
|
||||
if !strings.HasPrefix(network, "tcp") {
|
||||
return nil, os.ErrInvalid
|
||||
}
|
||||
var conn net.Conn
|
||||
conn, err := c.dialer.DialContext(ctx, "tcp", c.serverAddr)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
destinationAddress := destination.String()
|
||||
request := &http.Request{
|
||||
Method: http.MethodConnect,
|
||||
URL: &url.URL{
|
||||
Host: destinationAddress,
|
||||
},
|
||||
Host: destinationAddress,
|
||||
Header: http.Header{
|
||||
"Proxy-Connection": []string{"Keep-Alive"},
|
||||
},
|
||||
}
|
||||
if c.username != "" {
|
||||
auth := c.username + ":" + c.password
|
||||
request.Header.Add("Proxy-Authorization", "Basic "+base64.StdEncoding.EncodeToString([]byte(auth)))
|
||||
}
|
||||
err = request.Write(conn)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
reader := std_bufio.NewReader(conn)
|
||||
response, err := http.ReadResponse(reader, request)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
switch response.StatusCode {
|
||||
case http.StatusOK:
|
||||
if reader.Buffered() > 0 {
|
||||
buffer := buf.NewSize(reader.Buffered())
|
||||
_, err = buffer.ReadFrom(reader)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
conn = bufio.NewCachedConn(conn, buffer)
|
||||
}
|
||||
return conn, nil
|
||||
case http.StatusProxyAuthRequired:
|
||||
return nil, E.New("authentication required")
|
||||
case http.StatusMethodNotAllowed:
|
||||
return nil, E.New("method not allowed")
|
||||
default:
|
||||
return nil, E.New("unexpected status: ", response.Status)
|
||||
}
|
||||
}
|
||||
|
||||
func (c *Client) ListenPacket(ctx context.Context, destination M.Socksaddr) (net.PacketConn, error) {
|
||||
return nil, os.ErrInvalid
|
||||
}
|
Loading…
Add table
Add a link
Reference in a new issue