| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174 |
- package reverse
- import (
- "context"
- "time"
- "github.com/golang/protobuf/proto"
- "v2ray.com/core/common/mux"
- "v2ray.com/core/common/net"
- "v2ray.com/core/common/session"
- "v2ray.com/core/common/task"
- "v2ray.com/core/common/vio"
- "v2ray.com/core/features/routing"
- "v2ray.com/core/transport/pipe"
- )
- type Bridge struct {
- dispatcher routing.Dispatcher
- tag string
- domain string
- workers []*BridgeWorker
- monitorTask *task.Periodic
- }
- func NewBridge(config *BridgeConfig, dispatcher routing.Dispatcher) (*Bridge, error) {
- if len(config.Tag) == 0 {
- return nil, newError("bridge tag is empty")
- }
- if len(config.Domain) == 0 {
- return nil, newError("bridge domain is empty")
- }
- b := &Bridge{
- dispatcher: dispatcher,
- tag: config.Tag,
- domain: config.Domain,
- }
- b.monitorTask = &task.Periodic{
- Execute: b.monitor,
- Interval: time.Second * 2,
- }
- return b, nil
- }
- func (b *Bridge) monitor() error {
- var numConnections uint32
- var numWorker uint32
- for _, w := range b.workers {
- if w.IsActive() {
- numConnections += w.Connections()
- numWorker++
- }
- }
- if numWorker == 0 || numConnections/numWorker > 16 {
- worker, err := NewBridgeWorker(b.domain, b.tag, b.dispatcher)
- if err != nil {
- newError("failed to create bridge worker").Base(err).AtWarning().WriteToLog()
- return nil
- }
- b.workers = append(b.workers, worker)
- }
- return nil
- }
- func (b *Bridge) Start() error {
- return b.monitorTask.Start()
- }
- func (b *Bridge) Close() error {
- return b.monitorTask.Close()
- }
- type BridgeWorker struct {
- tag string
- worker *mux.ServerWorker
- dispatcher routing.Dispatcher
- state Control_State
- }
- func NewBridgeWorker(domain string, tag string, d routing.Dispatcher) (*BridgeWorker, error) {
- ctx := context.Background()
- ctx = session.ContextWithInbound(ctx, &session.Inbound{
- Tag: tag,
- })
- link, err := d.Dispatch(ctx, net.Destination{
- Network: net.Network_TCP,
- Address: net.DomainAddress(domain),
- Port: 0,
- })
- if err != nil {
- return nil, err
- }
- w := &BridgeWorker{
- dispatcher: d,
- tag: tag,
- }
- worker, err := mux.NewServerWorker(context.Background(), w, link)
- if err != nil {
- return nil, err
- }
- w.worker = worker
- return w, nil
- }
- func (w *BridgeWorker) Type() interface{} {
- return routing.DispatcherType()
- }
- func (w *BridgeWorker) Start() error {
- return nil
- }
- func (w *BridgeWorker) Close() error {
- return nil
- }
- func (w *BridgeWorker) IsActive() bool {
- return w.state == Control_ACTIVE
- }
- func (w *BridgeWorker) Connections() uint32 {
- return w.worker.ActiveConnections()
- }
- func (w *BridgeWorker) handleInternalConn(link vio.Link) {
- go func() {
- reader := link.Reader
- for {
- mb, err := reader.ReadMultiBuffer()
- if err != nil {
- break
- }
- for _, b := range mb {
- var ctl Control
- if err := proto.Unmarshal(b.Bytes(), &ctl); err != nil {
- newError("failed to parse proto message").Base(err).WriteToLog()
- break
- }
- if ctl.State != w.state {
- w.state = ctl.State
- }
- }
- }
- }()
- }
- func (w *BridgeWorker) Dispatch(ctx context.Context, dest net.Destination) (*vio.Link, error) {
- if !isInternalDomain(dest) {
- ctx = session.ContextWithInbound(ctx, &session.Inbound{
- Tag: w.tag,
- })
- return w.dispatcher.Dispatch(ctx, dest)
- }
- opt := []pipe.Option{pipe.WithSizeLimit(16 * 1024)}
- uplinkReader, uplinkWriter := pipe.New(opt...)
- downlinkReader, downlinkWriter := pipe.New(opt...)
- w.handleInternalConn(vio.Link{
- Reader: downlinkReader,
- Writer: uplinkWriter,
- })
- return &vio.Link{
- Reader: uplinkReader,
- Writer: downlinkWriter,
- }, nil
- }
|