2021-02-02 12:05:47 +01:00
|
|
|
package engine
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"encoding/json"
|
|
|
|
"errors"
|
|
|
|
"io/ioutil"
|
|
|
|
"net/http"
|
|
|
|
"net/http/httptest"
|
|
|
|
"os"
|
|
|
|
"path/filepath"
|
|
|
|
"strings"
|
|
|
|
"testing"
|
|
|
|
|
2022-01-03 13:53:23 +01:00
|
|
|
"github.com/ooni/probe-cli/v3/internal/model"
|
2021-02-02 12:05:47 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
func TestCreateAll(t *testing.T) {
|
|
|
|
if testing.Short() {
|
|
|
|
t.Skip("skip test in short mode")
|
|
|
|
}
|
|
|
|
sess := newSessionForTesting(t)
|
|
|
|
defer sess.Close()
|
|
|
|
for _, name := range AllExperiments() {
|
|
|
|
builder, err := sess.NewExperimentBuilder(name)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
exp := builder.NewExperiment()
|
|
|
|
good := (exp.Name() == name)
|
|
|
|
if !good {
|
2022-08-27 15:47:48 +02:00
|
|
|
// We have introduced the concept of versioned experiments in
|
|
|
|
// https://github.com/ooni/probe-cli/pull/882. This works like
|
|
|
|
// in brew: we append @vX.Y to the experiment name. So, here
|
|
|
|
// we're stripping the version specification and retry.
|
|
|
|
index := strings.Index(name, "@")
|
|
|
|
if index >= 0 {
|
|
|
|
name = name[:index]
|
|
|
|
if good := (exp.Name() == name); good {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
}
|
|
|
|
t.Fatal("unexpected experiment name", exp.Name(), name)
|
2021-02-02 12:05:47 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestRunDASH(t *testing.T) {
|
|
|
|
if testing.Short() {
|
|
|
|
t.Skip("skip test in short mode")
|
|
|
|
}
|
|
|
|
sess := newSessionForTesting(t)
|
|
|
|
defer sess.Close()
|
|
|
|
builder, err := sess.NewExperimentBuilder("dash")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if !builder.Interruptible() {
|
|
|
|
t.Fatal("dash not marked as interruptible")
|
|
|
|
}
|
|
|
|
runexperimentflow(t, builder.NewExperiment(), "")
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestRunExample(t *testing.T) {
|
|
|
|
if testing.Short() {
|
|
|
|
t.Skip("skip test in short mode")
|
|
|
|
}
|
|
|
|
sess := newSessionForTesting(t)
|
|
|
|
defer sess.Close()
|
|
|
|
builder, err := sess.NewExperimentBuilder("example")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
runexperimentflow(t, builder.NewExperiment(), "")
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestRunNdt7(t *testing.T) {
|
|
|
|
if testing.Short() {
|
|
|
|
t.Skip("skip test in short mode")
|
|
|
|
}
|
|
|
|
sess := newSessionForTesting(t)
|
|
|
|
defer sess.Close()
|
|
|
|
builder, err := sess.NewExperimentBuilder("ndt7")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if !builder.Interruptible() {
|
|
|
|
t.Fatal("ndt7 not marked as interruptible")
|
|
|
|
}
|
|
|
|
runexperimentflow(t, builder.NewExperiment(), "")
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestRunPsiphon(t *testing.T) {
|
|
|
|
if testing.Short() {
|
|
|
|
t.Skip("skip test in short mode")
|
|
|
|
}
|
|
|
|
sess := newSessionForTesting(t)
|
|
|
|
defer sess.Close()
|
|
|
|
builder, err := sess.NewExperimentBuilder("psiphon")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
runexperimentflow(t, builder.NewExperiment(), "")
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestRunSNIBlocking(t *testing.T) {
|
|
|
|
if testing.Short() {
|
|
|
|
t.Skip("skip test in short mode")
|
|
|
|
}
|
|
|
|
sess := newSessionForTesting(t)
|
|
|
|
defer sess.Close()
|
|
|
|
builder, err := sess.NewExperimentBuilder("sni_blocking")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
runexperimentflow(t, builder.NewExperiment(), "kernel.org")
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestRunTelegram(t *testing.T) {
|
|
|
|
if testing.Short() {
|
|
|
|
t.Skip("skip test in short mode")
|
|
|
|
}
|
|
|
|
sess := newSessionForTesting(t)
|
|
|
|
defer sess.Close()
|
|
|
|
builder, err := sess.NewExperimentBuilder("telegram")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
runexperimentflow(t, builder.NewExperiment(), "")
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestRunTor(t *testing.T) {
|
|
|
|
if testing.Short() {
|
|
|
|
t.Skip("skip test in short mode")
|
|
|
|
}
|
|
|
|
sess := newSessionForTesting(t)
|
|
|
|
defer sess.Close()
|
|
|
|
builder, err := sess.NewExperimentBuilder("tor")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
runexperimentflow(t, builder.NewExperiment(), "")
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestNeedsInput(t *testing.T) {
|
|
|
|
if testing.Short() {
|
|
|
|
t.Skip("skip test in short mode")
|
|
|
|
}
|
|
|
|
sess := newSessionForTesting(t)
|
|
|
|
defer sess.Close()
|
|
|
|
builder, err := sess.NewExperimentBuilder("web_connectivity")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2022-08-17 10:57:03 +02:00
|
|
|
if builder.InputPolicy() != model.InputOrQueryBackend {
|
2021-02-02 12:05:47 +01:00
|
|
|
t.Fatal("web_connectivity certainly needs input")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestSetCallbacks(t *testing.T) {
|
|
|
|
if testing.Short() {
|
|
|
|
t.Skip("skip test in short mode")
|
|
|
|
}
|
|
|
|
sess := newSessionForTesting(t)
|
|
|
|
defer sess.Close()
|
|
|
|
builder, err := sess.NewExperimentBuilder("example")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2022-07-08 11:51:59 +02:00
|
|
|
if err := builder.SetOptionAny("SleepTime", 0); err != nil {
|
2021-02-02 12:05:47 +01:00
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
register := ®isterCallbacksCalled{}
|
|
|
|
builder.SetCallbacks(register)
|
2022-07-08 12:29:23 +02:00
|
|
|
if _, err := builder.NewExperiment().MeasureWithContext(context.Background(), ""); err != nil {
|
2021-02-02 12:05:47 +01:00
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if register.onProgressCalled == false {
|
|
|
|
t.Fatal("OnProgress not called")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
type registerCallbacksCalled struct {
|
|
|
|
onProgressCalled bool
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *registerCallbacksCalled) OnProgress(percentage float64, message string) {
|
|
|
|
c.onProgressCalled = true
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestCreateInvalidExperiment(t *testing.T) {
|
|
|
|
if testing.Short() {
|
|
|
|
t.Skip("skip test in short mode")
|
|
|
|
}
|
|
|
|
sess := newSessionForTesting(t)
|
|
|
|
defer sess.Close()
|
|
|
|
builder, err := sess.NewExperimentBuilder("antani")
|
|
|
|
if err == nil {
|
|
|
|
t.Fatal("expected an error here")
|
|
|
|
}
|
|
|
|
if builder != nil {
|
|
|
|
t.Fatal("expected a nil builder here")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestMeasurementFailure(t *testing.T) {
|
|
|
|
if testing.Short() {
|
|
|
|
t.Skip("skip test in short mode")
|
|
|
|
}
|
|
|
|
sess := newSessionForTesting(t)
|
|
|
|
defer sess.Close()
|
|
|
|
builder, err := sess.NewExperimentBuilder("example")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2022-07-08 11:51:59 +02:00
|
|
|
if err := builder.SetOptionAny("ReturnError", true); err != nil {
|
2021-02-02 12:05:47 +01:00
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2022-07-08 12:29:23 +02:00
|
|
|
measurement, err := builder.NewExperiment().MeasureWithContext(context.Background(), "")
|
2021-02-02 12:05:47 +01:00
|
|
|
if err == nil {
|
|
|
|
t.Fatal("expected an error here")
|
|
|
|
}
|
|
|
|
if err.Error() != "mocked error" {
|
|
|
|
t.Fatal("unexpected error type")
|
|
|
|
}
|
2021-09-30 00:54:52 +02:00
|
|
|
if measurement != nil {
|
|
|
|
t.Fatal("expected nil measurement here")
|
2021-02-02 12:05:47 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestRunHHFM(t *testing.T) {
|
|
|
|
if testing.Short() {
|
|
|
|
t.Skip("skip test in short mode")
|
|
|
|
}
|
|
|
|
sess := newSessionForTesting(t)
|
|
|
|
defer sess.Close()
|
|
|
|
builder, err := sess.NewExperimentBuilder("http_header_field_manipulation")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
runexperimentflow(t, builder.NewExperiment(), "")
|
|
|
|
}
|
|
|
|
|
2022-08-17 10:57:03 +02:00
|
|
|
func runexperimentflow(t *testing.T, experiment model.Experiment, input string) {
|
2022-07-08 12:29:23 +02:00
|
|
|
ctx := context.Background()
|
|
|
|
err := experiment.OpenReportContext(ctx)
|
2021-02-02 12:05:47 +01:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if experiment.ReportID() == "" {
|
|
|
|
t.Fatal("reportID should not be empty here")
|
|
|
|
}
|
2022-07-08 12:29:23 +02:00
|
|
|
measurement, err := experiment.MeasureWithContext(ctx, input)
|
2021-02-02 12:05:47 +01:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
measurement.AddAnnotations(map[string]string{
|
|
|
|
"probe-engine-ci": "yes",
|
|
|
|
})
|
|
|
|
data, err := json.Marshal(measurement)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if data == nil {
|
|
|
|
t.Fatal("data is nil")
|
|
|
|
}
|
2022-07-08 12:29:23 +02:00
|
|
|
err = experiment.SubmitAndUpdateMeasurementContext(ctx, measurement)
|
2021-02-02 12:05:47 +01:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
err = experiment.SaveMeasurement(measurement, "/tmp/experiment.jsonl")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if experiment.KibiBytesSent() <= 0 {
|
|
|
|
t.Fatal("no data sent?!")
|
|
|
|
}
|
|
|
|
if experiment.KibiBytesReceived() <= 0 {
|
|
|
|
t.Fatal("no data received?!")
|
|
|
|
}
|
|
|
|
if _, err := experiment.GetSummaryKeys(measurement); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestSaveMeasurementErrors(t *testing.T) {
|
|
|
|
if testing.Short() {
|
|
|
|
t.Skip("skip test in short mode")
|
|
|
|
}
|
|
|
|
sess := newSessionForTesting(t)
|
|
|
|
defer sess.Close()
|
|
|
|
builder, err := sess.NewExperimentBuilder("example")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2022-07-08 12:29:23 +02:00
|
|
|
exp := builder.NewExperiment().(*experiment)
|
2021-02-02 12:05:47 +01:00
|
|
|
dirname, err := ioutil.TempDir("", "ooniprobe-engine-save-measurement")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
filename := filepath.Join(dirname, "report.jsonl")
|
|
|
|
m := new(model.Measurement)
|
2022-07-08 12:29:23 +02:00
|
|
|
err = exp.saveMeasurement(
|
2021-02-02 12:05:47 +01:00
|
|
|
m, filename, func(v interface{}) ([]byte, error) {
|
|
|
|
return nil, errors.New("mocked error")
|
|
|
|
}, os.OpenFile, func(fp *os.File, b []byte) (int, error) {
|
|
|
|
return fp.Write(b)
|
|
|
|
},
|
|
|
|
)
|
|
|
|
if err == nil {
|
|
|
|
t.Fatal("expected an error here")
|
|
|
|
}
|
2022-07-08 12:29:23 +02:00
|
|
|
err = exp.saveMeasurement(
|
2021-02-02 12:05:47 +01:00
|
|
|
m, filename, json.Marshal,
|
|
|
|
func(name string, flag int, perm os.FileMode) (*os.File, error) {
|
|
|
|
return nil, errors.New("mocked error")
|
|
|
|
}, func(fp *os.File, b []byte) (int, error) {
|
|
|
|
return fp.Write(b)
|
|
|
|
},
|
|
|
|
)
|
|
|
|
if err == nil {
|
|
|
|
t.Fatal("expected an error here")
|
|
|
|
}
|
2022-07-08 12:29:23 +02:00
|
|
|
err = exp.saveMeasurement(
|
2021-02-02 12:05:47 +01:00
|
|
|
m, filename, json.Marshal, os.OpenFile,
|
|
|
|
func(fp *os.File, b []byte) (int, error) {
|
|
|
|
return 0, errors.New("mocked error")
|
|
|
|
},
|
|
|
|
)
|
|
|
|
if err == nil {
|
|
|
|
t.Fatal("expected an error here")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestOpenReportIdempotent(t *testing.T) {
|
|
|
|
if testing.Short() {
|
|
|
|
t.Skip("skip test in short mode")
|
|
|
|
}
|
|
|
|
sess := newSessionForTesting(t)
|
|
|
|
defer sess.Close()
|
|
|
|
builder, err := sess.NewExperimentBuilder("example")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
exp := builder.NewExperiment()
|
|
|
|
if exp.ReportID() != "" {
|
|
|
|
t.Fatal("unexpected initial report ID")
|
|
|
|
}
|
2022-07-08 12:29:23 +02:00
|
|
|
ctx := context.Background()
|
|
|
|
if err := exp.SubmitAndUpdateMeasurementContext(ctx, &model.Measurement{}); err == nil {
|
2021-02-02 12:05:47 +01:00
|
|
|
t.Fatal("we should not be able to submit before OpenReport")
|
|
|
|
}
|
2022-07-08 12:29:23 +02:00
|
|
|
err = exp.OpenReportContext(ctx)
|
2021-02-02 12:05:47 +01:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
rid := exp.ReportID()
|
|
|
|
if rid == "" {
|
|
|
|
t.Fatal("invalid report ID")
|
|
|
|
}
|
2022-07-08 12:29:23 +02:00
|
|
|
err = exp.OpenReportContext(ctx)
|
2021-02-02 12:05:47 +01:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if rid != exp.ReportID() {
|
|
|
|
t.Fatal("OpenReport is not idempotent")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestOpenReportFailure(t *testing.T) {
|
|
|
|
if testing.Short() {
|
|
|
|
t.Skip("skip test in short mode")
|
|
|
|
}
|
|
|
|
server := httptest.NewServer(http.HandlerFunc(
|
|
|
|
func(w http.ResponseWriter, r *http.Request) {
|
|
|
|
w.WriteHeader(500)
|
|
|
|
},
|
|
|
|
))
|
|
|
|
defer server.Close()
|
|
|
|
sess := newSessionForTestingNoBackendsLookup(t)
|
|
|
|
defer sess.Close()
|
|
|
|
builder, err := sess.NewExperimentBuilder("example")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2022-07-08 12:29:23 +02:00
|
|
|
exp := builder.NewExperiment().(*experiment)
|
2022-01-03 13:53:23 +01:00
|
|
|
exp.session.selectedProbeService = &model.OOAPIService{
|
2021-02-02 12:05:47 +01:00
|
|
|
Address: server.URL,
|
|
|
|
Type: "https",
|
|
|
|
}
|
2022-07-08 12:29:23 +02:00
|
|
|
err = exp.OpenReportContext(context.Background())
|
2021-02-02 12:05:47 +01:00
|
|
|
if !strings.HasPrefix(err.Error(), "httpx: request failed") {
|
|
|
|
t.Fatal("not the error we expected")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestOpenReportNewClientFailure(t *testing.T) {
|
|
|
|
if testing.Short() {
|
|
|
|
t.Skip("skip test in short mode")
|
|
|
|
}
|
|
|
|
sess := newSessionForTestingNoBackendsLookup(t)
|
|
|
|
defer sess.Close()
|
|
|
|
builder, err := sess.NewExperimentBuilder("example")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2022-07-08 12:29:23 +02:00
|
|
|
exp := builder.NewExperiment().(*experiment)
|
2022-01-03 13:53:23 +01:00
|
|
|
exp.session.selectedProbeService = &model.OOAPIService{
|
2021-02-02 12:05:47 +01:00
|
|
|
Address: "antani:///",
|
|
|
|
Type: "antani",
|
|
|
|
}
|
2022-07-08 12:29:23 +02:00
|
|
|
err = exp.OpenReportContext(context.Background())
|
2021-02-02 12:05:47 +01:00
|
|
|
if err.Error() != "probe services: unsupported endpoint type" {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestSubmitAndUpdateMeasurementWithClosedReport(t *testing.T) {
|
|
|
|
if testing.Short() {
|
|
|
|
t.Skip("skip test in short mode")
|
|
|
|
}
|
|
|
|
sess := newSessionForTesting(t)
|
|
|
|
defer sess.Close()
|
|
|
|
builder, err := sess.NewExperimentBuilder("example")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
exp := builder.NewExperiment()
|
|
|
|
m := new(model.Measurement)
|
2022-07-08 12:29:23 +02:00
|
|
|
err = exp.SubmitAndUpdateMeasurementContext(context.Background(), m)
|
2021-02-02 12:05:47 +01:00
|
|
|
if err == nil {
|
|
|
|
t.Fatal("expected an error here")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestMeasureLookupLocationFailure(t *testing.T) {
|
|
|
|
if testing.Short() {
|
|
|
|
t.Skip("skip test in short mode")
|
|
|
|
}
|
|
|
|
sess := newSessionForTestingNoLookups(t)
|
|
|
|
defer sess.Close()
|
2022-07-08 12:29:23 +02:00
|
|
|
exp := newExperiment(sess, new(antaniMeasurer))
|
2021-02-02 12:05:47 +01:00
|
|
|
ctx, cancel := context.WithCancel(context.Background())
|
|
|
|
cancel() // so we fail immediately
|
|
|
|
if _, err := exp.MeasureWithContext(ctx, "xx"); err == nil {
|
|
|
|
t.Fatal("expected an error here")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestOpenReportNonHTTPS(t *testing.T) {
|
|
|
|
if testing.Short() {
|
|
|
|
t.Skip("skip test in short mode")
|
|
|
|
}
|
|
|
|
sess := newSessionForTestingNoLookups(t)
|
|
|
|
defer sess.Close()
|
2022-01-03 13:53:23 +01:00
|
|
|
sess.availableProbeServices = []model.OOAPIService{
|
2021-02-02 12:05:47 +01:00
|
|
|
{
|
|
|
|
Address: "antani",
|
|
|
|
Type: "mascetti",
|
|
|
|
},
|
|
|
|
}
|
2022-07-08 12:29:23 +02:00
|
|
|
exp := newExperiment(sess, new(antaniMeasurer))
|
|
|
|
if err := exp.OpenReportContext(context.Background()); err == nil {
|
2021-02-02 12:05:47 +01:00
|
|
|
t.Fatal("expected an error here")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
type antaniMeasurer struct{}
|
|
|
|
|
|
|
|
func (am *antaniMeasurer) ExperimentName() string {
|
|
|
|
return "antani"
|
|
|
|
}
|
|
|
|
|
|
|
|
func (am *antaniMeasurer) ExperimentVersion() string {
|
|
|
|
return "0.1.1"
|
|
|
|
}
|
|
|
|
|
2022-11-22 10:43:47 +01:00
|
|
|
func (am *antaniMeasurer) Run(ctx context.Context, args *model.ExperimentArgs) error {
|
2021-02-02 12:05:47 +01:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (am *antaniMeasurer) GetSummaryKeys(m *model.Measurement) (interface{}, error) {
|
|
|
|
return struct {
|
|
|
|
Failure *string `json:"failure"`
|
|
|
|
}{}, nil
|
|
|
|
}
|