base.go 2.4 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495
  1. package proxy
  2. import (
  3. "errors"
  4. "github.com/cnlh/nps/bridge"
  5. "github.com/cnlh/nps/lib/common"
  6. "github.com/cnlh/nps/lib/conn"
  7. "github.com/cnlh/nps/lib/file"
  8. "github.com/cnlh/nps/vender/github.com/astaxie/beego/logs"
  9. "net"
  10. "net/http"
  11. "sync"
  12. )
  13. type Service interface {
  14. Start() error
  15. Close() error
  16. }
  17. //BaseServer struct
  18. type BaseServer struct {
  19. id int
  20. bridge *bridge.Bridge
  21. task *file.Tunnel
  22. errorContent []byte
  23. sync.Mutex
  24. }
  25. func NewBaseServer(bridge *bridge.Bridge, task *file.Tunnel) *BaseServer {
  26. return &BaseServer{
  27. bridge: bridge,
  28. task: task,
  29. errorContent: nil,
  30. Mutex: sync.Mutex{},
  31. }
  32. }
  33. //add the flow
  34. func (s *BaseServer) FlowAdd(in, out int64) {
  35. s.Lock()
  36. defer s.Unlock()
  37. s.task.Flow.ExportFlow += out
  38. s.task.Flow.InletFlow += in
  39. }
  40. //change the flow
  41. func (s *BaseServer) FlowAddHost(host *file.Host, in, out int64) {
  42. s.Lock()
  43. defer s.Unlock()
  44. host.Flow.ExportFlow += out
  45. host.Flow.InletFlow += in
  46. }
  47. //write fail bytes to the connection
  48. func (s *BaseServer) writeConnFail(c net.Conn) {
  49. c.Write([]byte(common.ConnectionFailBytes))
  50. c.Write(s.errorContent)
  51. }
  52. //auth check
  53. func (s *BaseServer) auth(r *http.Request, c *conn.Conn, u, p string) error {
  54. if u != "" && p != "" && !common.CheckAuth(r, u, p) {
  55. c.Write([]byte(common.UnauthorizedBytes))
  56. c.Close()
  57. return errors.New("401 Unauthorized")
  58. }
  59. return nil
  60. }
  61. //check flow limit of the client ,and decrease the allow num of client
  62. func (s *BaseServer) CheckFlowAndConnNum(client *file.Client) error {
  63. if client.Flow.FlowLimit > 0 && (client.Flow.FlowLimit<<20) < (client.Flow.ExportFlow+client.Flow.InletFlow) {
  64. return errors.New("Traffic exceeded")
  65. }
  66. if !client.GetConn() {
  67. return errors.New("Connections exceed the current client limit")
  68. }
  69. return nil
  70. }
  71. //create a new connection and start bytes copying
  72. func (s *BaseServer) DealClient(c *conn.Conn, client *file.Client, addr string, rb []byte, tp string, f func(), flow *file.Flow, localProxy bool) error {
  73. link := conn.NewLink(tp, addr, client.Cnf.Crypt, client.Cnf.Compress, c.Conn.RemoteAddr().String(), localProxy)
  74. if target, err := s.bridge.SendLinkInfo(client.Id, link, s.task); err != nil {
  75. logs.Warn("get connection from client id %d error %s", client.Id, err.Error())
  76. c.Close()
  77. return err
  78. } else {
  79. if f != nil {
  80. f()
  81. }
  82. conn.CopyWaitGroup(target, c.Conn, link.Crypt, link.Compress, client.Rate, flow, true, rb)
  83. }
  84. return nil
  85. }