31e478b04e
* fix(pkg.go.dev): import a subpackage containing the assets We're trying to fix this issue that pkg.go.dev does not build. Thanks to @hellais for this very neat idea! Let's keep our fingers crossed and see whether it fixes! * feat: use embedded geoip databases Closes https://github.com/ooni/probe/issues/1372. Work done as part of https://github.com/ooni/probe/issues/1369. * fix(assetsx): add tests * feat: simplify and just vendor uncompressed DBs * remove tests that seems not necessary anymore * fix: run go mod tidy * Address https://github.com/ooni/probe-cli/pull/260/files#r605181364 * rewrite a test in a better way * fix: gently cleanup the legacy assetsdir Do not remove the whole directory with brute force. Just zap the files whose name we know. Then attempt to delete the legacy directory as well. If not empty, just fail. This is fine because it means the user has stored other files inside the directory. * fix: create .miniooni if missing
211 lines
5.8 KiB
Go
211 lines
5.8 KiB
Go
package engine
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"sync"
|
|
"testing"
|
|
|
|
"github.com/google/go-cmp/cmp"
|
|
"github.com/ooni/probe-cli/v3/internal/engine/geolocate"
|
|
"github.com/ooni/probe-cli/v3/internal/engine/model"
|
|
)
|
|
|
|
func (s *Session) GetAvailableProbeServices() []model.Service {
|
|
return s.getAvailableProbeServicesUnlocked()
|
|
}
|
|
|
|
func (s *Session) AppendAvailableProbeService(svc model.Service) {
|
|
s.availableProbeServices = append(s.availableProbeServices, svc)
|
|
}
|
|
|
|
func (s *Session) QueryProbeServicesCount() int64 {
|
|
return s.queryProbeServicesCount.Load()
|
|
}
|
|
|
|
// mockableProbeServicesClientForCheckIn allows us to mock the
|
|
// probeservices.Client used by Session.CheckIn.
|
|
type mockableProbeServicesClientForCheckIn struct {
|
|
// Config is the config passed to the call.
|
|
Config *model.CheckInConfig
|
|
|
|
// Results contains the results of the call. This field MUST be
|
|
// non-nil if and only if Error is nil.
|
|
Results *model.CheckInInfo
|
|
|
|
// Error indicates whether the call failed. This field MUST be
|
|
// non-nil if and only if Error is nil.
|
|
Error error
|
|
|
|
// mu provides mutual exclusion.
|
|
mu sync.Mutex
|
|
}
|
|
|
|
// CheckIn implements sessionProbeServicesClientForCheckIn.CheckIn.
|
|
func (c *mockableProbeServicesClientForCheckIn) CheckIn(
|
|
ctx context.Context, config model.CheckInConfig) (*model.CheckInInfo, error) {
|
|
defer c.mu.Unlock()
|
|
c.mu.Lock()
|
|
if c.Config != nil {
|
|
return nil, errors.New("called more than once")
|
|
}
|
|
c.Config = &config
|
|
if c.Results == nil && c.Error == nil {
|
|
return nil, errors.New("misconfigured mockableProbeServicesClientForCheckIn")
|
|
}
|
|
return c.Results, c.Error
|
|
}
|
|
|
|
func TestSessionCheckInSuccessful(t *testing.T) {
|
|
results := &model.CheckInInfo{
|
|
WebConnectivity: &model.CheckInInfoWebConnectivity{
|
|
ReportID: "xxx-x-xx",
|
|
URLs: []model.URLInfo{{
|
|
CategoryCode: "NEWS",
|
|
CountryCode: "IT",
|
|
URL: "https://www.repubblica.it/",
|
|
}, {
|
|
CategoryCode: "NEWS",
|
|
CountryCode: "IT",
|
|
URL: "https://www.unita.it/",
|
|
}},
|
|
},
|
|
}
|
|
mockedClnt := &mockableProbeServicesClientForCheckIn{
|
|
Results: results,
|
|
}
|
|
s := &Session{
|
|
location: &geolocate.Results{
|
|
ASN: 137,
|
|
CountryCode: "IT",
|
|
},
|
|
softwareName: "miniooni",
|
|
softwareVersion: "0.1.0-dev",
|
|
testMaybeLookupLocationContext: func(ctx context.Context) error {
|
|
return nil
|
|
},
|
|
testNewProbeServicesClientForCheckIn: func(
|
|
ctx context.Context) (sessionProbeServicesClientForCheckIn, error) {
|
|
return mockedClnt, nil
|
|
},
|
|
}
|
|
out, err := s.CheckIn(context.Background(), &model.CheckInConfig{})
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if diff := cmp.Diff(results, out); diff != "" {
|
|
t.Fatal(diff)
|
|
}
|
|
if mockedClnt.Config.Platform != s.Platform() {
|
|
t.Fatal("invalid Config.Platform")
|
|
}
|
|
if mockedClnt.Config.ProbeASN != "AS137" {
|
|
t.Fatal("invalid Config.ProbeASN")
|
|
}
|
|
if mockedClnt.Config.ProbeCC != "IT" {
|
|
t.Fatal("invalid Config.ProbeCC")
|
|
}
|
|
if mockedClnt.Config.RunType != "timed" {
|
|
t.Fatal("invalid Config.RunType")
|
|
}
|
|
if mockedClnt.Config.SoftwareName != "miniooni" {
|
|
t.Fatal("invalid Config.SoftwareName")
|
|
}
|
|
if mockedClnt.Config.SoftwareVersion != "0.1.0-dev" {
|
|
t.Fatal("invalid Config.SoftwareVersion")
|
|
}
|
|
if mockedClnt.Config.WebConnectivity.CategoryCodes == nil {
|
|
t.Fatal("invalid ...CategoryCodes")
|
|
}
|
|
}
|
|
|
|
func TestSessionCheckInCannotLookupLocation(t *testing.T) {
|
|
errMocked := errors.New("mocked error")
|
|
s := &Session{
|
|
testMaybeLookupLocationContext: func(ctx context.Context) error {
|
|
return errMocked
|
|
},
|
|
}
|
|
out, err := s.CheckIn(context.Background(), &model.CheckInConfig{})
|
|
if !errors.Is(err, errMocked) {
|
|
t.Fatal("no the error we expected", err)
|
|
}
|
|
if out != nil {
|
|
t.Fatal("expected nil result here")
|
|
}
|
|
}
|
|
|
|
func TestSessionCheckInCannotCreateProbeServicesClient(t *testing.T) {
|
|
errMocked := errors.New("mocked error")
|
|
s := &Session{
|
|
location: &geolocate.Results{
|
|
ASN: 137,
|
|
CountryCode: "IT",
|
|
},
|
|
softwareName: "miniooni",
|
|
softwareVersion: "0.1.0-dev",
|
|
testMaybeLookupLocationContext: func(ctx context.Context) error {
|
|
return nil
|
|
},
|
|
testNewProbeServicesClientForCheckIn: func(
|
|
ctx context.Context) (sessionProbeServicesClientForCheckIn, error) {
|
|
return nil, errMocked
|
|
},
|
|
}
|
|
out, err := s.CheckIn(context.Background(), &model.CheckInConfig{})
|
|
if !errors.Is(err, errMocked) {
|
|
t.Fatal("no the error we expected", err)
|
|
}
|
|
if out != nil {
|
|
t.Fatal("expected nil result here")
|
|
}
|
|
}
|
|
|
|
func TestLowercaseMaybeLookupLocationContextWithCancelledContext(t *testing.T) {
|
|
s := &Session{}
|
|
ctx, cancel := context.WithCancel(context.Background())
|
|
cancel() // immediately kill the context
|
|
err := s.maybeLookupLocationContext(ctx)
|
|
if !errors.Is(err, context.Canceled) {
|
|
t.Fatal("not the error we expected", err)
|
|
}
|
|
}
|
|
|
|
func TestNewProbeServicesClientForCheckIn(t *testing.T) {
|
|
s := &Session{}
|
|
ctx, cancel := context.WithCancel(context.Background())
|
|
cancel() // immediately kill the context
|
|
clnt, err := s.newProbeServicesClientForCheckIn(ctx)
|
|
if !errors.Is(err, context.Canceled) {
|
|
t.Fatal("not the error we expected", err)
|
|
}
|
|
if clnt != nil {
|
|
t.Fatal("expected nil client here")
|
|
}
|
|
}
|
|
|
|
func TestSessionNewSubmitterWithCancelledContext(t *testing.T) {
|
|
sess := newSessionForTesting(t)
|
|
ctx, cancel := context.WithCancel(context.Background())
|
|
cancel() // fail immediately
|
|
subm, err := sess.NewSubmitter(ctx)
|
|
if !errors.Is(err, context.Canceled) {
|
|
t.Fatal("not the error we expected", err)
|
|
}
|
|
if subm != nil {
|
|
t.Fatal("expected nil submitter here")
|
|
}
|
|
}
|
|
|
|
func TestSessionMaybeLookupLocationContextLookupLocationContextFailure(t *testing.T) {
|
|
errMocked := errors.New("mocked error")
|
|
sess := newSessionForTestingNoLookups(t)
|
|
sess.testLookupLocationContext = func(ctx context.Context) (*geolocate.Results, error) {
|
|
return nil, errMocked
|
|
}
|
|
err := sess.MaybeLookupLocationContext(context.Background())
|
|
if !errors.Is(err, errMocked) {
|
|
t.Fatal("not the error we expected", err)
|
|
}
|
|
}
|