Compare commits
6 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| d96af31f86 | |||
| cc55281f12 | |||
| c10d619df8 | |||
| 65f451e0c5 | |||
| ecd46700db | |||
| 2f588c77c4 |
@@ -0,0 +1,59 @@
|
|||||||
|
# https://github.com/home-assistant/builder
|
||||||
|
name: 'Builder'
|
||||||
|
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
tags: [ 'v*' ]
|
||||||
|
workflow_dispatch:
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
hassio:
|
||||||
|
name: Hassio Addon
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- name: Checkout the repository
|
||||||
|
uses: actions/checkout@v3
|
||||||
|
|
||||||
|
- name: Login to DockerHub
|
||||||
|
uses: docker/login-action@v2
|
||||||
|
with:
|
||||||
|
username: ${{ secrets.DOCKERHUB_USERNAME }}
|
||||||
|
password: ${{ secrets.DOCKERHUB_TOKEN }}
|
||||||
|
|
||||||
|
- name: Branch name
|
||||||
|
run: |
|
||||||
|
VERSION="${GITHUB_REF#refs/tags/v}"
|
||||||
|
echo "REPO=alexxit/go2rtc" >> $GITHUB_ENV
|
||||||
|
echo "TAG=${VERSION}" >> $GITHUB_ENV
|
||||||
|
echo "IMAGE=alexxit/go2rtc:${VERSION}" >> $GITHUB_ENV
|
||||||
|
|
||||||
|
- name: Build amd64
|
||||||
|
uses: home-assistant/builder@master
|
||||||
|
with:
|
||||||
|
args: --amd64 --target build/hassio --version $TAG-amd64 --no-latest --docker-hub-check
|
||||||
|
|
||||||
|
- name: Build i386
|
||||||
|
uses: home-assistant/builder@master
|
||||||
|
with:
|
||||||
|
args: --i386 --target build/hassio --version $TAG-i386 --no-latest --docker-hub-check
|
||||||
|
|
||||||
|
- name: Build aarch64
|
||||||
|
uses: home-assistant/builder@master
|
||||||
|
with:
|
||||||
|
args: --aarch64 --target build/hassio --version $TAG-aarch64 --no-latest --docker-hub-check
|
||||||
|
|
||||||
|
- name: Build armv7
|
||||||
|
uses: home-assistant/builder@master
|
||||||
|
with:
|
||||||
|
args: --armv7 --target build/hassio --version $TAG-armv7 --no-latest --docker-hub-check
|
||||||
|
|
||||||
|
- name: Docker manifest
|
||||||
|
run: |
|
||||||
|
# thanks to https://github.com/aler9/rtsp-simple-server/blob/main/Makefile
|
||||||
|
docker manifest create "${IMAGE}" \
|
||||||
|
"${IMAGE}-amd64" "${IMAGE}-i386" "${IMAGE}-aarch64" "${IMAGE}-armv7"
|
||||||
|
docker manifest push "${IMAGE}"
|
||||||
|
|
||||||
|
docker manifest create "${REPO}:latest" \
|
||||||
|
"${IMAGE}-amd64" "${IMAGE}-i386" "${IMAGE}-aarch64" "${IMAGE}-armv7"
|
||||||
|
docker manifest push "${REPO}:latest"
|
||||||
@@ -0,0 +1,11 @@
|
|||||||
|
ARG BUILD_FROM
|
||||||
|
FROM $BUILD_FROM
|
||||||
|
|
||||||
|
RUN apk add --no-cache git go
|
||||||
|
|
||||||
|
WORKDIR app
|
||||||
|
|
||||||
|
RUN git clone https://github.com/AlexxIT/go2rtc .
|
||||||
|
RUN CGO_ENABLED=0 go build -ldflags "-s -w" -trimpath
|
||||||
|
|
||||||
|
CMD [ "/app/go2rtc", "-config", "/config/go2rtc.yaml" ]
|
||||||
@@ -0,0 +1,6 @@
|
|||||||
|
# https://github.com/home-assistant/builder/blob/master/builder.sh
|
||||||
|
name: go2rtc
|
||||||
|
description: Ultimate camera streaming application
|
||||||
|
url: https://github.com/AlexxIT/go2rtc
|
||||||
|
image: alexxit/go2rtc
|
||||||
|
arch: [ amd64, aarch64, i386, armv7 ]
|
||||||
+10
-7
@@ -8,19 +8,22 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
var stackSkip = [][]byte{
|
var stackSkip = [][]byte{
|
||||||
// debug.go
|
// main.go
|
||||||
[]byte("github.com/AlexxIT/go2rtc/cmd/debug.handler"),
|
[]byte("main.main()"),
|
||||||
|
|
||||||
// cmd.go
|
|
||||||
[]byte("github.com/AlexxIT/go2rtc/cmd.Run"),
|
|
||||||
[]byte("created by os/signal.Notify"),
|
[]byte("created by os/signal.Notify"),
|
||||||
|
|
||||||
// api.go
|
// api/stack.go
|
||||||
|
[]byte("github.com/AlexxIT/go2rtc/cmd/api.stackHandler"),
|
||||||
|
|
||||||
|
// api/api.go
|
||||||
[]byte("created by github.com/AlexxIT/go2rtc/cmd/api.Init"),
|
[]byte("created by github.com/AlexxIT/go2rtc/cmd/api.Init"),
|
||||||
[]byte("created by net/http.(*connReader).startBackgroundRead"),
|
[]byte("created by net/http.(*connReader).startBackgroundRead"),
|
||||||
[]byte("created by net/http.(*Server).Serve"),
|
[]byte("created by net/http.(*Server).Serve"), // TODO: why two?
|
||||||
|
|
||||||
[]byte("created by github.com/AlexxIT/go2rtc/cmd/rtsp.Init"),
|
[]byte("created by github.com/AlexxIT/go2rtc/cmd/rtsp.Init"),
|
||||||
|
|
||||||
|
// webrtc/api.go
|
||||||
|
[]byte("created by github.com/pion/ice/v2.NewTCPMuxDefault"),
|
||||||
}
|
}
|
||||||
|
|
||||||
func stackHandler(w http.ResponseWriter, r *http.Request) {
|
func stackHandler(w http.ResponseWriter, r *http.Request) {
|
||||||
|
|||||||
+10
-1
@@ -1,6 +1,7 @@
|
|||||||
package app
|
package app
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"flag"
|
||||||
"github.com/rs/zerolog"
|
"github.com/rs/zerolog"
|
||||||
"gopkg.in/yaml.v3"
|
"gopkg.in/yaml.v3"
|
||||||
"io"
|
"io"
|
||||||
@@ -9,7 +10,15 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
func Init() {
|
func Init() {
|
||||||
data, _ = os.ReadFile("go2rtc.yaml")
|
config := flag.String(
|
||||||
|
"config",
|
||||||
|
"go2rtc.yaml",
|
||||||
|
"Path to go2rtc configuration file",
|
||||||
|
)
|
||||||
|
|
||||||
|
flag.Parse()
|
||||||
|
|
||||||
|
data, _ = os.ReadFile(*config)
|
||||||
|
|
||||||
var cfg struct {
|
var cfg struct {
|
||||||
Mod map[string]string `yaml:"log"`
|
Mod map[string]string `yaml:"log"`
|
||||||
|
|||||||
@@ -6,6 +6,7 @@ import (
|
|||||||
"fmt"
|
"fmt"
|
||||||
"github.com/AlexxIT/go2rtc/cmd/api"
|
"github.com/AlexxIT/go2rtc/cmd/api"
|
||||||
"github.com/AlexxIT/go2rtc/cmd/app"
|
"github.com/AlexxIT/go2rtc/cmd/app"
|
||||||
|
"github.com/AlexxIT/go2rtc/cmd/rtsp"
|
||||||
"github.com/AlexxIT/go2rtc/cmd/streams"
|
"github.com/AlexxIT/go2rtc/cmd/streams"
|
||||||
"github.com/AlexxIT/go2rtc/cmd/webrtc"
|
"github.com/AlexxIT/go2rtc/cmd/webrtc"
|
||||||
"github.com/AlexxIT/go2rtc/pkg/streamer"
|
"github.com/AlexxIT/go2rtc/pkg/streamer"
|
||||||
@@ -13,6 +14,7 @@ import (
|
|||||||
"net/http"
|
"net/http"
|
||||||
"os"
|
"os"
|
||||||
"path"
|
"path"
|
||||||
|
"strings"
|
||||||
)
|
)
|
||||||
|
|
||||||
func Init() {
|
func Init() {
|
||||||
@@ -80,6 +82,15 @@ func handler(w http.ResponseWriter, r *http.Request) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// TODO: fixme
|
||||||
|
if strings.HasPrefix(url, "rtsp://") {
|
||||||
|
port := ":" + rtsp.Port + "/"
|
||||||
|
i := strings.Index(url, port)
|
||||||
|
if i > 0 {
|
||||||
|
url = url[i+len(port):]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
stream := streams.Get(url)
|
stream := streams.Get(url)
|
||||||
str, err = webrtc.ExchangeSDP(stream, string(offer), r.UserAgent())
|
str, err = webrtc.ExchangeSDP(stream, string(offer), r.UserAgent())
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
|||||||
@@ -35,13 +35,14 @@ func Get(name string) *Stream {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func All() map[string]interface{} {
|
func All() map[string]interface{} {
|
||||||
active := map[string]interface{}{}
|
all := map[string]interface{}{}
|
||||||
for name, stream := range streams {
|
for name, stream := range streams {
|
||||||
if stream.Active() {
|
all[name] = stream
|
||||||
active[name] = stream
|
//if stream.Active() {
|
||||||
}
|
// all[name] = stream
|
||||||
|
//}
|
||||||
}
|
}
|
||||||
return active
|
return all
|
||||||
}
|
}
|
||||||
|
|
||||||
var log zerolog.Logger
|
var log zerolog.Logger
|
||||||
|
|||||||
@@ -0,0 +1,75 @@
|
|||||||
|
package webrtc
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/AlexxIT/go2rtc/cmd/api"
|
||||||
|
"github.com/AlexxIT/go2rtc/pkg/streamer"
|
||||||
|
"github.com/AlexxIT/go2rtc/pkg/webrtc"
|
||||||
|
"github.com/pion/sdp/v3"
|
||||||
|
"strings"
|
||||||
|
)
|
||||||
|
|
||||||
|
var candidates []string
|
||||||
|
|
||||||
|
func AddCandidate(address string) {
|
||||||
|
candidates = append(candidates, address)
|
||||||
|
}
|
||||||
|
|
||||||
|
func addCanditates(answer string) (string, error) {
|
||||||
|
if len(candidates) == 0 {
|
||||||
|
return answer, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
sd := &sdp.SessionDescription{}
|
||||||
|
if err := sd.Unmarshal([]byte(answer)); err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
|
||||||
|
md := sd.MediaDescriptions[0]
|
||||||
|
|
||||||
|
_, end := md.Attribute("end-of-candidates")
|
||||||
|
if end {
|
||||||
|
md.Attributes = md.Attributes[:len(md.Attributes)-1]
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, address := range candidates {
|
||||||
|
if strings.HasPrefix(address, "stun:") {
|
||||||
|
ip, err := webrtc.GetPublicIP()
|
||||||
|
if err != nil {
|
||||||
|
log.Warn().Err(err).Msg("[webrtc] public IP")
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
address = ip.String() + address[4:]
|
||||||
|
|
||||||
|
log.Debug().Str("addr", address).Msg("[webrtc] stun public address")
|
||||||
|
}
|
||||||
|
|
||||||
|
cand, err := webrtc.NewCandidate(address)
|
||||||
|
if err != nil {
|
||||||
|
log.Warn().Err(err).Msg("[webrtc] candidate")
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
md.WithPropertyAttribute(cand)
|
||||||
|
}
|
||||||
|
|
||||||
|
if end {
|
||||||
|
md.WithPropertyAttribute("end-of-candidates")
|
||||||
|
}
|
||||||
|
|
||||||
|
data, err := sd.Marshal()
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
|
||||||
|
return string(data), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func candidateHandler(ctx *api.Context, msg *streamer.Message) {
|
||||||
|
if ctx.Consumer == nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if conn := ctx.Consumer.(*webrtc.Conn); conn != nil {
|
||||||
|
log.Trace().Str("candidate", msg.Value.(string)).Msg("[webrtc] remote")
|
||||||
|
conn.Push(msg)
|
||||||
|
}
|
||||||
|
}
|
||||||
+8
-134
@@ -8,10 +8,7 @@ import (
|
|||||||
"github.com/AlexxIT/go2rtc/pkg/webrtc"
|
"github.com/AlexxIT/go2rtc/pkg/webrtc"
|
||||||
pion "github.com/pion/webrtc/v3"
|
pion "github.com/pion/webrtc/v3"
|
||||||
"github.com/rs/zerolog"
|
"github.com/rs/zerolog"
|
||||||
"io/ioutil"
|
|
||||||
"net"
|
"net"
|
||||||
"net/http"
|
|
||||||
"strings"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
func Init() {
|
func Init() {
|
||||||
@@ -56,112 +53,15 @@ func Init() {
|
|||||||
|
|
||||||
candidates = cfg.Mod.Candidates
|
candidates = cfg.Mod.Candidates
|
||||||
|
|
||||||
api.HandleFunc("/api/webrtc", apiHandler)
|
|
||||||
api.HandleFunc("/api/webrtc/camera", cameraHandler)
|
|
||||||
api.HandleWS(webrtc.MsgTypeOffer, offerHandler)
|
api.HandleWS(webrtc.MsgTypeOffer, offerHandler)
|
||||||
api.HandleWS(webrtc.MsgTypeCandidate, candidateHandler)
|
api.HandleWS(webrtc.MsgTypeCandidate, candidateHandler)
|
||||||
}
|
}
|
||||||
|
|
||||||
func AddCandidate(address string) {
|
|
||||||
candidates = append(candidates, address)
|
|
||||||
}
|
|
||||||
|
|
||||||
var Port string
|
var Port string
|
||||||
var log zerolog.Logger
|
var log zerolog.Logger
|
||||||
var candidates []string
|
|
||||||
|
|
||||||
var NewPConn func() (*pion.PeerConnection, error)
|
var NewPConn func() (*pion.PeerConnection, error)
|
||||||
|
|
||||||
func apiHandler(w http.ResponseWriter, r *http.Request) {
|
|
||||||
url := r.URL.Query().Get("url")
|
|
||||||
stream := streams.Get(url)
|
|
||||||
if stream == nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// get offer
|
|
||||||
offer, err := ioutil.ReadAll(r.Body)
|
|
||||||
if err != nil {
|
|
||||||
log.Error().Err(err).Msg("[webrtc] read offer")
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// create new webrtc instance
|
|
||||||
cons := new(webrtc.Conn)
|
|
||||||
cons.Conn, err = NewPConn()
|
|
||||||
if err != nil {
|
|
||||||
log.Error().Err(err).Msg("[webrtc] new conn")
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
cons.UserAgent = r.UserAgent()
|
|
||||||
cons.Listen(func(msg interface{}) {
|
|
||||||
if msg == streamer.StateNull {
|
|
||||||
stream.RemoveConsumer(cons)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
if err = stream.AddConsumer(cons); err != nil {
|
|
||||||
log.Warn().Err(err).Msg("[api.webrtc] add consumer")
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
cons.Init()
|
|
||||||
|
|
||||||
// exchange sdp with waiting all candidates
|
|
||||||
answer, err := cons.ExchangeSDP(string(offer), true)
|
|
||||||
|
|
||||||
// send SDP to client
|
|
||||||
if _, err = w.Write([]byte(answer)); err != nil {
|
|
||||||
log.Error().Err(err).Msg("[api.webrtc] send answer")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func cameraHandler(w http.ResponseWriter, r *http.Request) {
|
|
||||||
url := r.URL.Query().Get("url")
|
|
||||||
stream := streams.Get(url)
|
|
||||||
if stream == nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// get offer
|
|
||||||
offer, err := ioutil.ReadAll(r.Body)
|
|
||||||
if err != nil {
|
|
||||||
log.Error().Err(err).Msg("[webrtc] read offer")
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// create new webrtc instance
|
|
||||||
conn := new(webrtc.Conn)
|
|
||||||
conn.Conn, err = NewPConn()
|
|
||||||
if err != nil {
|
|
||||||
log.Error().Err(err).Msg("[webrtc] new conn")
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
conn.UserAgent = r.UserAgent()
|
|
||||||
conn.Listen(func(msg interface{}) {
|
|
||||||
switch msg.(type) {
|
|
||||||
case pion.PeerConnectionState:
|
|
||||||
if msg == pion.PeerConnectionStateDisconnected {
|
|
||||||
stream.RemoveConsumer(conn)
|
|
||||||
}
|
|
||||||
case streamer.Track:
|
|
||||||
//stream.AddProducer(conn)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
conn.Init()
|
|
||||||
|
|
||||||
// exchange sdp with waiting all candidates
|
|
||||||
answer, err := conn.ExchangeSDP(string(offer), true)
|
|
||||||
|
|
||||||
// send SDP to client
|
|
||||||
if _, err = w.Write([]byte(answer)); err != nil {
|
|
||||||
log.Error().Err(err).Msg("[api.webrtc] send answer")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func offerHandler(ctx *api.Context, msg *streamer.Message) {
|
func offerHandler(ctx *api.Context, msg *streamer.Message) {
|
||||||
name := ctx.Request.URL.Query().Get("url")
|
name := ctx.Request.URL.Query().Get("url")
|
||||||
stream := streams.Get(name)
|
stream := streams.Get(name)
|
||||||
@@ -216,7 +116,11 @@ func offerHandler(ctx *api.Context, msg *streamer.Message) {
|
|||||||
|
|
||||||
// exchange sdp without waiting all candidates
|
// exchange sdp without waiting all candidates
|
||||||
//answer, err := conn.ExchangeSDP(offer, false)
|
//answer, err := conn.ExchangeSDP(offer, false)
|
||||||
answer, err := conn.GetAnswer()
|
//answer, err := conn.GetAnswer()
|
||||||
|
answer, err := conn.GetCompleteAnswer()
|
||||||
|
if err == nil {
|
||||||
|
answer, err = addCanditates(answer)
|
||||||
|
}
|
||||||
log.Trace().Msgf("[webrtc] answer\n%s", answer)
|
log.Trace().Msgf("[webrtc] answer\n%s", answer)
|
||||||
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@@ -229,29 +133,6 @@ func offerHandler(ctx *api.Context, msg *streamer.Message) {
|
|||||||
Type: webrtc.MsgTypeAnswer, Value: answer,
|
Type: webrtc.MsgTypeAnswer, Value: answer,
|
||||||
})
|
})
|
||||||
|
|
||||||
for _, address := range candidates {
|
|
||||||
if strings.HasPrefix(address, "stun:") {
|
|
||||||
ip, err := webrtc.GetPublicIP()
|
|
||||||
if err != nil {
|
|
||||||
log.Warn().Err(err).Msg("[webrtc] public IP")
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
address = ip.String() + address[4:]
|
|
||||||
|
|
||||||
log.Debug().Str("addr", address).Msg("[webrtc] stun public address")
|
|
||||||
}
|
|
||||||
|
|
||||||
cand, err := webrtc.NewCandidate(address)
|
|
||||||
if err != nil {
|
|
||||||
log.Warn().Err(err).Msg("[webrtc] candidate")
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
|
|
||||||
conn.Fire(&streamer.Message{
|
|
||||||
Type: webrtc.MsgTypeCandidate, Value: cand,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
ctx.Consumer = conn
|
ctx.Consumer = conn
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -295,6 +176,9 @@ func ExchangeSDP(
|
|||||||
// exchange sdp without waiting all candidates
|
// exchange sdp without waiting all candidates
|
||||||
//answer, err := conn.ExchangeSDP(offer, false)
|
//answer, err := conn.ExchangeSDP(offer, false)
|
||||||
answer, err = conn.GetCompleteAnswer()
|
answer, err = conn.GetCompleteAnswer()
|
||||||
|
if err == nil {
|
||||||
|
answer, err = addCanditates(answer)
|
||||||
|
}
|
||||||
log.Trace().Msgf("[webrtc] answer\n%s", answer)
|
log.Trace().Msgf("[webrtc] answer\n%s", answer)
|
||||||
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@@ -303,13 +187,3 @@ func ExchangeSDP(
|
|||||||
|
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
func candidateHandler(ctx *api.Context, msg *streamer.Message) {
|
|
||||||
if ctx.Consumer == nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
if conn := ctx.Consumer.(*webrtc.Conn); conn != nil {
|
|
||||||
log.Trace().Str("candidate", msg.Value.(string)).Msg("[webrtc] remote")
|
|
||||||
conn.Push(msg)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -75,76 +75,6 @@ func (c *Conn) Init() {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *Conn) ExchangeSDP(offer string, complete bool) (answer string, err error) {
|
|
||||||
sdOffer := webrtc.SessionDescription{
|
|
||||||
Type: webrtc.SDPTypeOffer, SDP: offer,
|
|
||||||
}
|
|
||||||
if err = c.Conn.SetRemoteDescription(sdOffer); err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
//for _, tr := range c.Conn.GetTransceivers() {
|
|
||||||
// switch tr.Direction() {
|
|
||||||
// case webrtc.RTPTransceiverDirectionSendonly:
|
|
||||||
// // disable transceivers if we don't have track
|
|
||||||
// // make direction=inactive
|
|
||||||
// // don't really necessary, but anyway
|
|
||||||
// if tr.Sender() == nil {
|
|
||||||
// if err = tr.Stop(); err != nil {
|
|
||||||
// return
|
|
||||||
// }
|
|
||||||
// }
|
|
||||||
// case webrtc.RTPTransceiverDirectionRecvonly:
|
|
||||||
// // TODO: change codecs list
|
|
||||||
// caps := webrtc.RTPCodecCapability{
|
|
||||||
// MimeType: webrtc.MimeTypePCMU,
|
|
||||||
// ClockRate: 8000,
|
|
||||||
// }
|
|
||||||
// codecs := []webrtc.RTPCodecParameters{
|
|
||||||
// {RTPCodecCapability: caps},
|
|
||||||
// }
|
|
||||||
// if err = tr.SetCodecPreferences(codecs); err != nil {
|
|
||||||
// return
|
|
||||||
// }
|
|
||||||
// }
|
|
||||||
//}
|
|
||||||
|
|
||||||
var sdAnswer webrtc.SessionDescription
|
|
||||||
sdAnswer, err = c.Conn.CreateAnswer(nil)
|
|
||||||
if err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
//var sd *sdp.SessionDescription
|
|
||||||
//sd, err = sdAnswer.Unmarshal()
|
|
||||||
//for _, media := range sd.MediaDescriptions {
|
|
||||||
// if media.MediaName.Media != "audio" {
|
|
||||||
// continue
|
|
||||||
// }
|
|
||||||
// for i, attr := range media.Attributes {
|
|
||||||
// if attr.Key == "sendonly" {
|
|
||||||
// attr.Key = "inactive"
|
|
||||||
// media.Attributes[i] = attr
|
|
||||||
// break
|
|
||||||
// }
|
|
||||||
// }
|
|
||||||
//}
|
|
||||||
//var b []byte
|
|
||||||
//b, err = sd.Marshal()
|
|
||||||
//sdAnswer.SDP = string(b)
|
|
||||||
|
|
||||||
if err = c.Conn.SetLocalDescription(sdAnswer); err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
if complete {
|
|
||||||
<-webrtc.GatheringCompletePromise(c.Conn)
|
|
||||||
return c.Conn.LocalDescription().SDP, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
return sdAnswer.SDP, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *Conn) SetOffer(offer string) (err error) {
|
func (c *Conn) SetOffer(offer string) (err error) {
|
||||||
sdOffer := webrtc.SessionDescription{
|
sdOffer := webrtc.SessionDescription{
|
||||||
Type: webrtc.SDPTypeOffer, SDP: offer,
|
Type: webrtc.SDPTypeOffer, SDP: offer,
|
||||||
|
|||||||
@@ -47,6 +47,9 @@ func GetPublicIP() (net.IP, error) {
|
|||||||
if err = c.Do(message, func(e stun.Event) { res = e }); err != nil {
|
if err = c.Do(message, func(e stun.Event) { res = e }); err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
if err = c.Close(); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
if res.Error != nil {
|
if res.Error != nil {
|
||||||
return nil, res.Error
|
return nil, res.Error
|
||||||
|
|||||||
Reference in New Issue
Block a user