2023-10-25 09:48:57 +00:00
|
|
|
package webrtc
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"fmt"
|
|
|
|
"net/http"
|
|
|
|
"net/url"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/bluenviron/gortsplib/v4/pkg/format"
|
|
|
|
"github.com/pion/sdp/v3"
|
|
|
|
|
|
|
|
"github.com/bluenviron/mediamtx/internal/logger"
|
|
|
|
)
|
|
|
|
|
|
|
|
// WHIPClient is a WHIP client.
|
|
|
|
type WHIPClient struct {
|
|
|
|
HTTPClient *http.Client
|
|
|
|
URL *url.URL
|
|
|
|
Log logger.Writer
|
|
|
|
|
|
|
|
pc *PeerConnection
|
|
|
|
}
|
|
|
|
|
|
|
|
// Publish publishes tracks.
|
|
|
|
func (c *WHIPClient) Publish(
|
|
|
|
ctx context.Context,
|
|
|
|
videoTrack format.Format,
|
|
|
|
audioTrack format.Format,
|
|
|
|
) ([]*OutgoingTrack, error) {
|
|
|
|
iceServers, err := WHIPOptionsICEServers(ctx, c.HTTPClient, c.URL.String())
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2023-11-12 22:55:28 +00:00
|
|
|
api, err := NewAPI(APIConf{
|
|
|
|
LocalRandomUDP: true,
|
|
|
|
IPsFromInterfaces: true,
|
|
|
|
})
|
2023-10-25 09:48:57 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
c.pc = &PeerConnection{
|
|
|
|
ICEServers: iceServers,
|
|
|
|
API: api,
|
|
|
|
Publish: true,
|
|
|
|
Log: c.Log,
|
|
|
|
}
|
|
|
|
err = c.pc.Start()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
tracks, err := c.pc.SetupOutgoingTracks(videoTrack, audioTrack)
|
|
|
|
if err != nil {
|
|
|
|
c.pc.Close()
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
offer, err := c.pc.CreatePartialOffer()
|
|
|
|
if err != nil {
|
|
|
|
c.pc.Close()
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
res, err := PostOffer(ctx, c.HTTPClient, c.URL.String(), offer)
|
|
|
|
if err != nil {
|
|
|
|
c.pc.Close()
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
c.URL, err = c.URL.Parse(res.Location)
|
|
|
|
if err != nil {
|
|
|
|
c.pc.Close()
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
err = c.pc.SetAnswer(res.Answer)
|
|
|
|
if err != nil {
|
2024-03-10 10:33:00 +00:00
|
|
|
WHIPDeleteSession(context.Background(), c.HTTPClient, c.URL.String()) //nolint:errcheck
|
2023-10-25 09:48:57 +00:00
|
|
|
c.pc.Close()
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
t := time.NewTimer(webrtcHandshakeTimeout)
|
|
|
|
defer t.Stop()
|
|
|
|
|
|
|
|
outer:
|
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case ca := <-c.pc.NewLocalCandidate():
|
2024-03-06 21:38:36 +00:00
|
|
|
err := WHIPPatchCandidate(ctx, c.HTTPClient, c.URL.String(), offer, res.ETag, ca)
|
2023-10-25 09:48:57 +00:00
|
|
|
if err != nil {
|
2024-03-10 10:33:00 +00:00
|
|
|
WHIPDeleteSession(context.Background(), c.HTTPClient, c.URL.String()) //nolint:errcheck
|
2023-10-25 09:48:57 +00:00
|
|
|
c.pc.Close()
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
case <-c.pc.GatheringDone():
|
|
|
|
|
|
|
|
case <-c.pc.Connected():
|
|
|
|
break outer
|
|
|
|
|
|
|
|
case <-t.C:
|
2024-03-10 10:33:00 +00:00
|
|
|
WHIPDeleteSession(context.Background(), c.HTTPClient, c.URL.String()) //nolint:errcheck
|
2023-10-25 09:48:57 +00:00
|
|
|
c.pc.Close()
|
|
|
|
return nil, fmt.Errorf("deadline exceeded while waiting connection")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return tracks, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Read reads tracks.
|
|
|
|
func (c *WHIPClient) Read(ctx context.Context) ([]*IncomingTrack, error) {
|
|
|
|
iceServers, err := WHIPOptionsICEServers(ctx, c.HTTPClient, c.URL.String())
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2023-11-12 22:55:28 +00:00
|
|
|
api, err := NewAPI(APIConf{
|
|
|
|
LocalRandomUDP: true,
|
|
|
|
IPsFromInterfaces: true,
|
|
|
|
})
|
2023-10-25 09:48:57 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
c.pc = &PeerConnection{
|
|
|
|
ICEServers: iceServers,
|
|
|
|
API: api,
|
|
|
|
Publish: false,
|
|
|
|
Log: c.Log,
|
|
|
|
}
|
|
|
|
err = c.pc.Start()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
offer, err := c.pc.CreatePartialOffer()
|
|
|
|
if err != nil {
|
|
|
|
c.pc.Close()
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
res, err := PostOffer(ctx, c.HTTPClient, c.URL.String(), offer)
|
|
|
|
if err != nil {
|
|
|
|
c.pc.Close()
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
c.URL, err = c.URL.Parse(res.Location)
|
|
|
|
if err != nil {
|
|
|
|
c.pc.Close()
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
var sdp sdp.SessionDescription
|
|
|
|
err = sdp.Unmarshal([]byte(res.Answer.SDP))
|
|
|
|
if err != nil {
|
2024-03-10 10:33:00 +00:00
|
|
|
WHIPDeleteSession(context.Background(), c.HTTPClient, c.URL.String()) //nolint:errcheck
|
2023-10-25 09:48:57 +00:00
|
|
|
c.pc.Close()
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// check that there are at most two tracks
|
|
|
|
_, err = TrackCount(sdp.MediaDescriptions)
|
|
|
|
if err != nil {
|
2024-03-10 10:33:00 +00:00
|
|
|
WHIPDeleteSession(context.Background(), c.HTTPClient, c.URL.String()) //nolint:errcheck
|
2023-10-25 09:48:57 +00:00
|
|
|
c.pc.Close()
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
err = c.pc.SetAnswer(res.Answer)
|
|
|
|
if err != nil {
|
2024-03-10 10:33:00 +00:00
|
|
|
WHIPDeleteSession(context.Background(), c.HTTPClient, c.URL.String()) //nolint:errcheck
|
2023-10-25 09:48:57 +00:00
|
|
|
c.pc.Close()
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
t := time.NewTimer(webrtcHandshakeTimeout)
|
|
|
|
defer t.Stop()
|
|
|
|
|
|
|
|
outer:
|
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case ca := <-c.pc.NewLocalCandidate():
|
2024-03-06 21:38:36 +00:00
|
|
|
err := WHIPPatchCandidate(ctx, c.HTTPClient, c.URL.String(), offer, res.ETag, ca)
|
2023-10-25 09:48:57 +00:00
|
|
|
if err != nil {
|
2024-03-10 10:33:00 +00:00
|
|
|
WHIPDeleteSession(context.Background(), c.HTTPClient, c.URL.String()) //nolint:errcheck
|
2023-10-25 09:48:57 +00:00
|
|
|
c.pc.Close()
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
case <-c.pc.GatheringDone():
|
|
|
|
|
|
|
|
case <-c.pc.Connected():
|
|
|
|
break outer
|
|
|
|
|
|
|
|
case <-t.C:
|
2024-03-10 10:33:00 +00:00
|
|
|
WHIPDeleteSession(context.Background(), c.HTTPClient, c.URL.String()) //nolint:errcheck
|
2023-10-25 09:48:57 +00:00
|
|
|
c.pc.Close()
|
|
|
|
return nil, fmt.Errorf("deadline exceeded while waiting connection")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-03-10 10:33:00 +00:00
|
|
|
tracks, err := c.pc.GatherIncomingTracks(ctx, 0)
|
|
|
|
if err != nil {
|
|
|
|
WHIPDeleteSession(context.Background(), c.HTTPClient, c.URL.String()) //nolint:errcheck
|
|
|
|
c.pc.Close()
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return tracks, nil
|
2023-10-25 09:48:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Close closes the client.
|
|
|
|
func (c *WHIPClient) Close() error {
|
|
|
|
err := WHIPDeleteSession(context.Background(), c.HTTPClient, c.URL.String())
|
|
|
|
c.pc.Close()
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Wait waits for client errors.
|
|
|
|
func (c *WHIPClient) Wait(ctx context.Context) error {
|
|
|
|
select {
|
|
|
|
case <-c.pc.Disconnected():
|
|
|
|
return fmt.Errorf("peer connection closed")
|
|
|
|
|
|
|
|
case <-ctx.Done():
|
|
|
|
return fmt.Errorf("terminated")
|
|
|
|
}
|
|
|
|
}
|