2023-05-08 15:04:14 +00:00
|
|
|
package core
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"crypto/sha256"
|
|
|
|
"encoding/base64"
|
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"net/http"
|
|
|
|
"strings"
|
|
|
|
|
2023-08-26 16:54:28 +00:00
|
|
|
"github.com/bluenviron/gortsplib/v4/pkg/auth"
|
|
|
|
"github.com/bluenviron/gortsplib/v4/pkg/headers"
|
2023-05-08 15:04:14 +00:00
|
|
|
"github.com/google/uuid"
|
|
|
|
|
2023-05-16 14:14:20 +00:00
|
|
|
"github.com/bluenviron/mediamtx/internal/conf"
|
2023-12-08 18:17:17 +00:00
|
|
|
"github.com/bluenviron/mediamtx/internal/defs"
|
2023-05-08 15:04:14 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
func sha256Base64(in string) string {
|
|
|
|
h := sha256.New()
|
|
|
|
h.Write([]byte(in))
|
|
|
|
return base64.StdEncoding.EncodeToString(h.Sum(nil))
|
|
|
|
}
|
|
|
|
|
|
|
|
func checkCredential(right string, guess string) bool {
|
|
|
|
if strings.HasPrefix(right, "sha256:") {
|
|
|
|
return right[len("sha256:"):] == sha256Base64(guess)
|
|
|
|
}
|
|
|
|
|
|
|
|
return right == guess
|
|
|
|
}
|
|
|
|
|
2023-07-23 18:06:16 +00:00
|
|
|
func doExternalAuthentication(
|
2023-05-08 15:04:14 +00:00
|
|
|
ur string,
|
2023-12-08 18:17:17 +00:00
|
|
|
accessRequest defs.PathAccessRequest,
|
2023-05-08 15:04:14 +00:00
|
|
|
) error {
|
|
|
|
enc, _ := json.Marshal(struct {
|
|
|
|
IP string `json:"ip"`
|
|
|
|
User string `json:"user"`
|
|
|
|
Password string `json:"password"`
|
|
|
|
Path string `json:"path"`
|
|
|
|
Protocol string `json:"protocol"`
|
|
|
|
ID *uuid.UUID `json:"id"`
|
|
|
|
Action string `json:"action"`
|
|
|
|
Query string `json:"query"`
|
|
|
|
}{
|
2023-12-08 18:17:17 +00:00
|
|
|
IP: accessRequest.IP.String(),
|
|
|
|
User: accessRequest.User,
|
|
|
|
Password: accessRequest.Pass,
|
|
|
|
Path: accessRequest.Name,
|
|
|
|
Protocol: string(accessRequest.Proto),
|
|
|
|
ID: accessRequest.ID,
|
2023-05-08 15:04:14 +00:00
|
|
|
Action: func() string {
|
2023-12-08 18:17:17 +00:00
|
|
|
if accessRequest.Publish {
|
2023-05-08 15:04:14 +00:00
|
|
|
return "publish"
|
|
|
|
}
|
|
|
|
return "read"
|
|
|
|
}(),
|
2023-12-08 18:17:17 +00:00
|
|
|
Query: accessRequest.Query,
|
2023-05-08 15:04:14 +00:00
|
|
|
})
|
|
|
|
res, err := http.Post(ur, "application/json", bytes.NewReader(enc))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
defer res.Body.Close()
|
|
|
|
|
|
|
|
if res.StatusCode < 200 || res.StatusCode > 299 {
|
|
|
|
if resBody, err := io.ReadAll(res.Body); err == nil && len(resBody) != 0 {
|
2023-07-23 18:06:16 +00:00
|
|
|
return fmt.Errorf("server replied with code %d: %s", res.StatusCode, string(resBody))
|
2023-05-08 15:04:14 +00:00
|
|
|
}
|
2023-07-23 18:06:16 +00:00
|
|
|
return fmt.Errorf("server replied with code %d", res.StatusCode)
|
2023-05-08 15:04:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-07-23 18:06:16 +00:00
|
|
|
func doAuthentication(
|
2023-05-08 15:04:14 +00:00
|
|
|
externalAuthenticationURL string,
|
|
|
|
rtspAuthMethods conf.AuthMethods,
|
2023-10-07 21:32:15 +00:00
|
|
|
pathConf *conf.Path,
|
2023-12-08 18:17:17 +00:00
|
|
|
accessRequest defs.PathAccessRequest,
|
2023-05-08 15:04:14 +00:00
|
|
|
) error {
|
|
|
|
var rtspAuth headers.Authorization
|
2023-12-08 18:17:17 +00:00
|
|
|
if accessRequest.RTSPRequest != nil {
|
|
|
|
err := rtspAuth.Unmarshal(accessRequest.RTSPRequest.Header["Authorization"])
|
2023-05-08 15:04:14 +00:00
|
|
|
if err == nil && rtspAuth.Method == headers.AuthBasic {
|
2023-12-08 18:17:17 +00:00
|
|
|
accessRequest.User = rtspAuth.BasicUser
|
|
|
|
accessRequest.Pass = rtspAuth.BasicPass
|
2023-05-08 15:04:14 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if externalAuthenticationURL != "" {
|
2023-07-23 18:06:16 +00:00
|
|
|
err := doExternalAuthentication(
|
2023-05-08 15:04:14 +00:00
|
|
|
externalAuthenticationURL,
|
2023-10-18 09:50:26 +00:00
|
|
|
accessRequest,
|
2023-05-08 15:04:14 +00:00
|
|
|
)
|
|
|
|
if err != nil {
|
2024-01-03 20:13:20 +00:00
|
|
|
return defs.AuthenticationError{Message: fmt.Sprintf("external authentication failed: %s", err)}
|
2023-05-08 15:04:14 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
var pathIPs conf.IPsOrCIDRs
|
|
|
|
var pathUser string
|
|
|
|
var pathPass string
|
|
|
|
|
2023-12-08 18:17:17 +00:00
|
|
|
if accessRequest.Publish {
|
2023-05-08 15:04:14 +00:00
|
|
|
pathIPs = pathConf.PublishIPs
|
|
|
|
pathUser = string(pathConf.PublishUser)
|
|
|
|
pathPass = string(pathConf.PublishPass)
|
|
|
|
} else {
|
|
|
|
pathIPs = pathConf.ReadIPs
|
|
|
|
pathUser = string(pathConf.ReadUser)
|
|
|
|
pathPass = string(pathConf.ReadPass)
|
|
|
|
}
|
|
|
|
|
|
|
|
if pathIPs != nil {
|
2023-12-08 18:17:17 +00:00
|
|
|
if !ipEqualOrInRange(accessRequest.IP, pathIPs) {
|
2024-01-03 20:13:20 +00:00
|
|
|
return defs.AuthenticationError{Message: fmt.Sprintf("IP %s not allowed", accessRequest.IP)}
|
2023-05-08 15:04:14 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if pathUser != "" {
|
2023-12-08 18:17:17 +00:00
|
|
|
if accessRequest.RTSPRequest != nil && rtspAuth.Method == headers.AuthDigest {
|
2023-05-08 15:04:14 +00:00
|
|
|
err := auth.Validate(
|
2023-12-08 18:17:17 +00:00
|
|
|
accessRequest.RTSPRequest,
|
2023-05-08 15:04:14 +00:00
|
|
|
pathUser,
|
|
|
|
pathPass,
|
2023-12-08 18:17:17 +00:00
|
|
|
accessRequest.RTSPBaseURL,
|
2023-05-08 15:04:14 +00:00
|
|
|
rtspAuthMethods,
|
|
|
|
"IPCAM",
|
2023-12-08 18:17:17 +00:00
|
|
|
accessRequest.RTSPNonce)
|
2023-05-08 15:04:14 +00:00
|
|
|
if err != nil {
|
2024-01-03 20:13:20 +00:00
|
|
|
return defs.AuthenticationError{Message: err.Error()}
|
2023-05-08 15:04:14 +00:00
|
|
|
}
|
2023-12-08 18:17:17 +00:00
|
|
|
} else if !checkCredential(pathUser, accessRequest.User) ||
|
|
|
|
!checkCredential(pathPass, accessRequest.Pass) {
|
2024-01-03 20:13:20 +00:00
|
|
|
return defs.AuthenticationError{Message: "invalid credentials"}
|
2023-05-08 15:04:14 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|