| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131 |
- package core
- import (
- "context"
- "sync"
- "v2ray.com/core/common"
- "v2ray.com/core/common/buf"
- "v2ray.com/core/common/errors"
- "v2ray.com/core/common/net"
- )
- // Link is a utility for connecting between an inbound and an outbound proxy handler.
- type Link struct {
- Reader buf.Reader
- Writer buf.Writer
- }
- // Dispatcher is a feature that dispatches inbound requests to outbound handlers based on rules.
- // Dispatcher is required to be registered in a V2Ray instance to make V2Ray function properly.
- type Dispatcher interface {
- Feature
- // Dispatch returns a Ray for transporting data for the given request.
- Dispatch(ctx context.Context, dest net.Destination) (*Link, error)
- }
- type syncDispatcher struct {
- sync.RWMutex
- Dispatcher
- }
- func (d *syncDispatcher) Dispatch(ctx context.Context, dest net.Destination) (*Link, error) {
- d.RLock()
- defer d.RUnlock()
- if d.Dispatcher == nil {
- return nil, newError("Dispatcher not set.").AtError()
- }
- return d.Dispatcher.Dispatch(ctx, dest)
- }
- func (d *syncDispatcher) Start() error {
- d.RLock()
- defer d.RUnlock()
- if d.Dispatcher == nil {
- return newError("Dispatcher not set.").AtError()
- }
- return d.Dispatcher.Start()
- }
- func (d *syncDispatcher) Close() error {
- d.RLock()
- defer d.RUnlock()
- return common.Close(d.Dispatcher)
- }
- func (d *syncDispatcher) Set(disp Dispatcher) {
- if disp == nil {
- return
- }
- d.Lock()
- defer d.Unlock()
- common.Close(d.Dispatcher)
- d.Dispatcher = disp
- }
- var (
- // ErrNoClue is for the situation that existing information is not enough to make a decision. For example, Router may return this error when there is no suitable route.
- ErrNoClue = errors.New("not enough information for making a decision")
- )
- // Router is a feature to choose an outbound tag for the given request.
- type Router interface {
- Feature
- // PickRoute returns a tag of an OutboundHandler based on the given context.
- PickRoute(ctx context.Context) (string, error)
- }
- type syncRouter struct {
- sync.RWMutex
- Router
- }
- func (r *syncRouter) PickRoute(ctx context.Context) (string, error) {
- r.RLock()
- defer r.RUnlock()
- if r.Router == nil {
- return "", ErrNoClue
- }
- return r.Router.PickRoute(ctx)
- }
- func (r *syncRouter) Start() error {
- r.RLock()
- defer r.RUnlock()
- if r.Router == nil {
- return nil
- }
- return r.Router.Start()
- }
- func (r *syncRouter) Close() error {
- r.RLock()
- defer r.RUnlock()
- return common.Close(r.Router)
- }
- func (r *syncRouter) Set(router Router) {
- if router == nil {
- return
- }
- r.Lock()
- defer r.Unlock()
- common.Close(r.Router)
- r.Router = router
- }
|