Skip to content

Commit

Permalink
tests: add tests
Browse files Browse the repository at this point in the history
This adds a few tests to make sure multithreading continues
to work, and to test that we properly exit after hooks
throw exceptions. These tests run in CI.
  • Loading branch information
ndrewh committed Jul 8, 2024
1 parent 51adc16 commit 5c87db3
Show file tree
Hide file tree
Showing 7 changed files with 278 additions and 3 deletions.
24 changes: 21 additions & 3 deletions .github/workflows/build.yml
Original file line number Diff line number Diff line change
@@ -1,9 +1,10 @@
name: Create and publish a Docker image
name: Build, Test, and (if needed) Publish Image

on:
push:
branches:
- master
- dev
release:
types: ['published']

Expand Down Expand Up @@ -36,9 +37,26 @@ jobs:
registry: ghcr.io
username: ${{ github.actor }}
password: ${{ github.token }}
- uses: docker/[email protected]

- name: Build image
uses: docker/[email protected]
with:
load: true
push: false
tags: pyda_tmp
platforms: linux/amd64
provenance: false
cache-from: type=gha

- name: Test
run: |
docker run -e PYTHONUNBUFFERED=1 --rm --workdir /opt/pyda/tests --entrypoint python3 pyda_tmp run_tests.py
- name: Push image
uses: docker/[email protected]
if: github.event_name != 'pull_request'
with:
push: ${{ github.event_name != 'pull_request' }}
push: true
tags: ${{ steps.meta.outputs.tags }}
platforms: linux/amd64
provenance: false
Expand Down
32 changes: 32 additions & 0 deletions tests/err_hook.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,32 @@
from pyda import *
from pwnlib.elf.elf import ELF
from pwnlib.util.packing import u64
import string
import sys, time

p = process()

e = ELF(p.exe_path)
e.address = p.maps[p.exe_path].base

plt_map = { e.plt[x]: x for x in e.plt }

counter = 0
def lib_hook(p):
global counter
name = plt_map[p.regs.rip]
print(f"[thread {p.tid}] {name}")

counter += 1
if counter == 1000:
jsdkfjdsaklfadska

def thread_entry(p):
print(f"thread_entry for {p.tid}")

p.set_thread_entry(thread_entry)

for x in e.plt:
p.hook(e.plt[x], lib_hook)

p.run()
32 changes: 32 additions & 0 deletions tests/err_thread_entry.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,32 @@
from pyda import *
from pwnlib.elf.elf import ELF
from pwnlib.util.packing import u64
import string
import sys, time

p = process()

e = ELF(p.exe_path)
e.address = p.maps[p.exe_path].base

plt_map = { e.plt[x]: x for x in e.plt }

counter = 0
def lib_hook(p):
name = plt_map[p.regs.rip]
print(f"[thread {p.tid}] {name}")

def thread_entry(p):
global counter
print(f"thread_entry for {p.tid}")

counter += 1
if counter == 27:
jsdkfjdsaklfadska

p.set_thread_entry(thread_entry)

for x in e.plt:
p.hook(e.plt[x], lib_hook)

p.run()
128 changes: 128 additions & 0 deletions tests/run_tests.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,128 @@
import subprocess
from dataclasses import dataclass
from typing import Optional, Callable
from pathlib import Path
from tempfile import TemporaryDirectory

@dataclass
class ExpectedResult:
retcode: Optional[int] = None

# checker(stdout, stderr) -> bool
checkers: list[Callable[[bytes, bytes], bool]] = list

Res = ExpectedResult

def output_checker(stdout: bytes, stderr: bytes) -> bool:
try:
stdout.decode()
stderr.decode()
except:
return False

return True

def main():
res = True

# thread_1000.c tests whether we can handle a large number of threads
# with concurrent hooks
res &= run_test(
"thread_1000.c", "../examples/ltrace_multithreaded.py",
ExpectedResult(
retcode=0,
checkers=[
output_checker,
lambda o, e: o.count(b"malloc") == 20000,
lambda o, e: o.count(b"free") == 20000,
lambda o, e: all((o.count(f"[thread {i}]".encode('utf-8')) == 40 for i in range(2, 1002))),
]
)
)

