mirror of
https://github.com/aler9/gortsplib
synced 2025-09-27 03:25:52 +08:00
162 lines
2.8 KiB
Go
162 lines
2.8 KiB
Go
package gortsplib
|
|
|
|
import (
|
|
"bufio"
|
|
"fmt"
|
|
"io"
|
|
"sort"
|
|
"strconv"
|
|
)
|
|
|
|
const (
|
|
_RTSP_PROTO = "RTSP/1.0"
|
|
_MAX_HEADER_COUNT = 255
|
|
_MAX_HEADER_KEY_LENGTH = 255
|
|
_MAX_HEADER_VALUE_LENGTH = 255
|
|
_MAX_CONTENT_LENGTH = 4096
|
|
)
|
|
|
|
func readBytesLimited(rb *bufio.Reader, delim byte, n int) ([]byte, error) {
|
|
for i := 1; i <= n; i++ {
|
|
byts, err := rb.Peek(i)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if byts[len(byts)-1] == delim {
|
|
rb.Discard(len(byts))
|
|
return byts, nil
|
|
}
|
|
}
|
|
return nil, fmt.Errorf("buffer length exceeds %d", n)
|
|
}
|
|
|
|
func readByteEqual(rb *bufio.Reader, cmp byte) error {
|
|
byt, err := rb.ReadByte()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if byt != cmp {
|
|
return fmt.Errorf("expected '%c', got '%c'", cmp, byt)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func readHeaders(rb *bufio.Reader) (map[string]string, error) {
|
|
ret := make(map[string]string)
|
|
|
|
for {
|
|
byt, err := rb.ReadByte()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if byt == '\r' {
|
|
err := readByteEqual(rb, '\n')
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
break
|
|
}
|
|
|
|
if len(ret) >= _MAX_HEADER_COUNT {
|
|
return nil, fmt.Errorf("headers count exceeds %d", _MAX_HEADER_COUNT)
|
|
}
|
|
|
|
key := string([]byte{byt})
|
|
byts, err := readBytesLimited(rb, ':', _MAX_HEADER_KEY_LENGTH-1)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
key += string(byts[:len(byts)-1])
|
|
|
|
err = readByteEqual(rb, ' ')
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
byts, err = readBytesLimited(rb, '\r', _MAX_HEADER_VALUE_LENGTH)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
val := string(byts[:len(byts)-1])
|
|
|
|
if len(val) == 0 {
|
|
return nil, fmt.Errorf("empty header value")
|
|
}
|
|
|
|
err = readByteEqual(rb, '\n')
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
ret[key] = val
|
|
}
|
|
|
|
return ret, nil
|
|
}
|
|
|
|
func writeHeaders(wb *bufio.Writer, headers map[string]string) error {
|
|
// sort headers by key
|
|
// in order to obtain deterministic results
|
|
var keys []string
|
|
for key := range headers {
|
|
keys = append(keys, key)
|
|
}
|
|
sort.Strings(keys)
|
|
|
|
for _, key := range keys {
|
|
_, err := wb.Write([]byte(key + ": " + headers[key] + "\r\n"))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
_, err := wb.Write([]byte("\r\n"))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func readContent(rb *bufio.Reader, headers map[string]string) ([]byte, error) {
|
|
cls, ok := headers["Content-Length"]
|
|
if !ok {
|
|
return nil, nil
|
|
}
|
|
|
|
cl, err := strconv.ParseInt(cls, 10, 64)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("invalid Content-Length")
|
|
}
|
|
|
|
if cl > _MAX_CONTENT_LENGTH {
|
|
return nil, fmt.Errorf("Content-Length exceeds %d", _MAX_CONTENT_LENGTH)
|
|
}
|
|
|
|
ret := make([]byte, cl)
|
|
n, err := io.ReadFull(rb, ret)
|
|
if err != nil && n != len(ret) {
|
|
return nil, err
|
|
}
|
|
|
|
return ret, nil
|
|
}
|
|
|
|
func writeContent(wb *bufio.Writer, content []byte) error {
|
|
if len(content) == 0 {
|
|
return nil
|
|
}
|
|
|
|
_, err := wb.Write(content)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return nil
|
|
}
|