2024-01-03 21:57:33 +00:00
|
|
|
// SPDX-FileCopyrightText: 2023 The Pion community <https://pion.ly>
|
|
|
|
// SPDX-License-Identifier: MIT
|
|
|
|
|
2022-03-10 09:44:48 +00:00
|
|
|
package stun
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"net"
|
|
|
|
"strconv"
|
|
|
|
)
|
|
|
|
|
|
|
|
// MappedAddress represents MAPPED-ADDRESS attribute.
|
|
|
|
//
|
|
|
|
// This attribute is used only by servers for achieving backwards
|
|
|
|
// compatibility with RFC 3489 clients.
|
|
|
|
//
|
|
|
|
// RFC 5389 Section 15.1
|
|
|
|
type MappedAddress struct {
|
|
|
|
IP net.IP
|
|
|
|
Port int
|
|
|
|
}
|
|
|
|
|
|
|
|
// AlternateServer represents ALTERNATE-SERVER attribute.
|
|
|
|
//
|
|
|
|
// RFC 5389 Section 15.11
|
|
|
|
type AlternateServer struct {
|
|
|
|
IP net.IP
|
|
|
|
Port int
|
|
|
|
}
|
|
|
|
|
2024-01-03 21:57:33 +00:00
|
|
|
// ResponseOrigin represents RESPONSE-ORIGIN attribute.
|
|
|
|
//
|
|
|
|
// RFC 5780 Section 7.3
|
|
|
|
type ResponseOrigin struct {
|
|
|
|
IP net.IP
|
|
|
|
Port int
|
|
|
|
}
|
|
|
|
|
2022-03-10 09:44:48 +00:00
|
|
|
// OtherAddress represents OTHER-ADDRESS attribute.
|
|
|
|
//
|
|
|
|
// RFC 5780 Section 7.4
|
|
|
|
type OtherAddress struct {
|
|
|
|
IP net.IP
|
|
|
|
Port int
|
|
|
|
}
|
|
|
|
|
|
|
|
// AddTo adds ALTERNATE-SERVER attribute to message.
|
|
|
|
func (s *AlternateServer) AddTo(m *Message) error {
|
|
|
|
a := (*MappedAddress)(s)
|
2024-01-03 21:57:33 +00:00
|
|
|
return a.AddToAs(m, AttrAlternateServer)
|
2022-03-10 09:44:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// GetFrom decodes ALTERNATE-SERVER from message.
|
|
|
|
func (s *AlternateServer) GetFrom(m *Message) error {
|
|
|
|
a := (*MappedAddress)(s)
|
2024-01-03 21:57:33 +00:00
|
|
|
return a.GetFromAs(m, AttrAlternateServer)
|
2022-03-10 09:44:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (a MappedAddress) String() string {
|
|
|
|
return net.JoinHostPort(a.IP.String(), strconv.Itoa(a.Port))
|
|
|
|
}
|
|
|
|
|
2024-01-03 21:57:33 +00:00
|
|
|
// GetFromAs decodes MAPPED-ADDRESS value in message m as an attribute of type t.
|
|
|
|
func (a *MappedAddress) GetFromAs(m *Message, t AttrType) error {
|
2022-03-10 09:44:48 +00:00
|
|
|
v, err := m.Get(t)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if len(v) <= 4 {
|
|
|
|
return io.ErrUnexpectedEOF
|
|
|
|
}
|
|
|
|
family := bin.Uint16(v[0:2])
|
|
|
|
if family != familyIPv6 && family != familyIPv4 {
|
|
|
|
return newDecodeErr("xor-mapped address", "family",
|
|
|
|
fmt.Sprintf("bad value %d", family),
|
|
|
|
)
|
|
|
|
}
|
|
|
|
ipLen := net.IPv4len
|
|
|
|
if family == familyIPv6 {
|
|
|
|
ipLen = net.IPv6len
|
|
|
|
}
|
|
|
|
// Ensuring len(a.IP) == ipLen and reusing a.IP.
|
|
|
|
if len(a.IP) < ipLen {
|
|
|
|
a.IP = a.IP[:cap(a.IP)]
|
|
|
|
for len(a.IP) < ipLen {
|
|
|
|
a.IP = append(a.IP, 0)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
a.IP = a.IP[:ipLen]
|
|
|
|
for i := range a.IP {
|
|
|
|
a.IP[i] = 0
|
|
|
|
}
|
|
|
|
a.Port = int(bin.Uint16(v[2:4]))
|
|
|
|
copy(a.IP, v[4:])
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2024-01-03 21:57:33 +00:00
|
|
|
// AddToAs adds MAPPED-ADDRESS value to m as t attribute.
|
|
|
|
func (a *MappedAddress) AddToAs(m *Message, t AttrType) error {
|
2022-03-10 09:44:48 +00:00
|
|
|
var (
|
|
|
|
family = familyIPv4
|
|
|
|
ip = a.IP
|
|
|
|
)
|
|
|
|
if len(a.IP) == net.IPv6len {
|
|
|
|
if isIPv4(ip) {
|
|
|
|
ip = ip[12:16] // like in ip.To4()
|
|
|
|
} else {
|
|
|
|
family = familyIPv6
|
|
|
|
}
|
|
|
|
} else if len(ip) != net.IPv4len {
|
|
|
|
return ErrBadIPLength
|
|
|
|
}
|
|
|
|
value := make([]byte, 128)
|
|
|
|
value[0] = 0 // first 8 bits are zeroes
|
|
|
|
bin.PutUint16(value[0:2], family)
|
|
|
|
bin.PutUint16(value[2:4], uint16(a.Port))
|
|
|
|
copy(value[4:], ip)
|
|
|
|
m.Add(t, value[:4+len(ip)])
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// AddTo adds MAPPED-ADDRESS to message.
|
|
|
|
func (a *MappedAddress) AddTo(m *Message) error {
|
2024-01-03 21:57:33 +00:00
|
|
|
return a.AddToAs(m, AttrMappedAddress)
|
2022-03-10 09:44:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// GetFrom decodes MAPPED-ADDRESS from message.
|
|
|
|
func (a *MappedAddress) GetFrom(m *Message) error {
|
2024-01-03 21:57:33 +00:00
|
|
|
return a.GetFromAs(m, AttrMappedAddress)
|
2022-03-10 09:44:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// AddTo adds OTHER-ADDRESS attribute to message.
|
|
|
|
func (o *OtherAddress) AddTo(m *Message) error {
|
|
|
|
a := (*MappedAddress)(o)
|
2024-01-03 21:57:33 +00:00
|
|
|
return a.AddToAs(m, AttrOtherAddress)
|
2022-03-10 09:44:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// GetFrom decodes OTHER-ADDRESS from message.
|
|
|
|
func (o *OtherAddress) GetFrom(m *Message) error {
|
|
|
|
a := (*MappedAddress)(o)
|
2024-01-03 21:57:33 +00:00
|
|
|
return a.GetFromAs(m, AttrOtherAddress)
|
2022-03-10 09:44:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (o OtherAddress) String() string {
|
|
|
|
return net.JoinHostPort(o.IP.String(), strconv.Itoa(o.Port))
|
|
|
|
}
|
2024-01-03 21:57:33 +00:00
|
|
|
|
|
|
|
// AddTo adds RESPONSE-ORIGIN attribute to message.
|
|
|
|
func (o *ResponseOrigin) AddTo(m *Message) error {
|
|
|
|
a := (*MappedAddress)(o)
|
|
|
|
return a.AddToAs(m, AttrResponseOrigin)
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetFrom decodes RESPONSE-ORIGIN from message.
|
|
|
|
func (o *ResponseOrigin) GetFrom(m *Message) error {
|
|
|
|
a := (*MappedAddress)(o)
|
|
|
|
return a.GetFromAs(m, AttrResponseOrigin)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (o ResponseOrigin) String() string {
|
|
|
|
return net.JoinHostPort(o.IP.String(), strconv.Itoa(o.Port))
|
|
|
|
}
|