# thread_nojoin.c tests whether we can handle a large number of threads
# that do not get waited on (i.e. they are not joined). Mostly
# we just care about the return code and termination here.
res &= run_test(
"thread_nojoin.c", "../examples/ltrace_multithreaded.py",
ExpectedResult(
retcode=0,
checkers=[
output_checker,
]
)
)

# err_hook.py tests the case where a hook throws an exception
# NOTE: Hooks intentionally fail 'gracefully' and do not abort
res &= run_test(
"thread_1000.c", "err_hook.py",
ExpectedResult(
retcode=0,
checkers=[
output_checker,
lambda o, e: e.count(b"[Pyda] ERROR:") == 1,
]
)
)

# err_thread_entry.py tests the case where a hook throws an exception
# NOTE: Hooks intentionally fail 'gracefully' and do not abort
res &= run_test(
"thread_1000.c", "err_thread_entry.py",
ExpectedResult(
retcode=0,
checkers=[
output_checker,
lambda o, e: e.count(b"[Pyda] ERROR:") == 1,
]
)
)

if not res:
exit(1)

def run_test(c_file, python_file, expected_result):
# Compile to temporary directory
with TemporaryDirectory() as tmpdir:
tmpdir = Path(tmpdir)
c_path = Path(c_file)
p_path = Path(python_file)

c_exe = tmpdir / c_path.stem
compile_res = subprocess.run(['gcc', '-o', c_exe, c_path], capture_output=True)
if compile_res.returncode != 0:
print(f"Failed to compile {c_file}")
print(compile_res.stderr)
raise RuntimeError("Failed to compile test")

result_str = ""
try:
result = subprocess.run(f"pyda {p_path.resolve()} -- {c_exe.resolve()}", shell=True, timeout=10, capture_output=True)
except subprocess.TimeoutExpired:
result_str += " Timeout occurred. Did the test hang?\n"
result = None


if result:
# Check the results
if expected_result.retcode is not None:
if result.returncode != expected_result.retcode:
result_str += f" Expected return code {expected_result.retcode}, got {result.returncode}\n"

for (i, checker) in enumerate(expected_result.checkers):
if not checker(result.stdout, result.stderr):
result_str += f" Checker {i} failed\n"


if len(result_str) > 0:
print(f"[FAIL] {c_file} {python_file}")
print(result_str)
return False
else:
print(f"[OK] {c_file} {python_file}")
return True


if __name__ == '__main__':
main()
Empty file added tests/simple.c
Empty file.
35 changes: 35 additions & 0 deletions tests/thread_1000.c
Original file line number Diff line number Diff line change
@@ -0,0 +1,35 @@
#include <stdio.h>
#include <stdlib.h>
#include <pthread.h>

#define NTHREAD 1000

void* thread(void* arg) {
void *allocs[10];
for (int i=0; i<10; i++) {
allocs[i] = malloc(0x100);
}
for (int i=0; i<10; i++) {
free(allocs[i]);
}
for (int i=0; i<10; i++) {
allocs[i] = malloc(0x100);
}
for (int i=0; i<10; i++) {
free(allocs[i]);
}
}
int main(int argc, char** argv) {
pthread_t threads[NTHREAD];
for (int i=0; i<NTHREAD; i++) {
pthread_create(&threads[i], 0, thread, 0);
}

for (int i=0; i<NTHREAD; i++) {
void *ret;
pthread_join(threads[i], &ret);
}

// thread(NULL);
return 0;
}
30 changes: 30 additions & 0 deletions tests/thread_nojoin.c
Original file line number Diff line number Diff line change
@@ -0,0 +1,30 @@
#include <stdio.h>
#include <stdlib.h>
#include <pthread.h>

#define NTHREAD 1000

void* thread(void* arg) {
void *allocs[10];
for (int i=0; i<10; i++) {
allocs[i] = malloc(0x100);
}
for (int i=0; i<10; i++) {
free(allocs[i]);
}
for (int i=0; i<10; i++) {
allocs[i] = malloc(0x100);
}
for (int i=0; i<10; i++) {
free(allocs[i]);
}
}
int main(int argc, char** argv) {
pthread_t threads[NTHREAD];
for (int i=0; i<NTHREAD; i++) {
pthread_create(&threads[i], 0, thread, 0);
}

// thread(NULL);
return 0;
}

0 comments on commit 5c87db3

Please sign in to comment.