mirror of
https://github.com/aler9/gortsplib
synced 2025-10-05 15:16:51 +08:00
server: when a session is closed, close all associated connections
regardless of the fact that they use TCP or UDP
This commit is contained in:
124
server_test.go
124
server_test.go
@@ -252,8 +252,18 @@ func TestServerConnClose(t *testing.T) {
|
|||||||
nconn, err := net.Dial("tcp", "localhost:8554")
|
nconn, err := net.Dial("tcp", "localhost:8554")
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
defer nconn.Close()
|
defer nconn.Close()
|
||||||
|
conn := conn.NewConn(nconn)
|
||||||
|
|
||||||
<-nconnClosed
|
<-nconnClosed
|
||||||
|
|
||||||
|
_, err = writeReqReadRes(conn, base.Request{
|
||||||
|
Method: base.Options,
|
||||||
|
URL: mustParseURL("rtsp://localhost:8554/"),
|
||||||
|
Header: base.Header{
|
||||||
|
"CSeq": base.HeaderValue{"1"},
|
||||||
|
},
|
||||||
|
})
|
||||||
|
require.Error(t, err)
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestServerCSeq(t *testing.T) {
|
func TestServerCSeq(t *testing.T) {
|
||||||
@@ -795,21 +805,24 @@ func TestServerErrorInvalidSession(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestServerSessionClose(t *testing.T) {
|
func TestServerSessionClose(t *testing.T) {
|
||||||
sessionClosed := make(chan struct{})
|
stream := NewServerStream(Tracks{&TrackH264{
|
||||||
|
PayloadType: 96,
|
||||||
|
SPS: []byte{0x01, 0x02, 0x03, 0x04},
|
||||||
|
PPS: []byte{0x01, 0x02, 0x03, 0x04},
|
||||||
|
}})
|
||||||
|
defer stream.Close()
|
||||||
|
|
||||||
|
var session *ServerSession
|
||||||
|
|
||||||
s := &Server{
|
s := &Server{
|
||||||
Handler: &testServerHandler{
|
Handler: &testServerHandler{
|
||||||
onSessionOpen: func(ctx *ServerHandlerOnSessionOpenCtx) {
|
onSessionOpen: func(ctx *ServerHandlerOnSessionOpenCtx) {
|
||||||
ctx.Session.Close()
|
session = ctx.Session
|
||||||
ctx.Session.Close()
|
|
||||||
},
|
|
||||||
onSessionClose: func(ctx *ServerHandlerOnSessionCloseCtx) {
|
|
||||||
close(sessionClosed)
|
|
||||||
},
|
},
|
||||||
onSetup: func(ctx *ServerHandlerOnSetupCtx) (*base.Response, *ServerStream, error) {
|
onSetup: func(ctx *ServerHandlerOnSetupCtx) (*base.Response, *ServerStream, error) {
|
||||||
return &base.Response{
|
return &base.Response{
|
||||||
StatusCode: base.StatusOK,
|
StatusCode: base.StatusOK,
|
||||||
}, nil, nil
|
}, stream, nil
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
RTSPAddress: "localhost:8554",
|
RTSPAddress: "localhost:8554",
|
||||||
@@ -824,7 +837,7 @@ func TestServerSessionClose(t *testing.T) {
|
|||||||
defer nconn.Close()
|
defer nconn.Close()
|
||||||
conn := conn.NewConn(nconn)
|
conn := conn.NewConn(nconn)
|
||||||
|
|
||||||
err = conn.WriteRequest(&base.Request{
|
res, err := writeReqReadRes(conn, base.Request{
|
||||||
Method: base.Setup,
|
Method: base.Setup,
|
||||||
URL: mustParseURL("rtsp://localhost:8554/teststream/trackID=0"),
|
URL: mustParseURL("rtsp://localhost:8554/teststream/trackID=0"),
|
||||||
Header: base.Header{
|
Header: base.Header{
|
||||||
@@ -844,8 +857,19 @@ func TestServerSessionClose(t *testing.T) {
|
|||||||
},
|
},
|
||||||
})
|
})
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
|
require.Equal(t, base.StatusOK, res.StatusCode)
|
||||||
|
|
||||||
<-sessionClosed
|
session.Close()
|
||||||
|
session.Close()
|
||||||
|
|
||||||
|
_, err = writeReqReadRes(conn, base.Request{
|
||||||
|
Method: base.Options,
|
||||||
|
URL: mustParseURL("rtsp://localhost:8554/"),
|
||||||
|
Header: base.Header{
|
||||||
|
"CSeq": base.HeaderValue{"2"},
|
||||||
|
},
|
||||||
|
})
|
||||||
|
require.Error(t, err)
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestServerSessionAutoClose(t *testing.T) {
|
func TestServerSessionAutoClose(t *testing.T) {
|
||||||
@@ -855,13 +879,11 @@ func TestServerSessionAutoClose(t *testing.T) {
|
|||||||
t.Run(ca, func(t *testing.T) {
|
t.Run(ca, func(t *testing.T) {
|
||||||
sessionClosed := make(chan struct{})
|
sessionClosed := make(chan struct{})
|
||||||
|
|
||||||
track := &TrackH264{
|
stream := NewServerStream(Tracks{&TrackH264{
|
||||||
PayloadType: 96,
|
PayloadType: 96,
|
||||||
SPS: []byte{0x01, 0x02, 0x03, 0x04},
|
SPS: []byte{0x01, 0x02, 0x03, 0x04},
|
||||||
PPS: []byte{0x01, 0x02, 0x03, 0x04},
|
PPS: []byte{0x01, 0x02, 0x03, 0x04},
|
||||||
}
|
}})
|
||||||
|
|
||||||
stream := NewServerStream(Tracks{track})
|
|
||||||
defer stream.Close()
|
defer stream.Close()
|
||||||
|
|
||||||
s := &Server{
|
s := &Server{
|
||||||
@@ -920,6 +942,82 @@ func TestServerSessionAutoClose(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestServerSessionTeardown(t *testing.T) {
|
||||||
|
stream := NewServerStream(Tracks{&TrackH264{
|
||||||
|
PayloadType: 96,
|
||||||
|
SPS: []byte{0x01, 0x02, 0x03, 0x04},
|
||||||
|
PPS: []byte{0x01, 0x02, 0x03, 0x04},
|
||||||
|
}})
|
||||||
|
defer stream.Close()
|
||||||
|
|
||||||
|
s := &Server{
|
||||||
|
Handler: &testServerHandler{
|
||||||
|
onSetup: func(ctx *ServerHandlerOnSetupCtx) (*base.Response, *ServerStream, error) {
|
||||||
|
return &base.Response{
|
||||||
|
StatusCode: base.StatusOK,
|
||||||
|
}, stream, nil
|
||||||
|
},
|
||||||
|
},
|
||||||
|
RTSPAddress: "localhost:8554",
|
||||||
|
}
|
||||||
|
|
||||||
|
err := s.Start()
|
||||||
|
require.NoError(t, err)
|
||||||
|
defer s.Close()
|
||||||
|
|
||||||
|
nconn, err := net.Dial("tcp", "localhost:8554")
|
||||||
|
require.NoError(t, err)
|
||||||
|
defer nconn.Close()
|
||||||
|
conn := conn.NewConn(nconn)
|
||||||
|
|
||||||
|
res, err := writeReqReadRes(conn, base.Request{
|
||||||
|
Method: base.Setup,
|
||||||
|
URL: mustParseURL("rtsp://localhost:8554/teststream/trackID=0"),
|
||||||
|
Header: base.Header{
|
||||||
|
"CSeq": base.HeaderValue{"1"},
|
||||||
|
"Transport": headers.Transport{
|
||||||
|
Protocol: headers.TransportProtocolTCP,
|
||||||
|
Delivery: func() *headers.TransportDelivery {
|
||||||
|
v := headers.TransportDeliveryUnicast
|
||||||
|
return &v
|
||||||
|
}(),
|
||||||
|
Mode: func() *headers.TransportMode {
|
||||||
|
v := headers.TransportModePlay
|
||||||
|
return &v
|
||||||
|
}(),
|
||||||
|
InterleavedIDs: &[2]int{0, 1},
|
||||||
|
}.Marshal(),
|
||||||
|
},
|
||||||
|
})
|
||||||
|
require.NoError(t, err)
|
||||||
|
require.Equal(t, base.StatusOK, res.StatusCode)
|
||||||
|
|
||||||
|
var sx headers.Session
|
||||||
|
err = sx.Unmarshal(res.Header["Session"])
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
res, err = writeReqReadRes(conn, base.Request{
|
||||||
|
Method: base.Teardown,
|
||||||
|
URL: mustParseURL("rtsp://localhost:8554/"),
|
||||||
|
Header: base.Header{
|
||||||
|
"CSeq": base.HeaderValue{"2"},
|
||||||
|
"Session": base.HeaderValue{sx.Session},
|
||||||
|
},
|
||||||
|
})
|
||||||
|
require.NoError(t, err)
|
||||||
|
require.Equal(t, base.StatusOK, res.StatusCode)
|
||||||
|
|
||||||
|
res, err = writeReqReadRes(conn, base.Request{
|
||||||
|
Method: base.Options,
|
||||||
|
URL: mustParseURL("rtsp://localhost:8554/"),
|
||||||
|
Header: base.Header{
|
||||||
|
"CSeq": base.HeaderValue{"3"},
|
||||||
|
},
|
||||||
|
})
|
||||||
|
require.NoError(t, err)
|
||||||
|
require.Equal(t, base.StatusOK, res.StatusCode)
|
||||||
|
}
|
||||||
|
|
||||||
func TestServerErrorInvalidPath(t *testing.T) {
|
func TestServerErrorInvalidPath(t *testing.T) {
|
||||||
for _, ca := range []string{"inside session", "outside session"} {
|
for _, ca := range []string{"inside session", "outside session"} {
|
||||||
t.Run(ca, func(t *testing.T) {
|
t.Run(ca, func(t *testing.T) {
|
||||||
|
@@ -310,13 +310,14 @@ func (ss *ServerSession) run() {
|
|||||||
<-ss.writerDone
|
<-ss.writerDone
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// close all associated connections, both UDP and TCP
|
||||||
|
// except for the ones that called TEARDOWN
|
||||||
|
// (that are detached from the session just after the request)
|
||||||
for sc := range ss.conns {
|
for sc := range ss.conns {
|
||||||
if sc == ss.tcpConn {
|
sc.Close()
|
||||||
sc.Close()
|
|
||||||
|
|
||||||
// make sure that OnFrame() is never called after OnSessionClose()
|
// make sure that OnFrame() is never called after OnSessionClose()
|
||||||
<-sc.done
|
<-sc.done
|
||||||
}
|
|
||||||
|
|
||||||
select {
|
select {
|
||||||
case sc.sessionRemove <- ss:
|
case sc.sessionRemove <- ss:
|
||||||
@@ -379,6 +380,7 @@ func (ss *ServerSession) runInner() error {
|
|||||||
|
|
||||||
// after a TEARDOWN, session must be unpaired with the connection
|
// after a TEARDOWN, session must be unpaired with the connection
|
||||||
if req.req.Method == base.Teardown {
|
if req.req.Method == base.Teardown {
|
||||||
|
delete(ss.conns, req.sc)
|
||||||
returnedSession = nil
|
returnedSession = nil
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Reference in New Issue
Block a user