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:
aler9
2022-08-07 13:49:15 +02:00
parent 4940e8faeb
commit a2c6a7fb97
4 changed files with 99 additions and 44 deletions

View File

@@ -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

View File

@@ -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) {

View File

@@ -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 {

View File

@@ -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() {