mirror of
https://github.com/SagerNet/sing.git
synced 2025-04-04 20:37:40 +03:00
Refactor
This commit is contained in:
parent
98aaae2adc
commit
2e74275ceb
30 changed files with 938 additions and 61 deletions
|
@ -4,9 +4,9 @@ import (
|
||||||
"crypto/rand"
|
"crypto/rand"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
"sing/common/list"
|
|
||||||
|
|
||||||
"sing/common"
|
"sing/common"
|
||||||
|
"sing/common/list"
|
||||||
)
|
)
|
||||||
|
|
||||||
type Buffer struct {
|
type Buffer struct {
|
||||||
|
@ -25,6 +25,13 @@ func New() *Buffer {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func FullNew() *Buffer {
|
||||||
|
return &Buffer{
|
||||||
|
data: GetBytes(),
|
||||||
|
managed: true,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func StackNew() Buffer {
|
func StackNew() Buffer {
|
||||||
return Buffer{
|
return Buffer{
|
||||||
data: GetBytes(),
|
data: GetBytes(),
|
||||||
|
@ -226,6 +233,11 @@ func (b *Buffer) Reset() {
|
||||||
b.end = ReversedHeader
|
b.end = ReversedHeader
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (b *Buffer) FullReset() {
|
||||||
|
b.start = 0
|
||||||
|
b.end = 0
|
||||||
|
}
|
||||||
|
|
||||||
func (b *Buffer) Release() {
|
func (b *Buffer) Release() {
|
||||||
if b == nil || b.data == nil || !b.managed {
|
if b == nil || b.data == nil || !b.managed {
|
||||||
return
|
return
|
||||||
|
|
|
@ -5,6 +5,7 @@ import "sync"
|
||||||
const (
|
const (
|
||||||
ReversedHeader = 1024
|
ReversedHeader = 1024
|
||||||
BufferSize = 20 * 1024
|
BufferSize = 20 * 1024
|
||||||
|
UDPBufferSize = 16 * 1024
|
||||||
)
|
)
|
||||||
|
|
||||||
var pool = sync.Pool{
|
var pool = sync.Pool{
|
||||||
|
|
|
@ -6,15 +6,13 @@ import (
|
||||||
"time"
|
"time"
|
||||||
)
|
)
|
||||||
|
|
||||||
type ReadOnlyException struct {
|
type ReadOnlyException struct{}
|
||||||
}
|
|
||||||
|
|
||||||
func (e *ReadOnlyException) Error() string {
|
func (e *ReadOnlyException) Error() string {
|
||||||
return "read only connection"
|
return "read only connection"
|
||||||
}
|
}
|
||||||
|
|
||||||
type WriteOnlyException struct {
|
type WriteOnlyException struct{}
|
||||||
}
|
|
||||||
|
|
||||||
func (e *WriteOnlyException) Error() string {
|
func (e *WriteOnlyException) Error() string {
|
||||||
return "write only connection"
|
return "write only connection"
|
||||||
|
|
|
@ -2,8 +2,7 @@ package common
|
||||||
|
|
||||||
const EmptyString = ""
|
const EmptyString = ""
|
||||||
|
|
||||||
type DummyAddr struct {
|
type DummyAddr struct{}
|
||||||
}
|
|
||||||
|
|
||||||
func (d *DummyAddr) Network() string {
|
func (d *DummyAddr) Network() string {
|
||||||
return "dummy"
|
return "dummy"
|
||||||
|
|
|
@ -2,7 +2,7 @@
|
||||||
// Use of this source code is governed by a BSD-style
|
// Use of this source code is governed by a BSD-style
|
||||||
// license that can be found in the LICENSE file.
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
package genericsync
|
package gsync
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"sync"
|
"sync"
|
53
common/rw/copy.go
Normal file
53
common/rw/copy.go
Normal file
|
@ -0,0 +1,53 @@
|
||||||
|
package rw
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"io"
|
||||||
|
"net"
|
||||||
|
|
||||||
|
"sing/common"
|
||||||
|
"sing/common/buf"
|
||||||
|
"sing/common/task"
|
||||||
|
)
|
||||||
|
|
||||||
|
func CopyConn(ctx context.Context, conn net.Conn, outConn net.Conn) error {
|
||||||
|
return task.Run(ctx, func() error {
|
||||||
|
return common.Error(io.Copy(conn, outConn))
|
||||||
|
}, func() error {
|
||||||
|
return common.Error(io.Copy(outConn, conn))
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func CopyPacketConn(ctx context.Context, conn net.PacketConn, outPacketConn net.PacketConn) error {
|
||||||
|
return task.Run(ctx, func() error {
|
||||||
|
buffer := buf.FullNew()
|
||||||
|
defer buffer.Release()
|
||||||
|
for {
|
||||||
|
n, addr, err := conn.ReadFrom(buffer.FreeBytes())
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
buffer.Truncate(n)
|
||||||
|
_, err = outPacketConn.WriteTo(buffer.Bytes(), addr)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
buffer.FullReset()
|
||||||
|
}
|
||||||
|
}, func() error {
|
||||||
|
buffer := buf.FullNew()
|
||||||
|
defer buffer.Release()
|
||||||
|
for {
|
||||||
|
n, addr, err := outPacketConn.ReadFrom(buffer.FreeBytes())
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
buffer.Truncate(n)
|
||||||
|
_, err = conn.WriteTo(buffer.Bytes(), addr)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
buffer.FullReset()
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
|
@ -2,6 +2,7 @@ package rw
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"io"
|
"io"
|
||||||
|
|
||||||
"sing/common"
|
"sing/common"
|
||||||
"sing/common/buf"
|
"sing/common/buf"
|
||||||
"sing/common/list"
|
"sing/common/list"
|
||||||
|
|
66
common/session/context.go
Normal file
66
common/session/context.go
Normal file
|
@ -0,0 +1,66 @@
|
||||||
|
package session
|
||||||
|
|
||||||
|
import (
|
||||||
|
"net"
|
||||||
|
"strconv"
|
||||||
|
|
||||||
|
"sing/common/buf"
|
||||||
|
"sing/common/socksaddr"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Network int
|
||||||
|
|
||||||
|
const (
|
||||||
|
NetworkTCP Network = iota
|
||||||
|
NetworkUDP
|
||||||
|
)
|
||||||
|
|
||||||
|
type InstanceContext struct{}
|
||||||
|
|
||||||
|
type Context struct {
|
||||||
|
InstanceContext
|
||||||
|
Network Network
|
||||||
|
Source socksaddr.Addr
|
||||||
|
Destination socksaddr.Addr
|
||||||
|
SourcePort uint16
|
||||||
|
DestinationPort uint16
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c Context) DestinationNetAddr() string {
|
||||||
|
return net.JoinHostPort(c.Destination.String(), strconv.Itoa(int(c.DestinationPort)))
|
||||||
|
}
|
||||||
|
|
||||||
|
func AddressFromNetAddr(netAddr net.Addr) (addr socksaddr.Addr, port uint16) {
|
||||||
|
var ip net.IP
|
||||||
|
switch addr := netAddr.(type) {
|
||||||
|
case *net.TCPAddr:
|
||||||
|
ip = addr.IP
|
||||||
|
port = uint16(addr.Port)
|
||||||
|
case *net.UDPAddr:
|
||||||
|
ip = addr.IP
|
||||||
|
port = uint16(addr.Port)
|
||||||
|
}
|
||||||
|
return socksaddr.AddrFromIP(ip), port
|
||||||
|
}
|
||||||
|
|
||||||
|
type Conn struct {
|
||||||
|
Conn net.Conn
|
||||||
|
Context *Context
|
||||||
|
}
|
||||||
|
|
||||||
|
type PacketConn struct {
|
||||||
|
Conn net.PacketConn
|
||||||
|
Context *Context
|
||||||
|
}
|
||||||
|
|
||||||
|
type Packet struct {
|
||||||
|
Context *Context
|
||||||
|
Data *buf.Buffer
|
||||||
|
WriteBack func(buffer *buf.Buffer, addr *net.UDPAddr) error
|
||||||
|
Release func()
|
||||||
|
}
|
||||||
|
|
||||||
|
type Handler interface {
|
||||||
|
HandleConnection(conn *Conn)
|
||||||
|
HandlePacket(packet *Packet)
|
||||||
|
}
|
63
common/session/pool.go
Normal file
63
common/session/pool.go
Normal file
|
@ -0,0 +1,63 @@
|
||||||
|
package session
|
||||||
|
|
||||||
|
import (
|
||||||
|
"container/list"
|
||||||
|
"sync"
|
||||||
|
|
||||||
|
"sing/common"
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
connectionPool list.List
|
||||||
|
connectionPoolEnabled bool
|
||||||
|
connectionAccess sync.Mutex
|
||||||
|
)
|
||||||
|
|
||||||
|
func EnableConnectionPool() {
|
||||||
|
connectionPoolEnabled = true
|
||||||
|
}
|
||||||
|
|
||||||
|
func DisableConnectionPool() {
|
||||||
|
connectionAccess.Lock()
|
||||||
|
defer connectionAccess.Unlock()
|
||||||
|
connectionPoolEnabled = false
|
||||||
|
clearConnections()
|
||||||
|
}
|
||||||
|
|
||||||
|
func AddConnection(connection any) any {
|
||||||
|
if !connectionPoolEnabled {
|
||||||
|
return connection
|
||||||
|
}
|
||||||
|
connectionAccess.Lock()
|
||||||
|
defer connectionAccess.Unlock()
|
||||||
|
return connectionPool.PushBack(connection)
|
||||||
|
}
|
||||||
|
|
||||||
|
func RemoveConnection(anyElement any) {
|
||||||
|
element, ok := anyElement.(*list.Element)
|
||||||
|
if !ok {
|
||||||
|
common.Close(anyElement)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if element.Value == nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
common.Close(element.Value)
|
||||||
|
element.Value = nil
|
||||||
|
connectionAccess.Lock()
|
||||||
|
defer connectionAccess.Unlock()
|
||||||
|
connectionPool.Remove(element)
|
||||||
|
}
|
||||||
|
|
||||||
|
func ResetConnections() {
|
||||||
|
connectionAccess.Lock()
|
||||||
|
defer connectionAccess.Unlock()
|
||||||
|
clearConnections()
|
||||||
|
}
|
||||||
|
|
||||||
|
func clearConnections() {
|
||||||
|
for element := connectionPool.Front(); element != nil; element = element.Next() {
|
||||||
|
common.Close(element)
|
||||||
|
}
|
||||||
|
connectionPool.Init()
|
||||||
|
}
|
|
@ -10,6 +10,7 @@ type Addr interface {
|
||||||
Family() Family
|
Family() Family
|
||||||
Addr() netip.Addr
|
Addr() netip.Addr
|
||||||
Fqdn() string
|
Fqdn() string
|
||||||
|
String() string
|
||||||
}
|
}
|
||||||
|
|
||||||
func AddrFromIP(ip net.IP) Addr {
|
func AddrFromIP(ip net.IP) Addr {
|
||||||
|
@ -21,6 +22,19 @@ func AddrFromIP(ip net.IP) Addr {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func AddressFromNetAddr(netAddr net.Addr) (addr Addr, port uint16) {
|
||||||
|
var ip net.IP
|
||||||
|
switch addr := netAddr.(type) {
|
||||||
|
case *net.TCPAddr:
|
||||||
|
ip = addr.IP
|
||||||
|
port = uint16(addr.Port)
|
||||||
|
case *net.UDPAddr:
|
||||||
|
ip = addr.IP
|
||||||
|
port = uint16(addr.Port)
|
||||||
|
}
|
||||||
|
return AddrFromIP(ip), port
|
||||||
|
}
|
||||||
|
|
||||||
func AddrFromFqdn(fqdn string) Addr {
|
func AddrFromFqdn(fqdn string) Addr {
|
||||||
return AddrFqdn(fqdn)
|
return AddrFqdn(fqdn)
|
||||||
}
|
}
|
||||||
|
@ -39,6 +53,10 @@ func (a Addr4) Fqdn() string {
|
||||||
return ""
|
return ""
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (a Addr4) String() string {
|
||||||
|
return net.IP(a[:]).String()
|
||||||
|
}
|
||||||
|
|
||||||
type Addr16 [16]byte
|
type Addr16 [16]byte
|
||||||
|
|
||||||
func (a Addr16) Family() Family {
|
func (a Addr16) Family() Family {
|
||||||
|
@ -53,6 +71,10 @@ func (a Addr16) Fqdn() string {
|
||||||
return ""
|
return ""
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (a Addr16) String() string {
|
||||||
|
return net.IP(a[:]).String()
|
||||||
|
}
|
||||||
|
|
||||||
type AddrFqdn string
|
type AddrFqdn string
|
||||||
|
|
||||||
func (f AddrFqdn) Family() Family {
|
func (f AddrFqdn) Family() Family {
|
||||||
|
@ -66,3 +88,7 @@ func (f AddrFqdn) Addr() netip.Addr {
|
||||||
func (f AddrFqdn) Fqdn() string {
|
func (f AddrFqdn) Fqdn() string {
|
||||||
return string(f)
|
return string(f)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (f AddrFqdn) String() string {
|
||||||
|
return string(f)
|
||||||
|
}
|
||||||
|
|
|
@ -3,6 +3,7 @@ package socksaddr
|
||||||
import (
|
import (
|
||||||
"encoding/binary"
|
"encoding/binary"
|
||||||
"io"
|
"io"
|
||||||
|
|
||||||
"sing/common"
|
"sing/common"
|
||||||
"sing/common/exceptions"
|
"sing/common/exceptions"
|
||||||
"sing/common/rw"
|
"sing/common/rw"
|
||||||
|
|
36
common/task/task.go
Normal file
36
common/task/task.go
Normal file
|
@ -0,0 +1,36 @@
|
||||||
|
package task
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"sync"
|
||||||
|
|
||||||
|
"sing/common"
|
||||||
|
)
|
||||||
|
|
||||||
|
func Run(ctx context.Context, tasks ...func() error) error {
|
||||||
|
ctx, cancel := context.WithCancel(ctx)
|
||||||
|
wg := new(sync.WaitGroup)
|
||||||
|
wg.Add(len(tasks))
|
||||||
|
var retErr error
|
||||||
|
for _, task := range tasks {
|
||||||
|
task := task
|
||||||
|
go func() {
|
||||||
|
if err := task(); err != nil {
|
||||||
|
if !common.Done(ctx) {
|
||||||
|
retErr = err
|
||||||
|
}
|
||||||
|
cancel()
|
||||||
|
}
|
||||||
|
wg.Done()
|
||||||
|
}()
|
||||||
|
}
|
||||||
|
go func() {
|
||||||
|
wg.Wait()
|
||||||
|
cancel()
|
||||||
|
}()
|
||||||
|
<-ctx.Done()
|
||||||
|
if retErr != nil {
|
||||||
|
return retErr
|
||||||
|
}
|
||||||
|
return ctx.Err()
|
||||||
|
}
|
54
conf/config.go
Normal file
54
conf/config.go
Normal file
|
@ -0,0 +1,54 @@
|
||||||
|
package conf
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/json"
|
||||||
|
|
||||||
|
"sing/common/exceptions"
|
||||||
|
"sing/core"
|
||||||
|
"sing/transport"
|
||||||
|
"sing/transport/block"
|
||||||
|
"sing/transport/socks"
|
||||||
|
"sing/transport/system"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Config struct {
|
||||||
|
Inbounds []*InboundConfig `json:"inbounds,omitempty"`
|
||||||
|
Outbounds []*OutboundConfig `json:"outbounds,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type InboundConfig struct {
|
||||||
|
Type string `json:"type"`
|
||||||
|
Tag string `json:"tag,omitempty"`
|
||||||
|
Settings json.RawMessage `json:"settings,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c InboundConfig) Build(instance core.Instance) (transport.Inbound, error) {
|
||||||
|
switch c.Type {
|
||||||
|
case "socks":
|
||||||
|
config := new(socks.InboundConfig)
|
||||||
|
err := json.Unmarshal(c.Settings, config)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return socks.NewListener(instance, config)
|
||||||
|
}
|
||||||
|
return nil, exceptions.New("unknown inbound type ", c.Type)
|
||||||
|
}
|
||||||
|
|
||||||
|
type OutboundConfig struct {
|
||||||
|
Type string `json:"type"`
|
||||||
|
Settings json.RawMessage `json:"settings,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c OutboundConfig) Build(instance core.Instance) (transport.Outbound, error) {
|
||||||
|
var outbound transport.Outbound
|
||||||
|
switch c.Type {
|
||||||
|
case "system":
|
||||||
|
outbound = new(system.Outbound)
|
||||||
|
case "block":
|
||||||
|
outbound = new(block.Outbound)
|
||||||
|
default:
|
||||||
|
return nil, exceptions.New("unknown outbound type: ", c.Type)
|
||||||
|
}
|
||||||
|
return outbound, nil
|
||||||
|
}
|
94
core.go
Normal file
94
core.go
Normal file
|
@ -0,0 +1,94 @@
|
||||||
|
package sing
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"sing/common/session"
|
||||||
|
"sync"
|
||||||
|
|
||||||
|
"sing/common/gsync"
|
||||||
|
"sing/common/list"
|
||||||
|
"sing/core"
|
||||||
|
"sing/transport"
|
||||||
|
)
|
||||||
|
|
||||||
|
var _ core.Instance = (*Instance)(nil)
|
||||||
|
|
||||||
|
type Instance struct {
|
||||||
|
access sync.Mutex
|
||||||
|
ctx context.Context
|
||||||
|
cancel context.CancelFunc
|
||||||
|
inbounds list.List[*transport.InboundContext]
|
||||||
|
inboundByName gsync.Map[string, *transport.InboundContext]
|
||||||
|
outbounds list.List[*transport.OutboundContext]
|
||||||
|
outboundByName gsync.Map[string, *transport.OutboundContext]
|
||||||
|
defaultOutbound *transport.OutboundContext
|
||||||
|
}
|
||||||
|
|
||||||
|
func (i *Instance) AddInbound(inbound transport.Inbound, tag string) {
|
||||||
|
i.access.Lock()
|
||||||
|
defer i.access.Unlock()
|
||||||
|
|
||||||
|
ic := new(transport.InboundContext)
|
||||||
|
ic.Context = i.ctx
|
||||||
|
ic.Tag = tag
|
||||||
|
ic.Inbound = inbound
|
||||||
|
|
||||||
|
i.inbounds.InsertAfter(ic)
|
||||||
|
i.inboundByName.Store(tag, ic)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (i *Instance) Inbounds() *list.List[*transport.InboundContext] {
|
||||||
|
i.inboundByName.Range(func(tag string, inbound *transport.InboundContext) bool {
|
||||||
|
return true
|
||||||
|
})
|
||||||
|
return &i.inbounds
|
||||||
|
}
|
||||||
|
|
||||||
|
func (i *Instance) Inbound(tag string) *transport.InboundContext {
|
||||||
|
inbound, _ := i.inboundByName.Load(tag)
|
||||||
|
return inbound
|
||||||
|
}
|
||||||
|
|
||||||
|
func (i *Instance) Outbounds() *list.List[*transport.OutboundContext] {
|
||||||
|
return &i.outbounds
|
||||||
|
}
|
||||||
|
|
||||||
|
func (i *Instance) DefaultOutbound() *transport.OutboundContext {
|
||||||
|
i.access.Lock()
|
||||||
|
defer i.access.Unlock()
|
||||||
|
return i.defaultOutbound
|
||||||
|
}
|
||||||
|
|
||||||
|
func (i *Instance) Outbound(tag string) *transport.OutboundContext {
|
||||||
|
outbound, _ := i.outboundByName.Load(tag)
|
||||||
|
return outbound
|
||||||
|
}
|
||||||
|
|
||||||
|
func (i *Instance) HandleConnection(conn *session.Conn) {
|
||||||
|
i.defaultOutbound.Outbound.NewConnection(i.ctx, conn)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (i *Instance) HandlePacket(packet *session.Packet) {
|
||||||
|
}
|
||||||
|
|
||||||
|
type InstanceContext interface {
|
||||||
|
context.Context
|
||||||
|
Instance() *Instance
|
||||||
|
Load(key string) (any, bool)
|
||||||
|
Store(key string, value any)
|
||||||
|
}
|
||||||
|
|
||||||
|
type instanceContext struct {
|
||||||
|
context.Context
|
||||||
|
instance Instance
|
||||||
|
values gsync.Map[any, string]
|
||||||
|
}
|
||||||
|
|
||||||
|
func (i *instanceContext) Load(key string) (any, bool) {
|
||||||
|
return i.values.Load(key)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (i *instanceContext) Store(key string, value any) {
|
||||||
|
// TODO implement me
|
||||||
|
panic("implement me")
|
||||||
|
}
|
12
core/core.go
Normal file
12
core/core.go
Normal file
|
@ -0,0 +1,12 @@
|
||||||
|
package core
|
||||||
|
|
||||||
|
import (
|
||||||
|
"sing/common/session"
|
||||||
|
"sing/transport"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Instance interface {
|
||||||
|
session.Handler
|
||||||
|
transport.InboundManager
|
||||||
|
transport.OutboundManager
|
||||||
|
}
|
|
@ -8,14 +8,14 @@ import (
|
||||||
"log"
|
"log"
|
||||||
"net"
|
"net"
|
||||||
"os"
|
"os"
|
||||||
|
|
||||||
|
cObfs "github.com/Dreamacro/clash/transport/ssr/obfs"
|
||||||
|
cProtocol "github.com/Dreamacro/clash/transport/ssr/protocol"
|
||||||
"sing/common"
|
"sing/common"
|
||||||
"sing/common/buf"
|
"sing/common/buf"
|
||||||
"sing/common/socksaddr"
|
"sing/common/socksaddr"
|
||||||
"sing/protocol/shadowsocks"
|
"sing/protocol/shadowsocks"
|
||||||
_ "sing/protocol/shadowsocks/shadowstream"
|
_ "sing/protocol/shadowsocks/shadowstream"
|
||||||
|
|
||||||
cObfs "github.com/Dreamacro/clash/transport/ssr/obfs"
|
|
||||||
cProtocol "github.com/Dreamacro/clash/transport/ssr/protocol"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
|
@ -67,9 +67,9 @@ func main() {
|
||||||
|
|
||||||
key := shadowsocks.Key([]byte(password), cipher.KeySize())
|
key := shadowsocks.Key([]byte(password), cipher.KeySize())
|
||||||
|
|
||||||
if _, isAEAD := cipher.(*shadowsocks.AEADCipher); isAEAD {
|
/*if _, isAEAD := cipher.(*shadowsocks.AEADCipher); isAEAD {
|
||||||
log.Fatal("not a stream cipher: ", method)
|
log.Fatal("not a stream cipher: ", method)
|
||||||
}
|
}*/
|
||||||
|
|
||||||
ipAddr, err := net.ResolveIPAddr("ip", address)
|
ipAddr, err := net.ResolveIPAddr("ip", address)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
|
6
format.go
Normal file
6
format.go
Normal file
|
@ -0,0 +1,6 @@
|
||||||
|
package sing
|
||||||
|
|
||||||
|
//go:generate go install -v mvdan.cc/gofumpt@latest
|
||||||
|
//go:generate go install -v github.com/daixiang0/gci@latest
|
||||||
|
//go:generate gofumpt -l -w .
|
||||||
|
//go:generate gci -w .
|
6
go.mod
6
go.mod
|
@ -9,7 +9,7 @@ require (
|
||||||
github.com/dgryski/go-rc2 v0.0.0-20150621095337-8a9021637152
|
github.com/dgryski/go-rc2 v0.0.0-20150621095337-8a9021637152
|
||||||
github.com/geeksbaek/seed v0.0.0-20180909040025-2a7f5fb92e22
|
github.com/geeksbaek/seed v0.0.0-20180909040025-2a7f5fb92e22
|
||||||
github.com/kierdavis/cfb8 v0.0.0-20180105024805-3a17c36ee2f8
|
github.com/kierdavis/cfb8 v0.0.0-20180105024805-3a17c36ee2f8
|
||||||
golang.org/x/crypto v0.0.0-20220126234351-aa10faf2a1f8
|
golang.org/x/crypto v0.0.0-20220214200702-86341886e292
|
||||||
)
|
)
|
||||||
|
|
||||||
// for testing and example only
|
// for testing and example only
|
||||||
|
@ -36,7 +36,7 @@ require (
|
||||||
github.com/sirupsen/logrus v1.8.1 // indirect
|
github.com/sirupsen/logrus v1.8.1 // indirect
|
||||||
github.com/v2fly/ss-bloomring v0.0.0-20210312155135-28617310f63e // indirect
|
github.com/v2fly/ss-bloomring v0.0.0-20210312155135-28617310f63e // indirect
|
||||||
golang.org/x/mod v0.6.0-dev.0.20211013180041-c96bc1413d57 // indirect
|
golang.org/x/mod v0.6.0-dev.0.20211013180041-c96bc1413d57 // indirect
|
||||||
golang.org/x/net v0.0.0-20220127200216-cd36cc0744dd // indirect
|
golang.org/x/net v0.0.0-20220225172249-27dd8689420f // indirect
|
||||||
golang.org/x/sys v0.0.0-20220114195835-da31bd327af9 // indirect
|
golang.org/x/sys v0.0.0-20220227234510-4e6760a101f9 // indirect
|
||||||
google.golang.org/protobuf v1.27.1 // indirect
|
google.golang.org/protobuf v1.27.1 // indirect
|
||||||
)
|
)
|
||||||
|
|
6
go.sum
6
go.sum
|
@ -76,16 +76,22 @@ go4.org/unsafe/assume-no-moving-gc v0.0.0-20211027215541-db492cf91b37 h1:Tx9kY6y
|
||||||
golang.org/x/crypto v0.0.0-20210317152858-513c2a44f670/go.mod h1:T9bdIzuCu7OtxOm1hfPfRQxPLYneinmdGuTeoZ9dtd4=
|
golang.org/x/crypto v0.0.0-20210317152858-513c2a44f670/go.mod h1:T9bdIzuCu7OtxOm1hfPfRQxPLYneinmdGuTeoZ9dtd4=
|
||||||
golang.org/x/crypto v0.0.0-20220126234351-aa10faf2a1f8 h1:kACShD3qhmr/3rLmg1yXyt+N4HcwutKyPRB93s54TIU=
|
golang.org/x/crypto v0.0.0-20220126234351-aa10faf2a1f8 h1:kACShD3qhmr/3rLmg1yXyt+N4HcwutKyPRB93s54TIU=
|
||||||
golang.org/x/crypto v0.0.0-20220126234351-aa10faf2a1f8/go.mod h1:IxCIyHEi3zRg3s0A5j5BB6A9Jmi73HwBIUl50j+osU4=
|
golang.org/x/crypto v0.0.0-20220126234351-aa10faf2a1f8/go.mod h1:IxCIyHEi3zRg3s0A5j5BB6A9Jmi73HwBIUl50j+osU4=
|
||||||
|
golang.org/x/crypto v0.0.0-20220214200702-86341886e292 h1:f+lwQ+GtmgoY+A2YaQxlSOnDjXcQ7ZRLWOHbC6HtRqE=
|
||||||
|
golang.org/x/crypto v0.0.0-20220214200702-86341886e292/go.mod h1:IxCIyHEi3zRg3s0A5j5BB6A9Jmi73HwBIUl50j+osU4=
|
||||||
golang.org/x/mod v0.6.0-dev.0.20211013180041-c96bc1413d57 h1:LQmS1nU0twXLA96Kt7U9qtHJEbBk3z6Q0V4UXjZkpr4=
|
golang.org/x/mod v0.6.0-dev.0.20211013180041-c96bc1413d57 h1:LQmS1nU0twXLA96Kt7U9qtHJEbBk3z6Q0V4UXjZkpr4=
|
||||||
golang.org/x/mod v0.6.0-dev.0.20211013180041-c96bc1413d57/go.mod h1:3p9vT2HGsQu2K1YbXdKPJLVgG5VJdoTa1poYQBtP1AY=
|
golang.org/x/mod v0.6.0-dev.0.20211013180041-c96bc1413d57/go.mod h1:3p9vT2HGsQu2K1YbXdKPJLVgG5VJdoTa1poYQBtP1AY=
|
||||||
golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg=
|
golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg=
|
||||||
golang.org/x/net v0.0.0-20220127200216-cd36cc0744dd h1:O7DYs+zxREGLKzKoMQrtrEacpb0ZVXA5rIwylE2Xchk=
|
golang.org/x/net v0.0.0-20220127200216-cd36cc0744dd h1:O7DYs+zxREGLKzKoMQrtrEacpb0ZVXA5rIwylE2Xchk=
|
||||||
golang.org/x/net v0.0.0-20220127200216-cd36cc0744dd/go.mod h1:CfG3xpIq0wQ8r1q4Su4UZFWDARRcnwPjda9FqA0JpMk=
|
golang.org/x/net v0.0.0-20220127200216-cd36cc0744dd/go.mod h1:CfG3xpIq0wQ8r1q4Su4UZFWDARRcnwPjda9FqA0JpMk=
|
||||||
|
golang.org/x/net v0.0.0-20220225172249-27dd8689420f h1:oA4XRj0qtSt8Yo1Zms0CUlsT3KG69V2UGQWPBxujDmc=
|
||||||
|
golang.org/x/net v0.0.0-20220225172249-27dd8689420f/go.mod h1:CfG3xpIq0wQ8r1q4Su4UZFWDARRcnwPjda9FqA0JpMk=
|
||||||
golang.org/x/sync v0.0.0-20210220032951-036812b2e83c h1:5KslGYwFpkhGh+Q16bwMP3cOontH8FOep7tGV86Y7SQ=
|
golang.org/x/sync v0.0.0-20210220032951-036812b2e83c h1:5KslGYwFpkhGh+Q16bwMP3cOontH8FOep7tGV86Y7SQ=
|
||||||
golang.org/x/sys v0.0.0-20191026070338-33540a1f6037/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
golang.org/x/sys v0.0.0-20191026070338-33540a1f6037/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||||
golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||||
golang.org/x/sys v0.0.0-20220114195835-da31bd327af9 h1:XfKQ4OlFl8okEOr5UvAqFRVj8pY/4yfcXrddB8qAbU0=
|
golang.org/x/sys v0.0.0-20220114195835-da31bd327af9 h1:XfKQ4OlFl8okEOr5UvAqFRVj8pY/4yfcXrddB8qAbU0=
|
||||||
golang.org/x/sys v0.0.0-20220114195835-da31bd327af9/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
golang.org/x/sys v0.0.0-20220114195835-da31bd327af9/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||||
|
golang.org/x/sys v0.0.0-20220227234510-4e6760a101f9 h1:nhht2DYV/Sn3qOayu8lM+cU1ii9sTLUeBQwQQfUHtrs=
|
||||||
|
golang.org/x/sys v0.0.0-20220227234510-4e6760a101f9/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||||
golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo=
|
golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo=
|
||||||
golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ=
|
golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ=
|
||||||
golang.org/x/text v0.3.7 h1:olpwvP2KacW1ZWvsR7uQhoyTYvKAupfQrRGBFM352Gk=
|
golang.org/x/text v0.3.7 h1:olpwvP2KacW1ZWvsR7uQhoyTYvKAupfQrRGBFM352Gk=
|
||||||
|
|
|
@ -2,6 +2,7 @@ package shadowsocks
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"io"
|
"io"
|
||||||
|
|
||||||
"sing/common/buf"
|
"sing/common/buf"
|
||||||
"sing/common/exceptions"
|
"sing/common/exceptions"
|
||||||
"sing/common/list"
|
"sing/common/list"
|
||||||
|
@ -18,8 +19,10 @@ type Cipher interface {
|
||||||
|
|
||||||
type CipherCreator func() Cipher
|
type CipherCreator func() Cipher
|
||||||
|
|
||||||
var cipherList *list.List[string]
|
var (
|
||||||
var cipherMap map[string]CipherCreator
|
cipherList *list.List[string]
|
||||||
|
cipherMap map[string]CipherCreator
|
||||||
|
)
|
||||||
|
|
||||||
func init() {
|
func init() {
|
||||||
cipherList = new(list.List[string])
|
cipherList = new(list.List[string])
|
||||||
|
|
|
@ -5,7 +5,6 @@ import (
|
||||||
"bytes"
|
"bytes"
|
||||||
"io"
|
"io"
|
||||||
"net"
|
"net"
|
||||||
"sing/common/rw"
|
|
||||||
"strings"
|
"strings"
|
||||||
"sync"
|
"sync"
|
||||||
"testing"
|
"testing"
|
||||||
|
@ -17,6 +16,7 @@ import (
|
||||||
"sing/common"
|
"sing/common"
|
||||||
"sing/common/buf"
|
"sing/common/buf"
|
||||||
"sing/common/crypto"
|
"sing/common/crypto"
|
||||||
|
"sing/common/rw"
|
||||||
"sing/common/socksaddr"
|
"sing/common/socksaddr"
|
||||||
"sing/protocol/shadowsocks"
|
"sing/protocol/shadowsocks"
|
||||||
_ "sing/protocol/shadowsocks/shadowstream"
|
_ "sing/protocol/shadowsocks/shadowstream"
|
||||||
|
@ -195,7 +195,6 @@ func benchmarkShadowsocksCipher(b *testing.B, method string, data int) {
|
||||||
} else {
|
} else {
|
||||||
writer.Write(buffer.Bytes())
|
writer.Write(buffer.Bytes())
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func testShadowsocksClientTCPWithCipher(t *testing.T, cipherType vs.CipherType, cipherName string) {
|
func testShadowsocksClientTCPWithCipher(t *testing.T, cipherType vs.CipherType, cipherName string) {
|
||||||
|
|
|
@ -5,6 +5,7 @@ import (
|
||||||
"io"
|
"io"
|
||||||
|
|
||||||
"sing/common"
|
"sing/common"
|
||||||
|
"sing/common/buf"
|
||||||
"sing/common/exceptions"
|
"sing/common/exceptions"
|
||||||
"sing/common/rw"
|
"sing/common/rw"
|
||||||
"sing/common/socksaddr"
|
"sing/common/socksaddr"
|
||||||
|
@ -316,28 +317,12 @@ type AssociatePacket struct {
|
||||||
Data []byte
|
Data []byte
|
||||||
}
|
}
|
||||||
|
|
||||||
func WriteAssociatePacket(writer io.Writer, packet *AssociatePacket) error {
|
func DecodeAssociatePacket(buffer *buf.Buffer) (*AssociatePacket, error) {
|
||||||
err := rw.WriteZeroN(writer, 2)
|
if buffer.Len() < 5 {
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
err = rw.WriteByte(writer, packet.Fragment)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
err = AddressSerializer.WriteAddressAndPort(writer, packet.Addr, packet.Port)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
return rw.WriteBytes(writer, packet.Data)
|
|
||||||
}
|
|
||||||
|
|
||||||
func DecodeAssociatePacket(buffer []byte) (*AssociatePacket, error) {
|
|
||||||
if len(buffer) < 5 {
|
|
||||||
return nil, exceptions.New("insufficient length")
|
return nil, exceptions.New("insufficient length")
|
||||||
}
|
}
|
||||||
fragment := buffer[2]
|
fragment := buffer.Byte(2)
|
||||||
reader := bytes.NewReader(buffer)
|
reader := bytes.NewReader(buffer.Bytes())
|
||||||
err := common.Error(reader.Seek(3, io.SeekStart))
|
err := common.Error(reader.Seek(3, io.SeekStart))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
|
@ -346,39 +331,29 @@ func DecodeAssociatePacket(buffer []byte) (*AssociatePacket, error) {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
index := len(buffer) - reader.Len()
|
buffer.Advance(reader.Len())
|
||||||
data := buffer[index:]
|
|
||||||
packet := &AssociatePacket{
|
packet := &AssociatePacket{
|
||||||
Fragment: fragment,
|
Fragment: fragment,
|
||||||
Addr: addr,
|
Addr: addr,
|
||||||
Port: port,
|
Port: port,
|
||||||
Data: data,
|
Data: buffer.Bytes(),
|
||||||
}
|
}
|
||||||
return packet, nil
|
return packet, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func ReadAssociatePacket(reader io.Reader) (*AssociatePacket, error) {
|
func EncodeAssociatePacket(packet *AssociatePacket, buffer *buf.Buffer) error {
|
||||||
err := rw.SkipN(reader, 2)
|
err := rw.WriteZeroN(buffer, 2)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return err
|
||||||
}
|
}
|
||||||
fragment, err := rw.ReadByte(reader)
|
err = rw.WriteByte(buffer, packet.Fragment)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return err
|
||||||
}
|
}
|
||||||
addr, port, err := AddressSerializer.ReadAddressAndPort(reader)
|
err = AddressSerializer.WriteAddressAndPort(buffer, packet.Addr, packet.Port)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return err
|
||||||
}
|
}
|
||||||
data, err := io.ReadAll(reader)
|
_, err = buffer.Write(packet.Data)
|
||||||
if err != nil {
|
return err
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
packet := &AssociatePacket{
|
|
||||||
Fragment: fragment,
|
|
||||||
Addr: addr,
|
|
||||||
Port: port,
|
|
||||||
Data: data,
|
|
||||||
}
|
|
||||||
return packet, nil
|
|
||||||
}
|
}
|
||||||
|
|
30
transport/block/outbound.go
Normal file
30
transport/block/outbound.go
Normal file
|
@ -0,0 +1,30 @@
|
||||||
|
package block
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
|
||||||
|
"sing/common/session"
|
||||||
|
"sing/transport"
|
||||||
|
)
|
||||||
|
|
||||||
|
var _ transport.Outbound = (*Outbound)(nil)
|
||||||
|
|
||||||
|
type Outbound struct {
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *Outbound) Init(*transport.OutboundContext) {
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *Outbound) Close() error {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (o *Outbound) NewConnection(ctx context.Context, conn *session.Conn) error {
|
||||||
|
conn.Conn.Close()
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (o *Outbound) NewPacketConnection(ctx context.Context, packetConn *session.PacketConn) error {
|
||||||
|
packetConn.Conn.Close()
|
||||||
|
return nil
|
||||||
|
}
|
25
transport/inbound.go
Normal file
25
transport/inbound.go
Normal file
|
@ -0,0 +1,25 @@
|
||||||
|
package transport
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
|
||||||
|
"sing/common/list"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Inbound interface {
|
||||||
|
Init(ctx *InboundContext)
|
||||||
|
Start() error
|
||||||
|
Close() error
|
||||||
|
}
|
||||||
|
|
||||||
|
type InboundContext struct {
|
||||||
|
Context context.Context
|
||||||
|
Tag string
|
||||||
|
Inbound Inbound
|
||||||
|
}
|
||||||
|
|
||||||
|
type InboundManager interface {
|
||||||
|
AddInbound(inbound Inbound, tag string)
|
||||||
|
Inbounds() *list.List[*InboundContext]
|
||||||
|
Inbound(tag string) *InboundContext
|
||||||
|
}
|
28
transport/outbound.go
Normal file
28
transport/outbound.go
Normal file
|
@ -0,0 +1,28 @@
|
||||||
|
package transport
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
|
||||||
|
"sing/common/list"
|
||||||
|
"sing/common/session"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Outbound interface {
|
||||||
|
Init(ctx *OutboundContext)
|
||||||
|
Close() error
|
||||||
|
NewConnection(ctx context.Context, conn *session.Conn) error
|
||||||
|
NewPacketConnection(ctx context.Context, packetConn *session.PacketConn) error
|
||||||
|
}
|
||||||
|
|
||||||
|
type OutboundContext struct {
|
||||||
|
Context context.Context
|
||||||
|
Tag string
|
||||||
|
Outbound Outbound
|
||||||
|
}
|
||||||
|
|
||||||
|
type OutboundManager interface {
|
||||||
|
AddOutbound(outbound Outbound, tag string)
|
||||||
|
Outbounds() *list.List[*OutboundContext]
|
||||||
|
Outbound(tag string) *OutboundContext
|
||||||
|
DefaultOutbound() *OutboundContext
|
||||||
|
}
|
187
transport/socks/inbound.go
Normal file
187
transport/socks/inbound.go
Normal file
|
@ -0,0 +1,187 @@
|
||||||
|
package socks
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"io"
|
||||||
|
"net"
|
||||||
|
|
||||||
|
"github.com/sirupsen/logrus"
|
||||||
|
"net/netip"
|
||||||
|
"sing/common"
|
||||||
|
"sing/common/buf"
|
||||||
|
"sing/common/exceptions"
|
||||||
|
"sing/common/session"
|
||||||
|
"sing/common/socksaddr"
|
||||||
|
"sing/protocol/socks"
|
||||||
|
"sing/transport"
|
||||||
|
"sing/transport/system"
|
||||||
|
)
|
||||||
|
|
||||||
|
var _ transport.Inbound = (*Inbound)(nil)
|
||||||
|
|
||||||
|
type Inbound struct {
|
||||||
|
lAddr netip.AddrPort
|
||||||
|
username, password string
|
||||||
|
tcpListener *system.TCPListener
|
||||||
|
udpListener *system.UDPListener
|
||||||
|
handler session.Handler
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *Inbound) Init(ctx *transport.InboundContext) {
|
||||||
|
}
|
||||||
|
|
||||||
|
type InboundConfig struct {
|
||||||
|
Listen string `json:"listen"`
|
||||||
|
Port uint16 `json:"port"`
|
||||||
|
Username string `json:"username,omitempty"`
|
||||||
|
Password string `json:"password,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewListener(handler session.Handler, config *InboundConfig) (*Inbound, error) {
|
||||||
|
addr, err := netip.ParseAddr(config.Listen)
|
||||||
|
if err != nil {
|
||||||
|
return nil, exceptions.Cause(err, "invalid listen address: ", config.Listen)
|
||||||
|
}
|
||||||
|
lAddr := netip.AddrPortFrom(addr, config.Port)
|
||||||
|
inbound := new(Inbound)
|
||||||
|
inbound.username, inbound.password = config.Username, config.Password
|
||||||
|
inbound.handler = handler
|
||||||
|
inbound.tcpListener = system.NewTCPListener(lAddr, inbound)
|
||||||
|
inbound.udpListener = system.NewUDPListener(lAddr, inbound)
|
||||||
|
return inbound, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *Inbound) Start() error {
|
||||||
|
err := h.tcpListener.Start()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return h.udpListener.Start()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *Inbound) HandleTCP(conn net.Conn) error {
|
||||||
|
authRequest, err := socks.ReadAuthRequest(conn)
|
||||||
|
if err != nil {
|
||||||
|
return exceptions.Cause(err, "read socks auth request")
|
||||||
|
}
|
||||||
|
if h.username != "" {
|
||||||
|
if bytes.IndexByte(authRequest.Methods, socks.AuthTypeNotRequired) > 0 {
|
||||||
|
err = socks.WriteAuthResponse(conn, &socks.AuthResponse{
|
||||||
|
Version: authRequest.Version,
|
||||||
|
Method: socks.AuthTypeNotRequired,
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
return exceptions.Cause(err, "write socks auth response")
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
socks.WriteAuthResponse(conn, &socks.AuthResponse{
|
||||||
|
Version: authRequest.Version,
|
||||||
|
Method: socks.AuthTypeNoAcceptedMethods,
|
||||||
|
})
|
||||||
|
return exceptions.New("no accepted methods, requested = ", authRequest.Methods, ", except no auth")
|
||||||
|
}
|
||||||
|
} else if bytes.IndexByte(authRequest.Methods, socks.AuthTypeNotRequired) == -1 {
|
||||||
|
socks.WriteAuthResponse(conn, &socks.AuthResponse{
|
||||||
|
Version: authRequest.Version,
|
||||||
|
Method: socks.AuthTypeNoAcceptedMethods,
|
||||||
|
})
|
||||||
|
return exceptions.New("no accepted methods, requested = ", authRequest.Methods, ", except password")
|
||||||
|
} else {
|
||||||
|
err = socks.WriteAuthResponse(conn, &socks.AuthResponse{
|
||||||
|
Version: authRequest.Version,
|
||||||
|
Method: socks.AuthTypeUsernamePassword,
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
return exceptions.Cause(err, "write socks auth response: ", err)
|
||||||
|
}
|
||||||
|
usernamePasswordRequest, err := socks.ReadUsernamePasswordAuthRequest(conn)
|
||||||
|
if err != nil {
|
||||||
|
return exceptions.Cause(err, "read username-password request")
|
||||||
|
}
|
||||||
|
if usernamePasswordRequest.Username != h.username {
|
||||||
|
socks.WriteUsernamePasswordAuthResponse(conn, &socks.UsernamePasswordAuthResponse{Status: socks.UsernamePasswordStatusFailure})
|
||||||
|
return exceptions.New("auth failed: excepted username ", h.username, ", got ", usernamePasswordRequest.Username)
|
||||||
|
} else if usernamePasswordRequest.Password != h.password {
|
||||||
|
socks.WriteUsernamePasswordAuthResponse(conn, &socks.UsernamePasswordAuthResponse{Status: socks.UsernamePasswordStatusFailure})
|
||||||
|
return exceptions.New("auth failed: excepted password ", h.password, ", got ", usernamePasswordRequest.Password)
|
||||||
|
}
|
||||||
|
err = socks.WriteUsernamePasswordAuthResponse(conn, &socks.UsernamePasswordAuthResponse{Status: socks.UsernamePasswordStatusSuccess})
|
||||||
|
if err != nil {
|
||||||
|
return exceptions.Cause(err, "write username-password response")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
request, err := socks.ReadRequest(conn)
|
||||||
|
if err != nil {
|
||||||
|
return exceptions.Cause(err, "read request")
|
||||||
|
}
|
||||||
|
switch request.Command {
|
||||||
|
case socks.CommandBind:
|
||||||
|
socks.WriteResponse(conn, &socks.Response{
|
||||||
|
Version: request.Version,
|
||||||
|
ReplyCode: socks.ReplyCodeUnsupported,
|
||||||
|
})
|
||||||
|
return exceptions.New("bind unsupported")
|
||||||
|
case socks.CommandUDPAssociate:
|
||||||
|
addr, port := session.AddressFromNetAddr(h.udpListener.LocalAddr())
|
||||||
|
err = socks.WriteResponse(conn, &socks.Response{
|
||||||
|
Version: request.Version,
|
||||||
|
ReplyCode: socks.ReplyCodeSuccess,
|
||||||
|
BindAddr: addr,
|
||||||
|
BindPort: port,
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
return exceptions.Cause(err, "write response")
|
||||||
|
}
|
||||||
|
io.Copy(io.Discard, conn)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
context := new(session.Context)
|
||||||
|
context.Network = session.NetworkTCP
|
||||||
|
context.Source, context.SourcePort = socksaddr.AddressFromNetAddr(conn.RemoteAddr())
|
||||||
|
context.Destination, context.DestinationPort = request.Addr, request.Port
|
||||||
|
h.handler.HandleConnection(&session.Conn{
|
||||||
|
Conn: conn,
|
||||||
|
Context: context,
|
||||||
|
})
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *Inbound) HandleUDP(buffer *buf.Buffer, sourceAddr net.Addr) error {
|
||||||
|
associatePacket, err := socks.DecodeAssociatePacket(buffer)
|
||||||
|
if err != nil {
|
||||||
|
return exceptions.Cause(err, "decode associate packet")
|
||||||
|
}
|
||||||
|
context := new(session.Context)
|
||||||
|
context.Network = session.NetworkUDP
|
||||||
|
context.Source, context.SourcePort = socksaddr.AddressFromNetAddr(sourceAddr)
|
||||||
|
context.Destination, context.DestinationPort = associatePacket.Addr, associatePacket.Port
|
||||||
|
h.handler.HandlePacket(&session.Packet{
|
||||||
|
Context: context,
|
||||||
|
Data: buffer,
|
||||||
|
Release: nil,
|
||||||
|
WriteBack: func(buffer *buf.Buffer, addr *net.UDPAddr) error {
|
||||||
|
header := new(socks.AssociatePacket)
|
||||||
|
header.Addr, header.Port = socksaddr.AddressFromNetAddr(addr)
|
||||||
|
header.Data = buffer.Bytes()
|
||||||
|
packet := buf.FullNew()
|
||||||
|
defer packet.Release()
|
||||||
|
err := socks.EncodeAssociatePacket(header, packet)
|
||||||
|
buffer.Release()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return common.Error(h.udpListener.WriteTo(packet.Bytes(), sourceAddr))
|
||||||
|
},
|
||||||
|
})
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *Inbound) OnError(err error) {
|
||||||
|
logrus.Warn("socks: ", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *Inbound) Close() error {
|
||||||
|
h.tcpListener.Close()
|
||||||
|
h.udpListener.Close()
|
||||||
|
return nil
|
||||||
|
}
|
30
transport/system/control.go
Normal file
30
transport/system/control.go
Normal file
|
@ -0,0 +1,30 @@
|
||||||
|
package system
|
||||||
|
|
||||||
|
import "syscall"
|
||||||
|
|
||||||
|
var ControlFunc func(fd uintptr) error
|
||||||
|
|
||||||
|
func Control(conn syscall.Conn) error {
|
||||||
|
if ControlFunc == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
rawConn, err := conn.SyscallConn()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return ControlRaw(rawConn)
|
||||||
|
}
|
||||||
|
|
||||||
|
func ControlRaw(conn syscall.RawConn) error {
|
||||||
|
if ControlFunc == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
var rawFd uintptr
|
||||||
|
err := conn.Control(func(fd uintptr) {
|
||||||
|
rawFd = fd
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return ControlFunc(rawFd)
|
||||||
|
}
|
52
transport/system/outbound.go
Normal file
52
transport/system/outbound.go
Normal file
|
@ -0,0 +1,52 @@
|
||||||
|
package system
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"net"
|
||||||
|
"sing/transport"
|
||||||
|
"syscall"
|
||||||
|
|
||||||
|
"sing/common/rw"
|
||||||
|
"sing/common/session"
|
||||||
|
)
|
||||||
|
|
||||||
|
var _ transport.Outbound = (*Outbound)(nil)
|
||||||
|
|
||||||
|
type Outbound struct{}
|
||||||
|
|
||||||
|
func (h *Outbound) Init(*transport.OutboundContext) {
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *Outbound) Close() error {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *Outbound) NewConnection(ctx context.Context, conn *session.Conn) error {
|
||||||
|
dialer := net.Dialer{
|
||||||
|
Control: func(network, address string, c syscall.RawConn) error {
|
||||||
|
return ControlRaw(c)
|
||||||
|
},
|
||||||
|
}
|
||||||
|
outConn, err := dialer.DialContext(ctx, "tcp", conn.Context.DestinationNetAddr())
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
connElement := session.AddConnection(outConn)
|
||||||
|
defer session.RemoveConnection(connElement)
|
||||||
|
return rw.CopyConn(ctx, conn.Conn, outConn)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *Outbound) NewPacketConnection(ctx context.Context, packetConn *session.PacketConn) error {
|
||||||
|
dialer := net.Dialer{
|
||||||
|
Control: func(network, address string, c syscall.RawConn) error {
|
||||||
|
return ControlRaw(c)
|
||||||
|
},
|
||||||
|
}
|
||||||
|
outConn, err := dialer.DialContext(ctx, "udp", packetConn.Context.DestinationNetAddr())
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
connElement := session.AddConnection(outConn)
|
||||||
|
defer session.RemoveConnection(connElement)
|
||||||
|
return rw.CopyPacketConn(ctx, packetConn.Conn, outConn.(net.PacketConn))
|
||||||
|
}
|
58
transport/system/tcp.go
Normal file
58
transport/system/tcp.go
Normal file
|
@ -0,0 +1,58 @@
|
||||||
|
package system
|
||||||
|
|
||||||
|
import (
|
||||||
|
"net"
|
||||||
|
|
||||||
|
"net/netip"
|
||||||
|
)
|
||||||
|
|
||||||
|
type TCPHandler interface {
|
||||||
|
HandleTCP(conn net.Conn) error
|
||||||
|
OnError(err error)
|
||||||
|
}
|
||||||
|
|
||||||
|
type TCPListener struct {
|
||||||
|
Listen netip.AddrPort
|
||||||
|
Handler TCPHandler
|
||||||
|
*net.TCPListener
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewTCPListener(listen netip.AddrPort, handler TCPHandler) *TCPListener {
|
||||||
|
return &TCPListener{
|
||||||
|
Listen: listen,
|
||||||
|
Handler: handler,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *TCPListener) Start() error {
|
||||||
|
tcpListener, err := net.ListenTCP("tcp", net.TCPAddrFromAddrPort(l.Listen))
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
l.TCPListener = tcpListener
|
||||||
|
go l.loop()
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *TCPListener) Close() error {
|
||||||
|
if l == nil || l.TCPListener == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
return l.TCPListener.Close()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *TCPListener) loop() {
|
||||||
|
for {
|
||||||
|
tcpConn, err := l.Accept()
|
||||||
|
if err != nil {
|
||||||
|
l.Close()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
go func() {
|
||||||
|
err := l.Handler.HandleTCP(tcpConn)
|
||||||
|
if err != nil {
|
||||||
|
l.Handler.OnError(err)
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
}
|
||||||
|
}
|
62
transport/system/udp.go
Normal file
62
transport/system/udp.go
Normal file
|
@ -0,0 +1,62 @@
|
||||||
|
package system
|
||||||
|
|
||||||
|
import (
|
||||||
|
"net"
|
||||||
|
|
||||||
|
"net/netip"
|
||||||
|
"sing/common/buf"
|
||||||
|
)
|
||||||
|
|
||||||
|
type UDPHandler interface {
|
||||||
|
HandleUDP(buffer *buf.Buffer, sourceAddr net.Addr) error
|
||||||
|
OnError(err error)
|
||||||
|
}
|
||||||
|
|
||||||
|
type UDPListener struct {
|
||||||
|
Listen netip.AddrPort
|
||||||
|
Handler UDPHandler
|
||||||
|
*net.UDPConn
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewUDPListener(listen netip.AddrPort, handler UDPHandler) *UDPListener {
|
||||||
|
return &UDPListener{
|
||||||
|
Listen: listen,
|
||||||
|
Handler: handler,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *UDPListener) Start() error {
|
||||||
|
udpConn, err := net.ListenUDP("udp", net.UDPAddrFromAddrPort(l.Listen))
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
l.UDPConn = udpConn
|
||||||
|
go l.loop()
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *UDPListener) Close() error {
|
||||||
|
if l == nil || l.UDPConn == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
return l.UDPConn.Close()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *UDPListener) loop() {
|
||||||
|
for {
|
||||||
|
buffer := buf.New()
|
||||||
|
n, addr, err := l.ReadFromUDP(buffer.Extend(buf.UDPBufferSize))
|
||||||
|
if err != nil {
|
||||||
|
buffer.Release()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
buffer.Truncate(n)
|
||||||
|
go func() {
|
||||||
|
err := l.Handler.HandleUDP(buffer, addr)
|
||||||
|
if err != nil {
|
||||||
|
buffer.Release()
|
||||||
|
l.Handler.OnError(err)
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
}
|
||||||
|
}
|
Loading…
Add table
Add a link
Reference in a new issue