import subprocess
import os
import io
import json
import sys
import urllib
import pytest
sourcedir = os.path.dirname(os.path.dirname(os.path.realpath(__file__)))
pytest.register_assert_rewrite("tests.assertions")
def make_dsn(httpserver, auth="uiaeosnrtdy", id=123456):
url = urllib.parse.urlsplit(httpserver.url_for("/{}".format(id)))
host = url.netloc.replace("localhost", "127.0.0.1")
return urllib.parse.urlunsplit(
(url.scheme, "{}@{}".format(auth, host), url.path, url.query, url.fragment,)
)
def run(cwd, exe, args, env=dict(os.environ), **kwargs):
__tracebackhide__ = True
if os.environ.get("ANDROID_API"):
is_pipe = kwargs.get("stdout") == subprocess.PIPE
kwargs["stdout"] = subprocess.PIPE
child = subprocess.run(
[
"{}/platform-tools/adb".format(os.environ["ANDROID_HOME"]),
"shell",
"cd /data/local/tmp && LD_LIBRARY_PATH=. ./{} {}; echo -n ret:$?".format(
exe, " ".join(args)
),
],
**kwargs,
)
stdout = child.stdout
child.returncode = int(stdout[stdout.rfind(b"ret:") :][4:])
child.stdout = stdout[: stdout.rfind(b"ret:")]
if not is_pipe:
sys.stdout.buffer.write(child.stdout)
if kwargs.get("check") and child.returncode:
raise subprocess.CalledProcessError(
child.returncode, child.args, output=child.stdout, stderr=child.stderr
)
return child
cmd = [
"./{}".format(exe) if sys.platform != "win32" else "{}\\{}.exe".format(cwd, exe)
]
if "asan" in os.environ.get("RUN_ANALYZER", ""):
env["ASAN_OPTIONS"] = "detect_leaks=1"
env["LSAN_OPTIONS"] = "suppressions={}".format(
os.path.join(sourcedir, "tests", "leaks.txt")
)
if "llvm-cov" in os.environ.get("RUN_ANALYZER", ""):
continuous = "%c" if sys.platform == "darwin" else ""
env["LLVM_PROFILE_FILE"] = f"coverage-%p{continuous}.profraw"
if "kcov" in os.environ.get("RUN_ANALYZER", ""):
coverage_dir = os.path.join(cwd, "coverage")
cmd = [
"kcov",
"--include-path={}".format(os.path.join(sourcedir, "src")),
coverage_dir,
*cmd,
]
if "valgrind" in os.environ.get("RUN_ANALYZER", ""):
cmd = [
"valgrind",
"--suppressions={}".format(
os.path.join(sourcedir, "tests", "valgrind.txt")
),
"--error-exitcode=33",
"--leak-check=yes",
*cmd,
]
try:
return subprocess.run([*cmd, *args], cwd=cwd, env=env, **kwargs)
except subprocess.CalledProcessError:
raise pytest.fail.Exception(
"running command failed: {cmd} {args}".format(
cmd=" ".join(cmd), args=" ".join(args)
)
) from None
def check_output(*args, **kwargs):
stdout = run(*args, check=True, stdout=subprocess.PIPE, **kwargs).stdout
stdout = stdout.replace(b"\r\n", b"\n")
return stdout
class Envelope(object):
def __init__(
self,
headers=None, items=None, ):
if headers is not None:
headers = dict(headers)
self.headers = headers or {}
if items is None:
items = []
else:
items = list(items)
self.items = items
def get_event(self):
for item in self.items:
event = item.get_event()
if event is not None:
return event
return None
def __iter__(self):
return iter(self.items)
@classmethod
def deserialize_from(
cls, f ):
headers = json.loads(f.readline())
items = []
while 1:
item = Item.deserialize_from(f)
if item is None:
break
items.append(item)
return cls(headers=headers, items=items)
@classmethod
def deserialize(
cls, bytes ):
return cls.deserialize_from(io.BytesIO(bytes))
def __repr__(self):
return "<Envelope headers=%r items=%r>" % (self.headers, self.items)
class PayloadRef(object):
def __init__(
self,
bytes=None, json=None, ):
self.json = json
self.bytes = bytes
def __repr__(self):
return "<Payload bytes=%r json=%r>" % (self.bytes, self.json)
class Item(object):
def __init__(
self,
payload, headers=None, ):
if headers is not None:
headers = dict(headers)
elif headers is None:
headers = {}
self.headers = headers
if isinstance(payload, bytes):
payload = PayloadRef(bytes=payload)
else:
payload = payload
self.payload = payload
def get_event(self):
if self.headers.get("type") == "event" and self.payload.json is not None:
return self.payload.json
return None
@classmethod
def deserialize_from(
cls, f ):
line = f.readline().rstrip()
if not line:
return None
headers = json.loads(line)
length = headers["length"]
payload = f.read(length)
if headers.get("type") == "event" or headers.get("type") == "session":
rv = cls(headers=headers, payload=PayloadRef(json=json.loads(payload)))
else:
rv = cls(headers=headers, payload=payload)
f.readline()
return rv
@classmethod
def deserialize(
cls, bytes ):
return cls.deserialize_from(io.BytesIO(bytes))
def __repr__(self):
return "<Item headers=%r payload=%r>" % (self.headers, self.payload,)