2024-09-08 12:06:14 +02:00
|
|
|
package messages
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
2024-09-11 16:20:30 +02:00
|
|
|
MaxHandshakeSize = 212
|
|
|
|
MaxHandshakeRespSize = 8192
|
|
|
|
|
|
|
|
CurrentProtocolVersion = 1
|
|
|
|
|
|
|
|
MsgTypeUnknown MsgType = 0
|
|
|
|
// Deprecated: Use MsgTypeAuth instead.
|
|
|
|
MsgTypeHello MsgType = 1
|
|
|
|
// Deprecated: Use MsgTypeAuthResponse instead.
|
2024-09-08 12:06:14 +02:00
|
|
|
MsgTypeHelloResponse MsgType = 2
|
|
|
|
MsgTypeTransport MsgType = 3
|
|
|
|
MsgTypeClose MsgType = 4
|
|
|
|
MsgTypeHealthCheck MsgType = 5
|
2024-09-11 16:20:30 +02:00
|
|
|
MsgTypeAuth = 6
|
|
|
|
MsgTypeAuthResponse = 7
|
2024-09-08 12:06:14 +02:00
|
|
|
|
|
|
|
SizeOfVersionByte = 1
|
|
|
|
SizeOfMsgType = 1
|
|
|
|
|
|
|
|
SizeOfProtoHeader = SizeOfVersionByte + SizeOfMsgType
|
|
|
|
|
|
|
|
sizeOfMagicByte = 4
|
|
|
|
|
|
|
|
headerSizeTransport = IDSize
|
2024-09-11 16:20:30 +02:00
|
|
|
|
2024-09-08 12:06:14 +02:00
|
|
|
headerSizeHello = sizeOfMagicByte + IDSize
|
|
|
|
headerSizeHelloResp = 0
|
|
|
|
|
2024-09-11 16:20:30 +02:00
|
|
|
headerSizeAuth = sizeOfMagicByte + IDSize
|
|
|
|
headerSizeAuthResp = 0
|
2024-09-08 12:06:14 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
ErrInvalidMessageLength = errors.New("invalid message length")
|
|
|
|
ErrUnsupportedVersion = errors.New("unsupported version")
|
|
|
|
|
|
|
|
magicHeader = []byte{0x21, 0x12, 0xA4, 0x42}
|
|
|
|
|
|
|
|
healthCheckMsg = []byte{byte(CurrentProtocolVersion), byte(MsgTypeHealthCheck)}
|
|
|
|
)
|
|
|
|
|
|
|
|
type MsgType byte
|
|
|
|
|
|
|
|
func (m MsgType) String() string {
|
|
|
|
switch m {
|
|
|
|
case MsgTypeHello:
|
|
|
|
return "hello"
|
|
|
|
case MsgTypeHelloResponse:
|
|
|
|
return "hello response"
|
2024-09-11 16:20:30 +02:00
|
|
|
case MsgTypeAuth:
|
|
|
|
return "auth"
|
|
|
|
case MsgTypeAuthResponse:
|
|
|
|
return "auth response"
|
2024-09-08 12:06:14 +02:00
|
|
|
case MsgTypeTransport:
|
|
|
|
return "transport"
|
|
|
|
case MsgTypeClose:
|
|
|
|
return "close"
|
|
|
|
case MsgTypeHealthCheck:
|
|
|
|
return "health check"
|
|
|
|
default:
|
|
|
|
return "unknown"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// ValidateVersion checks if the given version is supported by the protocol
|
|
|
|
func ValidateVersion(msg []byte) (int, error) {
|
|
|
|
if len(msg) < SizeOfVersionByte {
|
|
|
|
return 0, ErrInvalidMessageLength
|
|
|
|
}
|
|
|
|
version := int(msg[0])
|
|
|
|
if version != CurrentProtocolVersion {
|
|
|
|
return 0, fmt.Errorf("%d: %w", version, ErrUnsupportedVersion)
|
|
|
|
}
|
|
|
|
return version, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// DetermineClientMessageType determines the message type from the first the message
|
|
|
|
func DetermineClientMessageType(msg []byte) (MsgType, error) {
|
|
|
|
if len(msg) < SizeOfMsgType {
|
|
|
|
return 0, ErrInvalidMessageLength
|
|
|
|
}
|
|
|
|
|
|
|
|
msgType := MsgType(msg[0])
|
|
|
|
switch msgType {
|
|
|
|
case
|
|
|
|
MsgTypeHello,
|
2024-09-11 16:20:30 +02:00
|
|
|
MsgTypeAuth,
|
2024-09-08 12:06:14 +02:00
|
|
|
MsgTypeTransport,
|
|
|
|
MsgTypeClose,
|
|
|
|
MsgTypeHealthCheck:
|
|
|
|
return msgType, nil
|
|
|
|
default:
|
|
|
|
return MsgTypeUnknown, fmt.Errorf("invalid msg type %d", msgType)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// DetermineServerMessageType determines the message type from the first the message
|
|
|
|
func DetermineServerMessageType(msg []byte) (MsgType, error) {
|
|
|
|
if len(msg) < SizeOfMsgType {
|
|
|
|
return 0, ErrInvalidMessageLength
|
|
|
|
}
|
|
|
|
|
|
|
|
msgType := MsgType(msg[0])
|
|
|
|
switch msgType {
|
|
|
|
case
|
|
|
|
MsgTypeHelloResponse,
|
2024-09-11 16:20:30 +02:00
|
|
|
MsgTypeAuthResponse,
|
2024-09-08 12:06:14 +02:00
|
|
|
MsgTypeTransport,
|
|
|
|
MsgTypeClose,
|
|
|
|
MsgTypeHealthCheck:
|
|
|
|
return msgType, nil
|
|
|
|
default:
|
|
|
|
return MsgTypeUnknown, fmt.Errorf("invalid msg type %d", msgType)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-09-11 16:20:30 +02:00
|
|
|
// Deprecated: Use MarshalAuthMsg instead.
|
2024-09-08 12:06:14 +02:00
|
|
|
// MarshalHelloMsg initial hello message
|
|
|
|
// The Hello message is the first message sent by a client after establishing a connection with the Relay server. This
|
|
|
|
// message is used to authenticate the client with the server. The authentication is done using an HMAC method.
|
|
|
|
// The protocol does not limit to use HMAC, it can be any other method. If the authentication failed the server will
|
|
|
|
// close the network connection without any response.
|
|
|
|
func MarshalHelloMsg(peerID []byte, additions []byte) ([]byte, error) {
|
|
|
|
if len(peerID) != IDSize {
|
|
|
|
return nil, fmt.Errorf("invalid peerID length: %d", len(peerID))
|
|
|
|
}
|
|
|
|
|
|
|
|
msg := make([]byte, SizeOfProtoHeader+sizeOfMagicByte, SizeOfProtoHeader+headerSizeHello+len(additions))
|
|
|
|
|
|
|
|
msg[0] = byte(CurrentProtocolVersion)
|
|
|
|
msg[1] = byte(MsgTypeHello)
|
|
|
|
|
|
|
|
copy(msg[SizeOfProtoHeader:SizeOfProtoHeader+sizeOfMagicByte], magicHeader)
|
|
|
|
|
|
|
|
msg = append(msg, peerID...)
|
|
|
|
msg = append(msg, additions...)
|
|
|
|
|
|
|
|
return msg, nil
|
|
|
|
}
|
|
|
|
|
2024-09-11 16:20:30 +02:00
|
|
|
// Deprecated: Use UnmarshalAuthMsg instead.
|
2024-09-08 12:06:14 +02:00
|
|
|
// UnmarshalHelloMsg extracts peerID and the additional data from the hello message. The Additional data is used to
|
|
|
|
// authenticate the client with the server.
|
|
|
|
func UnmarshalHelloMsg(msg []byte) ([]byte, []byte, error) {
|
|
|
|
if len(msg) < headerSizeHello {
|
|
|
|
return nil, nil, ErrInvalidMessageLength
|
|
|
|
}
|
|
|
|
if !bytes.Equal(msg[:sizeOfMagicByte], magicHeader) {
|
|
|
|
return nil, nil, errors.New("invalid magic header")
|
|
|
|
}
|
|
|
|
|
|
|
|
return msg[sizeOfMagicByte:headerSizeHello], msg[headerSizeHello:], nil
|
|
|
|
}
|
|
|
|
|
2024-09-11 16:20:30 +02:00
|
|
|
// Deprecated: Use MarshalAuthResponse instead.
|
2024-09-08 12:06:14 +02:00
|
|
|
// MarshalHelloResponse creates a response message to the hello message.
|
|
|
|
// In case of success connection the server response with a Hello Response message. This message contains the server's
|
|
|
|
// instance URL. This URL will be used by choose the common Relay server in case if the peers are in different Relay
|
|
|
|
// servers.
|
|
|
|
func MarshalHelloResponse(additionalData []byte) ([]byte, error) {
|
|
|
|
msg := make([]byte, SizeOfProtoHeader, SizeOfProtoHeader+headerSizeHelloResp+len(additionalData))
|
|
|
|
|
|
|
|
msg[0] = byte(CurrentProtocolVersion)
|
|
|
|
msg[1] = byte(MsgTypeHelloResponse)
|
|
|
|
|
|
|
|
msg = append(msg, additionalData...)
|
|
|
|
|
|
|
|
return msg, nil
|
|
|
|
}
|
|
|
|
|
2024-09-11 16:20:30 +02:00
|
|
|
// Deprecated: Use UnmarshalAuthResponse instead.
|
2024-09-08 12:06:14 +02:00
|
|
|
// UnmarshalHelloResponse extracts the additional data from the hello response message.
|
|
|
|
func UnmarshalHelloResponse(msg []byte) ([]byte, error) {
|
|
|
|
if len(msg) < headerSizeHelloResp {
|
|
|
|
return nil, ErrInvalidMessageLength
|
|
|
|
}
|
|
|
|
return msg, nil
|
|
|
|
}
|
|
|
|
|
2024-09-11 16:20:30 +02:00
|
|
|
// MarshalAuthMsg initial authentication message
|
|
|
|
// The Auth message is the first message sent by a client after establishing a connection with the Relay server. This
|
|
|
|
// message is used to authenticate the client with the server. The authentication is done using an HMAC method.
|
|
|
|
// The protocol does not limit to use HMAC, it can be any other method. If the authentication failed the server will
|
|
|
|
// close the network connection without any response.
|
|
|
|
func MarshalAuthMsg(peerID []byte, authPayload []byte) ([]byte, error) {
|
|
|
|
if len(peerID) != IDSize {
|
|
|
|
return nil, fmt.Errorf("invalid peerID length: %d", len(peerID))
|
|
|
|
}
|
|
|
|
|
|
|
|
msg := make([]byte, SizeOfProtoHeader+sizeOfMagicByte, SizeOfProtoHeader+headerSizeAuth+len(authPayload))
|
|
|
|
|
|
|
|
msg[0] = byte(CurrentProtocolVersion)
|
|
|
|
msg[1] = byte(MsgTypeAuth)
|
|
|
|
|
|
|
|
copy(msg[SizeOfProtoHeader:SizeOfProtoHeader+sizeOfMagicByte], magicHeader)
|
|
|
|
|
|
|
|
msg = append(msg, peerID...)
|
|
|
|
msg = append(msg, authPayload...)
|
|
|
|
|
|
|
|
return msg, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// UnmarshalAuthMsg extracts peerID and the auth payload from the message
|
|
|
|
func UnmarshalAuthMsg(msg []byte) ([]byte, []byte, error) {
|
|
|
|
if len(msg) < headerSizeAuth {
|
|
|
|
return nil, nil, ErrInvalidMessageLength
|
|
|
|
}
|
|
|
|
if !bytes.Equal(msg[:sizeOfMagicByte], magicHeader) {
|
|
|
|
return nil, nil, errors.New("invalid magic header")
|
|
|
|
}
|
|
|
|
|
|
|
|
return msg[sizeOfMagicByte:headerSizeAuth], msg[headerSizeAuth:], nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// MarshalAuthResponse creates a response message to the auth.
|
|
|
|
// In case of success connection the server response with a AuthResponse message. This message contains the server's
|
|
|
|
// instance URL. This URL will be used by choose the common Relay server in case if the peers are in different Relay
|
|
|
|
// servers.
|
|
|
|
func MarshalAuthResponse(address string) ([]byte, error) {
|
|
|
|
ab := []byte(address)
|
|
|
|
msg := make([]byte, SizeOfProtoHeader, SizeOfProtoHeader+headerSizeAuthResp+len(ab))
|
|
|
|
|
|
|
|
msg[0] = byte(CurrentProtocolVersion)
|
|
|
|
msg[1] = byte(MsgTypeAuthResponse)
|
|
|
|
|
|
|
|
msg = append(msg, ab...)
|
|
|
|
|
|
|
|
if len(msg) > MaxHandshakeRespSize {
|
|
|
|
return nil, fmt.Errorf("invalid message length: %d", len(msg))
|
|
|
|
}
|
|
|
|
|
|
|
|
return msg, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// UnmarshalAuthResponse it is a confirmation message to auth success
|
|
|
|
func UnmarshalAuthResponse(msg []byte) (string, error) {
|
|
|
|
if len(msg) < headerSizeAuthResp+1 {
|
|
|
|
return "", ErrInvalidMessageLength
|
|
|
|
}
|
|
|
|
return string(msg), nil
|
|
|
|
}
|
|
|
|
|
2024-09-08 12:06:14 +02:00
|
|
|
// MarshalCloseMsg creates a close message.
|
|
|
|
// The close message is used to close the connection gracefully between the client and the server. The server and the
|
|
|
|
// client can send this message. After receiving this message, the server or client will close the connection.
|
|
|
|
func MarshalCloseMsg() []byte {
|
|
|
|
msg := make([]byte, SizeOfProtoHeader)
|
|
|
|
|
|
|
|
msg[0] = byte(CurrentProtocolVersion)
|
|
|
|
msg[1] = byte(MsgTypeClose)
|
|
|
|
|
|
|
|
return msg
|
|
|
|
}
|
|
|
|
|
|
|
|
// MarshalTransportMsg creates a transport message.
|
|
|
|
// The transport message is used to exchange data between peers. The message contains the data to be exchanged and the
|
|
|
|
// destination peer hashed ID.
|
|
|
|
func MarshalTransportMsg(peerID []byte, payload []byte) ([]byte, error) {
|
|
|
|
if len(peerID) != IDSize {
|
|
|
|
return nil, fmt.Errorf("invalid peerID length: %d", len(peerID))
|
|
|
|
}
|
|
|
|
|
|
|
|
msg := make([]byte, SizeOfProtoHeader+headerSizeTransport, SizeOfProtoHeader+headerSizeTransport+len(payload))
|
|
|
|
|
|
|
|
msg[0] = byte(CurrentProtocolVersion)
|
|
|
|
msg[1] = byte(MsgTypeTransport)
|
|
|
|
|
|
|
|
copy(msg[SizeOfProtoHeader:], peerID)
|
|
|
|
|
|
|
|
msg = append(msg, payload...)
|
|
|
|
|
|
|
|
return msg, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// UnmarshalTransportMsg extracts the peerID and the payload from the transport message.
|
|
|
|
func UnmarshalTransportMsg(buf []byte) ([]byte, []byte, error) {
|
|
|
|
if len(buf) < headerSizeTransport {
|
|
|
|
return nil, nil, ErrInvalidMessageLength
|
|
|
|
}
|
|
|
|
|
|
|
|
return buf[:headerSizeTransport], buf[headerSizeTransport:], nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// UnmarshalTransportID extracts the peerID from the transport message.
|
|
|
|
func UnmarshalTransportID(buf []byte) ([]byte, error) {
|
|
|
|
if len(buf) < headerSizeTransport {
|
|
|
|
return nil, ErrInvalidMessageLength
|
|
|
|
}
|
|
|
|
return buf[:headerSizeTransport], nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// UpdateTransportMsg updates the peerID in the transport message.
|
|
|
|
// With this function the server can reuse the given byte slice to update the peerID in the transport message. So do
|
|
|
|
// need to allocate a new byte slice.
|
|
|
|
func UpdateTransportMsg(msg []byte, peerID []byte) error {
|
|
|
|
if len(msg) < len(peerID) {
|
|
|
|
return ErrInvalidMessageLength
|
|
|
|
}
|
|
|
|
copy(msg, peerID)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// MarshalHealthcheck creates a health check message.
|
|
|
|
// Health check message is sent by the server periodically. The client will respond with a health check response
|
|
|
|
// message. If the client does not respond to the health check message, the server will close the connection.
|
|
|
|
func MarshalHealthcheck() []byte {
|
|
|
|
return healthCheckMsg
|
|
|
|
}
|