1
0

https.go 5.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185
  1. package proxy
  2. import (
  3. "github.com/cnlh/nps/bridge"
  4. "github.com/cnlh/nps/lib/cache"
  5. "github.com/cnlh/nps/lib/common"
  6. "github.com/cnlh/nps/lib/conn"
  7. "github.com/cnlh/nps/lib/crypt"
  8. "github.com/cnlh/nps/lib/file"
  9. "github.com/cnlh/nps/vender/github.com/astaxie/beego"
  10. "github.com/cnlh/nps/vender/github.com/astaxie/beego/logs"
  11. "github.com/pkg/errors"
  12. "net"
  13. "net/http"
  14. "net/url"
  15. "sync"
  16. )
  17. type HttpsServer struct {
  18. httpServer
  19. listener net.Listener
  20. httpsListenerMap sync.Map
  21. }
  22. func NewHttpsServer(l net.Listener, bridge *bridge.Bridge, useCache bool, cacheLen int) *HttpsServer {
  23. https := &HttpsServer{listener: l}
  24. https.bridge = bridge
  25. https.useCache = useCache
  26. if useCache {
  27. https.cache = cache.New(cacheLen)
  28. }
  29. return https
  30. }
  31. //start https server
  32. func (https *HttpsServer) Start() error {
  33. if b, err := beego.AppConfig.Bool("https_just_proxy"); err == nil && b {
  34. conn.Accept(https.listener, func(c net.Conn) {
  35. https.handleHttps(c)
  36. })
  37. } else {
  38. //start the default listener
  39. certFile := beego.AppConfig.String("https_default_cert_file")
  40. keyFile := beego.AppConfig.String("https_default_key_file")
  41. if common.FileExists(certFile) && common.FileExists(keyFile) {
  42. l := NewHttpsListener(https.listener)
  43. https.NewHttps(l, certFile, keyFile)
  44. https.httpsListenerMap.Store("default", l)
  45. }
  46. conn.Accept(https.listener, func(c net.Conn) {
  47. serverName, rb := GetServerNameFromClientHello(c)
  48. //if the clientHello does not contains sni ,use the default ssl certificate
  49. if serverName == "" {
  50. serverName = "default"
  51. }
  52. var l *HttpsListener
  53. if v, ok := https.httpsListenerMap.Load(serverName); ok {
  54. l = v.(*HttpsListener)
  55. } else {
  56. r := buildHttpsRequest(serverName)
  57. if host, err := file.GetDb().GetInfoByHost(serverName, r); err != nil {
  58. c.Close()
  59. logs.Notice("the url %s can't be parsed!,remote addr %s", serverName, c.RemoteAddr().String())
  60. return
  61. } else {
  62. if !common.FileExists(host.CertFilePath) || !common.FileExists(host.KeyFilePath) {
  63. //if the host cert file or key file is not set ,use the default file
  64. if v, ok := https.httpsListenerMap.Load("default"); ok {
  65. l = v.(*HttpsListener)
  66. } else {
  67. c.Close()
  68. logs.Error("the key %s cert %s file is not exist", host.KeyFilePath, host.CertFilePath)
  69. return
  70. }
  71. } else {
  72. l = NewHttpsListener(https.listener)
  73. https.NewHttps(l, host.CertFilePath, host.KeyFilePath)
  74. https.httpsListenerMap.Store(serverName, l)
  75. }
  76. }
  77. }
  78. acceptConn := conn.NewConn(c)
  79. acceptConn.Rb = rb
  80. l.acceptConn <- acceptConn
  81. })
  82. }
  83. return nil
  84. }
  85. // close
  86. func (https *HttpsServer) Close() error {
  87. return https.listener.Close()
  88. }
  89. // new https server by cert and key file
  90. func (https *HttpsServer) NewHttps(l net.Listener, certFile string, keyFile string) {
  91. go func() {
  92. logs.Error(https.NewServer(0, "https").ServeTLS(l, certFile, keyFile))
  93. }()
  94. }
  95. //handle the https which is just proxy to other client
  96. func (https *HttpsServer) handleHttps(c net.Conn) {
  97. hostName, rb := GetServerNameFromClientHello(c)
  98. var targetAddr string
  99. r := buildHttpsRequest(hostName)
  100. var host *file.Host
  101. var err error
  102. if host, err = file.GetDb().GetInfoByHost(hostName, r); err != nil {
  103. c.Close()
  104. logs.Notice("the url %s can't be parsed!", hostName)
  105. return
  106. }
  107. if err := https.CheckFlowAndConnNum(host.Client); err != nil {
  108. logs.Warn("client id %d, host id %d, error %s, when https connection", host.Client.Id, host.Id, err.Error())
  109. c.Close()
  110. return
  111. }
  112. defer host.Client.AddConn()
  113. if err = https.auth(r, conn.NewConn(c), host.Client.Cnf.U, host.Client.Cnf.P); err != nil {
  114. logs.Warn("auth error", err, r.RemoteAddr)
  115. return
  116. }
  117. if targetAddr, err = host.Target.GetRandomTarget(); err != nil {
  118. logs.Warn(err.Error())
  119. }
  120. logs.Trace("new https connection,clientId %d,host %s,remote address %s", host.Client.Id, r.Host, c.RemoteAddr().String())
  121. https.DealClient(conn.NewConn(c), host.Client, targetAddr, rb, common.CONN_TCP, nil, host.Flow, host.Target.LocalProxy)
  122. }
  123. type HttpsListener struct {
  124. acceptConn chan *conn.Conn
  125. parentListener net.Listener
  126. }
  127. // https listener
  128. func NewHttpsListener(l net.Listener) *HttpsListener {
  129. return &HttpsListener{parentListener: l, acceptConn: make(chan *conn.Conn)}
  130. }
  131. // accept
  132. func (httpsListener *HttpsListener) Accept() (net.Conn, error) {
  133. httpsConn := <-httpsListener.acceptConn
  134. if httpsConn == nil {
  135. return nil, errors.New("get connection error")
  136. }
  137. return httpsConn, nil
  138. }
  139. // close
  140. func (httpsListener *HttpsListener) Close() error {
  141. return nil
  142. }
  143. // addr
  144. func (httpsListener *HttpsListener) Addr() net.Addr {
  145. return httpsListener.parentListener.Addr()
  146. }
  147. // get server name from connection by read client hello bytes
  148. func GetServerNameFromClientHello(c net.Conn) (string, []byte) {
  149. buf := make([]byte, 4096)
  150. data := make([]byte, 4096)
  151. n, err := c.Read(buf)
  152. if err != nil {
  153. return "", nil
  154. }
  155. if n < 42 {
  156. return "", nil
  157. }
  158. copy(data, buf[:n])
  159. clientHello := new(crypt.ClientHelloMsg)
  160. clientHello.Unmarshal(data[5:n])
  161. return clientHello.GetServerName(), buf[:n]
  162. }
  163. // build https request
  164. func buildHttpsRequest(hostName string) *http.Request {
  165. r := new(http.Request)
  166. r.RequestURI = "/"
  167. r.URL = new(url.URL)
  168. r.URL.Scheme = "https"
  169. r.Host = hostName
  170. return r
  171. }