mirror of
https://github.com/aler9/gortsplib
synced 2025-10-05 07:06:58 +08:00
merge TrackPCMA and TrackPCMU into Track G711
This commit is contained in:
@@ -50,12 +50,11 @@ Features:
|
|||||||
* VP8
|
* VP8
|
||||||
* VP9
|
* VP9
|
||||||
* Audio
|
* Audio
|
||||||
|
* G711
|
||||||
* G722
|
* G722
|
||||||
* LPCM
|
* LPCM
|
||||||
* MPEG4-audio (AAC)
|
* MPEG4-audio (AAC)
|
||||||
* Opus
|
* Opus
|
||||||
* PCMA
|
|
||||||
* PCMU
|
|
||||||
* Parse RTSP elements: requests, responses, SDP
|
* Parse RTSP elements: requests, responses, SDP
|
||||||
* Parse H264 elements and formats: Annex-B, AVCC, anti-competition, DTS
|
* Parse H264 elements and formats: Annex-B, AVCC, anti-competition, DTS
|
||||||
* Parse MPEG4-audio (AAC) element and formats: ADTS, MPEG4-audio configurations
|
* Parse MPEG4-audio (AAC) element and formats: ADTS, MPEG4-audio configurations
|
||||||
@@ -79,7 +78,6 @@ Features:
|
|||||||
* [client-read-codec-mpeg4audio](examples/client-read-codec-mpeg4audio/main.go)
|
* [client-read-codec-mpeg4audio](examples/client-read-codec-mpeg4audio/main.go)
|
||||||
* [client-read-codec-opus](examples/client-read-codec-opus/main.go)
|
* [client-read-codec-opus](examples/client-read-codec-opus/main.go)
|
||||||
* [client-read-codec-pcma](examples/client-read-codec-pcma/main.go)
|
* [client-read-codec-pcma](examples/client-read-codec-pcma/main.go)
|
||||||
* [client-read-codec-pcmu](examples/client-read-codec-pcmu/main.go)
|
|
||||||
* [client-read-codec-vp8](examples/client-read-codec-vp8/main.go)
|
* [client-read-codec-vp8](examples/client-read-codec-vp8/main.go)
|
||||||
* [client-read-codec-vp9](examples/client-read-codec-vp9/main.go)
|
* [client-read-codec-vp9](examples/client-read-codec-vp9/main.go)
|
||||||
* [client-read-partial](examples/client-read-partial/main.go)
|
* [client-read-partial](examples/client-read-partial/main.go)
|
||||||
@@ -93,7 +91,6 @@ Features:
|
|||||||
* [client-publish-codec-mpeg4audio](examples/client-publish-codec-mpeg4audio/main.go)
|
* [client-publish-codec-mpeg4audio](examples/client-publish-codec-mpeg4audio/main.go)
|
||||||
* [client-publish-codec-opus](examples/client-publish-codec-opus/main.go)
|
* [client-publish-codec-opus](examples/client-publish-codec-opus/main.go)
|
||||||
* [client-publish-codec-pcma](examples/client-publish-codec-pcma/main.go)
|
* [client-publish-codec-pcma](examples/client-publish-codec-pcma/main.go)
|
||||||
* [client-publish-codec-pcmu](examples/client-publish-codec-pcmu/main.go)
|
|
||||||
* [client-publish-codec-vp8](examples/client-publish-codec-vp8/main.go)
|
* [client-publish-codec-vp8](examples/client-publish-codec-vp8/main.go)
|
||||||
* [client-publish-codec-vp9](examples/client-publish-codec-vp9/main.go)
|
* [client-publish-codec-vp9](examples/client-publish-codec-vp9/main.go)
|
||||||
* [client-publish-options](examples/client-publish-options/main.go)
|
* [client-publish-options](examples/client-publish-options/main.go)
|
||||||
|
@@ -34,7 +34,7 @@ func main() {
|
|||||||
log.Println("stream connected")
|
log.Println("stream connected")
|
||||||
|
|
||||||
// create a PCMA track
|
// create a PCMA track
|
||||||
track := &gortsplib.TrackPCMA{}
|
track := &gortsplib.TrackG711{}
|
||||||
|
|
||||||
c := gortsplib.Client{}
|
c := gortsplib.Client{}
|
||||||
|
|
@@ -1,69 +0,0 @@
|
|||||||
package main
|
|
||||||
|
|
||||||
import (
|
|
||||||
"log"
|
|
||||||
"net"
|
|
||||||
|
|
||||||
"github.com/aler9/gortsplib"
|
|
||||||
"github.com/pion/rtp"
|
|
||||||
)
|
|
||||||
|
|
||||||
// This example shows how to
|
|
||||||
// 1. generate RTP/PCMU packets with GStreamer
|
|
||||||
// 2. connect to a RTSP server, announce a PCMU track
|
|
||||||
// 3. route the packets from GStreamer to the server
|
|
||||||
|
|
||||||
func main() {
|
|
||||||
// open a listener to receive RTP/PCMU packets
|
|
||||||
pc, err := net.ListenPacket("udp", "localhost:9000")
|
|
||||||
if err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
defer pc.Close()
|
|
||||||
|
|
||||||
log.Println("Waiting for a RTP/PCMU stream on UDP port 9000 - you can send one with GStreamer:\n" +
|
|
||||||
"gst-launch-1.0 audiotestsrc freq=300 ! audioconvert ! audioresample ! audio/x-raw,rate=8000" +
|
|
||||||
" ! mulawenc ! rtppcmupay ! udpsink host=127.0.0.1 port=9000")
|
|
||||||
|
|
||||||
// wait for first packet
|
|
||||||
buf := make([]byte, 2048)
|
|
||||||
n, _, err := pc.ReadFrom(buf)
|
|
||||||
if err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
log.Println("stream connected")
|
|
||||||
|
|
||||||
// create a PCMU track
|
|
||||||
track := &gortsplib.TrackPCMU{}
|
|
||||||
|
|
||||||
c := gortsplib.Client{}
|
|
||||||
|
|
||||||
// connect to the server and start publishing the track
|
|
||||||
err = c.StartPublishing("rtsp://localhost:8554/mystream",
|
|
||||||
gortsplib.Tracks{track})
|
|
||||||
if err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
defer c.Close()
|
|
||||||
|
|
||||||
var pkt rtp.Packet
|
|
||||||
for {
|
|
||||||
// parse RTP packet
|
|
||||||
err = pkt.Unmarshal(buf[:n])
|
|
||||||
if err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// route RTP packet to the server
|
|
||||||
err = c.WritePacketRTP(0, &pkt)
|
|
||||||
if err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// read another RTP packet from source
|
|
||||||
n, _, err = pc.ReadFrom(buf)
|
|
||||||
if err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
@@ -35,9 +35,9 @@ func main() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// find the PCMA track
|
// find the PCMA track
|
||||||
track := func() *gortsplib.TrackPCMA {
|
track := func() *gortsplib.TrackG711 {
|
||||||
for _, track := range tracks {
|
for _, track := range tracks {
|
||||||
if tt, ok := track.(*gortsplib.TrackPCMA); ok {
|
if tt, ok := track.(*gortsplib.TrackG711); ok {
|
||||||
return tt
|
return tt
|
||||||
}
|
}
|
||||||
}
|
}
|
@@ -1,73 +0,0 @@
|
|||||||
package main
|
|
||||||
|
|
||||||
import (
|
|
||||||
"log"
|
|
||||||
|
|
||||||
"github.com/aler9/gortsplib"
|
|
||||||
"github.com/aler9/gortsplib/pkg/url"
|
|
||||||
)
|
|
||||||
|
|
||||||
// This example shows how to
|
|
||||||
// 1. connect to a RTSP server and read all tracks on a path
|
|
||||||
// 2. check if there's a PCMU track
|
|
||||||
// 3. get PCMU frames of that track
|
|
||||||
|
|
||||||
func main() {
|
|
||||||
c := gortsplib.Client{}
|
|
||||||
|
|
||||||
// parse URL
|
|
||||||
u, err := url.Parse("rtsp://localhost:8554/mystream")
|
|
||||||
if err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// connect to the server
|
|
||||||
err = c.Start(u.Scheme, u.Host)
|
|
||||||
if err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
defer c.Close()
|
|
||||||
|
|
||||||
// find published tracks
|
|
||||||
tracks, baseURL, _, err := c.Describe(u)
|
|
||||||
if err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// find the PCMU track
|
|
||||||
track := func() *gortsplib.TrackPCMU {
|
|
||||||
for _, track := range tracks {
|
|
||||||
if tt, ok := track.(*gortsplib.TrackPCMU); ok {
|
|
||||||
return tt
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}()
|
|
||||||
if track == nil {
|
|
||||||
panic("PCMU track not found")
|
|
||||||
}
|
|
||||||
|
|
||||||
// setup decoder
|
|
||||||
dec := track.CreateDecoder()
|
|
||||||
|
|
||||||
// called when a RTP packet arrives
|
|
||||||
c.OnPacketRTP = func(ctx *gortsplib.ClientOnPacketRTPCtx) {
|
|
||||||
// decode an PCMU packet from the RTP packet
|
|
||||||
op, _, err := dec.Decode(ctx.Packet)
|
|
||||||
if err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// print
|
|
||||||
log.Printf("received PCMU frame of size %d\n", len(op))
|
|
||||||
}
|
|
||||||
|
|
||||||
// setup and read the PCMU track only
|
|
||||||
err = c.SetupAndPlay(gortsplib.Tracks{track}, baseURL)
|
|
||||||
if err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// wait until a fatal error
|
|
||||||
panic(c.Wait())
|
|
||||||
}
|
|
7
track.go
7
track.go
@@ -122,11 +122,8 @@ func newTrackFromMediaDescription(md *psdp.MediaDescription) (Track, error) {
|
|||||||
|
|
||||||
case md.MediaName.Media == "audio":
|
case md.MediaName.Media == "audio":
|
||||||
switch {
|
switch {
|
||||||
case payloadType == 0:
|
case payloadType == 0, payloadType == 8:
|
||||||
return newTrackPCMUFromMediaDescription(control, clock)
|
return newTrackG711FromMediaDescription(control, payloadType, clock)
|
||||||
|
|
||||||
case payloadType == 8:
|
|
||||||
return newTrackPCMAFromMediaDescription(control, clock)
|
|
||||||
|
|
||||||
case payloadType == 9:
|
case payloadType == 9:
|
||||||
return newTrackG722FromMediaDescription(control, clock)
|
return newTrackG722FromMediaDescription(control, clock)
|
||||||
|
@@ -1,4 +1,4 @@
|
|||||||
package gortsplib //nolint:dupl
|
package gortsplib
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
@@ -9,22 +9,27 @@ import (
|
|||||||
"github.com/aler9/gortsplib/pkg/rtpcodecs/rtpsimpleaudio"
|
"github.com/aler9/gortsplib/pkg/rtpcodecs/rtpsimpleaudio"
|
||||||
)
|
)
|
||||||
|
|
||||||
// TrackPCMA is a PCMA track.
|
// TrackG711 is a PCMA track.
|
||||||
type TrackPCMA struct {
|
type TrackG711 struct {
|
||||||
|
// whether to use mu-law. Otherwise, A-law is used.
|
||||||
|
MULaw bool
|
||||||
|
|
||||||
trackBase
|
trackBase
|
||||||
}
|
}
|
||||||
|
|
||||||
func newTrackPCMAFromMediaDescription(
|
func newTrackG711FromMediaDescription(
|
||||||
control string,
|
control string,
|
||||||
|
payloadType uint8,
|
||||||
clock string,
|
clock string,
|
||||||
) (*TrackPCMA, error,
|
) (*TrackG711, error,
|
||||||
) {
|
) {
|
||||||
tmp := strings.Split(clock, "/")
|
tmp := strings.Split(clock, "/")
|
||||||
if len(tmp) == 2 && tmp[1] != "1" {
|
if len(tmp) == 2 && tmp[1] != "1" {
|
||||||
return nil, fmt.Errorf("PCMA tracks can have only one channel")
|
return nil, fmt.Errorf("G711 tracks can have only one channel")
|
||||||
}
|
}
|
||||||
|
|
||||||
return &TrackPCMA{
|
return &TrackG711{
|
||||||
|
MULaw: (payloadType == 0),
|
||||||
trackBase: trackBase{
|
trackBase: trackBase{
|
||||||
control: control,
|
control: control,
|
||||||
},
|
},
|
||||||
@@ -32,27 +37,37 @@ func newTrackPCMAFromMediaDescription(
|
|||||||
}
|
}
|
||||||
|
|
||||||
// String returns the track codec.
|
// String returns the track codec.
|
||||||
func (t *TrackPCMA) String() string {
|
func (t *TrackG711) String() string {
|
||||||
return "PCMA"
|
return "G711"
|
||||||
}
|
}
|
||||||
|
|
||||||
// ClockRate returns the track clock rate.
|
// ClockRate returns the track clock rate.
|
||||||
func (t *TrackPCMA) ClockRate() int {
|
func (t *TrackG711) ClockRate() int {
|
||||||
return 8000
|
return 8000
|
||||||
}
|
}
|
||||||
|
|
||||||
// MediaDescription returns the track media description in SDP format.
|
// MediaDescription returns the track media description in SDP format.
|
||||||
func (t *TrackPCMA) MediaDescription() *psdp.MediaDescription {
|
func (t *TrackG711) MediaDescription() *psdp.MediaDescription {
|
||||||
|
var formats []string
|
||||||
|
var rtpmap string
|
||||||
|
if t.MULaw {
|
||||||
|
formats = []string{"0"}
|
||||||
|
rtpmap = "0 PCMU/8000"
|
||||||
|
} else {
|
||||||
|
formats = []string{"8"}
|
||||||
|
rtpmap = "8 PCMA/8000"
|
||||||
|
}
|
||||||
|
|
||||||
return &psdp.MediaDescription{
|
return &psdp.MediaDescription{
|
||||||
MediaName: psdp.MediaName{
|
MediaName: psdp.MediaName{
|
||||||
Media: "audio",
|
Media: "audio",
|
||||||
Protos: []string{"RTP", "AVP"},
|
Protos: []string{"RTP", "AVP"},
|
||||||
Formats: []string{"8"},
|
Formats: formats,
|
||||||
},
|
},
|
||||||
Attributes: []psdp.Attribute{
|
Attributes: []psdp.Attribute{
|
||||||
{
|
{
|
||||||
Key: "rtpmap",
|
Key: "rtpmap",
|
||||||
Value: "8 PCMA/8000",
|
Value: rtpmap,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
Key: "control",
|
Key: "control",
|
||||||
@@ -62,14 +77,15 @@ func (t *TrackPCMA) MediaDescription() *psdp.MediaDescription {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (t *TrackPCMA) clone() Track {
|
func (t *TrackG711) clone() Track {
|
||||||
return &TrackPCMA{
|
return &TrackG711{
|
||||||
|
MULaw: t.MULaw,
|
||||||
trackBase: t.trackBase,
|
trackBase: t.trackBase,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// CreateDecoder creates a decoder able to decode the content of the track.
|
// CreateDecoder creates a decoder able to decode the content of the track.
|
||||||
func (t *TrackPCMA) CreateDecoder() *rtpsimpleaudio.Decoder {
|
func (t *TrackG711) CreateDecoder() *rtpsimpleaudio.Decoder {
|
||||||
d := &rtpsimpleaudio.Decoder{
|
d := &rtpsimpleaudio.Decoder{
|
||||||
SampleRate: 8000,
|
SampleRate: 8000,
|
||||||
}
|
}
|
||||||
@@ -78,9 +94,16 @@ func (t *TrackPCMA) CreateDecoder() *rtpsimpleaudio.Decoder {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// CreateEncoder creates an encoder able to encode the content of the track.
|
// CreateEncoder creates an encoder able to encode the content of the track.
|
||||||
func (t *TrackPCMA) CreateEncoder() *rtpsimpleaudio.Encoder {
|
func (t *TrackG711) CreateEncoder() *rtpsimpleaudio.Encoder {
|
||||||
|
var payloadType uint8
|
||||||
|
if t.MULaw {
|
||||||
|
payloadType = 0
|
||||||
|
} else {
|
||||||
|
payloadType = 8
|
||||||
|
}
|
||||||
|
|
||||||
e := &rtpsimpleaudio.Encoder{
|
e := &rtpsimpleaudio.Encoder{
|
||||||
PayloadType: 8,
|
PayloadType: payloadType,
|
||||||
SampleRate: 8000,
|
SampleRate: 8000,
|
||||||
}
|
}
|
||||||
e.Init()
|
e.Init()
|
71
track_g711_test.go
Normal file
71
track_g711_test.go
Normal file
@@ -0,0 +1,71 @@
|
|||||||
|
package gortsplib
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
psdp "github.com/pion/sdp/v3"
|
||||||
|
"github.com/stretchr/testify/require"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestTrackG711Attributes(t *testing.T) {
|
||||||
|
track := &TrackG711{}
|
||||||
|
require.Equal(t, "G711", track.String())
|
||||||
|
require.Equal(t, 8000, track.ClockRate())
|
||||||
|
require.Equal(t, "", track.GetControl())
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestTrackG711Clone(t *testing.T) {
|
||||||
|
track := &TrackG711{}
|
||||||
|
|
||||||
|
clone := track.clone()
|
||||||
|
require.NotSame(t, track, clone)
|
||||||
|
require.Equal(t, track, clone)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestTrackG711MediaDescription(t *testing.T) {
|
||||||
|
t.Run("pcma", func(t *testing.T) {
|
||||||
|
track := &TrackG711{}
|
||||||
|
|
||||||
|
require.Equal(t, &psdp.MediaDescription{
|
||||||
|
MediaName: psdp.MediaName{
|
||||||
|
Media: "audio",
|
||||||
|
Protos: []string{"RTP", "AVP"},
|
||||||
|
Formats: []string{"8"},
|
||||||
|
},
|
||||||
|
Attributes: []psdp.Attribute{
|
||||||
|
{
|
||||||
|
Key: "rtpmap",
|
||||||
|
Value: "8 PCMA/8000",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Key: "control",
|
||||||
|
Value: "",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}, track.MediaDescription())
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("pcmu", func(t *testing.T) {
|
||||||
|
track := &TrackG711{
|
||||||
|
MULaw: true,
|
||||||
|
}
|
||||||
|
|
||||||
|
require.Equal(t, &psdp.MediaDescription{
|
||||||
|
MediaName: psdp.MediaName{
|
||||||
|
Media: "audio",
|
||||||
|
Protos: []string{"RTP", "AVP"},
|
||||||
|
Formats: []string{"0"},
|
||||||
|
},
|
||||||
|
Attributes: []psdp.Attribute{
|
||||||
|
{
|
||||||
|
Key: "rtpmap",
|
||||||
|
Value: "0 PCMU/8000",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Key: "control",
|
||||||
|
Value: "",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}, track.MediaDescription())
|
||||||
|
})
|
||||||
|
}
|
@@ -1,45 +0,0 @@
|
|||||||
package gortsplib //nolint:dupl
|
|
||||||
|
|
||||||
import (
|
|
||||||
"testing"
|
|
||||||
|
|
||||||
psdp "github.com/pion/sdp/v3"
|
|
||||||
"github.com/stretchr/testify/require"
|
|
||||||
)
|
|
||||||
|
|
||||||
func TestTrackPCMAAttributes(t *testing.T) {
|
|
||||||
track := &TrackPCMA{}
|
|
||||||
require.Equal(t, "PCMA", track.String())
|
|
||||||
require.Equal(t, 8000, track.ClockRate())
|
|
||||||
require.Equal(t, "", track.GetControl())
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestTrackPCMAClone(t *testing.T) {
|
|
||||||
track := &TrackPCMA{}
|
|
||||||
|
|
||||||
clone := track.clone()
|
|
||||||
require.NotSame(t, track, clone)
|
|
||||||
require.Equal(t, track, clone)
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestTrackPCMAMediaDescription(t *testing.T) {
|
|
||||||
track := &TrackPCMA{}
|
|
||||||
|
|
||||||
require.Equal(t, &psdp.MediaDescription{
|
|
||||||
MediaName: psdp.MediaName{
|
|
||||||
Media: "audio",
|
|
||||||
Protos: []string{"RTP", "AVP"},
|
|
||||||
Formats: []string{"8"},
|
|
||||||
},
|
|
||||||
Attributes: []psdp.Attribute{
|
|
||||||
{
|
|
||||||
Key: "rtpmap",
|
|
||||||
Value: "8 PCMA/8000",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
Key: "control",
|
|
||||||
Value: "",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
}, track.MediaDescription())
|
|
||||||
}
|
|
@@ -1,88 +0,0 @@
|
|||||||
package gortsplib //nolint:dupl
|
|
||||||
|
|
||||||
import (
|
|
||||||
"fmt"
|
|
||||||
"strings"
|
|
||||||
|
|
||||||
psdp "github.com/pion/sdp/v3"
|
|
||||||
|
|
||||||
"github.com/aler9/gortsplib/pkg/rtpcodecs/rtpsimpleaudio"
|
|
||||||
)
|
|
||||||
|
|
||||||
// TrackPCMU is a PCMU track.
|
|
||||||
type TrackPCMU struct {
|
|
||||||
trackBase
|
|
||||||
}
|
|
||||||
|
|
||||||
func newTrackPCMUFromMediaDescription(
|
|
||||||
control string,
|
|
||||||
clock string,
|
|
||||||
) (*TrackPCMU, error,
|
|
||||||
) {
|
|
||||||
tmp := strings.SplitN(clock, "/", 2)
|
|
||||||
if len(tmp) == 2 && tmp[1] != "1" {
|
|
||||||
return nil, fmt.Errorf("PCMU tracks can have only one channel")
|
|
||||||
}
|
|
||||||
|
|
||||||
return &TrackPCMU{
|
|
||||||
trackBase: trackBase{
|
|
||||||
control: control,
|
|
||||||
},
|
|
||||||
}, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// String returns the track codec.
|
|
||||||
func (t *TrackPCMU) String() string {
|
|
||||||
return "PCMU"
|
|
||||||
}
|
|
||||||
|
|
||||||
// ClockRate returns the track clock rate.
|
|
||||||
func (t *TrackPCMU) ClockRate() int {
|
|
||||||
return 8000
|
|
||||||
}
|
|
||||||
|
|
||||||
// MediaDescription returns the track media description in SDP format.
|
|
||||||
func (t *TrackPCMU) MediaDescription() *psdp.MediaDescription {
|
|
||||||
return &psdp.MediaDescription{
|
|
||||||
MediaName: psdp.MediaName{
|
|
||||||
Media: "audio",
|
|
||||||
Protos: []string{"RTP", "AVP"},
|
|
||||||
Formats: []string{"0"},
|
|
||||||
},
|
|
||||||
Attributes: []psdp.Attribute{
|
|
||||||
{
|
|
||||||
Key: "rtpmap",
|
|
||||||
Value: "0 PCMU/8000",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
Key: "control",
|
|
||||||
Value: t.control,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (t *TrackPCMU) clone() Track {
|
|
||||||
return &TrackPCMU{
|
|
||||||
trackBase: t.trackBase,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// CreateDecoder creates a decoder able to decode the content of the track.
|
|
||||||
func (t *TrackPCMU) CreateDecoder() *rtpsimpleaudio.Decoder {
|
|
||||||
d := &rtpsimpleaudio.Decoder{
|
|
||||||
SampleRate: 8000,
|
|
||||||
}
|
|
||||||
d.Init()
|
|
||||||
return d
|
|
||||||
}
|
|
||||||
|
|
||||||
// CreateEncoder creates an encoder able to encode the content of the track.
|
|
||||||
func (t *TrackPCMU) CreateEncoder() *rtpsimpleaudio.Encoder {
|
|
||||||
e := &rtpsimpleaudio.Encoder{
|
|
||||||
PayloadType: 0,
|
|
||||||
SampleRate: 8000,
|
|
||||||
}
|
|
||||||
e.Init()
|
|
||||||
return e
|
|
||||||
}
|
|
@@ -1,45 +0,0 @@
|
|||||||
package gortsplib //nolint:dupl
|
|
||||||
|
|
||||||
import (
|
|
||||||
"testing"
|
|
||||||
|
|
||||||
psdp "github.com/pion/sdp/v3"
|
|
||||||
"github.com/stretchr/testify/require"
|
|
||||||
)
|
|
||||||
|
|
||||||
func TestTrackPCMUAttributes(t *testing.T) {
|
|
||||||
track := &TrackPCMU{}
|
|
||||||
require.Equal(t, "PCMU", track.String())
|
|
||||||
require.Equal(t, 8000, track.ClockRate())
|
|
||||||
require.Equal(t, "", track.GetControl())
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestTrackPCMUClone(t *testing.T) {
|
|
||||||
track := &TrackPCMU{}
|
|
||||||
|
|
||||||
clone := track.clone()
|
|
||||||
require.NotSame(t, track, clone)
|
|
||||||
require.Equal(t, track, clone)
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestTrackPCMUMediaDescription(t *testing.T) {
|
|
||||||
track := &TrackPCMU{}
|
|
||||||
|
|
||||||
require.Equal(t, &psdp.MediaDescription{
|
|
||||||
MediaName: psdp.MediaName{
|
|
||||||
Media: "audio",
|
|
||||||
Protos: []string{"RTP", "AVP"},
|
|
||||||
Formats: []string{"0"},
|
|
||||||
},
|
|
||||||
Attributes: []psdp.Attribute{
|
|
||||||
{
|
|
||||||
Key: "rtpmap",
|
|
||||||
Value: "0 PCMU/8000",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
Key: "control",
|
|
||||||
Value: "",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
}, track.MediaDescription())
|
|
||||||
}
|
|
@@ -25,7 +25,7 @@ func TestTrackNewFromMediaDescription(t *testing.T) {
|
|||||||
Formats: []string{"8"},
|
Formats: []string{"8"},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
&TrackPCMA{},
|
&TrackG711{},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"pcmu",
|
"pcmu",
|
||||||
@@ -36,7 +36,9 @@ func TestTrackNewFromMediaDescription(t *testing.T) {
|
|||||||
Formats: []string{"0"},
|
Formats: []string{"0"},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
&TrackPCMU{},
|
&TrackG711{
|
||||||
|
MULaw: true,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"g722",
|
"g722",
|
||||||
|
@@ -81,7 +81,8 @@ func TestTracksReadSkipGenericTracksWithoutClockRate(t *testing.T) {
|
|||||||
PPS: []byte{0x68, 0xee, 0x01, 0x9e, 0x2c},
|
PPS: []byte{0x68, 0xee, 0x01, 0x9e, 0x2c},
|
||||||
PacketizationMode: 1,
|
PacketizationMode: 1,
|
||||||
},
|
},
|
||||||
&TrackPCMU{
|
&TrackG711{
|
||||||
|
MULaw: true,
|
||||||
trackBase: trackBase{
|
trackBase: trackBase{
|
||||||
control: "rtsp://10.0.100.50/profile5/media.smp/trackID=a",
|
control: "rtsp://10.0.100.50/profile5/media.smp/trackID=a",
|
||||||
},
|
},
|
||||||
|
Reference in New Issue
Block a user