2020-10-08 14:38:33 +02:00
|
|
|
package group
|
2020-09-30 00:33:23 +02:00
|
|
|
|
|
|
|
import (
|
2020-11-29 17:00:12 +01:00
|
|
|
"encoding/json"
|
2024-03-03 13:34:18 +01:00
|
|
|
"errors"
|
2021-02-14 19:44:19 +01:00
|
|
|
"fmt"
|
2024-08-11 23:57:11 +02:00
|
|
|
"sort"
|
2020-09-30 00:33:23 +02:00
|
|
|
"testing"
|
|
|
|
"time"
|
2025-01-04 18:49:46 +01:00
|
|
|
|
|
|
|
"github.com/pion/webrtc/v4"
|
2020-09-30 00:33:23 +02:00
|
|
|
)
|
|
|
|
|
2021-02-14 20:09:30 +01:00
|
|
|
func TestGroup(t *testing.T) {
|
|
|
|
groups.groups = nil
|
2021-04-22 17:58:30 +02:00
|
|
|
Add("group", &Description{})
|
|
|
|
Add("group/subgroup", &Description{Public: true})
|
2021-02-14 20:09:30 +01:00
|
|
|
if len(groups.groups) != 2 {
|
|
|
|
t.Errorf("Expected 2, got %v", len(groups.groups))
|
|
|
|
}
|
|
|
|
g := Get("group")
|
|
|
|
g2 := Get("group/subgroup")
|
|
|
|
if g == nil {
|
|
|
|
t.Fatalf("Couldn't get group")
|
|
|
|
}
|
|
|
|
if g2 == nil {
|
|
|
|
t.Fatalf("Couldn't get group/subgroup")
|
|
|
|
}
|
|
|
|
if name := g.Name(); name != "group" {
|
|
|
|
t.Errorf("Name: expected group1, got %v", name)
|
|
|
|
}
|
|
|
|
if locked, _ := g.Locked(); locked {
|
|
|
|
t.Errorf("Locked: expected false, got %v", locked)
|
|
|
|
}
|
2021-05-08 16:09:22 +02:00
|
|
|
api, err := g.API()
|
|
|
|
if err != nil || api == nil {
|
|
|
|
t.Errorf("Couldn't get API: %v", err)
|
2021-02-14 20:09:30 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if names := GetNames(); len(names) != 2 {
|
|
|
|
t.Errorf("Expected 2, got %v", names)
|
|
|
|
}
|
|
|
|
|
|
|
|
if subs := GetSubGroups("group"); len(subs) != 0 {
|
|
|
|
t.Errorf("Expected [], got %v", subs)
|
|
|
|
}
|
|
|
|
|
2024-02-24 12:09:10 +01:00
|
|
|
if public := GetPublic(nil); len(public) != 1 || public[0].Name != "group/subgroup" {
|
2021-05-22 16:58:09 +02:00
|
|
|
t.Errorf("Expected group/subgroup, got %v", public)
|
2021-02-14 20:09:30 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-02-14 19:44:19 +01:00
|
|
|
func TestChatHistory(t *testing.T) {
|
|
|
|
g := Group{
|
2021-04-22 17:58:30 +02:00
|
|
|
description: &Description{},
|
2021-02-14 19:44:19 +01:00
|
|
|
}
|
2023-03-22 17:41:16 +01:00
|
|
|
user := "user"
|
2021-02-14 19:44:19 +01:00
|
|
|
for i := 0; i < 2*maxChatHistory; i++ {
|
2024-08-11 23:57:11 +02:00
|
|
|
g.AddToChatHistory(
|
|
|
|
fmt.Sprintf("id-%v", i),
|
|
|
|
fmt.Sprintf("source-%v", i%4),
|
|
|
|
&user, time.Now(), "",
|
2021-02-14 19:44:19 +01:00
|
|
|
fmt.Sprintf("%v", i),
|
|
|
|
)
|
|
|
|
}
|
|
|
|
h := g.GetChatHistory()
|
|
|
|
if len(h) != maxChatHistory {
|
|
|
|
t.Errorf("Expected %v, got %v", maxChatHistory, len(g.history))
|
|
|
|
}
|
|
|
|
for i, s := range h {
|
2024-08-11 23:57:11 +02:00
|
|
|
j := i + maxChatHistory
|
|
|
|
if s.Id != fmt.Sprintf("id-%v", j) {
|
|
|
|
t.Errorf("Expected %v, got %v", j, s.Id)
|
|
|
|
}
|
|
|
|
if s.Source != fmt.Sprintf("source-%v", j%4) {
|
|
|
|
t.Errorf("Expected %v, got %v", j%4, s.Id)
|
2021-02-14 19:44:19 +01:00
|
|
|
}
|
2024-08-11 23:57:11 +02:00
|
|
|
if s.Value.(string) != fmt.Sprintf("%v", j) {
|
|
|
|
t.Errorf("Expected %v, got %v", j, s.Value)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
l := len(h)
|
|
|
|
j := maxChatHistory + 4
|
|
|
|
g.ClearChatHistory(
|
|
|
|
fmt.Sprintf("id-%v", j), fmt.Sprintf("source-%v", j%4),
|
|
|
|
)
|
|
|
|
if lh := len(g.GetChatHistory()); lh != l-1 {
|
|
|
|
t.Errorf("Expected %v, got %v", l-1, lh)
|
|
|
|
}
|
|
|
|
g.ClearChatHistory("", fmt.Sprintf("source-%v", j%4))
|
|
|
|
if lh := len(g.GetChatHistory()); lh != l*3/4 {
|
|
|
|
t.Errorf("Expected %v, got %v", l*3/4, lh)
|
|
|
|
}
|
2024-11-08 13:08:11 +01:00
|
|
|
g.ClearChatHistory("", "")
|
2024-08-11 23:57:11 +02:00
|
|
|
if lh := len(g.GetChatHistory()); lh != 0 {
|
|
|
|
t.Errorf("Expected 0, got %v", lh)
|
2021-02-14 19:44:19 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-01-02 18:36:09 +01:00
|
|
|
func permissionsEqual(a, b []string) bool {
|
|
|
|
// nil case
|
|
|
|
if len(a) == 0 && len(b) == 0 {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
if len(a) != len(b) {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
aa := append([]string(nil), a...)
|
|
|
|
sort.Slice(aa, func(i, j int) bool {
|
|
|
|
return aa[i] < aa[j]
|
|
|
|
})
|
|
|
|
bb := append([]string(nil), b...)
|
|
|
|
sort.Slice(bb, func(i, j int) bool {
|
|
|
|
return bb[i] < bb[j]
|
|
|
|
})
|
|
|
|
for i := range aa {
|
|
|
|
if aa[i] != bb[i] {
|
|
|
|
return false
|
|
|
|
}
|
2020-11-29 17:00:12 +01:00
|
|
|
}
|
2024-01-02 18:36:09 +01:00
|
|
|
return true
|
2020-11-29 17:00:12 +01:00
|
|
|
}
|
2021-02-06 23:30:45 +01:00
|
|
|
|
2023-03-22 17:41:16 +01:00
|
|
|
var jch = "jch"
|
|
|
|
var john = "john"
|
|
|
|
var james = "james"
|
|
|
|
var paul = "paul"
|
2024-01-02 18:36:09 +01:00
|
|
|
var peter = "peter"
|
2024-11-08 13:08:11 +01:00
|
|
|
var admin = "admin"
|
|
|
|
var stt = "speech-to-text"
|
2023-03-22 17:41:16 +01:00
|
|
|
|
2021-10-27 04:15:44 +02:00
|
|
|
var badClients = []ClientCredentials{
|
2023-03-22 17:41:16 +01:00
|
|
|
{Username: &jch, Password: "foo"},
|
|
|
|
{Username: &john, Password: "foo"},
|
|
|
|
{Username: &james, Password: "foo"},
|
2021-02-06 23:30:45 +01:00
|
|
|
}
|
|
|
|
|
2021-10-27 04:15:44 +02:00
|
|
|
type credPerm struct {
|
|
|
|
c ClientCredentials
|
2022-02-19 23:43:44 +01:00
|
|
|
p []string
|
2021-02-06 23:30:45 +01:00
|
|
|
}
|
|
|
|
|
2024-11-08 13:08:11 +01:00
|
|
|
var desc2JSON = `
|
|
|
|
{
|
|
|
|
"max-history-age": 10,
|
|
|
|
"auto-subgroups": true,
|
|
|
|
"users": {
|
|
|
|
"jch": {"password": "topsecret", "permissions": "op"},
|
|
|
|
"john": {"password": "secret", "permissions": "present"},
|
|
|
|
"james": {"password": "secret2", "permissions": "message"},
|
|
|
|
"peter": {"password": "secret4"},
|
|
|
|
"admin": {"password": "admin", "permissions": "admin"},
|
|
|
|
"speech-to-text": {"password": {"type": "wildcard"},
|
|
|
|
"permissions": "caption"}
|
|
|
|
},
|
|
|
|
"wildcard-user":
|
|
|
|
{"permissions": "message", "password": {"type":"wildcard"}}
|
|
|
|
}`
|
|
|
|
|
2021-10-27 04:15:44 +02:00
|
|
|
var goodClients = []credPerm{
|
2021-02-06 23:30:45 +01:00
|
|
|
{
|
2023-03-22 17:41:16 +01:00
|
|
|
ClientCredentials{Username: &jch, Password: "topsecret"},
|
2024-11-08 13:08:11 +01:00
|
|
|
[]string{"op", "present", "message", "caption", "token"},
|
2021-02-06 23:30:45 +01:00
|
|
|
},
|
|
|
|
{
|
2023-03-22 17:41:16 +01:00
|
|
|
ClientCredentials{Username: &john, Password: "secret"},
|
2024-05-08 16:00:18 +02:00
|
|
|
[]string{"present", "message"},
|
2021-02-06 23:30:45 +01:00
|
|
|
},
|
|
|
|
{
|
2024-01-02 18:36:09 +01:00
|
|
|
ClientCredentials{Username: &james, Password: "secret2"},
|
2024-05-08 16:00:18 +02:00
|
|
|
[]string{"message"},
|
2021-02-06 23:30:45 +01:00
|
|
|
},
|
|
|
|
{
|
2024-01-02 18:36:09 +01:00
|
|
|
ClientCredentials{Username: &paul, Password: "secret3"},
|
2024-05-08 16:00:18 +02:00
|
|
|
[]string{"message"},
|
2021-02-06 23:30:45 +01:00
|
|
|
},
|
|
|
|
{
|
2024-01-02 18:36:09 +01:00
|
|
|
ClientCredentials{Username: &peter, Password: "secret4"},
|
2023-04-03 20:09:22 +02:00
|
|
|
[]string{},
|
2021-02-06 23:30:45 +01:00
|
|
|
},
|
2024-11-08 13:08:11 +01:00
|
|
|
{
|
|
|
|
ClientCredentials{Username: &admin, Password: "admin"},
|
|
|
|
[]string{"admin"},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
ClientCredentials{Username: &stt},
|
|
|
|
[]string{"caption"},
|
|
|
|
},
|
2021-02-06 23:30:45 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestPermissions(t *testing.T) {
|
2022-09-09 14:13:24 +02:00
|
|
|
var g Group
|
2024-11-08 13:08:11 +01:00
|
|
|
err := json.Unmarshal([]byte(desc2JSON), &g.description)
|
2021-02-06 23:30:45 +01:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("unmarshal: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, c := range badClients {
|
2023-03-22 17:41:16 +01:00
|
|
|
t.Run("bad "+*c.Username, func(t *testing.T) {
|
2024-03-03 13:34:18 +01:00
|
|
|
var autherr *NotAuthorisedError
|
2022-09-09 14:13:24 +02:00
|
|
|
_, p, err := g.GetPermission(c)
|
2024-03-03 13:34:18 +01:00
|
|
|
if !errors.As(err, &autherr) {
|
2021-02-06 23:30:45 +01:00
|
|
|
t.Errorf("GetPermission %v: %v %v", c, err, p)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, cp := range goodClients {
|
2023-03-22 17:41:16 +01:00
|
|
|
t.Run("good "+*cp.c.Username, func(t *testing.T) {
|
2022-09-09 14:13:24 +02:00
|
|
|
u, p, err := g.GetPermission(cp.c)
|
2021-02-06 23:30:45 +01:00
|
|
|
if err != nil {
|
|
|
|
t.Errorf("GetPermission %v: %v", cp.c, err)
|
2023-03-22 17:41:16 +01:00
|
|
|
} else if u != *cp.c.Username ||
|
2024-01-02 18:36:09 +01:00
|
|
|
!permissionsEqual(p, cp.p) {
|
2022-02-20 15:32:18 +01:00
|
|
|
t.Errorf("%v: got %v %v, expected %v",
|
|
|
|
cp.c, u, p, cp.p)
|
2021-02-06 23:30:45 +01:00
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
2021-08-02 15:53:33 +02:00
|
|
|
|
2024-01-02 18:36:09 +01:00
|
|
|
func TestExtraPermissions(t *testing.T) {
|
|
|
|
j := `
|
|
|
|
{
|
|
|
|
"users": {
|
|
|
|
"jch": {"password": "topsecret", "permissions": "op"},
|
|
|
|
"john": {"password": "secret", "permissions": "present"},
|
|
|
|
"james": {"password": "secret2", "permissions": "observe"}
|
|
|
|
}
|
|
|
|
}`
|
|
|
|
|
|
|
|
var d Description
|
|
|
|
err := json.Unmarshal([]byte(j), &d)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("unmarshal: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
doit := func(u string, p []string) {
|
2024-11-08 13:08:11 +01:00
|
|
|
t.Helper()
|
2024-01-02 18:36:09 +01:00
|
|
|
pu := d.Users[u].Permissions.Permissions(&d)
|
|
|
|
if !permissionsEqual(pu, p) {
|
|
|
|
t.Errorf("%v: expected %v, got %v", u, p, pu)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-11-08 13:08:11 +01:00
|
|
|
doit("jch", []string{"op", "token", "present", "message", "caption"})
|
2024-05-08 16:00:18 +02:00
|
|
|
doit("john", []string{"present", "message"})
|
2024-01-02 18:36:09 +01:00
|
|
|
doit("james", []string{})
|
|
|
|
|
|
|
|
d.AllowRecording = true
|
|
|
|
d.UnrestrictedTokens = false
|
|
|
|
|
2024-11-08 13:08:11 +01:00
|
|
|
doit("jch", []string{
|
|
|
|
"op", "record", "token", "present", "message", "caption",
|
|
|
|
})
|
2024-05-08 16:00:18 +02:00
|
|
|
doit("john", []string{"present", "message"})
|
2024-01-02 18:36:09 +01:00
|
|
|
doit("james", []string{})
|
|
|
|
|
|
|
|
d.AllowRecording = false
|
|
|
|
d.UnrestrictedTokens = true
|
|
|
|
|
2024-11-08 13:08:11 +01:00
|
|
|
doit("jch", []string{"op", "token", "present", "message", "caption"})
|
2024-05-08 16:00:18 +02:00
|
|
|
doit("john", []string{"token", "present", "message"})
|
2024-01-02 18:36:09 +01:00
|
|
|
doit("james", []string{})
|
|
|
|
|
|
|
|
d.AllowRecording = true
|
|
|
|
d.UnrestrictedTokens = true
|
|
|
|
|
2024-11-08 13:08:11 +01:00
|
|
|
doit("jch", []string{
|
|
|
|
"op", "record", "token", "present", "message", "caption",
|
|
|
|
})
|
2024-05-08 16:00:18 +02:00
|
|
|
doit("john", []string{"token", "present", "message"})
|
2024-01-02 18:36:09 +01:00
|
|
|
doit("james", []string{})
|
|
|
|
}
|
|
|
|
|
2023-03-22 04:04:22 +01:00
|
|
|
func TestUsernameTaken(t *testing.T) {
|
|
|
|
var g Group
|
2024-11-08 13:08:11 +01:00
|
|
|
err := json.Unmarshal([]byte(desc2JSON), &g.description)
|
2023-03-22 04:04:22 +01:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("unmarshal: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if g.UserExists("") {
|
|
|
|
t.Error("UserExists(\"\") is true, expected false")
|
|
|
|
}
|
|
|
|
if !g.UserExists("john") {
|
|
|
|
t.Error("UserExists(john) is false")
|
|
|
|
}
|
|
|
|
if !g.UserExists("john") {
|
|
|
|
t.Error("UserExists(james) is false")
|
|
|
|
}
|
|
|
|
if g.UserExists("paul") {
|
|
|
|
t.Error("UserExists(paul) is true")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-08-02 15:53:33 +02:00
|
|
|
func TestFmtpValue(t *testing.T) {
|
|
|
|
type fmtpTest struct {
|
|
|
|
fmtp string
|
|
|
|
key string
|
|
|
|
value string
|
|
|
|
}
|
|
|
|
fmtpTests := []fmtpTest{
|
|
|
|
{"", "foo", ""},
|
|
|
|
{"profile-id=0", "profile-id", "0"},
|
|
|
|
{"profile-id=0", "foo", ""},
|
|
|
|
{"level-asymmetry-allowed=1;packetization-mode=1;profile-level-id=42001f", "profile-level-id", "42001f"},
|
|
|
|
{"foo=1;bar=2;quux=3", "foo", "1"},
|
|
|
|
{"foo=1;bar=2;quux=3", "bar", "2"},
|
|
|
|
{"foo=1;bar=2;quux=3", "fu", ""},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, test := range fmtpTests {
|
|
|
|
v := fmtpValue(test.fmtp, test.key)
|
|
|
|
if v != test.value {
|
|
|
|
t.Errorf("fmtpValue(%v, %v) = %v, expected %v",
|
|
|
|
test.fmtp, test.key, v, test.value,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-08-24 00:31:46 +02:00
|
|
|
|
|
|
|
func TestValidGroupName(t *testing.T) {
|
|
|
|
type nameTest struct {
|
|
|
|
name string
|
|
|
|
result bool
|
|
|
|
}
|
|
|
|
tests := []nameTest{
|
|
|
|
{"", false},
|
|
|
|
{"/", false},
|
|
|
|
{"/foo", false},
|
|
|
|
{"foo/", false},
|
|
|
|
{"./foo", false},
|
|
|
|
{"foo/.", false},
|
|
|
|
{"../foo", false},
|
|
|
|
{"foo/..", false},
|
|
|
|
{"foo/./bar", false},
|
|
|
|
{"foo/../bar", false},
|
|
|
|
{"foo", true},
|
|
|
|
{"foo/bar", true},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, test := range tests {
|
|
|
|
r := validGroupName(test.name)
|
|
|
|
if r != test.result {
|
|
|
|
t.Errorf("Valid %v: got %v, expected %v",
|
|
|
|
test.name, r, test.result)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2025-01-04 18:49:46 +01:00
|
|
|
|
|
|
|
func TestPayloadTypeDiscontiguous(t *testing.T) {
|
|
|
|
names := []string{
|
|
|
|
"vp8", "vp9", "av1", "h264",
|
|
|
|
"opus", "g722", "pcmu", "pcma",
|
|
|
|
}
|
|
|
|
|
|
|
|
m := make(map[webrtc.PayloadType]bool)
|
|
|
|
|
|
|
|
for _, n := range names {
|
|
|
|
codec, err := codecsFromName(n)
|
|
|
|
if err != nil {
|
|
|
|
t.Errorf("%v: %v", n, err)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
pt, err := CodecPayloadType(codec[0].RTPCodecCapability)
|
|
|
|
if err != nil {
|
|
|
|
t.Errorf("%v: %v", codec, err)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
m[pt] = true
|
|
|
|
}
|
|
|
|
|
|
|
|
for pt := range m {
|
|
|
|
if m[pt+1] {
|
|
|
|
t.Errorf("%v and %v are both defined", pt, pt+1)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|