mirror of https://github.com/v2ray/v2ray-core
handle inbound detour in vmess inbound
parent
573d0a9c29
commit
b0adb24003
|
@ -10,6 +10,7 @@ import (
|
||||||
"github.com/v2ray/v2ray-core/common/log"
|
"github.com/v2ray/v2ray-core/common/log"
|
||||||
v2net "github.com/v2ray/v2ray-core/common/net"
|
v2net "github.com/v2ray/v2ray-core/common/net"
|
||||||
"github.com/v2ray/v2ray-core/common/retry"
|
"github.com/v2ray/v2ray-core/common/retry"
|
||||||
|
"github.com/v2ray/v2ray-core/proxy"
|
||||||
)
|
)
|
||||||
|
|
||||||
type DokodemoDoor struct {
|
type DokodemoDoor struct {
|
||||||
|
@ -22,6 +23,7 @@ type DokodemoDoor struct {
|
||||||
space app.Space
|
space app.Space
|
||||||
tcpListener *net.TCPListener
|
tcpListener *net.TCPListener
|
||||||
udpConn *net.UDPConn
|
udpConn *net.UDPConn
|
||||||
|
listeningPort v2net.Port
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewDokodemoDoor(space app.Space, config *Config) *DokodemoDoor {
|
func NewDokodemoDoor(space app.Space, config *Config) *DokodemoDoor {
|
||||||
|
@ -33,6 +35,10 @@ func NewDokodemoDoor(space app.Space, config *Config) *DokodemoDoor {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (this *DokodemoDoor) Port() v2net.Port {
|
||||||
|
return this.listeningPort
|
||||||
|
}
|
||||||
|
|
||||||
func (this *DokodemoDoor) Close() {
|
func (this *DokodemoDoor) Close() {
|
||||||
this.accepting = false
|
this.accepting = false
|
||||||
if this.tcpListener != nil {
|
if this.tcpListener != nil {
|
||||||
|
@ -50,6 +56,14 @@ func (this *DokodemoDoor) Close() {
|
||||||
}
|
}
|
||||||
|
|
||||||
func (this *DokodemoDoor) Listen(port v2net.Port) error {
|
func (this *DokodemoDoor) Listen(port v2net.Port) error {
|
||||||
|
if this.accepting {
|
||||||
|
if this.listeningPort == port {
|
||||||
|
return nil
|
||||||
|
} else {
|
||||||
|
return proxy.ErrorAlreadyListening
|
||||||
|
}
|
||||||
|
}
|
||||||
|
this.listeningPort = port
|
||||||
this.accepting = true
|
this.accepting = true
|
||||||
|
|
||||||
if this.config.Network.HasNetwork(v2net.TCPNetwork) {
|
if this.config.Network.HasNetwork(v2net.TCPNetwork) {
|
||||||
|
|
|
@ -7,4 +7,5 @@ import (
|
||||||
var (
|
var (
|
||||||
InvalidAuthentication = errors.New("Invalid authentication.")
|
InvalidAuthentication = errors.New("Invalid authentication.")
|
||||||
InvalidProtocolVersion = errors.New("Invalid protocol version.")
|
InvalidProtocolVersion = errors.New("Invalid protocol version.")
|
||||||
|
ErrorAlreadyListening = errors.New("Already listening on another port.")
|
||||||
)
|
)
|
||||||
|
|
|
@ -15,6 +15,7 @@ import (
|
||||||
v2net "github.com/v2ray/v2ray-core/common/net"
|
v2net "github.com/v2ray/v2ray-core/common/net"
|
||||||
"github.com/v2ray/v2ray-core/common/retry"
|
"github.com/v2ray/v2ray-core/common/retry"
|
||||||
"github.com/v2ray/v2ray-core/common/serial"
|
"github.com/v2ray/v2ray-core/common/serial"
|
||||||
|
"github.com/v2ray/v2ray-core/proxy"
|
||||||
"github.com/v2ray/v2ray-core/transport/ray"
|
"github.com/v2ray/v2ray-core/transport/ray"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -24,6 +25,7 @@ type HttpProxyServer struct {
|
||||||
space app.Space
|
space app.Space
|
||||||
config *Config
|
config *Config
|
||||||
tcpListener *net.TCPListener
|
tcpListener *net.TCPListener
|
||||||
|
listeningPort v2net.Port
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewHttpProxyServer(space app.Space, config *Config) *HttpProxyServer {
|
func NewHttpProxyServer(space app.Space, config *Config) *HttpProxyServer {
|
||||||
|
@ -33,6 +35,10 @@ func NewHttpProxyServer(space app.Space, config *Config) *HttpProxyServer {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (this *HttpProxyServer) Port() v2net.Port {
|
||||||
|
return this.listeningPort
|
||||||
|
}
|
||||||
|
|
||||||
func (this *HttpProxyServer) Close() {
|
func (this *HttpProxyServer) Close() {
|
||||||
this.accepting = false
|
this.accepting = false
|
||||||
if this.tcpListener != nil {
|
if this.tcpListener != nil {
|
||||||
|
@ -44,6 +50,15 @@ func (this *HttpProxyServer) Close() {
|
||||||
}
|
}
|
||||||
|
|
||||||
func (this *HttpProxyServer) Listen(port v2net.Port) error {
|
func (this *HttpProxyServer) Listen(port v2net.Port) error {
|
||||||
|
if this.accepting {
|
||||||
|
if this.listeningPort == port {
|
||||||
|
return nil
|
||||||
|
} else {
|
||||||
|
return proxy.ErrorAlreadyListening
|
||||||
|
}
|
||||||
|
}
|
||||||
|
this.listeningPort = port
|
||||||
|
|
||||||
tcpListener, err := net.ListenTCP("tcp", &net.TCPAddr{
|
tcpListener, err := net.ListenTCP("tcp", &net.TCPAddr{
|
||||||
Port: int(port.Value()),
|
Port: int(port.Value()),
|
||||||
IP: []byte{0, 0, 0, 0},
|
IP: []byte{0, 0, 0, 0},
|
||||||
|
|
|
@ -13,6 +13,8 @@ type InboundConnectionHandler interface {
|
||||||
Listen(port v2net.Port) error
|
Listen(port v2net.Port) error
|
||||||
// Close stops the handler to accepting anymore inbound connections.
|
// Close stops the handler to accepting anymore inbound connections.
|
||||||
Close()
|
Close()
|
||||||
|
// Port returns the port that the handler is listening on.
|
||||||
|
Port() v2net.Port
|
||||||
}
|
}
|
||||||
|
|
||||||
// An OutboundConnectionHandler handles outbound network connection for V2Ray.
|
// An OutboundConnectionHandler handles outbound network connection for V2Ray.
|
||||||
|
|
|
@ -31,6 +31,7 @@ type SocksServer struct {
|
||||||
tcpListener *net.TCPListener
|
tcpListener *net.TCPListener
|
||||||
udpConn *net.UDPConn
|
udpConn *net.UDPConn
|
||||||
udpAddress v2net.Destination
|
udpAddress v2net.Destination
|
||||||
|
listeningPort v2net.Port
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewSocksServer(space app.Space, config *Config) *SocksServer {
|
func NewSocksServer(space app.Space, config *Config) *SocksServer {
|
||||||
|
@ -40,6 +41,10 @@ func NewSocksServer(space app.Space, config *Config) *SocksServer {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (this *SocksServer) Port() v2net.Port {
|
||||||
|
return this.listeningPort
|
||||||
|
}
|
||||||
|
|
||||||
func (this *SocksServer) Close() {
|
func (this *SocksServer) Close() {
|
||||||
this.accepting = false
|
this.accepting = false
|
||||||
if this.tcpListener != nil {
|
if this.tcpListener != nil {
|
||||||
|
@ -57,6 +62,15 @@ func (this *SocksServer) Close() {
|
||||||
}
|
}
|
||||||
|
|
||||||
func (this *SocksServer) Listen(port v2net.Port) error {
|
func (this *SocksServer) Listen(port v2net.Port) error {
|
||||||
|
if this.accepting {
|
||||||
|
if this.listeningPort == port {
|
||||||
|
return nil
|
||||||
|
} else {
|
||||||
|
return proxy.ErrorAlreadyListening
|
||||||
|
}
|
||||||
|
}
|
||||||
|
this.listeningPort = port
|
||||||
|
|
||||||
listener, err := net.ListenTCP("tcp", &net.TCPAddr{
|
listener, err := net.ListenTCP("tcp", &net.TCPAddr{
|
||||||
IP: []byte{0, 0, 0, 0},
|
IP: []byte{0, 0, 0, 0},
|
||||||
Port: int(port),
|
Port: int(port),
|
||||||
|
|
|
@ -9,17 +9,21 @@ import (
|
||||||
)
|
)
|
||||||
|
|
||||||
type InboundConnectionHandler struct {
|
type InboundConnectionHandler struct {
|
||||||
Port v2net.Port
|
port v2net.Port
|
||||||
Space app.Space
|
Space app.Space
|
||||||
ConnInput io.Reader
|
ConnInput io.Reader
|
||||||
ConnOutput io.Writer
|
ConnOutput io.Writer
|
||||||
}
|
}
|
||||||
|
|
||||||
func (this *InboundConnectionHandler) Listen(port v2net.Port) error {
|
func (this *InboundConnectionHandler) Listen(port v2net.Port) error {
|
||||||
this.Port = port
|
this.port = port
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (this *InboundConnectionHandler) Port() v2net.Port {
|
||||||
|
return this.port
|
||||||
|
}
|
||||||
|
|
||||||
func (this *InboundConnectionHandler) Close() {
|
func (this *InboundConnectionHandler) Close() {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,7 +2,6 @@ package command
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"io"
|
"io"
|
||||||
"time"
|
|
||||||
|
|
||||||
v2net "github.com/v2ray/v2ray-core/common/net"
|
v2net "github.com/v2ray/v2ray-core/common/net"
|
||||||
"github.com/v2ray/v2ray-core/common/serial"
|
"github.com/v2ray/v2ray-core/common/serial"
|
||||||
|
@ -26,40 +25,29 @@ type SwitchAccount struct {
|
||||||
Port v2net.Port
|
Port v2net.Port
|
||||||
ID *uuid.UUID
|
ID *uuid.UUID
|
||||||
AlterIds serial.Uint16Literal
|
AlterIds serial.Uint16Literal
|
||||||
ValidUntil time.Time
|
ValidSec serial.Uint16Literal
|
||||||
}
|
}
|
||||||
|
|
||||||
func (this *SwitchAccount) Marshal(writer io.Writer) (int, error) {
|
func (this *SwitchAccount) Marshal(writer io.Writer) {
|
||||||
outBytes := 0
|
|
||||||
hostStr := ""
|
hostStr := ""
|
||||||
if this.Host != nil {
|
if this.Host != nil {
|
||||||
hostStr = this.Host.String()
|
hostStr = this.Host.String()
|
||||||
}
|
}
|
||||||
writer.Write([]byte{byte(len(hostStr))})
|
writer.Write([]byte{byte(len(hostStr))})
|
||||||
outBytes++
|
|
||||||
|
|
||||||
if len(hostStr) > 0 {
|
if len(hostStr) > 0 {
|
||||||
writer.Write([]byte(hostStr))
|
writer.Write([]byte(hostStr))
|
||||||
outBytes += len(hostStr)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
writer.Write(this.Port.Bytes())
|
writer.Write(this.Port.Bytes())
|
||||||
outBytes += 2
|
|
||||||
|
|
||||||
idBytes := this.ID.Bytes()
|
idBytes := this.ID.Bytes()
|
||||||
writer.Write(idBytes)
|
writer.Write(idBytes)
|
||||||
outBytes += len(idBytes)
|
|
||||||
|
|
||||||
writer.Write(this.AlterIds.Bytes())
|
writer.Write(this.AlterIds.Bytes())
|
||||||
outBytes += 2
|
|
||||||
|
|
||||||
timestamp := this.ValidUntil.Unix()
|
|
||||||
timeBytes := serial.Int64Literal(timestamp).Bytes()
|
|
||||||
|
|
||||||
|
timeBytes := this.ValidSec.Bytes()
|
||||||
writer.Write(timeBytes)
|
writer.Write(timeBytes)
|
||||||
outBytes += len(timeBytes)
|
|
||||||
|
|
||||||
return outBytes, nil
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (this *SwitchAccount) Unmarshal(data []byte) error {
|
func (this *SwitchAccount) Unmarshal(data []byte) error {
|
||||||
|
@ -84,9 +72,9 @@ func (this *SwitchAccount) Unmarshal(data []byte) error {
|
||||||
}
|
}
|
||||||
this.AlterIds = serial.ParseUint16(data[alterIdStart : alterIdStart+2])
|
this.AlterIds = serial.ParseUint16(data[alterIdStart : alterIdStart+2])
|
||||||
timeStart := alterIdStart + 2
|
timeStart := alterIdStart + 2
|
||||||
if len(data) < timeStart+8 {
|
if len(data) < timeStart+2 {
|
||||||
return transport.CorruptedPacket
|
return transport.CorruptedPacket
|
||||||
}
|
}
|
||||||
this.ValidUntil = time.Unix(serial.BytesLiteral(data[timeStart:timeStart+8]).Int64Value(), 0)
|
this.ValidSec = serial.ParseUint16(data[timeStart : timeStart+2])
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,7 +3,6 @@ package command_test
|
||||||
import (
|
import (
|
||||||
"bytes"
|
"bytes"
|
||||||
"testing"
|
"testing"
|
||||||
"time"
|
|
||||||
|
|
||||||
netassert "github.com/v2ray/v2ray-core/common/net/testing/assert"
|
netassert "github.com/v2ray/v2ray-core/common/net/testing/assert"
|
||||||
"github.com/v2ray/v2ray-core/common/uuid"
|
"github.com/v2ray/v2ray-core/common/uuid"
|
||||||
|
@ -19,16 +18,14 @@ func TestSwitchAccount(t *testing.T) {
|
||||||
Port: 1234,
|
Port: 1234,
|
||||||
ID: uuid.New(),
|
ID: uuid.New(),
|
||||||
AlterIds: 1024,
|
AlterIds: 1024,
|
||||||
ValidUntil: time.Now(),
|
ValidSec: 8080,
|
||||||
}
|
}
|
||||||
|
|
||||||
cmd, err := CreateResponseCommand(1)
|
cmd, err := CreateResponseCommand(1)
|
||||||
assert.Error(err).IsNil()
|
assert.Error(err).IsNil()
|
||||||
|
|
||||||
buffer := bytes.NewBuffer(make([]byte, 0, 1024))
|
buffer := bytes.NewBuffer(make([]byte, 0, 1024))
|
||||||
nBytes, err := sa.Marshal(buffer)
|
sa.Marshal(buffer)
|
||||||
assert.Error(err).IsNil()
|
|
||||||
assert.Int(nBytes).Equals(buffer.Len())
|
|
||||||
|
|
||||||
cmd.Unmarshal(buffer.Bytes())
|
cmd.Unmarshal(buffer.Bytes())
|
||||||
sa2, ok := cmd.(*SwitchAccount)
|
sa2, ok := cmd.(*SwitchAccount)
|
||||||
|
@ -36,5 +33,5 @@ func TestSwitchAccount(t *testing.T) {
|
||||||
netassert.Port(sa.Port).Equals(sa2.Port)
|
netassert.Port(sa.Port).Equals(sa2.Port)
|
||||||
assert.String(sa.ID).Equals(sa2.ID.String())
|
assert.String(sa.ID).Equals(sa2.ID.String())
|
||||||
assert.Uint16(sa.AlterIds.Value()).Equals(sa2.AlterIds.Value())
|
assert.Uint16(sa.AlterIds.Value()).Equals(sa2.AlterIds.Value())
|
||||||
assert.Int64(sa.ValidUntil.Unix()).Equals(sa2.ValidUntil.Unix())
|
assert.Uint16(sa.ValidSec.Value()).Equals(sa2.ValidSec.Value())
|
||||||
}
|
}
|
||||||
|
|
|
@ -10,7 +10,7 @@ var (
|
||||||
)
|
)
|
||||||
|
|
||||||
type Command interface {
|
type Command interface {
|
||||||
Marshal(io.Writer) (int, error)
|
Marshal(io.Writer)
|
||||||
Unmarshal([]byte) error
|
Unmarshal([]byte) error
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -26,20 +26,16 @@ type CacheDns struct {
|
||||||
Address v2net.Address
|
Address v2net.Address
|
||||||
}
|
}
|
||||||
|
|
||||||
func (this *CacheDns) Marshal(writer io.Writer) (int, error) {
|
func (this *CacheDns) Marshal(writer io.Writer) {
|
||||||
if this.Address.IsIPv4() {
|
if this.Address.IsIPv4() {
|
||||||
writer.Write([]byte{typeIPv4})
|
writer.Write([]byte{typeIPv4})
|
||||||
writer.Write(this.Address.IP())
|
writer.Write(this.Address.IP())
|
||||||
return 5, nil
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if this.Address.IsIPv6() {
|
if this.Address.IsIPv6() {
|
||||||
writer.Write([]byte{typeIPv6})
|
writer.Write([]byte{typeIPv6})
|
||||||
writer.Write(this.Address.IP())
|
writer.Write(this.Address.IP())
|
||||||
return 17, nil
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0, ErrDomainAddress
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (this *CacheDns) Unmarshal(data []byte) error {
|
func (this *CacheDns) Unmarshal(data []byte) error {
|
||||||
|
|
|
@ -21,12 +21,10 @@ func TestCacheDnsIPv4(t *testing.T) {
|
||||||
buffer := alloc.NewBuffer().Clear()
|
buffer := alloc.NewBuffer().Clear()
|
||||||
defer buffer.Release()
|
defer buffer.Release()
|
||||||
|
|
||||||
nBytes, err := cd.Marshal(buffer)
|
cd.Marshal(buffer)
|
||||||
assert.Error(err).IsNil()
|
|
||||||
assert.Int(nBytes).Equals(buffer.Len())
|
|
||||||
|
|
||||||
cd2 := &CacheDns{}
|
cd2 := &CacheDns{}
|
||||||
err = cd2.Unmarshal(buffer.Value)
|
err := cd2.Unmarshal(buffer.Value)
|
||||||
assert.Error(err).IsNil()
|
assert.Error(err).IsNil()
|
||||||
netassert.Address(cd.Address).Equals(cd2.Address)
|
netassert.Address(cd.Address).Equals(cd2.Address)
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,45 @@
|
||||||
|
package inbound
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/v2ray/v2ray-core/common/alloc"
|
||||||
|
"github.com/v2ray/v2ray-core/common/serial"
|
||||||
|
"github.com/v2ray/v2ray-core/proxy/vmess/command"
|
||||||
|
)
|
||||||
|
|
||||||
|
func (this *VMessInboundHandler) generateCommand(buffer *alloc.Buffer) {
|
||||||
|
cmd := byte(0)
|
||||||
|
commandBytes := alloc.NewSmallBuffer().Clear()
|
||||||
|
defer commandBytes.Release()
|
||||||
|
|
||||||
|
if this.features != nil && this.features.Detour != nil {
|
||||||
|
tag := this.features.Detour.ToTag
|
||||||
|
if this.space.HasInboundHandlerManager() {
|
||||||
|
handlerManager := this.space.InboundHandlerManager()
|
||||||
|
handler, availableSec := handlerManager.GetHandler(tag)
|
||||||
|
inboundHandler, ok := handler.(*VMessInboundHandler)
|
||||||
|
if ok {
|
||||||
|
user := inboundHandler.GetUser()
|
||||||
|
availableSecUint16 := uint16(65535)
|
||||||
|
if availableSec < 65535 {
|
||||||
|
availableSecUint16 = uint16(availableSec)
|
||||||
|
}
|
||||||
|
|
||||||
|
saCmd := &command.SwitchAccount{
|
||||||
|
Port: inboundHandler.Port(),
|
||||||
|
ID: user.ID.UUID(),
|
||||||
|
AlterIds: serial.Uint16Literal(len(user.AlterIDs)),
|
||||||
|
ValidSec: serial.Uint16Literal(availableSecUint16),
|
||||||
|
}
|
||||||
|
saCmd.Marshal(commandBytes)
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if commandBytes.Len() > 256 {
|
||||||
|
buffer.AppendBytes(byte(0), byte(0))
|
||||||
|
} else {
|
||||||
|
buffer.AppendBytes(cmd, byte(commandBytes.Len()))
|
||||||
|
buffer.Append(commandBytes.Value)
|
||||||
|
}
|
||||||
|
}
|
|
@ -24,15 +24,15 @@ type VMessInboundHandler struct {
|
||||||
sync.Mutex
|
sync.Mutex
|
||||||
space app.Space
|
space app.Space
|
||||||
clients protocol.UserSet
|
clients protocol.UserSet
|
||||||
|
user *vmess.User
|
||||||
accepting bool
|
accepting bool
|
||||||
listener *net.TCPListener
|
listener *net.TCPListener
|
||||||
|
features *FeaturesConfig
|
||||||
|
listeningPort v2net.Port
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewVMessInboundHandler(space app.Space, clients protocol.UserSet) *VMessInboundHandler {
|
func (this *VMessInboundHandler) Port() v2net.Port {
|
||||||
return &VMessInboundHandler{
|
return this.listeningPort
|
||||||
space: space,
|
|
||||||
clients: clients,
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (this *VMessInboundHandler) Close() {
|
func (this *VMessInboundHandler) Close() {
|
||||||
|
@ -45,11 +45,20 @@ func (this *VMessInboundHandler) Close() {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (this *VMessInboundHandler) AddUser(user vmess.User) {
|
func (this *VMessInboundHandler) GetUser() *vmess.User {
|
||||||
|
return this.user
|
||||||
}
|
}
|
||||||
|
|
||||||
func (this *VMessInboundHandler) Listen(port v2net.Port) error {
|
func (this *VMessInboundHandler) Listen(port v2net.Port) error {
|
||||||
|
if this.accepting {
|
||||||
|
if this.listeningPort == port {
|
||||||
|
return nil
|
||||||
|
} else {
|
||||||
|
return proxy.ErrorAlreadyListening
|
||||||
|
}
|
||||||
|
}
|
||||||
|
this.listeningPort = port
|
||||||
|
|
||||||
listener, err := net.ListenTCP("tcp", &net.TCPAddr{
|
listener, err := net.ListenTCP("tcp", &net.TCPAddr{
|
||||||
IP: []byte{0, 0, 0, 0},
|
IP: []byte{0, 0, 0, 0},
|
||||||
Port: int(port),
|
Port: int(port),
|
||||||
|
@ -175,6 +184,11 @@ func init() {
|
||||||
allowedClients.AddUser(user)
|
allowedClients.AddUser(user)
|
||||||
}
|
}
|
||||||
|
|
||||||
return NewVMessInboundHandler(space, allowedClients), nil
|
return &VMessInboundHandler{
|
||||||
|
space: space,
|
||||||
|
clients: allowedClients,
|
||||||
|
features: config.Features,
|
||||||
|
user: config.AllowedUsers[0],
|
||||||
|
}, nil
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue