commander.go 2.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121
  1. package commander
  2. import (
  3. "context"
  4. "net"
  5. "sync"
  6. "github.com/xtls/xray-core/common"
  7. "github.com/xtls/xray-core/common/errors"
  8. "github.com/xtls/xray-core/common/signal/done"
  9. core "github.com/xtls/xray-core/core"
  10. "github.com/xtls/xray-core/features/outbound"
  11. "google.golang.org/grpc"
  12. )
  13. // Commander is a Xray feature that provides gRPC methods to external clients.
  14. type Commander struct {
  15. sync.Mutex
  16. server *grpc.Server
  17. services []Service
  18. ohm outbound.Manager
  19. tag string
  20. listen string
  21. }
  22. // NewCommander creates a new Commander based on the given config.
  23. func NewCommander(ctx context.Context, config *Config) (*Commander, error) {
  24. c := &Commander{
  25. tag: config.Tag,
  26. listen: config.Listen,
  27. }
  28. common.Must(core.RequireFeatures(ctx, func(om outbound.Manager) {
  29. c.ohm = om
  30. }))
  31. for _, rawConfig := range config.Service {
  32. config, err := rawConfig.GetInstance()
  33. if err != nil {
  34. return nil, err
  35. }
  36. rawService, err := common.CreateObject(ctx, config)
  37. if err != nil {
  38. return nil, err
  39. }
  40. service, ok := rawService.(Service)
  41. if !ok {
  42. return nil, errors.New("not a Service.")
  43. }
  44. c.services = append(c.services, service)
  45. }
  46. return c, nil
  47. }
  48. // Type implements common.HasType.
  49. func (c *Commander) Type() interface{} {
  50. return (*Commander)(nil)
  51. }
  52. // Start implements common.Runnable.
  53. func (c *Commander) Start() error {
  54. c.Lock()
  55. c.server = grpc.NewServer()
  56. for _, service := range c.services {
  57. service.Register(c.server)
  58. }
  59. c.Unlock()
  60. var listen = func(listener net.Listener) {
  61. if err := c.server.Serve(listener); err != nil {
  62. errors.LogErrorInner(context.Background(), err, "failed to start grpc server")
  63. }
  64. }
  65. if len(c.listen) > 0 {
  66. if l, err := net.Listen("tcp", c.listen); err != nil {
  67. errors.LogErrorInner(context.Background(), err, "API server failed to listen on ", c.listen)
  68. return err
  69. } else {
  70. errors.LogInfo(context.Background(), "API server listening on ", l.Addr())
  71. go listen(l)
  72. }
  73. return nil
  74. }
  75. listener := &OutboundListener{
  76. buffer: make(chan net.Conn, 4),
  77. done: done.New(),
  78. }
  79. go listen(listener)
  80. if err := c.ohm.RemoveHandler(context.Background(), c.tag); err != nil {
  81. errors.LogInfoInner(context.Background(), err, "failed to remove existing handler")
  82. }
  83. return c.ohm.AddHandler(context.Background(), &Outbound{
  84. tag: c.tag,
  85. listener: listener,
  86. })
  87. }
  88. // Close implements common.Closable.
  89. func (c *Commander) Close() error {
  90. c.Lock()
  91. defer c.Unlock()
  92. if c.server != nil {
  93. c.server.Stop()
  94. c.server = nil
  95. }
  96. return nil
  97. }
  98. func init() {
  99. common.Must(common.RegisterConfig((*Config)(nil), func(ctx context.Context, cfg interface{}) (interface{}, error) {
  100. return NewCommander(ctx, cfg.(*Config))
  101. }))
  102. }