182 lines
6.1 KiB
Go
182 lines
6.1 KiB
Go
// Package telegram contains the Telegram network experiment.
|
|
//
|
|
// See https://github.com/ooni/spec/blob/master/nettests/ts-020-telegram.md.
|
|
package telegram
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/ooni/probe-cli/v3/internal/engine/experiment/urlgetter"
|
|
"github.com/ooni/probe-cli/v3/internal/model"
|
|
"github.com/ooni/probe-cli/v3/internal/netxlite"
|
|
)
|
|
|
|
const (
|
|
testName = "telegram"
|
|
testVersion = "0.2.0"
|
|
)
|
|
|
|
// Config contains the telegram experiment config.
|
|
type Config struct{}
|
|
|
|
// TestKeys contains telegram test keys.
|
|
type TestKeys struct {
|
|
urlgetter.TestKeys
|
|
TelegramHTTPBlocking bool `json:"telegram_http_blocking"`
|
|
TelegramTCPBlocking bool `json:"telegram_tcp_blocking"`
|
|
TelegramWebFailure *string `json:"telegram_web_failure"`
|
|
TelegramWebStatus string `json:"telegram_web_status"`
|
|
}
|
|
|
|
// NewTestKeys creates new telegram TestKeys.
|
|
func NewTestKeys() *TestKeys {
|
|
return &TestKeys{
|
|
TelegramHTTPBlocking: true,
|
|
TelegramTCPBlocking: true,
|
|
TelegramWebFailure: nil,
|
|
TelegramWebStatus: "ok",
|
|
}
|
|
}
|
|
|
|
// Update updates the TestKeys using the given MultiOutput result.
|
|
func (tk *TestKeys) Update(v urlgetter.MultiOutput) {
|
|
// update the easy to update entries first
|
|
tk.NetworkEvents = append(tk.NetworkEvents, v.TestKeys.NetworkEvents...)
|
|
tk.Queries = append(tk.Queries, v.TestKeys.Queries...)
|
|
tk.Requests = append(tk.Requests, v.TestKeys.Requests...)
|
|
tk.TCPConnect = append(tk.TCPConnect, v.TestKeys.TCPConnect...)
|
|
tk.TLSHandshakes = append(tk.TLSHandshakes, v.TestKeys.TLSHandshakes...)
|
|
// then process access points
|
|
if v.Input.Config.Method != "GET" {
|
|
if v.TestKeys.Failure == nil {
|
|
tk.TelegramHTTPBlocking = false
|
|
tk.TelegramTCPBlocking = false
|
|
return // found successful access point connection
|
|
}
|
|
if v.TestKeys.FailedOperation == nil || *v.TestKeys.FailedOperation != netxlite.ConnectOperation {
|
|
tk.TelegramTCPBlocking = false
|
|
}
|
|
return
|
|
}
|
|
// now take care of web
|
|
if tk.TelegramWebStatus != "ok" {
|
|
return // we already flipped the state
|
|
}
|
|
if v.TestKeys.Failure != nil {
|
|
tk.TelegramWebStatus = "blocked"
|
|
tk.TelegramWebFailure = v.TestKeys.Failure
|
|
return
|
|
}
|
|
title := `<title>Telegram Web</title>`
|
|
if strings.Contains(v.TestKeys.HTTPResponseBody, title) == false {
|
|
failureString := "telegram_missing_title_error"
|
|
tk.TelegramWebFailure = &failureString
|
|
tk.TelegramWebStatus = "blocked"
|
|
return
|
|
}
|
|
return
|
|
}
|
|
|
|
// Measurer performs the measurement
|
|
type Measurer struct {
|
|
// Config contains the experiment settings. If empty we
|
|
// will be using default settings.
|
|
Config Config
|
|
|
|
// Getter is an optional getter to be used for testing.
|
|
Getter urlgetter.MultiGetter
|
|
}
|
|
|
|
// ExperimentName implements ExperimentMeasurer.ExperimentName
|
|
func (m Measurer) ExperimentName() string {
|
|
return testName
|
|
}
|
|
|
|
// ExperimentVersion implements ExperimentMeasurer.ExperimentVersion
|
|
func (m Measurer) ExperimentVersion() string {
|
|
return testVersion
|
|
}
|
|
|
|
// Run implements ExperimentMeasurer.Run
|
|
func (m Measurer) Run(ctx context.Context, args *model.ExperimentArgs) error {
|
|
callbacks := args.Callbacks
|
|
measurement := args.Measurement
|
|
sess := args.Session
|
|
|
|
ctx, cancel := context.WithTimeout(ctx, 60*time.Second)
|
|
defer cancel()
|
|
urlgetter.RegisterExtensions(measurement)
|
|
inputs := []urlgetter.MultiInput{
|
|
{Target: "http://149.154.175.50/", Config: urlgetter.Config{Method: "POST"}},
|
|
{Target: "http://149.154.167.51/", Config: urlgetter.Config{Method: "POST"}},
|
|
{Target: "http://149.154.175.100/", Config: urlgetter.Config{Method: "POST"}},
|
|
{Target: "http://149.154.167.91/", Config: urlgetter.Config{Method: "POST"}},
|
|
{Target: "http://149.154.171.5/", Config: urlgetter.Config{Method: "POST"}},
|
|
{Target: "http://95.161.76.100/", Config: urlgetter.Config{Method: "POST"}},
|
|
|
|
// Note: the following list contains the same endpoints as above with HTTP (not a typo using
|
|
// https would not work here) _and_ port 443.
|
|
{Target: "http://149.154.175.50:443/", Config: urlgetter.Config{Method: "POST"}},
|
|
{Target: "http://149.154.167.51:443/", Config: urlgetter.Config{Method: "POST"}},
|
|
{Target: "http://149.154.175.100:443/", Config: urlgetter.Config{Method: "POST"}},
|
|
{Target: "http://149.154.167.91:443/", Config: urlgetter.Config{Method: "POST"}},
|
|
{Target: "http://149.154.171.5:443/", Config: urlgetter.Config{Method: "POST"}},
|
|
{Target: "http://95.161.76.100:443/", Config: urlgetter.Config{Method: "POST"}},
|
|
|
|
// Here we need to provide the method explicitly. See
|
|
// https://github.com/ooni/probe-engine/issues/827.
|
|
{Target: "http://web.telegram.org/", Config: urlgetter.Config{
|
|
Method: "GET",
|
|
FailOnHTTPError: true,
|
|
}},
|
|
{Target: "https://web.telegram.org/", Config: urlgetter.Config{
|
|
Method: "GET",
|
|
FailOnHTTPError: true,
|
|
}},
|
|
}
|
|
multi := urlgetter.Multi{Begin: time.Now(), Getter: m.Getter, Session: sess}
|
|
testkeys := NewTestKeys()
|
|
testkeys.Agent = "redirect"
|
|
measurement.TestKeys = testkeys
|
|
for entry := range multi.Collect(ctx, inputs, "telegram", callbacks) {
|
|
testkeys.Update(entry)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// NewExperimentMeasurer creates a new ExperimentMeasurer.
|
|
func NewExperimentMeasurer(config Config) model.ExperimentMeasurer {
|
|
return Measurer{Config: config}
|
|
}
|
|
|
|
// SummaryKeys contains summary keys for this experiment.
|
|
//
|
|
// Note that this structure is part of the ABI contract with ooniprobe
|
|
// therefore we should be careful when changing it.
|
|
type SummaryKeys struct {
|
|
HTTPBlocking bool `json:"telegram_http_blocking"`
|
|
TCPBlocking bool `json:"telegram_tcp_blocking"`
|
|
WebBlocking bool `json:"telegram_web_blocking"`
|
|
IsAnomaly bool `json:"-"`
|
|
}
|
|
|
|
// GetSummaryKeys implements model.ExperimentMeasurer.GetSummaryKeys.
|
|
func (m Measurer) GetSummaryKeys(measurement *model.Measurement) (interface{}, error) {
|
|
sk := SummaryKeys{IsAnomaly: false}
|
|
tk, ok := measurement.TestKeys.(*TestKeys)
|
|
if !ok {
|
|
return sk, errors.New("invalid test keys type")
|
|
}
|
|
tcpBlocking := tk.TelegramTCPBlocking
|
|
httpBlocking := tk.TelegramHTTPBlocking
|
|
webBlocking := tk.TelegramWebFailure != nil
|
|
sk.TCPBlocking = tcpBlocking
|
|
sk.HTTPBlocking = httpBlocking
|
|
sk.WebBlocking = webBlocking
|
|
sk.IsAnomaly = webBlocking || httpBlocking || tcpBlocking
|
|
return sk, nil
|
|
}
|