[forwardport] fix(qa): adapt to new wcth (#691)

This diff forward ports 36ba3630c9002db0bd79e3a7e49641ce6b665471,
whose original commit message follows:

- - -

This diff contains minimal changes to make webconnectivity QA
WAI with the new Web Connectivity test helper.

It seems we're currently doing round robin between the old and
the new implementation, so I needed to locally pin my probes
to use the new implementation by changing the code. But, obviously,
I don't want to commit this code.

Likewise, in my working environment, I need to build the docker
container using `docker buildx build --platform linux/amd64`, but
I am not sure whether to commit this code.

While there, I noticed there was a missing QA test for the case
in which we're passing through a transparent HTTP proxy. I noticed
as well that the test that said it was passing through such a
proxy was actually using a transparent TLS proxy. I remediated
this by ensuring we have a test for both cases.

The other major change in the suite is that, when using the new TH,
there's uncommon headers intersection in some tests, so we have
had a flip from headers not matching to headers matching.

Finally, some formatting changes because I did re-run black.

These changes should be enough to call it a day with respect to
QA (see https://github.com/ooni/probe/issues/2016#issuecomment-1033813344).

This diff WILL need to be forward ported to master.

(I don't know whether the GitHub QA will converge after these changes
and I suspect it won't because of the test helper round robin.)
This commit is contained in:
Simone Basso 2022-02-09 23:09:37 +01:00 committed by GitHub
parent 7bbd36a434
commit e72263dacb
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
2 changed files with 87 additions and 46 deletions

2
.gitignore vendored
View File

@ -1,11 +1,13 @@
/apitool /apitool
/apitool.exe /apitool.exe
/*.asc /*.asc
/badproxy.pem
/coverage.cov /coverage.cov
/*.deb /*.deb
/DEBIAN_INSTALLED_PACKAGE.txt /DEBIAN_INSTALLED_PACKAGE.txt
/debops-ci /debops-ci
.DS_Store .DS_Store
/jafar
/*.jsonl /*.jsonl
/miniooni /miniooni
/miniooni.exe /miniooni.exe

View File

@ -7,15 +7,9 @@
network conditions and verifies that the measurement is consistent network conditions and verifies that the measurement is consistent
with the expectations, by parsing the resulting JSONL. """ with the expectations, by parsing the resulting JSONL. """
import contextlib
import json
import os
import shlex
import socket import socket
import subprocess
import sys import sys
import time import time
import urllib.parse
sys.path.insert(0, ".") sys.path.insert(0, ".")
import common import common
@ -24,28 +18,42 @@ import common
def execute_jafar_and_return_validated_test_keys( def execute_jafar_and_return_validated_test_keys(
ooni_exe, outfile, experiment_args, tag, args ooni_exe, outfile, experiment_args, tag, args
): ):
""" Executes jafar and returns the validated parsed test keys, or throws """Executes jafar and returns the validated parsed test keys, or throws
an AssertionError if the result is not valid. """ an AssertionError if the result is not valid."""
tk = common.execute_jafar_and_miniooni( tk = common.execute_jafar_and_miniooni(
ooni_exe, outfile, experiment_args, tag, args ooni_exe, outfile, experiment_args, tag, args
) )
print("dns_experiment_failure", tk["dns_experiment_failure"], file=sys.stderr)
print("dns_consistency", tk["dns_consistency"], file=sys.stderr)
print("control_failure", tk["control_failure"], file=sys.stderr)
print("http_experiment_failure", tk["http_experiment_failure"], file=sys.stderr)
print("tk_body_length_match", tk["body_length_match"], file=sys.stderr)
print("body_proportion", tk["body_proportion"], file=sys.stderr)
print("status_code_match", tk["status_code_match"], file=sys.stderr)
print("headers_match", tk["headers_match"], file=sys.stderr)
print("title_match", tk["title_match"], file=sys.stderr)
print("blocking", tk["blocking"], file=sys.stderr)
print("accessible", tk["accessible"], file=sys.stderr)
print("x_status", tk["x_status"], file=sys.stderr)
return tk return tk
def assert_status_flags_are(ooni_exe, tk, desired): def assert_status_flags_are(ooni_exe, tk, desired):
""" Checks whether the status flags are what we expect them to """Checks whether the status flags are what we expect them to
be when we're running miniooni. This check only makes sense be when we're running miniooni. This check only makes sense
with miniooni b/c status flags are a miniooni extension. """ with miniooni b/c status flags are a miniooni extension."""
if "miniooni" not in ooni_exe: if "miniooni" not in ooni_exe:
return return
assert tk["x_status"] == desired assert tk["x_status"] == desired
def webconnectivity_https_ok_with_control_failure(ooni_exe, outfile): def webconnectivity_https_ok_with_control_failure(ooni_exe, outfile):
""" Successful HTTPS measurement but control failure. """ """Successful HTTPS measurement but control failure."""
args = [ args = [
"-iptables-reset-keyword", "-iptables-reset-keyword",
"wcth.ooni.io", "wcth.ooni.io",
"-iptables-reset-keyword",
"th.ooni.org",
] ]
tk = execute_jafar_and_return_validated_test_keys( tk = execute_jafar_and_return_validated_test_keys(
ooni_exe, ooni_exe,
@ -73,10 +81,12 @@ def webconnectivity_https_ok_with_control_failure(ooni_exe, outfile):
def webconnectivity_http_ok_with_control_failure(ooni_exe, outfile): def webconnectivity_http_ok_with_control_failure(ooni_exe, outfile):
""" Successful HTTP measurement but control failure. """ """Successful HTTP measurement but control failure."""
args = [ args = [
"-iptables-reset-keyword", "-iptables-reset-keyword",
"wcth.ooni.io", "wcth.ooni.io",
"-iptables-reset-keyword",
"th.ooni.org",
] ]
tk = execute_jafar_and_return_validated_test_keys( tk = execute_jafar_and_return_validated_test_keys(
ooni_exe, ooni_exe,
@ -100,7 +110,33 @@ def webconnectivity_http_ok_with_control_failure(ooni_exe, outfile):
def webconnectivity_transparent_http_proxy(ooni_exe, outfile): def webconnectivity_transparent_http_proxy(ooni_exe, outfile):
""" Test case where we pass through a transparent HTTP proxy """ """Test case where we pass through a transparent HTTP proxy"""
args = []
args.append("-iptables-hijack-http-to")
args.append("127.0.0.1:80")
tk = execute_jafar_and_return_validated_test_keys(
ooni_exe,
outfile,
"-i http://example.org web_connectivity",
"webconnectivity_transparent_http_proxy",
args,
)
assert tk["dns_experiment_failure"] == None
assert tk["dns_consistency"] == "consistent"
assert tk["control_failure"] == None
assert tk["http_experiment_failure"] == None
assert tk["body_length_match"] == True
assert tk["body_proportion"] == 1
assert tk["status_code_match"] == True
assert tk["headers_match"] == True
assert tk["title_match"] == True
assert tk["blocking"] == False
assert tk["accessible"] == True
assert_status_flags_are(ooni_exe, tk, 2)
def webconnectivity_transparent_https_proxy(ooni_exe, outfile):
"""Test case where we pass through a transparent HTTPS proxy"""
args = [] args = []
args.append("-iptables-hijack-https-to") args.append("-iptables-hijack-https-to")
args.append("127.0.0.1:443") args.append("127.0.0.1:443")
@ -108,7 +144,7 @@ def webconnectivity_transparent_http_proxy(ooni_exe, outfile):
ooni_exe, ooni_exe,
outfile, outfile,
"-i https://example.org web_connectivity", "-i https://example.org web_connectivity",
"webconnectivity_transparent_http_proxy", "webconnectivity_transparent_https_proxy",
args, args,
) )
assert tk["dns_experiment_failure"] == None assert tk["dns_experiment_failure"] == None
@ -126,7 +162,7 @@ def webconnectivity_transparent_http_proxy(ooni_exe, outfile):
def webconnectivity_dns_hijacking(ooni_exe, outfile): def webconnectivity_dns_hijacking(ooni_exe, outfile):
""" Test case where there is DNS hijacking towards a transparent proxy. """ """Test case where there is DNS hijacking towards a transparent proxy."""
args = [] args = []
args.append("-iptables-hijack-dns-to") args.append("-iptables-hijack-dns-to")
args.append("127.0.0.1:53") args.append("127.0.0.1:53")
@ -154,11 +190,13 @@ def webconnectivity_dns_hijacking(ooni_exe, outfile):
def webconnectivity_control_unreachable_and_using_http(ooni_exe, outfile): def webconnectivity_control_unreachable_and_using_http(ooni_exe, outfile):
""" Test case where the control is unreachable and we're using the """Test case where the control is unreachable and we're using the
plaintext HTTP protocol rather than HTTPS """ plaintext HTTP protocol rather than HTTPS"""
args = [] args = []
args.append("-iptables-reset-keyword") args.append("-iptables-reset-keyword")
args.append("wcth.ooni.io") args.append("wcth.ooni.io")
args.append("-iptables-reset-keyword")
args.append("th.ooni.org")
tk = execute_jafar_and_return_validated_test_keys( tk = execute_jafar_and_return_validated_test_keys(
ooni_exe, ooni_exe,
outfile, outfile,
@ -181,7 +219,7 @@ def webconnectivity_control_unreachable_and_using_http(ooni_exe, outfile):
def webconnectivity_nonexistent_domain(ooni_exe, outfile): def webconnectivity_nonexistent_domain(ooni_exe, outfile):
""" Test case where the domain does not exist """ """Test case where the domain does not exist"""
args = [] args = []
tk = execute_jafar_and_return_validated_test_keys( tk = execute_jafar_and_return_validated_test_keys(
ooni_exe, ooni_exe,
@ -223,7 +261,7 @@ def webconnectivity_nonexistent_domain(ooni_exe, outfile):
def webconnectivity_tcpip_blocking_with_consistent_dns(ooni_exe, outfile): def webconnectivity_tcpip_blocking_with_consistent_dns(ooni_exe, outfile):
""" Test case where there's TCP/IP blocking w/ consistent DNS """ """Test case where there's TCP/IP blocking w/ consistent DNS"""
ip = socket.gethostbyname("nexa.polito.it") ip = socket.gethostbyname("nexa.polito.it")
args = [ args = [
"-iptables-drop-ip", "-iptables-drop-ip",
@ -251,7 +289,7 @@ def webconnectivity_tcpip_blocking_with_consistent_dns(ooni_exe, outfile):
def webconnectivity_tcpip_blocking_with_inconsistent_dns(ooni_exe, outfile): def webconnectivity_tcpip_blocking_with_inconsistent_dns(ooni_exe, outfile):
""" Test case where there's TCP/IP blocking w/ inconsistent DNS """ """Test case where there's TCP/IP blocking w/ inconsistent DNS"""
def runner(port): def runner(port):
args = [ args = [
@ -286,8 +324,8 @@ def webconnectivity_tcpip_blocking_with_inconsistent_dns(ooni_exe, outfile):
def webconnectivity_http_connection_refused_with_consistent_dns(ooni_exe, outfile): def webconnectivity_http_connection_refused_with_consistent_dns(ooni_exe, outfile):
""" Test case where there's TCP/IP blocking w/ consistent DNS that occurs """Test case where there's TCP/IP blocking w/ consistent DNS that occurs
while we're following the chain of redirects. """ while we're following the chain of redirects."""
# We use a bit.ly link redirecting to nexa.polito.it. We block the IP address # We use a bit.ly link redirecting to nexa.polito.it. We block the IP address
# used by nexa.polito.it. So the error should happen in the redirect chain. # used by nexa.polito.it. So the error should happen in the redirect chain.
ip = socket.gethostbyname("nexa.polito.it") ip = socket.gethostbyname("nexa.polito.it")
@ -317,8 +355,8 @@ def webconnectivity_http_connection_refused_with_consistent_dns(ooni_exe, outfil
def webconnectivity_http_connection_reset_with_consistent_dns(ooni_exe, outfile): def webconnectivity_http_connection_reset_with_consistent_dns(ooni_exe, outfile):
""" Test case where there's RST-based blocking blocking w/ consistent DNS that """Test case where there's RST-based blocking blocking w/ consistent DNS that
occurs while we're following the chain of redirects. """ occurs while we're following the chain of redirects."""
# We use a bit.ly link redirecting to nexa.polito.it. We block the Host header # We use a bit.ly link redirecting to nexa.polito.it. We block the Host header
# used for nexa.polito.it. So the error should happen in the redirect chain. # used for nexa.polito.it. So the error should happen in the redirect chain.
args = [ args = [
@ -347,8 +385,8 @@ def webconnectivity_http_connection_reset_with_consistent_dns(ooni_exe, outfile)
def webconnectivity_http_nxdomain_with_consistent_dns(ooni_exe, outfile): def webconnectivity_http_nxdomain_with_consistent_dns(ooni_exe, outfile):
""" Test case where there's a redirection and the redirected request cannot """Test case where there's a redirection and the redirected request cannot
continue because a NXDOMAIN error occurs. """ continue because a NXDOMAIN error occurs."""
# We use a bit.ly link redirecting to nexa.polito.it. We block the DNS request # We use a bit.ly link redirecting to nexa.polito.it. We block the DNS request
# for nexa.polito.it. So the error should happen in the redirect chain. # for nexa.polito.it. So the error should happen in the redirect chain.
args = [ args = [
@ -382,8 +420,8 @@ def webconnectivity_http_nxdomain_with_consistent_dns(ooni_exe, outfile):
def webconnectivity_http_eof_error_with_consistent_dns(ooni_exe, outfile): def webconnectivity_http_eof_error_with_consistent_dns(ooni_exe, outfile):
""" Test case where there's a redirection and the redirected request cannot """Test case where there's a redirection and the redirected request cannot
continue because an eof_error error occurs. """ continue because an eof_error error occurs."""
# We use a bit.ly link redirecting to nexa.polito.it. We block the HTTP request # We use a bit.ly link redirecting to nexa.polito.it. We block the HTTP request
# for nexa.polito.it using the cleartext bad proxy. So the error should happen in # for nexa.polito.it using the cleartext bad proxy. So the error should happen in
# the redirect chain and should be EOF. # the redirect chain and should be EOF.
@ -417,8 +455,8 @@ def webconnectivity_http_eof_error_with_consistent_dns(ooni_exe, outfile):
def webconnectivity_http_generic_timeout_error_with_consistent_dns(ooni_exe, outfile): def webconnectivity_http_generic_timeout_error_with_consistent_dns(ooni_exe, outfile):
""" Test case where there's a redirection and the redirected request cannot """Test case where there's a redirection and the redirected request cannot
continue because a generic_timeout_error error occurs. """ continue because a generic_timeout_error error occurs."""
# We use a bit.ly link redirecting to nexa.polito.it. We block the HTTP request # We use a bit.ly link redirecting to nexa.polito.it. We block the HTTP request
# for nexa.polito.it by dropping packets using DPI. So the error should happen in # for nexa.polito.it by dropping packets using DPI. So the error should happen in
# the redirect chain and should be timeout. # the redirect chain and should be timeout.
@ -452,8 +490,8 @@ def webconnectivity_http_generic_timeout_error_with_consistent_dns(ooni_exe, out
def webconnectivity_http_connection_reset_with_inconsistent_dns(ooni_exe, outfile): def webconnectivity_http_connection_reset_with_inconsistent_dns(ooni_exe, outfile):
""" Test case where there's inconsistent DNS and the connection is RST when """Test case where there's inconsistent DNS and the connection is RST when
we're executing HTTP code. """ we're executing HTTP code."""
args = [ args = [
"-iptables-reset-keyword", "-iptables-reset-keyword",
"nexa.polito.it", "nexa.polito.it",
@ -484,7 +522,7 @@ def webconnectivity_http_connection_reset_with_inconsistent_dns(ooni_exe, outfil
def webconnectivity_http_successful_website(ooni_exe, outfile): def webconnectivity_http_successful_website(ooni_exe, outfile):
""" Test case where we succeed with an HTTP only webpage """ """Test case where we succeed with an HTTP only webpage"""
args = [] args = []
tk = execute_jafar_and_return_validated_test_keys( tk = execute_jafar_and_return_validated_test_keys(
ooni_exe, ooni_exe,
@ -508,7 +546,7 @@ def webconnectivity_http_successful_website(ooni_exe, outfile):
def webconnectivity_https_successful_website(ooni_exe, outfile): def webconnectivity_https_successful_website(ooni_exe, outfile):
""" Test case where we succeed with an HTTPS only webpage """ """Test case where we succeed with an HTTPS only webpage"""
args = [] args = []
tk = execute_jafar_and_return_validated_test_keys( tk = execute_jafar_and_return_validated_test_keys(
ooni_exe, ooni_exe,
@ -532,7 +570,7 @@ def webconnectivity_https_successful_website(ooni_exe, outfile):
def webconnectivity_http_diff_with_inconsistent_dns(ooni_exe, outfile): def webconnectivity_http_diff_with_inconsistent_dns(ooni_exe, outfile):
""" Test case where we get an http-diff and the DNS is inconsistent """ """Test case where we get an http-diff and the DNS is inconsistent"""
args = [ args = [
"-iptables-hijack-dns-to", "-iptables-hijack-dns-to",
"127.0.0.1:53", "127.0.0.1:53",
@ -555,7 +593,7 @@ def webconnectivity_http_diff_with_inconsistent_dns(ooni_exe, outfile):
assert tk["body_length_match"] == False assert tk["body_length_match"] == False
assert tk["body_proportion"] < 1 assert tk["body_proportion"] < 1
assert tk["status_code_match"] == False assert tk["status_code_match"] == False
assert tk["headers_match"] == False assert tk["headers_match"] == True
assert tk["title_match"] == False assert tk["title_match"] == False
assert tk["blocking"] == "dns" assert tk["blocking"] == "dns"
assert tk["accessible"] == False assert tk["accessible"] == False
@ -563,7 +601,7 @@ def webconnectivity_http_diff_with_inconsistent_dns(ooni_exe, outfile):
def webconnectivity_http_diff_with_consistent_dns(ooni_exe, outfile): def webconnectivity_http_diff_with_consistent_dns(ooni_exe, outfile):
""" Test case where we get an http-diff and the DNS is consistent """ """Test case where we get an http-diff and the DNS is consistent"""
args = [ args = [
"-iptables-hijack-http-to", "-iptables-hijack-http-to",
"127.0.0.1:80", "127.0.0.1:80",
@ -584,7 +622,7 @@ def webconnectivity_http_diff_with_consistent_dns(ooni_exe, outfile):
assert tk["body_length_match"] == False assert tk["body_length_match"] == False
assert tk["body_proportion"] < 1 assert tk["body_proportion"] < 1
assert tk["status_code_match"] == False assert tk["status_code_match"] == False
assert tk["headers_match"] == False assert tk["headers_match"] == True
assert tk["title_match"] == False assert tk["title_match"] == False
assert tk["blocking"] == "http-diff" assert tk["blocking"] == "http-diff"
assert tk["accessible"] == False assert tk["accessible"] == False
@ -592,7 +630,7 @@ def webconnectivity_http_diff_with_consistent_dns(ooni_exe, outfile):
def webconnectivity_https_expired_certificate(ooni_exe, outfile): def webconnectivity_https_expired_certificate(ooni_exe, outfile):
""" Test case where the domain's certificate is expired """ """Test case where the domain's certificate is expired"""
args = [] args = []
tk = execute_jafar_and_return_validated_test_keys( tk = execute_jafar_and_return_validated_test_keys(
ooni_exe, ooni_exe,
@ -629,7 +667,7 @@ def webconnectivity_https_expired_certificate(ooni_exe, outfile):
def webconnectivity_https_wrong_host(ooni_exe, outfile): def webconnectivity_https_wrong_host(ooni_exe, outfile):
""" Test case where the hostname is wrong for the certificate """ """Test case where the hostname is wrong for the certificate"""
args = [] args = []
tk = execute_jafar_and_return_validated_test_keys( tk = execute_jafar_and_return_validated_test_keys(
ooni_exe, ooni_exe,
@ -666,7 +704,7 @@ def webconnectivity_https_wrong_host(ooni_exe, outfile):
def webconnectivity_https_self_signed(ooni_exe, outfile): def webconnectivity_https_self_signed(ooni_exe, outfile):
""" Test case where the certificate is self signed """ """Test case where the certificate is self signed"""
args = [] args = []
tk = execute_jafar_and_return_validated_test_keys( tk = execute_jafar_and_return_validated_test_keys(
ooni_exe, ooni_exe,
@ -703,7 +741,7 @@ def webconnectivity_https_self_signed(ooni_exe, outfile):
def webconnectivity_https_untrusted_root(ooni_exe, outfile): def webconnectivity_https_untrusted_root(ooni_exe, outfile):
""" Test case where the certificate has an untrusted root """ """Test case where the certificate has an untrusted root"""
args = [] args = []
tk = execute_jafar_and_return_validated_test_keys( tk = execute_jafar_and_return_validated_test_keys(
ooni_exe, ooni_exe,
@ -740,7 +778,7 @@ def webconnectivity_https_untrusted_root(ooni_exe, outfile):
def webconnectivity_dns_blocking_nxdomain(ooni_exe, outfile): def webconnectivity_dns_blocking_nxdomain(ooni_exe, outfile):
""" Test case where there is blocking using NXDOMAIN """ """Test case where there is blocking using NXDOMAIN"""
args = [ args = [
"-iptables-hijack-dns-to", "-iptables-hijack-dns-to",
"127.0.0.1:53", "127.0.0.1:53",
@ -779,8 +817,8 @@ def webconnectivity_dns_blocking_nxdomain(ooni_exe, outfile):
def webconnectivity_https_unknown_authority_with_inconsistent_dns(ooni_exe, outfile): def webconnectivity_https_unknown_authority_with_inconsistent_dns(ooni_exe, outfile):
""" Test case where the DNS is sending us towards a website where """Test case where the DNS is sending us towards a website where
we're served an invalid certificate """ we're served an invalid certificate"""
args = [ args = [
"-iptables-hijack-dns-to", "-iptables-hijack-dns-to",
"127.0.0.1:53", "127.0.0.1:53",
@ -824,6 +862,7 @@ def main():
webconnectivity_https_ok_with_control_failure, webconnectivity_https_ok_with_control_failure,
webconnectivity_http_ok_with_control_failure, webconnectivity_http_ok_with_control_failure,
webconnectivity_transparent_http_proxy, webconnectivity_transparent_http_proxy,
webconnectivity_transparent_https_proxy,
webconnectivity_dns_hijacking, webconnectivity_dns_hijacking,
webconnectivity_control_unreachable_and_using_http, webconnectivity_control_unreachable_and_using_http,
webconnectivity_nonexistent_domain, webconnectivity_nonexistent_domain,