2024-06-05 20:10:03 +00:00
|
|
|
// SPDX-FileCopyrightText: 2023 The Pion community <https://pion.ly>
|
|
|
|
// SPDX-License-Identifier: MIT
|
|
|
|
|
2022-03-10 09:44:48 +00:00
|
|
|
package sctp
|
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
)
|
|
|
|
|
|
|
|
/*
|
|
|
|
chunkHeartbeat represents an SCTP Chunk of type HEARTBEAT
|
|
|
|
|
|
|
|
An endpoint should send this chunk to its peer endpoint to probe the
|
|
|
|
reachability of a particular destination transport address defined in
|
|
|
|
the present association.
|
|
|
|
|
|
|
|
The parameter field contains the Heartbeat Information, which is a
|
|
|
|
variable-length opaque data structure understood only by the sender.
|
|
|
|
|
2024-05-15 23:15:00 +00:00
|
|
|
0 1 2 3
|
|
|
|
0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
|
|
|
|
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
|
|
|
| Type = 4 | Chunk Flags | Heartbeat Length |
|
|
|
|
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
|
|
|
| |
|
|
|
|
| Heartbeat Information TLV (Variable-Length) |
|
|
|
|
| |
|
|
|
|
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
2022-03-10 09:44:48 +00:00
|
|
|
|
|
|
|
Defined as a variable-length parameter using the format described
|
|
|
|
in Section 3.2.1, i.e.:
|
|
|
|
|
|
|
|
Variable Parameters Status Type Value
|
|
|
|
-------------------------------------------------------------
|
|
|
|
heartbeat Info Mandatory 1
|
|
|
|
*/
|
|
|
|
type chunkHeartbeat struct {
|
|
|
|
chunkHeader
|
|
|
|
params []param
|
|
|
|
}
|
|
|
|
|
2024-05-15 23:15:00 +00:00
|
|
|
// Heartbeat chunk errors
|
2022-03-10 09:44:48 +00:00
|
|
|
var (
|
2024-05-15 23:15:00 +00:00
|
|
|
ErrChunkTypeNotHeartbeat = errors.New("ChunkType is not of type HEARTBEAT")
|
|
|
|
ErrHeartbeatNotLongEnoughInfo = errors.New("heartbeat is not long enough to contain Heartbeat Info")
|
|
|
|
ErrParseParamTypeFailed = errors.New("failed to parse param type")
|
|
|
|
ErrHeartbeatParam = errors.New("heartbeat should only have HEARTBEAT param")
|
|
|
|
ErrHeartbeatChunkUnmarshal = errors.New("failed unmarshalling param in Heartbeat Chunk")
|
2022-03-10 09:44:48 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
func (h *chunkHeartbeat) unmarshal(raw []byte) error {
|
|
|
|
if err := h.chunkHeader.unmarshal(raw); err != nil {
|
|
|
|
return err
|
|
|
|
} else if h.typ != ctHeartbeat {
|
2024-05-15 23:15:00 +00:00
|
|
|
return fmt.Errorf("%w: actually is %s", ErrChunkTypeNotHeartbeat, h.typ.String())
|
2022-03-10 09:44:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if len(raw) <= chunkHeaderSize {
|
2024-05-15 23:15:00 +00:00
|
|
|
return fmt.Errorf("%w: %d", ErrHeartbeatNotLongEnoughInfo, len(raw))
|
2022-03-10 09:44:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pType, err := parseParamType(raw[chunkHeaderSize:])
|
|
|
|
if err != nil {
|
2024-05-15 23:15:00 +00:00
|
|
|
return fmt.Errorf("%w: %v", ErrParseParamTypeFailed, err) //nolint:errorlint
|
2022-03-10 09:44:48 +00:00
|
|
|
}
|
|
|
|
if pType != heartbeatInfo {
|
2024-05-15 23:15:00 +00:00
|
|
|
return fmt.Errorf("%w: instead have %s", ErrHeartbeatParam, pType.String())
|
2022-03-10 09:44:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
p, err := buildParam(pType, raw[chunkHeaderSize:])
|
|
|
|
if err != nil {
|
2024-05-15 23:15:00 +00:00
|
|
|
return fmt.Errorf("%w: %v", ErrHeartbeatChunkUnmarshal, err) //nolint:errorlint
|
2022-03-10 09:44:48 +00:00
|
|
|
}
|
|
|
|
h.params = append(h.params, p)
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (h *chunkHeartbeat) Marshal() ([]byte, error) {
|
2024-05-15 23:15:00 +00:00
|
|
|
return nil, ErrUnimplemented
|
2022-03-10 09:44:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (h *chunkHeartbeat) check() (abort bool, err error) {
|
|
|
|
return false, nil
|
|
|
|
}
|