golanggohlsrtmpwebrtcmedia-serverobs-studiortcprtmp-proxyrtmp-serverrtprtsprtsp-proxyrtsp-relayrtsp-serversrtstreamingwebrtc-proxy
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
454 lines
11 KiB
454 lines
11 KiB
package core |
|
|
|
import ( |
|
"bytes" |
|
"crypto/tls" |
|
"encoding/json" |
|
"fmt" |
|
"io" |
|
"net/http" |
|
"os" |
|
"testing" |
|
"time" |
|
|
|
"github.com/aler9/gortsplib" |
|
"github.com/stretchr/testify/require" |
|
) |
|
|
|
func httpRequest(method string, ur string, in interface{}, out interface{}) error { |
|
buf, err := func() (io.Reader, error) { |
|
if in == nil { |
|
return nil, nil |
|
} |
|
|
|
byts, err := json.Marshal(in) |
|
if err != nil { |
|
return nil, err |
|
} |
|
|
|
return bytes.NewBuffer(byts), nil |
|
}() |
|
if err != nil { |
|
return err |
|
} |
|
|
|
req, err := http.NewRequest(method, ur, buf) |
|
if err != nil { |
|
return err |
|
} |
|
|
|
res, err := http.DefaultClient.Do(req) |
|
if err != nil { |
|
return err |
|
} |
|
defer res.Body.Close() |
|
|
|
if res.StatusCode != http.StatusOK { |
|
return fmt.Errorf("bad status code: %d", res.StatusCode) |
|
} |
|
|
|
if out == nil { |
|
return nil |
|
} |
|
|
|
return json.NewDecoder(res.Body).Decode(out) |
|
} |
|
|
|
func TestAPIConfigGet(t *testing.T) { |
|
p, ok := newInstance("api: yes\n") |
|
require.Equal(t, true, ok) |
|
defer p.close() |
|
|
|
var out map[string]interface{} |
|
err := httpRequest(http.MethodGet, "http://localhost:9997/v1/config/get", nil, &out) |
|
require.NoError(t, err) |
|
require.Equal(t, true, out["api"]) |
|
} |
|
|
|
func TestAPIConfigSet(t *testing.T) { |
|
p, ok := newInstance("api: yes\n") |
|
require.Equal(t, true, ok) |
|
defer p.close() |
|
|
|
err := httpRequest(http.MethodPost, "http://localhost:9997/v1/config/set", map[string]interface{}{ |
|
"rtmpDisable": true, |
|
"readTimeout": "7s", |
|
"protocols": []string{"tcp"}, |
|
}, nil) |
|
require.NoError(t, err) |
|
|
|
time.Sleep(500 * time.Millisecond) |
|
|
|
var out map[string]interface{} |
|
err = httpRequest(http.MethodGet, "http://localhost:9997/v1/config/get", nil, &out) |
|
require.NoError(t, err) |
|
require.Equal(t, true, out["rtmpDisable"]) |
|
require.Equal(t, "7s", out["readTimeout"]) |
|
require.Equal(t, []interface{}{"tcp"}, out["protocols"]) |
|
} |
|
|
|
func TestAPIConfigPathsAdd(t *testing.T) { |
|
p, ok := newInstance("api: yes\n") |
|
require.Equal(t, true, ok) |
|
defer p.close() |
|
|
|
err := httpRequest(http.MethodPost, "http://localhost:9997/v1/config/paths/add/my/path", map[string]interface{}{ |
|
"source": "rtsp://127.0.0.1:9999/mypath", |
|
"sourceOnDemand": true, |
|
}, nil) |
|
require.NoError(t, err) |
|
|
|
var out map[string]interface{} |
|
err = httpRequest(http.MethodGet, "http://localhost:9997/v1/config/get", nil, &out) |
|
require.NoError(t, err) |
|
require.Equal(t, "rtsp://127.0.0.1:9999/mypath", |
|
out["paths"].(map[string]interface{})["my/path"].(map[string]interface{})["source"]) |
|
} |
|
|
|
func TestAPIConfigPathsEdit(t *testing.T) { |
|
p, ok := newInstance("api: yes\n") |
|
require.Equal(t, true, ok) |
|
defer p.close() |
|
|
|
err := httpRequest(http.MethodPost, "http://localhost:9997/v1/config/paths/add/my/path", map[string]interface{}{ |
|
"source": "rtsp://127.0.0.1:9999/mypath", |
|
"sourceOnDemand": true, |
|
}, nil) |
|
require.NoError(t, err) |
|
|
|
err = httpRequest(http.MethodPost, "http://localhost:9997/v1/config/paths/edit/my/path", map[string]interface{}{ |
|
"source": "rtsp://127.0.0.1:9998/mypath", |
|
"sourceOnDemand": true, |
|
}, nil) |
|
require.NoError(t, err) |
|
|
|
var out struct { |
|
Paths map[string]struct { |
|
Source string `json:"source"` |
|
} `json:"paths"` |
|
} |
|
err = httpRequest(http.MethodGet, "http://localhost:9997/v1/config/get", nil, &out) |
|
require.NoError(t, err) |
|
require.Equal(t, "rtsp://127.0.0.1:9998/mypath", out.Paths["my/path"].Source) |
|
} |
|
|
|
func TestAPIConfigPathsRemove(t *testing.T) { |
|
p, ok := newInstance("api: yes\n") |
|
require.Equal(t, true, ok) |
|
defer p.close() |
|
|
|
err := httpRequest(http.MethodPost, "http://localhost:9997/v1/config/paths/add/my/path", map[string]interface{}{ |
|
"source": "rtsp://127.0.0.1:9999/mypath", |
|
"sourceOnDemand": true, |
|
}, nil) |
|
require.NoError(t, err) |
|
|
|
err = httpRequest(http.MethodPost, "http://localhost:9997/v1/config/paths/remove/my/path", nil, nil) |
|
require.NoError(t, err) |
|
|
|
var out struct { |
|
Paths map[string]interface{} `json:"paths"` |
|
} |
|
err = httpRequest(http.MethodGet, "http://localhost:9997/v1/config/get", nil, &out) |
|
require.NoError(t, err) |
|
_, ok = out.Paths["my/path"] |
|
require.Equal(t, false, ok) |
|
} |
|
|
|
func TestAPIPathsList(t *testing.T) { |
|
serverCertFpath, err := writeTempFile(serverCert) |
|
require.NoError(t, err) |
|
defer os.Remove(serverCertFpath) |
|
|
|
serverKeyFpath, err := writeTempFile(serverKey) |
|
require.NoError(t, err) |
|
defer os.Remove(serverKeyFpath) |
|
|
|
p, ok := newInstance("api: yes\n" + |
|
"encryption: optional\n" + |
|
"serverCert: " + serverCertFpath + "\n" + |
|
"serverKey: " + serverKeyFpath + "\n" + |
|
"paths:\n" + |
|
" mypath:\n") |
|
require.Equal(t, true, ok) |
|
defer p.close() |
|
|
|
var out struct { |
|
Items map[string]struct { |
|
Source struct { |
|
Type string `json:"type"` |
|
} `json:"source"` |
|
} `json:"items"` |
|
} |
|
err = httpRequest(http.MethodGet, "http://localhost:9997/v1/paths/list", nil, &out) |
|
require.NoError(t, err) |
|
_, ok = out.Items["mypath"] |
|
require.Equal(t, true, ok) |
|
|
|
track, err := gortsplib.NewTrackH264(96, |
|
[]byte{0x01, 0x02, 0x03, 0x04}, []byte{0x01, 0x02, 0x03, 0x04}, nil) |
|
require.NoError(t, err) |
|
|
|
func() { |
|
source := gortsplib.Client{} |
|
|
|
err = source.StartPublishing("rtsp://localhost:8554/mypath", |
|
gortsplib.Tracks{track}) |
|
require.NoError(t, err) |
|
defer source.Close() |
|
|
|
err = httpRequest(http.MethodGet, "http://localhost:9997/v1/paths/list", nil, &out) |
|
require.NoError(t, err) |
|
require.Equal(t, "rtspSession", out.Items["mypath"].Source.Type) |
|
}() |
|
|
|
func() { |
|
source := gortsplib.Client{ |
|
TLSConfig: &tls.Config{InsecureSkipVerify: true}, |
|
} |
|
|
|
err := source.StartPublishing("rtsps://localhost:8322/mypath", |
|
gortsplib.Tracks{track}) |
|
require.NoError(t, err) |
|
defer source.Close() |
|
|
|
err = httpRequest(http.MethodGet, "http://localhost:9997/v1/paths/list", nil, &out) |
|
require.NoError(t, err) |
|
require.Equal(t, "rtspsSession", out.Items["mypath"].Source.Type) |
|
}() |
|
} |
|
|
|
func TestAPIList(t *testing.T) { |
|
serverCertFpath, err := writeTempFile(serverCert) |
|
require.NoError(t, err) |
|
defer os.Remove(serverCertFpath) |
|
|
|
serverKeyFpath, err := writeTempFile(serverKey) |
|
require.NoError(t, err) |
|
defer os.Remove(serverKeyFpath) |
|
|
|
for _, ca := range []string{ |
|
"rtsp", |
|
"rtsps", |
|
"rtmp", |
|
"hls", |
|
} { |
|
t.Run(ca, func(t *testing.T) { |
|
conf := "api: yes\n" |
|
|
|
if ca == "rtsps" { |
|
conf += "protocols: [tcp]\n" + |
|
"encryption: strict\n" + |
|
"serverCert: " + serverCertFpath + "\n" + |
|
"serverKey: " + serverKeyFpath + "\n" |
|
} |
|
|
|
conf += "paths:\n" + |
|
" all:\n" |
|
|
|
p, ok := newInstance(conf) |
|
require.Equal(t, true, ok) |
|
defer p.close() |
|
|
|
track, err := gortsplib.NewTrackH264(96, |
|
[]byte{0x01, 0x02, 0x03, 0x04}, []byte{0x01, 0x02, 0x03, 0x04}, nil) |
|
require.NoError(t, err) |
|
|
|
switch ca { |
|
case "rtsp": |
|
source := gortsplib.Client{} |
|
|
|
err := source.StartPublishing("rtsp://localhost:8554/mypath", |
|
gortsplib.Tracks{track}) |
|
require.NoError(t, err) |
|
defer source.Close() |
|
|
|
case "rtsps": |
|
source := gortsplib.Client{ |
|
TLSConfig: &tls.Config{InsecureSkipVerify: true}, |
|
} |
|
|
|
err := source.StartPublishing("rtsps://localhost:8322/mypath", |
|
gortsplib.Tracks{track}) |
|
require.NoError(t, err) |
|
defer source.Close() |
|
|
|
case "rtmp": |
|
cnt1, err := newContainer("ffmpeg", "source", []string{ |
|
"-re", |
|
"-stream_loop", "-1", |
|
"-i", "emptyvideo.mkv", |
|
"-c", "copy", |
|
"-f", "flv", |
|
"rtmp://localhost:1935/test1/test2", |
|
}) |
|
require.NoError(t, err) |
|
defer cnt1.close() |
|
|
|
case "hls": |
|
source := gortsplib.Client{} |
|
|
|
err := source.StartPublishing("rtsp://localhost:8554/mypath", |
|
gortsplib.Tracks{track}) |
|
require.NoError(t, err) |
|
defer source.Close() |
|
|
|
func() { |
|
res, err := http.Get("http://localhost:8888/mypath/index.m3u8") |
|
require.NoError(t, err) |
|
defer res.Body.Close() |
|
require.Equal(t, 200, res.StatusCode) |
|
}() |
|
} |
|
|
|
switch ca { |
|
case "rtsp", "rtsps", "rtmp": |
|
var pa string |
|
switch ca { |
|
case "rtsp": |
|
pa = "rtspsessions" |
|
|
|
case "rtsps": |
|
pa = "rtspssessions" |
|
|
|
case "rtmp": |
|
pa = "rtmpconns" |
|
} |
|
|
|
var out struct { |
|
Items map[string]struct { |
|
State string `json:"state"` |
|
} `json:"items"` |
|
} |
|
err = httpRequest(http.MethodGet, "http://localhost:9997/v1/"+pa+"/list", nil, &out) |
|
require.NoError(t, err) |
|
|
|
var firstID string |
|
for k := range out.Items { |
|
firstID = k |
|
} |
|
|
|
require.Equal(t, "publish", out.Items[firstID].State) |
|
|
|
case "hls": |
|
var out struct { |
|
Items map[string]struct { |
|
LastRequest string `json:"lastRequest"` |
|
} `json:"items"` |
|
} |
|
err = httpRequest(http.MethodGet, "http://localhost:9997/v1/hlsmuxers/list", nil, &out) |
|
require.NoError(t, err) |
|
|
|
var firstID string |
|
for k := range out.Items { |
|
firstID = k |
|
} |
|
|
|
require.NotEqual(t, "", out.Items[firstID].LastRequest) |
|
} |
|
}) |
|
} |
|
} |
|
|
|
func TestAPIKick(t *testing.T) { |
|
serverCertFpath, err := writeTempFile(serverCert) |
|
require.NoError(t, err) |
|
defer os.Remove(serverCertFpath) |
|
|
|
serverKeyFpath, err := writeTempFile(serverKey) |
|
require.NoError(t, err) |
|
defer os.Remove(serverKeyFpath) |
|
|
|
for _, ca := range []string{ |
|
"rtsp", |
|
"rtsps", |
|
"rtmp", |
|
} { |
|
t.Run(ca, func(t *testing.T) { |
|
conf := "api: yes\n" |
|
|
|
if ca == "rtsps" { |
|
conf += "protocols: [tcp]\n" + |
|
"encryption: strict\n" + |
|
"serverCert: " + serverCertFpath + "\n" + |
|
"serverKey: " + serverKeyFpath + "\n" |
|
} |
|
|
|
conf += "paths:\n" + |
|
" all:\n" |
|
|
|
p, ok := newInstance(conf) |
|
require.Equal(t, true, ok) |
|
defer p.close() |
|
|
|
track, err := gortsplib.NewTrackH264(96, |
|
[]byte{0x01, 0x02, 0x03, 0x04}, []byte{0x01, 0x02, 0x03, 0x04}, nil) |
|
require.NoError(t, err) |
|
|
|
switch ca { |
|
case "rtsp": |
|
source := gortsplib.Client{} |
|
|
|
err := source.StartPublishing("rtsp://localhost:8554/mypath", |
|
gortsplib.Tracks{track}) |
|
require.NoError(t, err) |
|
defer source.Close() |
|
|
|
case "rtsps": |
|
source := gortsplib.Client{ |
|
TLSConfig: &tls.Config{InsecureSkipVerify: true}, |
|
} |
|
|
|
err := source.StartPublishing("rtsps://localhost:8322/mypath", |
|
gortsplib.Tracks{track}) |
|
require.NoError(t, err) |
|
defer source.Close() |
|
|
|
case "rtmp": |
|
cnt1, err := newContainer("ffmpeg", "source", []string{ |
|
"-re", |
|
"-stream_loop", "-1", |
|
"-i", "emptyvideo.mkv", |
|
"-c", "copy", |
|
"-f", "flv", |
|
"rtmp://localhost:1935/test1/test2", |
|
}) |
|
require.NoError(t, err) |
|
defer cnt1.close() |
|
} |
|
|
|
var pa string |
|
switch ca { |
|
case "rtsp": |
|
pa = "rtspsessions" |
|
|
|
case "rtsps": |
|
pa = "rtspssessions" |
|
|
|
case "rtmp": |
|
pa = "rtmpconns" |
|
} |
|
|
|
var out1 struct { |
|
Items map[string]struct{} `json:"items"` |
|
} |
|
err = httpRequest(http.MethodGet, "http://localhost:9997/v1/"+pa+"/list", nil, &out1) |
|
require.NoError(t, err) |
|
|
|
var firstID string |
|
for k := range out1.Items { |
|
firstID = k |
|
} |
|
|
|
err = httpRequest(http.MethodPost, "http://localhost:9997/v1/"+pa+"/kick/"+firstID, nil, nil) |
|
require.NoError(t, err) |
|
|
|
var out2 struct { |
|
Items map[string]struct{} `json:"items"` |
|
} |
|
err = httpRequest(http.MethodGet, "http://localhost:9997/v1/"+pa+"/list", nil, &out2) |
|
require.NoError(t, err) |
|
require.Equal(t, 0, len(out2.Items)) |
|
}) |
|
} |
|
}
|
|
|