123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990 |
- package vhost
- import (
- "crypto/tls"
- "io"
- "net"
- "time"
- libnet "github.com/fatedier/golib/net"
- )
- type HTTPSMuxer struct {
- *Muxer
- }
- func NewHTTPSMuxer(listener net.Listener, timeout time.Duration) (*HTTPSMuxer, error) {
- mux, err := NewMuxer(listener, GetHTTPSHostname, timeout)
- mux.SetFailHookFunc(vhostFailed)
- if err != nil {
- return nil, err
- }
- return &HTTPSMuxer{mux}, err
- }
- func GetHTTPSHostname(c net.Conn) (_ net.Conn, _ map[string]string, err error) {
- reqInfoMap := make(map[string]string, 0)
- sc, rd := libnet.NewSharedConn(c)
- clientHello, err := readClientHello(rd)
- if err != nil {
- return nil, reqInfoMap, err
- }
- reqInfoMap["Host"] = clientHello.ServerName
- reqInfoMap["Scheme"] = "https"
- return sc, reqInfoMap, nil
- }
- func readClientHello(reader io.Reader) (*tls.ClientHelloInfo, error) {
- var hello *tls.ClientHelloInfo
-
-
-
- err := tls.Server(readOnlyConn{reader: reader}, &tls.Config{
- GetConfigForClient: func(argHello *tls.ClientHelloInfo) (*tls.Config, error) {
- hello = &tls.ClientHelloInfo{}
- *hello = *argHello
- return nil, nil
- },
- }).Handshake()
- if hello == nil {
- return nil, err
- }
- return hello, nil
- }
- func vhostFailed(c net.Conn) {
-
- _ = tls.Server(c, &tls.Config{}).Handshake()
- c.Close()
- }
- type readOnlyConn struct {
- reader io.Reader
- }
- func (conn readOnlyConn) Read(p []byte) (int, error) { return conn.reader.Read(p) }
- func (conn readOnlyConn) Write(_ []byte) (int, error) { return 0, io.ErrClosedPipe }
- func (conn readOnlyConn) Close() error { return nil }
- func (conn readOnlyConn) LocalAddr() net.Addr { return nil }
- func (conn readOnlyConn) RemoteAddr() net.Addr { return nil }
- func (conn readOnlyConn) SetDeadline(_ time.Time) error { return nil }
- func (conn readOnlyConn) SetReadDeadline(_ time.Time) error { return nil }
- func (conn readOnlyConn) SetWriteDeadline(_ time.Time) error { return nil }
|