| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111 |
- package udp
- import (
- "context"
- "sync"
- "time"
- "v2ray.com/core/app/dispatcher"
- "v2ray.com/core/app/log"
- "v2ray.com/core/common/buf"
- "v2ray.com/core/common/net"
- "v2ray.com/core/common/signal"
- "v2ray.com/core/transport/ray"
- )
- type ResponseCallback func(payload *buf.Buffer)
- type connEntry struct {
- inbound ray.InboundRay
- timer signal.ActivityUpdater
- cancel context.CancelFunc
- }
- type Dispatcher struct {
- sync.RWMutex
- conns map[net.Destination]*connEntry
- dispatcher dispatcher.Interface
- }
- func NewDispatcher(dispatcher dispatcher.Interface) *Dispatcher {
- return &Dispatcher{
- conns: make(map[net.Destination]*connEntry),
- dispatcher: dispatcher,
- }
- }
- func (v *Dispatcher) RemoveRay(dest net.Destination) {
- v.Lock()
- defer v.Unlock()
- if conn, found := v.conns[dest]; found {
- conn.inbound.InboundInput().Close()
- conn.inbound.InboundOutput().Close()
- delete(v.conns, dest)
- }
- }
- func (v *Dispatcher) getInboundRay(dest net.Destination, callback ResponseCallback) *connEntry {
- v.Lock()
- defer v.Unlock()
- if entry, found := v.conns[dest]; found {
- return entry
- }
- log.Trace(newError("establishing new connection for ", dest))
- ctx, cancel := context.WithCancel(context.Background())
- removeRay := func() {
- cancel()
- v.RemoveRay(dest)
- }
- timer := signal.CancelAfterInactivity(ctx, removeRay, time.Second*4)
- inboundRay, _ := v.dispatcher.Dispatch(ctx, dest)
- entry := &connEntry{
- inbound: inboundRay,
- timer: timer,
- cancel: removeRay,
- }
- v.conns[dest] = entry
- go handleInput(ctx, entry, callback)
- return entry
- }
- func (v *Dispatcher) Dispatch(ctx context.Context, destination net.Destination, payload *buf.Buffer, callback ResponseCallback) {
- // TODO: Add user to destString
- log.Trace(newError("dispatch request to: ", destination).AtDebug())
- conn := v.getInboundRay(destination, callback)
- outputStream := conn.inbound.InboundInput()
- if outputStream != nil {
- if err := outputStream.WriteMultiBuffer(buf.NewMultiBufferValue(payload)); err != nil {
- log.Trace(newError("failed to write first UDP payload").Base(err))
- conn.cancel()
- return
- }
- }
- }
- func handleInput(ctx context.Context, conn *connEntry, callback ResponseCallback) {
- input := conn.inbound.InboundOutput()
- timer := conn.timer
- for {
- select {
- case <-ctx.Done():
- return
- default:
- }
- mb, err := input.ReadMultiBuffer()
- if err != nil {
- log.Trace(newError("failed to handl UDP input").Base(err))
- conn.cancel()
- return
- }
- timer.Update()
- for _, b := range mb {
- callback(b)
- }
- }
- }
|