Skip to content

Commit

Permalink
Add run command
Browse files Browse the repository at this point in the history
  • Loading branch information
gaogaotiantian committed May 1, 2024
1 parent ac4bfd4 commit 84ac2f9
Show file tree
Hide file tree
Showing 6 changed files with 216 additions and 11 deletions.
129 changes: 124 additions & 5 deletions src/coredumpy/coredumpy.py
Original file line number Diff line number Diff line change
Expand Up @@ -10,17 +10,104 @@
import os
import pdb
import platform
import sys
import tokenize
import textwrap
import types
import warnings
from types import CodeType
from typing import Callable, Optional, Union

from .patch import patch_all
from .py_object_proxy import PyObjectProxy
from .utils import get_dump_filename


class _ExecutableTarget:
filename: str
code: CodeType | str
namespace: dict


class _ScriptTarget(_ExecutableTarget):
def __init__(self, target):
self._target = os.path.realpath(target)

if not os.path.exists(self._target):
print(f'Error: {target} does not exist')
sys.exit(1)
if os.path.isdir(self._target):
print(f'Error: {target} is a directory')
sys.exit(1)

# If safe_path(-P) is not set, sys.path[0] is the directory
# of pdb, and we should replace it with the directory of the script
if not sys.flags.safe_path:
sys.path[0] = os.path.dirname(self._target)

def __repr__(self):
return self._target

@property
def filename(self):
return self._target

@property
def code(self):
# Open the file each time because the file may be modified
import io
with io.open_code(self._target) as fp:
return f"exec(compile({fp.read()!r}, {self._target!r}, 'exec'))"

@property
def namespace(self):
return dict(
__name__='__main__',
__file__=self._target,
__builtins__=__builtins__,
__spec__=None,
)


class _ModuleTarget(_ExecutableTarget):
def __init__(self, target):
self._target = target

import runpy
try:
_, self._spec, self._code = runpy._get_module_details(self._target)
sys.path.insert(0, os.getcwd())
except ImportError as e:
print(f"ImportError: {e}")
sys.exit(1)
except Exception:
import traceback
traceback.print_exc()
sys.exit(1)

def __repr__(self):
return self._target

@property
def filename(self):
return self._code.co_filename

@property
def code(self):
return self._code

@property
def namespace(self):
return dict(
__name__='__main__',
__file__=os.path.normcase(os.path.abspath(self.filename)),
__package__=self._spec.parent,
__loader__=self._spec.loader,
__spec__=self._spec,
__builtins__=__builtins__,
)


