a4cf473ee9
* chore: run go-generate See https://github.com/ooni/probe/issues/1468 * chore: update all the dependencies Unclear to me why `go get -u -v ./...` did not actually update all of them and I needed to spell out each of them and force to update by going `go get -u -v $pkg@latest` ¯\_(ツ)_/¯. Part of https://github.com/ooni/probe/issues/1468 * fix(c/o/i/d/actions_test.go): ensure we check for return value Part of https://github.com/ooni/probe/issues/1468 * chore: update the user agents we use See https://github.com/ooni/probe/issues/1468 * chore: set version to 3.10.0-beta.3 See https://github.com/ooni/probe/issues/1468 * chore: use probe-assets v0.3.1 Part of https://github.com/ooni/probe/issues/1468
389 lines
7.8 KiB
Go
389 lines
7.8 KiB
Go
package database
|
|
|
|
import (
|
|
"database/sql"
|
|
"encoding/json"
|
|
"fmt"
|
|
"io/ioutil"
|
|
"os"
|
|
"testing"
|
|
|
|
db "upper.io/db.v3"
|
|
)
|
|
|
|
type locationInfo struct {
|
|
asn uint
|
|
countryCode string
|
|
ip string
|
|
networkName string
|
|
resolverIP string
|
|
}
|
|
|
|
func (lp *locationInfo) ProbeASN() uint {
|
|
return lp.asn
|
|
}
|
|
|
|
func (lp *locationInfo) ProbeASNString() string {
|
|
return fmt.Sprintf("AS%d", lp.asn)
|
|
}
|
|
|
|
func (lp *locationInfo) ProbeCC() string {
|
|
return lp.countryCode
|
|
}
|
|
|
|
func (lp *locationInfo) ProbeIP() string {
|
|
return lp.ip
|
|
}
|
|
|
|
func (lp *locationInfo) ProbeNetworkName() string {
|
|
return lp.networkName
|
|
}
|
|
|
|
func (lp *locationInfo) ResolverIP() string {
|
|
return lp.resolverIP
|
|
}
|
|
|
|
func TestMeasurementWorkflow(t *testing.T) {
|
|
tmpfile, err := ioutil.TempFile("", "dbtest")
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer os.Remove(tmpfile.Name())
|
|
|
|
tmpdir, err := ioutil.TempDir("", "oonitest")
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer os.RemoveAll(tmpdir)
|
|
|
|
sess, err := Connect(tmpfile.Name())
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
location := locationInfo{
|
|
asn: 0,
|
|
countryCode: "IT",
|
|
networkName: "Unknown",
|
|
}
|
|
network, err := CreateNetwork(sess, &location)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
result, err := CreateResult(sess, tmpdir, "websites", network.ID)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
reportID := sql.NullString{String: "", Valid: false}
|
|
testName := "antani"
|
|
resultID := result.ID
|
|
msmtFilePath := tmpdir
|
|
urlID := sql.NullInt64{Int64: 0, Valid: false}
|
|
|
|
m1, err := CreateMeasurement(sess, reportID, testName, msmtFilePath, 0, resultID, urlID)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
m1.IsUploaded = true
|
|
err = sess.Collection("measurements").Find("measurement_id", m1.ID).Update(m1)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
var m2 Measurement
|
|
err = sess.Collection("measurements").Find("measurement_id", m1.ID).One(&m2)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
m2.IsUploaded = false
|
|
err = sess.Collection("measurements").Find("measurement_id", m2.ID).Update(m2)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
if m2.ResultID != m1.ResultID {
|
|
t.Error("result_id mismatch")
|
|
}
|
|
err = UpdateUploadedStatus(sess, result)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
var r Result
|
|
err = sess.Collection("measurements").Find("result_id", result.ID).One(&r)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if r.IsUploaded == true {
|
|
t.Error("result should be marked as not uploaded")
|
|
}
|
|
|
|
done, incomplete, err := ListResults(sess)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
if len(incomplete) != 1 {
|
|
t.Error("there should be 1 incomplete measurement")
|
|
}
|
|
if len(done) != 0 {
|
|
t.Error("there should be 0 done measurements")
|
|
}
|
|
|
|
msmts, err := ListMeasurements(sess, resultID)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if msmts[0].Network.NetworkType != "wifi" {
|
|
t.Error("network_type should be wifi")
|
|
}
|
|
}
|
|
|
|
func TestDeleteResult(t *testing.T) {
|
|
tmpfile, err := ioutil.TempFile("", "dbtest")
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer os.Remove(tmpfile.Name())
|
|
|
|
tmpdir, err := ioutil.TempDir("", "oonitest")
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer os.RemoveAll(tmpdir)
|
|
|
|
sess, err := Connect(tmpfile.Name())
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
location := locationInfo{
|
|
asn: 0,
|
|
countryCode: "IT",
|
|
networkName: "Unknown",
|
|
}
|
|
network, err := CreateNetwork(sess, &location)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
result, err := CreateResult(sess, tmpdir, "websites", network.ID)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
reportID := sql.NullString{String: "", Valid: false}
|
|
testName := "antani"
|
|
resultID := result.ID
|
|
msmtFilePath := tmpdir
|
|
urlID := sql.NullInt64{Int64: 0, Valid: false}
|
|
|
|
m1, err := CreateMeasurement(sess, reportID, testName, msmtFilePath, 0, resultID, urlID)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
var m2 Measurement
|
|
err = sess.Collection("measurements").Find("measurement_id", m1.ID).One(&m2)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
if m2.ResultID != m1.ResultID {
|
|
t.Error("result_id mismatch")
|
|
}
|
|
|
|
err = DeleteResult(sess, resultID)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
totalResults, err := sess.Collection("results").Find().Count()
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
totalMeasurements, err := sess.Collection("measurements").Find().Count()
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if totalResults != 0 {
|
|
t.Fatal("results should be zero")
|
|
}
|
|
if totalMeasurements != 0 {
|
|
t.Fatal("measurements should be zero")
|
|
}
|
|
|
|
err = DeleteResult(sess, 20)
|
|
if err != db.ErrNoMoreRows {
|
|
t.Fatal(err)
|
|
}
|
|
}
|
|
|
|
func TestNetworkCreate(t *testing.T) {
|
|
tmpfile, err := ioutil.TempFile("", "dbtest")
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer os.Remove(tmpfile.Name())
|
|
|
|
sess, err := Connect(tmpfile.Name())
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
l1 := locationInfo{
|
|
asn: 2,
|
|
countryCode: "IT",
|
|
networkName: "Antaninet",
|
|
}
|
|
|
|
l2 := locationInfo{
|
|
asn: 3,
|
|
countryCode: "IT",
|
|
networkName: "Fufnet",
|
|
}
|
|
|
|
_, err = CreateNetwork(sess, &l1)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
_, err = CreateNetwork(sess, &l2)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
}
|
|
|
|
func TestURLCreation(t *testing.T) {
|
|
tmpfile, err := ioutil.TempFile("", "dbtest")
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer os.Remove(tmpfile.Name())
|
|
|
|
sess, err := Connect(tmpfile.Name())
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
newID1, err := CreateOrUpdateURL(sess, "https://google.com", "GMB", "XX")
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
newID2, err := CreateOrUpdateURL(sess, "https://google.com", "SRCH", "XX")
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
newID3, err := CreateOrUpdateURL(sess, "https://facebook.com", "GRP", "XX")
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
newID4, err := CreateOrUpdateURL(sess, "https://facebook.com", "GMP", "XX")
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
newID5, err := CreateOrUpdateURL(sess, "https://google.com", "SRCH", "XX")
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
if newID2 != newID1 {
|
|
t.Error("inserting the same URL with different category code should produce the same result")
|
|
}
|
|
|
|
if newID3 == newID1 {
|
|
t.Error("inserting different URL should produce different ids")
|
|
}
|
|
|
|
if newID4 != newID3 {
|
|
t.Error("inserting the same URL with different category code should produce the same result")
|
|
}
|
|
|
|
if newID5 != newID1 {
|
|
t.Error("the ID of google should still be the same")
|
|
}
|
|
}
|
|
|
|
func TestPerformanceTestKeys(t *testing.T) {
|
|
var tk PerformanceTestKeys
|
|
|
|
ndtS := "{\"download\":100.0,\"upload\":20.0,\"ping\":2.2}"
|
|
dashS := "{\"median_bitrate\":102.0}"
|
|
if err := json.Unmarshal([]byte(ndtS), &tk); err != nil {
|
|
t.Fatal("failed to parse ndtS")
|
|
}
|
|
if err := json.Unmarshal([]byte(dashS), &tk); err != nil {
|
|
t.Fatal("failed to parse dashS")
|
|
}
|
|
if tk.Bitrate != 102.0 {
|
|
t.Fatalf("error Bitrate %f", tk.Bitrate)
|
|
}
|
|
if tk.Download != 100.0 {
|
|
t.Fatalf("error Download %f", tk.Download)
|
|
}
|
|
}
|
|
|
|
func TestGetMeasurementJSON(t *testing.T) {
|
|
tmpfile, err := ioutil.TempFile("", "dbtest")
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer os.Remove(tmpfile.Name())
|
|
|
|
tmpdir, err := ioutil.TempDir("", "oonitest")
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer os.RemoveAll(tmpdir)
|
|
|
|
sess, err := Connect(tmpfile.Name())
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
location := locationInfo{
|
|
asn: 0,
|
|
countryCode: "IT",
|
|
networkName: "Unknown",
|
|
}
|
|
network, err := CreateNetwork(sess, &location)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
result, err := CreateResult(sess, tmpdir, "websites", network.ID)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
reportID := sql.NullString{String: "20210111T085144Z_ndt_RU_3216_n1_qMVnP0PTX7ObUSmD", Valid: true}
|
|
testName := "antani"
|
|
resultID := result.ID
|
|
msmtFilePath := tmpdir
|
|
urlID := sql.NullInt64{Int64: 0, Valid: false}
|
|
|
|
msmt, err := CreateMeasurement(sess, reportID, testName, msmtFilePath, 0, resultID, urlID)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
msmt.IsUploaded = true
|
|
err = sess.Collection("measurements").Find("measurement_id", msmt.ID).Update(msmt)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
tk, err := GetMeasurementJSON(sess, msmt.ID)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if tk["probe_asn"] != "AS3216" {
|
|
t.Error("inconsistent measurement downloaded")
|
|
}
|
|
}
|