mirror of
https://github.com/strukturag/nextcloud-spreed-signaling.git
synced 2025-03-14 11:32:46 +00:00
208 lines
5.2 KiB
Go
208 lines
5.2 KiB
Go
/**
|
|
* Standalone signaling server for the Nextcloud Spreed app.
|
|
* Copyright (C) 2019 struktur AG
|
|
*
|
|
* @author Joachim Bauch <bauch@struktur.de>
|
|
*
|
|
* @license GNU AGPL version 3 or any later version
|
|
*
|
|
* This program is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU Affero General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU Affero General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Affero General Public License
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
package signaling
|
|
|
|
import (
|
|
"archive/tar"
|
|
"compress/gzip"
|
|
"fmt"
|
|
"io"
|
|
"net"
|
|
"net/http"
|
|
"os"
|
|
"strings"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/stretchr/testify/assert"
|
|
"github.com/stretchr/testify/require"
|
|
)
|
|
|
|
func testGeoLookupReader(t *testing.T, reader *GeoLookup) {
|
|
tests := map[string]string{
|
|
// Example from maxminddb-golang code.
|
|
"81.2.69.142": "GB",
|
|
// Local addresses don't have a country assigned.
|
|
"127.0.0.1": "",
|
|
}
|
|
|
|
for ip, expected := range tests {
|
|
ip := ip
|
|
expected := expected
|
|
t.Run(ip, func(t *testing.T) {
|
|
country, err := reader.LookupCountry(net.ParseIP(ip))
|
|
if !assert.NoError(t, err, "Could not lookup %s", ip) {
|
|
return
|
|
}
|
|
|
|
assert.Equal(t, expected, country, "Unexpected country for %s", ip)
|
|
})
|
|
}
|
|
}
|
|
|
|
func GetGeoIpUrlForTest(t *testing.T) string {
|
|
t.Helper()
|
|
|
|
var geoIpUrl string
|
|
if os.Getenv("USE_DB_IP_GEOIP_DATABASE") != "" {
|
|
now := time.Now().UTC()
|
|
geoIpUrl = fmt.Sprintf("https://download.db-ip.com/free/dbip-country-lite-%d-%.2d.mmdb.gz", now.Year(), now.Month())
|
|
}
|
|
if geoIpUrl == "" {
|
|
license := os.Getenv("MAXMIND_GEOLITE2_LICENSE")
|
|
if license == "" {
|
|
t.Skip("No MaxMind GeoLite2 license was set in MAXMIND_GEOLITE2_LICENSE environment variable.")
|
|
}
|
|
geoIpUrl = GetGeoIpDownloadUrl(license)
|
|
}
|
|
return geoIpUrl
|
|
}
|
|
|
|
func TestGeoLookup(t *testing.T) {
|
|
CatchLogForTest(t)
|
|
require := require.New(t)
|
|
reader, err := NewGeoLookupFromUrl(GetGeoIpUrlForTest(t))
|
|
require.NoError(err)
|
|
defer reader.Close()
|
|
|
|
require.NoError(reader.Update())
|
|
|
|
testGeoLookupReader(t, reader)
|
|
}
|
|
|
|
func TestGeoLookupCaching(t *testing.T) {
|
|
CatchLogForTest(t)
|
|
require := require.New(t)
|
|
reader, err := NewGeoLookupFromUrl(GetGeoIpUrlForTest(t))
|
|
require.NoError(err)
|
|
defer reader.Close()
|
|
|
|
require.NoError(reader.Update())
|
|
|
|
// Updating the second time will most likely return a "304 Not Modified".
|
|
// Make sure this doesn't trigger an error.
|
|
require.NoError(reader.Update())
|
|
}
|
|
|
|
func TestGeoLookupContinent(t *testing.T) {
|
|
tests := map[string][]string{
|
|
"AU": {"OC"},
|
|
"DE": {"EU"},
|
|
"RU": {"EU"},
|
|
"": nil,
|
|
"INVALID ": nil,
|
|
}
|
|
|
|
for country, expected := range tests {
|
|
country := country
|
|
expected := expected
|
|
t.Run(country, func(t *testing.T) {
|
|
continents := LookupContinents(country)
|
|
if !assert.Equal(t, len(expected), len(continents), "Continents didn't match for %s: got %s, expected %s", country, continents, expected) {
|
|
return
|
|
}
|
|
for idx, c := range expected {
|
|
if !assert.Equal(t, c, continents[idx], "Continents didn't match for %s: got %s, expected %s", country, continents, expected) {
|
|
break
|
|
}
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestGeoLookupCloseEmpty(t *testing.T) {
|
|
CatchLogForTest(t)
|
|
reader, err := NewGeoLookupFromUrl("ignore-url")
|
|
require.NoError(t, err)
|
|
reader.Close()
|
|
}
|
|
|
|
func TestGeoLookupFromFile(t *testing.T) {
|
|
CatchLogForTest(t)
|
|
require := require.New(t)
|
|
geoIpUrl := GetGeoIpUrlForTest(t)
|
|
|
|
resp, err := http.Get(geoIpUrl)
|
|
require.NoError(err)
|
|
defer resp.Body.Close()
|
|
|
|
body := resp.Body
|
|
url := geoIpUrl
|
|
if strings.HasSuffix(geoIpUrl, ".gz") {
|
|
body, err = gzip.NewReader(body)
|
|
require.NoError(err)
|
|
url = strings.TrimSuffix(url, ".gz")
|
|
}
|
|
|
|
tmpfile, err := os.CreateTemp("", "geoipdb")
|
|
require.NoError(err)
|
|
t.Cleanup(func() {
|
|
os.Remove(tmpfile.Name())
|
|
})
|
|
|
|
foundDatabase := false
|
|
if strings.HasSuffix(url, ".tar") || strings.HasSuffix(url, "=tar") {
|
|
tarfile := tar.NewReader(body)
|
|
for {
|
|
header, err := tarfile.Next()
|
|
if err == io.EOF {
|
|
break
|
|
}
|
|
require.NoError(err)
|
|
|
|
if !strings.HasSuffix(header.Name, ".mmdb") {
|
|
continue
|
|
}
|
|
|
|
if _, err := io.Copy(tmpfile, tarfile); err != nil {
|
|
tmpfile.Close()
|
|
require.NoError(err)
|
|
}
|
|
require.NoError(tmpfile.Close())
|
|
foundDatabase = true
|
|
break
|
|
}
|
|
} else {
|
|
if _, err := io.Copy(tmpfile, body); err != nil {
|
|
tmpfile.Close()
|
|
require.NoError(err)
|
|
}
|
|
require.NoError(tmpfile.Close())
|
|
foundDatabase = true
|
|
}
|
|
|
|
require.True(foundDatabase, "Did not find GeoIP database in download from %s", geoIpUrl)
|
|
|
|
reader, err := NewGeoLookupFromFile(tmpfile.Name())
|
|
require.NoError(err)
|
|
defer reader.Close()
|
|
|
|
testGeoLookupReader(t, reader)
|
|
}
|
|
|
|
func TestIsValidContinent(t *testing.T) {
|
|
for country, continents := range ContinentMap {
|
|
for _, continent := range continents {
|
|
assert.True(t, IsValidContinent(continent), "Continent %s of country %s is not valid", continent, country)
|
|
}
|
|
}
|
|
}
|