mirror of
https://github.com/aler9/gortsplib
synced 2025-09-27 19:42:11 +08:00
server: change behavior in case of unhandled methods
(https://github.com/aler9/rtsp-simple-server/issues/1066) Return 501 and keep connection open instead of returning 400 and closing the connection.
This commit is contained in:
@@ -48,16 +48,6 @@ func (e ErrServerCSeqMissing) Error() string {
|
|||||||
return "CSeq is missing"
|
return "CSeq is missing"
|
||||||
}
|
}
|
||||||
|
|
||||||
// ErrServerUnhandledRequest is an error that can be returned by a server.
|
|
||||||
type ErrServerUnhandledRequest struct {
|
|
||||||
Request *base.Request
|
|
||||||
}
|
|
||||||
|
|
||||||
// Error implements the error interface.
|
|
||||||
func (e ErrServerUnhandledRequest) Error() string {
|
|
||||||
return fmt.Sprintf("unhandled request: %v %v", e.Request.Method, e.Request.URL)
|
|
||||||
}
|
|
||||||
|
|
||||||
// ErrServerInvalidState is an error that can be returned by a server.
|
// ErrServerInvalidState is an error that can be returned by a server.
|
||||||
type ErrServerInvalidState struct {
|
type ErrServerInvalidState struct {
|
||||||
AllowedList []fmt.Stringer
|
AllowedList []fmt.Stringer
|
||||||
|
125
server_test.go
125
server_test.go
@@ -316,38 +316,103 @@ func TestServerErrorCSeqMissing(t *testing.T) {
|
|||||||
<-connClosed
|
<-connClosed
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestServerErrorInvalidMethod(t *testing.T) {
|
type testServerErrMethodNotImplemented struct {
|
||||||
connClosed := make(chan struct{})
|
stream *ServerStream
|
||||||
|
}
|
||||||
|
|
||||||
s := &Server{
|
func (s *testServerErrMethodNotImplemented) OnSetup(
|
||||||
Handler: &testServerHandler{
|
ctx *ServerHandlerOnSetupCtx,
|
||||||
onConnClose: func(ctx *ServerHandlerOnConnCloseCtx) {
|
) (*base.Response, *ServerStream, error) {
|
||||||
require.EqualError(t, ctx.Error, "unhandled request: INVALID rtsp://localhost:8554/")
|
return &base.Response{
|
||||||
close(connClosed)
|
StatusCode: base.StatusOK,
|
||||||
},
|
}, s.stream, nil
|
||||||
},
|
}
|
||||||
RTSPAddress: "localhost:8554",
|
|
||||||
|
func TestServerErrorMethodNotImplemented(t *testing.T) {
|
||||||
|
for _, ca := range []string{"outside session", "inside session"} {
|
||||||
|
t.Run(ca, func(t *testing.T) {
|
||||||
|
track := &TrackH264{
|
||||||
|
PayloadType: 96,
|
||||||
|
SPS: []byte{0x01, 0x02, 0x03, 0x04},
|
||||||
|
PPS: []byte{0x01, 0x02, 0x03, 0x04},
|
||||||
|
}
|
||||||
|
|
||||||
|
stream := NewServerStream(Tracks{track})
|
||||||
|
defer stream.Close()
|
||||||
|
|
||||||
|
s := &Server{
|
||||||
|
Handler: &testServerErrMethodNotImplemented{stream},
|
||||||
|
RTSPAddress: "localhost:8554",
|
||||||
|
}
|
||||||
|
|
||||||
|
err := s.Start()
|
||||||
|
require.NoError(t, err)
|
||||||
|
defer s.Close()
|
||||||
|
|
||||||
|
conn, err := net.Dial("tcp", "localhost:8554")
|
||||||
|
require.NoError(t, err)
|
||||||
|
defer conn.Close()
|
||||||
|
br := bufio.NewReader(conn)
|
||||||
|
|
||||||
|
var sx headers.Session
|
||||||
|
|
||||||
|
if ca == "inside session" {
|
||||||
|
res, err := writeReqReadRes(conn, br, 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)
|
||||||
|
|
||||||
|
err = sx.Unmarshal(res.Header["Session"])
|
||||||
|
require.NoError(t, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
headers := base.Header{
|
||||||
|
"CSeq": base.HeaderValue{"2"},
|
||||||
|
}
|
||||||
|
if ca == "inside session" {
|
||||||
|
headers["Session"] = base.HeaderValue{sx.Session}
|
||||||
|
}
|
||||||
|
|
||||||
|
res, err := writeReqReadRes(conn, br, base.Request{
|
||||||
|
Method: base.SetParameter,
|
||||||
|
URL: mustParseURL("rtsp://localhost:8554/teststream/trackID=0"),
|
||||||
|
Header: headers,
|
||||||
|
})
|
||||||
|
require.NoError(t, err)
|
||||||
|
require.Equal(t, base.StatusNotImplemented, res.StatusCode)
|
||||||
|
|
||||||
|
headers = base.Header{
|
||||||
|
"CSeq": base.HeaderValue{"3"},
|
||||||
|
}
|
||||||
|
if ca == "inside session" {
|
||||||
|
headers["Session"] = base.HeaderValue{sx.Session}
|
||||||
|
}
|
||||||
|
|
||||||
|
res, err = writeReqReadRes(conn, br, base.Request{
|
||||||
|
Method: base.Options,
|
||||||
|
URL: mustParseURL("rtsp://localhost:8554/teststream/trackID=0"),
|
||||||
|
Header: headers,
|
||||||
|
})
|
||||||
|
require.NoError(t, err)
|
||||||
|
require.Equal(t, base.StatusOK, res.StatusCode)
|
||||||
|
})
|
||||||
}
|
}
|
||||||
err := s.Start()
|
|
||||||
require.NoError(t, err)
|
|
||||||
defer s.Close()
|
|
||||||
|
|
||||||
conn, err := net.Dial("tcp", "localhost:8554")
|
|
||||||
require.NoError(t, err)
|
|
||||||
defer conn.Close()
|
|
||||||
br := bufio.NewReader(conn)
|
|
||||||
|
|
||||||
res, err := writeReqReadRes(conn, br, base.Request{
|
|
||||||
Method: "INVALID",
|
|
||||||
URL: mustParseURL("rtsp://localhost:8554/"),
|
|
||||||
Header: base.Header{
|
|
||||||
"CSeq": base.HeaderValue{"1"},
|
|
||||||
},
|
|
||||||
})
|
|
||||||
require.NoError(t, err)
|
|
||||||
require.Equal(t, base.StatusBadRequest, res.StatusCode)
|
|
||||||
|
|
||||||
<-connClosed
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestServerErrorTCPTwoConnOneSession(t *testing.T) {
|
func TestServerErrorTCPTwoConnOneSession(t *testing.T) {
|
||||||
|
@@ -505,8 +505,8 @@ func (sc *ServerConn) handleRequest(req *base.Request) (*base.Response, error) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
return &base.Response{
|
return &base.Response{
|
||||||
StatusCode: base.StatusBadRequest,
|
StatusCode: base.StatusNotImplemented,
|
||||||
}, liberrors.ErrServerUnhandledRequest{Request: req}
|
}, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (sc *ServerConn) handleRequestOuter(req *base.Request) error {
|
func (sc *ServerConn) handleRequestOuter(req *base.Request) error {
|
||||||
|
@@ -1133,8 +1133,8 @@ func (ss *ServerSession) handleRequest(sc *ServerConn, req *base.Request) (*base
|
|||||||
}
|
}
|
||||||
|
|
||||||
return &base.Response{
|
return &base.Response{
|
||||||
StatusCode: base.StatusBadRequest,
|
StatusCode: base.StatusNotImplemented,
|
||||||
}, liberrors.ErrServerUnhandledRequest{Request: req}
|
}, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (ss *ServerSession) runWriter() {
|
func (ss *ServerSession) runWriter() {
|
||||||
|
Reference in New Issue
Block a user