class Coredumpy:
@classmethod
def dump(cls,
Expand Down Expand Up @@ -48,15 +135,15 @@ def dump(cls,
assert inner_frame is not None
frame = inner_frame.f_back
curr_frame = frame

with warnings.catch_warnings():
warnings.simplefilter("ignore")
PyObjectProxy.add_object(frame)

while frame:
filename = frame.f_code.co_filename

if filename not in files:
files.add(filename)

with warnings.catch_warnings():
warnings.simplefilter("ignore")
PyObjectProxy.add_object(frame)
frame = frame.f_back

output_file = get_dump_filename(curr_frame, path, directory)
Expand Down Expand Up @@ -121,6 +208,37 @@ def peek(cls, path):
if data["description"]:
print(textwrap.indent(data["description"], " "))

@classmethod
def run(cls, options):
if options.module:
file = options.module
target = _ModuleTarget(file)
else:
if not options.args:
print("Error: no script specified")
sys.exit(1)
file = options.args.pop(0)
target = _ScriptTarget(file)

sys.argv[:] = [file] + options.args

import __main__
__main__.__dict__.clear()
__main__.__dict__.update(target.namespace)

cmd = target.code

if isinstance(cmd, str):
cmd = compile(cmd, "<string>", "exec")

from .except_hook import patch_except
patch_except(path=options.path, directory=options.directory)

from .unittest_hook import patch_unittest
patch_unittest(path=options.path, directory=options.directory)

exec(cmd, __main__.__dict__, __main__.__dict__)

@classmethod
def get_metadata(cls):
from coredumpy import __version__
Expand All @@ -140,3 +258,4 @@ def get_metadata(cls):
dump = Coredumpy.dump
load = Coredumpy.load
peek = Coredumpy.peek
run = Coredumpy.run
14 changes: 11 additions & 3 deletions src/coredumpy/main.py
Original file line number Diff line number Diff line change
Expand Up @@ -5,18 +5,24 @@
import argparse
import os

from .coredumpy import load, peek
from .coredumpy import load, peek, run


def main():
parser = argparse.ArgumentParser()
subparsers = parser.add_subparsers(dest="command")

subparsers_run = subparsers.add_parser("run", help="Run a file/module with coredumpy enabled.")
subparsers_run.add_argument("-m", metavar="module", dest="module")
subparsers_run.add_argument("--path", help="The path of dump file", default=None)
subparsers_run.add_argument("--directory", help="The directory of dump file", default=None)
subparsers_run.add_argument("args", nargs="*")

subparsers_load = subparsers.add_parser("load", help="Load a dump file.")
subparsers_load.add_argument("file", type=str, help="The dump file to load.")

subparsers_load = subparsers.add_parser("peek", help="Peek a dump file.")
subparsers_load.add_argument("files", help="The dump file to load.", nargs="+")
subparsers_peek = subparsers.add_parser("peek", help="Peek a dump file.")
subparsers_peek.add_argument("files", help="The dump file to load.", nargs="+")

args = parser.parse_args()

Expand All @@ -42,3 +48,5 @@ def main():
pass
else:
print(f"File {file} not found.")
elif args.command == "run":
run(args)
3 changes: 3 additions & 0 deletions src/coredumpy/py_object_proxy.py
Original file line number Diff line number Diff line change
Expand Up @@ -53,6 +53,9 @@ def _add_object(cls, obj):
id_str = str(id(obj))
if id_str not in cls._objects:
cls._objects[id_str] = {"type": "_coredumpy_unknown"}
if obj is cls._objects or obj is cls._pending_objects:
# Avoid changing the dict while dumping
return
cls._pending_objects.put((cls._current_recursion_depth + 1, obj))

@classmethod
Expand Down
18 changes: 15 additions & 3 deletions tests/base.py
Original file line number Diff line number Diff line change
Expand Up @@ -13,13 +13,17 @@


class TestBase(unittest.TestCase):
def run_test(self, script, dumppath, commands):
def run_test(self, script, dumppath, commands, use_cli_run=False):
script = textwrap.dedent(script)
with tempfile.TemporaryDirectory() as tmpdir:
with open(f"{tmpdir}/script.py", "w") as f:
f.write(script)
subprocess.run(normalize_commands([sys.executable, f"{tmpdir}/script.py"]),
stdout=subprocess.DEVNULL, stderr=subprocess.DEVNULL)
if use_cli_run:
subprocess.run(normalize_commands(["coredumpy", "run", f"{tmpdir}/script.py", "--path", dumppath]),
stdout=subprocess.DEVNULL, stderr=subprocess.DEVNULL)
else:
subprocess.run(normalize_commands([sys.executable, f"{tmpdir}/script.py"]),
stdout=subprocess.DEVNULL, stderr=subprocess.DEVNULL)

process = subprocess.Popen(normalize_commands(["coredumpy", "load", dumppath]),
stdin=subprocess.PIPE, stdout=subprocess.PIPE, stderr=subprocess.PIPE)
Expand Down Expand Up @@ -47,6 +51,14 @@ def run_script(self, script, expected_returncode=0):
f"script failed with return code {process.returncode}\n{stderr}")
return stdout, stderr

def run_run(self, args):
process = subprocess.Popen(normalize_commands(["coredumpy", "run"] + args),
stdin=subprocess.PIPE, stdout=subprocess.PIPE, stderr=subprocess.PIPE)
stdout, stderr = process.communicate()
stdout = stdout.decode(errors='backslashreplace')
stderr = stderr.decode(errors='backslashreplace')
return stdout, stderr

def run_peek(self, paths):
process = subprocess.Popen(normalize_commands(["coredumpy", "peek"] + paths),
stdin=subprocess.PIPE, stdout=subprocess.PIPE, stderr=subprocess.PIPE)
Expand Down
31 changes: 31 additions & 0 deletions tests/test_basic.py
Original file line number Diff line number Diff line change
Expand Up @@ -67,6 +67,37 @@ def g(arg):
self.assertIn("return 1 / arg", stdout)
self.assertIn("0", stdout)

def test_cli(self):
script = """
def g(arg):
return 1 / arg
def f(x):
a = 142857
g(x)
f(0)
"""
stdout, _ = self.run_test(script, "coredumpy_dump", [
"w",
"p arg",
"u",
"p a",
"q"
], use_cli_run=True)

self.assertIn("-> f(0)", stdout)
self.assertIn("-> g(x)", stdout)
self.assertIn("142857", stdout)

def test_cli_invalid(self):
stdout, _ = self.run_run(["notexist.py"])
self.assertIn("Error", stdout)

stdout, _ = self.run_run([os.path.dirname(__file__)])
self.assertIn("Error", stdout)

stdout, _ = self.run_run(["-m", "nonexistmodule"])
self.assertIn("Error", stdout)

def test_peek(self):
with tempfile.TemporaryDirectory() as tmpdir:
script = f"""
Expand Down
32 changes: 32 additions & 0 deletions tests/test_unittest.py
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@

import os
import tempfile
import textwrap


from .base import TestBase
Expand Down Expand Up @@ -34,3 +35,34 @@ def test_error(self):
self.assertNotIn("test_pass", stderr)
self.assertEqual(stdout.count(tempdir), 3)
self.assertEqual(len(os.listdir(tempdir)), 3)

def test_unittest_with_cli(self):
with tempfile.TemporaryDirectory() as tempdir:
script = textwrap.dedent("""
import unittest
class TestUnittest(unittest.TestCase):
def test_bool(self):
self.assertTrue(False)
def test_eq(self):
self.assertEqual(1, 2)
def test_pass(self):
self.assertEqual(1, 1)
def test_error(self):
raise ValueError()
""")
with open(f"{tempdir}/script.py", "w") as f:
f.write(script)

try:
curdir = os.getcwd()
os.chdir(tempdir)
stdout, stderr = self.run_run(["-m", "unittest", "script",
"--directory", os.path.join(tempdir, "dump")])
finally:
os.chdir(curdir)
self.assertIn("FAIL: test_bool", stderr)
self.assertIn("FAIL: test_eq", stderr)
self.assertIn("ERROR: test_error", stderr)
self.assertNotIn("test_pass", stderr)
self.assertEqual(stdout.count(tempdir), 3)
self.assertEqual(len(os.listdir(os.path.join(tempdir, "dump"))), 3)

0 comments on commit 84ac2f9

Please sign in to comment.