1
0
mirror of https://github.com/v2fly/v2ray-core.git synced 2024-06-30 02:55:23 +00:00
v2fly/common/mux/frame.go

146 lines
3.4 KiB
Go
Raw Normal View History

2017-02-07 20:11:47 +00:00
package mux
import (
2018-11-02 14:01:33 +00:00
"encoding/binary"
2018-09-26 11:01:12 +00:00
"io"
2021-02-16 20:31:50 +00:00
"github.com/v2fly/v2ray-core/v4/common"
"github.com/v2fly/v2ray-core/v4/common/bitmask"
"github.com/v2fly/v2ray-core/v4/common/buf"
"github.com/v2fly/v2ray-core/v4/common/net"
"github.com/v2fly/v2ray-core/v4/common/protocol"
"github.com/v2fly/v2ray-core/v4/common/serial"
2017-02-07 20:11:47 +00:00
)
type SessionStatus byte
const (
2017-04-12 09:52:10 +00:00
SessionStatusNew SessionStatus = 0x01
SessionStatusKeep SessionStatus = 0x02
SessionStatusEnd SessionStatus = 0x03
SessionStatusKeepAlive SessionStatus = 0x04
2017-02-07 20:11:47 +00:00
)
2017-03-31 22:53:01 +00:00
const (
2018-04-04 19:33:33 +00:00
OptionData bitmask.Byte = 0x01
OptionError bitmask.Byte = 0x02
2017-03-31 22:53:01 +00:00
)
2017-02-07 20:11:47 +00:00
type TargetNetwork byte
const (
TargetNetworkTCP TargetNetwork = 0x01
2017-04-02 07:48:30 +00:00
TargetNetworkUDP TargetNetwork = 0x02
2017-02-07 20:11:47 +00:00
)
2018-02-23 23:57:54 +00:00
var addrParser = protocol.NewAddressParser(
protocol.AddressFamilyByte(byte(protocol.AddressTypeIPv4), net.AddressFamilyIPv4),
protocol.AddressFamilyByte(byte(protocol.AddressTypeDomain), net.AddressFamilyDomain),
protocol.AddressFamilyByte(byte(protocol.AddressTypeIPv6), net.AddressFamilyIPv6),
protocol.PortThenAddress(),
)
2017-03-26 23:47:01 +00:00
/*
Frame format
2 bytes - length
2 bytes - session id
1 bytes - status
2017-03-31 22:53:01 +00:00
1 bytes - option
2017-03-26 23:47:01 +00:00
1 byte - network
2 bytes - port
n bytes - address
*/
2017-02-07 20:11:47 +00:00
type FrameMetadata struct {
Target net.Destination
2017-10-21 17:29:27 +00:00
SessionID uint16
2017-10-21 18:40:31 +00:00
Option bitmask.Byte
SessionStatus SessionStatus
2017-02-07 20:11:47 +00:00
}
2018-02-23 23:57:54 +00:00
func (f FrameMetadata) WriteTo(b *buf.Buffer) error {
2018-11-14 21:11:05 +00:00
lenBytes := b.Extend(2)
2018-02-23 23:57:54 +00:00
len0 := b.Len()
2018-11-14 21:11:05 +00:00
sessionBytes := b.Extend(2)
binary.BigEndian.PutUint16(sessionBytes, f.SessionID)
2018-02-23 23:57:54 +00:00
2018-11-14 21:11:05 +00:00
common.Must(b.WriteByte(byte(f.SessionStatus)))
common.Must(b.WriteByte(byte(f.Option)))
2018-02-23 23:57:54 +00:00
if f.SessionStatus == SessionStatusNew {
switch f.Target.Network {
case net.Network_TCP:
2018-11-14 21:11:05 +00:00
common.Must(b.WriteByte(byte(TargetNetworkTCP)))
2018-02-23 23:57:54 +00:00
case net.Network_UDP:
2018-11-14 21:11:05 +00:00
common.Must(b.WriteByte(byte(TargetNetworkUDP)))
2017-02-07 20:11:47 +00:00
}
2017-04-03 10:55:46 +00:00
2018-02-23 23:57:54 +00:00
if err := addrParser.WriteAddressPort(b, f.Target.Address, f.Target.Port); err != nil {
return err
}
2017-02-07 20:11:47 +00:00
}
2018-02-23 23:57:54 +00:00
len1 := b.Len()
2018-11-02 14:01:33 +00:00
binary.BigEndian.PutUint16(lenBytes, uint16(len1-len0))
2018-02-23 23:57:54 +00:00
return nil
2017-02-07 20:11:47 +00:00
}
2018-10-14 06:05:23 +00:00
// Unmarshal reads FrameMetadata from the given reader.
func (f *FrameMetadata) Unmarshal(reader io.Reader) error {
2018-11-03 12:03:02 +00:00
metaLen, err := serial.ReadUint16(reader)
2018-09-26 11:01:12 +00:00
if err != nil {
return err
}
if metaLen > 512 {
return newError("invalid metalen ", metaLen).AtError()
}
b := buf.New()
defer b.Release()
2018-11-02 14:01:33 +00:00
if _, err := b.ReadFullFrom(reader, int32(metaLen)); err != nil {
2018-09-26 11:01:12 +00:00
return err
}
2018-10-14 06:05:23 +00:00
return f.UnmarshalFromBuffer(b)
2018-09-26 11:01:12 +00:00
}
2018-10-14 06:05:23 +00:00
// UnmarshalFromBuffer reads a FrameMetadata from the given buffer.
2018-05-26 23:19:05 +00:00
// Visible for testing only.
2018-10-14 06:05:23 +00:00
func (f *FrameMetadata) UnmarshalFromBuffer(b *buf.Buffer) error {
2018-02-23 23:57:54 +00:00
if b.Len() < 4 {
2018-09-26 11:01:12 +00:00
return newError("insufficient buffer: ", b.Len())
2017-02-07 20:11:47 +00:00
}
2018-11-02 14:47:58 +00:00
f.SessionID = binary.BigEndian.Uint16(b.BytesTo(2))
2018-09-26 11:01:12 +00:00
f.SessionStatus = SessionStatus(b.Byte(2))
f.Option = bitmask.Byte(b.Byte(3))
f.Target.Network = net.Network_Unknown
2017-02-07 20:11:47 +00:00
if f.SessionStatus == SessionStatusNew {
2018-11-14 21:55:33 +00:00
if b.Len() < 8 {
return newError("insufficient buffer: ", b.Len())
}
2018-02-23 23:57:54 +00:00
network := TargetNetwork(b.Byte(4))
b.Advance(5)
2018-02-23 23:57:54 +00:00
addr, port, err := addrParser.ReadAddressPort(nil, b)
if err != nil {
2018-09-26 11:01:12 +00:00
return newError("failed to parse address and port").Base(err)
2017-02-07 20:11:47 +00:00
}
2018-02-23 23:57:54 +00:00
2017-02-07 20:11:47 +00:00
switch network {
case TargetNetworkTCP:
f.Target = net.TCPDestination(addr, port)
2017-04-02 07:48:30 +00:00
case TargetNetworkUDP:
2017-02-07 20:11:47 +00:00
f.Target = net.UDPDestination(addr, port)
2017-04-03 11:07:12 +00:00
default:
2018-09-26 11:01:12 +00:00
return newError("unknown network type: ", network)
2017-02-07 20:11:47 +00:00
}
}
2018-09-26 11:01:12 +00:00
return nil
2017-02-07 20:11:47 +00:00
}