63df85db6d
This extends the dendrite monolith for pinecone to integrate the s&f features into the mobile apps. Also makes a few tweaks to federation queueing/statistics to make some edge cases more robust.
323 lines
8.4 KiB
Go
323 lines
8.4 KiB
Go
// Copyright 2022 The Matrix.org Foundation C.I.C.
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
package gobind
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"net"
|
|
"strings"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/matrix-org/dendrite/federationapi/api"
|
|
relayServerAPI "github.com/matrix-org/dendrite/relayapi/api"
|
|
"github.com/matrix-org/gomatrixserverlib"
|
|
"github.com/sirupsen/logrus"
|
|
"github.com/stretchr/testify/assert"
|
|
"gotest.tools/v3/poll"
|
|
)
|
|
|
|
var TestBuf = []byte{0, 1, 2, 3, 4, 5, 6, 7, 8, 9}
|
|
|
|
type TestNetConn struct {
|
|
net.Conn
|
|
shouldFail bool
|
|
}
|
|
|
|
func (t *TestNetConn) Read(b []byte) (int, error) {
|
|
if t.shouldFail {
|
|
return 0, fmt.Errorf("Failed")
|
|
} else {
|
|
n := copy(b, TestBuf)
|
|
return n, nil
|
|
}
|
|
}
|
|
|
|
func (t *TestNetConn) Write(b []byte) (int, error) {
|
|
if t.shouldFail {
|
|
return 0, fmt.Errorf("Failed")
|
|
} else {
|
|
return len(b), nil
|
|
}
|
|
}
|
|
|
|
func (t *TestNetConn) Close() error {
|
|
if t.shouldFail {
|
|
return fmt.Errorf("Failed")
|
|
} else {
|
|
return nil
|
|
}
|
|
}
|
|
|
|
func TestConduitStoresPort(t *testing.T) {
|
|
conduit := Conduit{port: 7}
|
|
assert.Equal(t, 7, conduit.Port())
|
|
}
|
|
|
|
func TestConduitRead(t *testing.T) {
|
|
conduit := Conduit{conn: &TestNetConn{}}
|
|
b := make([]byte, len(TestBuf))
|
|
bytes, err := conduit.Read(b)
|
|
assert.NoError(t, err)
|
|
assert.Equal(t, len(TestBuf), bytes)
|
|
assert.Equal(t, TestBuf, b)
|
|
}
|
|
|
|
func TestConduitReadCopy(t *testing.T) {
|
|
conduit := Conduit{conn: &TestNetConn{}}
|
|
result, err := conduit.ReadCopy()
|
|
assert.NoError(t, err)
|
|
assert.Equal(t, TestBuf, result)
|
|
}
|
|
|
|
func TestConduitWrite(t *testing.T) {
|
|
conduit := Conduit{conn: &TestNetConn{}}
|
|
bytes, err := conduit.Write(TestBuf)
|
|
assert.NoError(t, err)
|
|
assert.Equal(t, len(TestBuf), bytes)
|
|
}
|
|
|
|
func TestConduitClose(t *testing.T) {
|
|
conduit := Conduit{conn: &TestNetConn{}}
|
|
err := conduit.Close()
|
|
assert.NoError(t, err)
|
|
assert.True(t, conduit.closed.Load())
|
|
}
|
|
|
|
func TestConduitReadClosed(t *testing.T) {
|
|
conduit := Conduit{conn: &TestNetConn{}}
|
|
err := conduit.Close()
|
|
assert.NoError(t, err)
|
|
b := make([]byte, len(TestBuf))
|
|
_, err = conduit.Read(b)
|
|
assert.Error(t, err)
|
|
}
|
|
|
|
func TestConduitReadCopyClosed(t *testing.T) {
|
|
conduit := Conduit{conn: &TestNetConn{}}
|
|
err := conduit.Close()
|
|
assert.NoError(t, err)
|
|
_, err = conduit.ReadCopy()
|
|
assert.Error(t, err)
|
|
}
|
|
|
|
func TestConduitWriteClosed(t *testing.T) {
|
|
conduit := Conduit{conn: &TestNetConn{}}
|
|
err := conduit.Close()
|
|
assert.NoError(t, err)
|
|
_, err = conduit.Write(TestBuf)
|
|
assert.Error(t, err)
|
|
}
|
|
|
|
func TestConduitReadCopyFails(t *testing.T) {
|
|
conduit := Conduit{conn: &TestNetConn{shouldFail: true}}
|
|
_, err := conduit.ReadCopy()
|
|
assert.Error(t, err)
|
|
}
|
|
|
|
var testRelayServers = []gomatrixserverlib.ServerName{"relay1", "relay2"}
|
|
|
|
type FakeFedAPI struct {
|
|
api.FederationInternalAPI
|
|
}
|
|
|
|
func (f *FakeFedAPI) P2PQueryRelayServers(ctx context.Context, req *api.P2PQueryRelayServersRequest, res *api.P2PQueryRelayServersResponse) error {
|
|
res.RelayServers = testRelayServers
|
|
return nil
|
|
}
|
|
|
|
type FakeRelayAPI struct {
|
|
relayServerAPI.RelayInternalAPI
|
|
}
|
|
|
|
func (r *FakeRelayAPI) PerformRelayServerSync(ctx context.Context, userID gomatrixserverlib.UserID, relayServer gomatrixserverlib.ServerName) error {
|
|
return nil
|
|
}
|
|
|
|
func TestRelayRetrieverInitialization(t *testing.T) {
|
|
retriever := RelayServerRetriever{
|
|
Context: context.Background(),
|
|
ServerName: "server",
|
|
relayServersQueried: make(map[gomatrixserverlib.ServerName]bool),
|
|
FederationAPI: &FakeFedAPI{},
|
|
RelayAPI: &FakeRelayAPI{},
|
|
}
|
|
|
|
retriever.InitializeRelayServers(logrus.WithField("test", "relay"))
|
|
relayServers := retriever.GetQueriedServerStatus()
|
|
assert.Equal(t, 2, len(relayServers))
|
|
}
|
|
|
|
func TestRelayRetrieverSync(t *testing.T) {
|
|
retriever := RelayServerRetriever{
|
|
Context: context.Background(),
|
|
ServerName: "server",
|
|
relayServersQueried: make(map[gomatrixserverlib.ServerName]bool),
|
|
FederationAPI: &FakeFedAPI{},
|
|
RelayAPI: &FakeRelayAPI{},
|
|
}
|
|
|
|
retriever.InitializeRelayServers(logrus.WithField("test", "relay"))
|
|
relayServers := retriever.GetQueriedServerStatus()
|
|
assert.Equal(t, 2, len(relayServers))
|
|
|
|
stopRelayServerSync := make(chan bool)
|
|
go retriever.SyncRelayServers(stopRelayServerSync)
|
|
|
|
check := func(log poll.LogT) poll.Result {
|
|
relayServers := retriever.GetQueriedServerStatus()
|
|
for _, queried := range relayServers {
|
|
if !queried {
|
|
return poll.Continue("waiting for all servers to be queried")
|
|
}
|
|
}
|
|
|
|
stopRelayServerSync <- true
|
|
return poll.Success()
|
|
}
|
|
poll.WaitOn(t, check, poll.WithTimeout(5*time.Second), poll.WithDelay(100*time.Millisecond))
|
|
}
|
|
|
|
func TestMonolithStarts(t *testing.T) {
|
|
monolith := DendriteMonolith{}
|
|
monolith.Start()
|
|
monolith.PublicKey()
|
|
monolith.Stop()
|
|
}
|
|
|
|
func TestMonolithSetRelayServers(t *testing.T) {
|
|
testCases := []struct {
|
|
name string
|
|
nodeID string
|
|
relays string
|
|
expectedRelays string
|
|
expectSelf bool
|
|
}{
|
|
{
|
|
name: "assorted valid, invalid, empty & self keys",
|
|
nodeID: "@valid:abcdef123456abcdef123456abcdef123456abcdef123456abcdef123456abcd",
|
|
relays: "@valid:123456123456abcdef123456abcdef123456abcdef123456abcdef123456abcd,@invalid:notakey,,",
|
|
expectedRelays: "123456123456abcdef123456abcdef123456abcdef123456abcdef123456abcd",
|
|
expectSelf: true,
|
|
},
|
|
{
|
|
name: "invalid node key",
|
|
nodeID: "@invalid:notakey",
|
|
relays: "@valid:123456123456abcdef123456abcdef123456abcdef123456abcdef123456abcd,@invalid:notakey,,",
|
|
expectedRelays: "",
|
|
expectSelf: false,
|
|
},
|
|
{
|
|
name: "node is self",
|
|
nodeID: "self",
|
|
relays: "@valid:123456123456abcdef123456abcdef123456abcdef123456abcdef123456abcd,@invalid:notakey,,",
|
|
expectedRelays: "123456123456abcdef123456abcdef123456abcdef123456abcdef123456abcd",
|
|
expectSelf: false,
|
|
},
|
|
}
|
|
|
|
for _, tc := range testCases {
|
|
monolith := DendriteMonolith{}
|
|
monolith.Start()
|
|
|
|
inputRelays := tc.relays
|
|
expectedRelays := tc.expectedRelays
|
|
if tc.expectSelf {
|
|
inputRelays += "," + monolith.PublicKey()
|
|
expectedRelays += "," + monolith.PublicKey()
|
|
}
|
|
nodeID := tc.nodeID
|
|
if nodeID == "self" {
|
|
nodeID = monolith.PublicKey()
|
|
}
|
|
|
|
monolith.SetRelayServers(nodeID, inputRelays)
|
|
relays := monolith.GetRelayServers(nodeID)
|
|
monolith.Stop()
|
|
|
|
if !containSameKeys(strings.Split(relays, ","), strings.Split(expectedRelays, ",")) {
|
|
t.Fatalf("%s: expected %s got %s", tc.name, expectedRelays, relays)
|
|
}
|
|
}
|
|
}
|
|
|
|
func containSameKeys(expected []string, actual []string) bool {
|
|
if len(expected) != len(actual) {
|
|
return false
|
|
}
|
|
|
|
for _, expectedKey := range expected {
|
|
hasMatch := false
|
|
for _, actualKey := range actual {
|
|
if actualKey == expectedKey {
|
|
hasMatch = true
|
|
}
|
|
}
|
|
|
|
if !hasMatch {
|
|
return false
|
|
}
|
|
}
|
|
|
|
return true
|
|
}
|
|
|
|
func TestParseServerKey(t *testing.T) {
|
|
testCases := []struct {
|
|
name string
|
|
serverKey string
|
|
expectedErr bool
|
|
expectedKey gomatrixserverlib.ServerName
|
|
}{
|
|
{
|
|
name: "valid userid as key",
|
|
serverKey: "@valid:abcdef123456abcdef123456abcdef123456abcdef123456abcdef123456abcd",
|
|
expectedErr: false,
|
|
expectedKey: "abcdef123456abcdef123456abcdef123456abcdef123456abcdef123456abcd",
|
|
},
|
|
{
|
|
name: "valid key",
|
|
serverKey: "abcdef123456abcdef123456abcdef123456abcdef123456abcdef123456abcd",
|
|
expectedErr: false,
|
|
expectedKey: "abcdef123456abcdef123456abcdef123456abcdef123456abcdef123456abcd",
|
|
},
|
|
{
|
|
name: "invalid userid key",
|
|
serverKey: "@invalid:notakey",
|
|
expectedErr: true,
|
|
expectedKey: "",
|
|
},
|
|
{
|
|
name: "invalid key",
|
|
serverKey: "@invalid:notakey",
|
|
expectedErr: true,
|
|
expectedKey: "",
|
|
},
|
|
}
|
|
|
|
for _, tc := range testCases {
|
|
key, err := getServerKeyFromString(tc.serverKey)
|
|
if tc.expectedErr && err == nil {
|
|
t.Fatalf("%s: expected an error", tc.name)
|
|
} else if !tc.expectedErr && err != nil {
|
|
t.Fatalf("%s: didn't expect an error: %s", tc.name, err.Error())
|
|
}
|
|
if tc.expectedKey != key {
|
|
t.Fatalf("%s: keys not equal. expected: %s got: %s", tc.name, tc.expectedKey, key)
|
|
}
|
|
}
|
|
}
|