summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorNed Batchelder <ned@nedbatchelder.com>2009-04-04 08:34:23 -0400
committerNed Batchelder <ned@nedbatchelder.com>2009-04-04 08:34:23 -0400
commit3bf521fdf5cd8d4715d03f22d20d82012deb8ddc (patch)
tree80799bfd61cd9813556e463289a44693b297f063
parent281e88d349332b4e9f06edd2a073baface6b56ce (diff)
downloadpython-coveragepy-git-3bf521fdf5cd8d4715d03f22d20d82012deb8ddc.tar.gz
Even better execution of main files.
-rw-r--r--coverage/cmdline.py6
-rw-r--r--coverage/execfile.py30
-rw-r--r--test/test_execfile.py31
-rw-r--r--test/try_execfile.py22
4 files changed, 75 insertions, 14 deletions
diff --git a/coverage/cmdline.py b/coverage/cmdline.py
index 8ddb5a45..a6844467 100644
--- a/coverage/cmdline.py
+++ b/coverage/cmdline.py
@@ -126,10 +126,10 @@ class CoverageScript:
if not args:
help_fn("Nothing to do.")
return ERR
- # Create the runtime environment the script on the cmdline expects.
- sys.argv = args
+
+ # Run the script.
self.coverage.start()
- run_python_file(sys.argv[0])
+ run_python_file(args[0], args)
self.coverage.stop()
if settings.get('combine'):
diff --git a/coverage/execfile.py b/coverage/execfile.py
index bcf5bf4f..4820c8e1 100644
--- a/coverage/execfile.py
+++ b/coverage/execfile.py
@@ -1,11 +1,27 @@
"""Execute files of Python code."""
-import os, sys
+import imp, os, sys
-def run_python_file(filename):
- mod_globals = {
- '__name__': '__main__',
- '__file__': filename,
- }
+def run_python_file(filename, args):
+ """Run a python source file as if it were the main program on the python
+ command line.
+
+ `filename` is the path to the file to execute, must be a .py file.
+ `args` is the argument array to present as sys.argv.
+
+ """
+ # Most code that does this does it in a way that leaves __main__ or __file__
+ # with the wrong values. Importing the code as __main__ gets all of this
+ # right automatically.
+ #
+ # One difference from python.exe: if I run foo.py from the command line, it
+ # always uses foo.py. With this code, it might find foo.pyc instead.
+
+ sys.argv = args
sys.path[0] = os.path.dirname(filename)
- execfile(filename, mod_globals)
+
+ try:
+ src = open(filename)
+ imp.load_module('__main__', src, filename, (".py", "r", imp.PY_SOURCE))
+ finally:
+ src.close()
diff --git a/test/test_execfile.py b/test/test_execfile.py
index 90340f57..6ad2f58e 100644
--- a/test/test_execfile.py
+++ b/test/test_execfile.py
@@ -3,17 +3,42 @@ import cStringIO, os, sys, unittest
here = os.path.dirname(__file__)
+class Tee(object):
+ def __init__(self, *files):
+ self.files = files
+
+ def write(self, data):
+ for f in self.files:
+ f.write(data)
+
class RunTests(unittest.TestCase):
def setUp(self):
self.oldstdout = sys.stdout
- self.stdout = sys.stdout = cStringIO.StringIO()
+ self.stdout = cStringIO.StringIO()
+ sys.stdout = Tee(sys.stdout, self.stdout)
def tearDown(self):
self.stdout = self.oldstdout
def test_run_python_file(self):
tryfile = os.path.join(here, "try_execfile.py")
- run_python_file(tryfile)
+ run_python_file(tryfile, [tryfile, "arg1", "arg2"])
mod_globs = eval(self.stdout.getvalue())
+
+ # The file should think it is __main__
self.assertEqual(mod_globs['__name__'], "__main__")
- self.assertEqual(os.path.basename(mod_globs['__file__']), "try_execfile.py")
+
+ # It should seem to come from a file named try_execfile
+ dunder_file = os.path.splitext(os.path.basename(mod_globs['__file__']))[0]
+ self.assertEqual(dunder_file, "try_execfile")
+
+ # It should have its correct module data.
+ self.assertEqual(mod_globs['__doc__'], "Test file for run_python_file.")
+ self.assertEqual(mod_globs['DATA'], "xyzzy")
+ self.assertEqual(mod_globs['FN_VAL'], "my_fn('fooey')")
+
+ # It must be self-importable as __main__.
+ self.assertEqual(mod_globs['__main__.DATA'], "xyzzy")
+
+ # Argv should have the proper values.
+ self.assertEqual(mod_globs['argv'], [tryfile, "arg1", "arg2"])
diff --git a/test/try_execfile.py b/test/try_execfile.py
index 523823e4..f66b9c5b 100644
--- a/test/try_execfile.py
+++ b/test/try_execfile.py
@@ -1,5 +1,25 @@
+"""Test file for run_python_file."""
+
+import pprint, sys
+
+DATA = "xyzzy"
+
+import __main__
+
+def my_function(a):
+ return "my_fn(%r)" % a
+
+FN_VAL = my_function("fooey")
+
globals_to_check = {
'__name__': __name__,
'__file__': __file__,
+ '__doc__': __doc__,
+ 'DATA': DATA,
+ 'FN_VAL': FN_VAL,
+ '__main__.DATA': getattr(__main__, "DATA", "nothing"),
+ 'argv': sys.argv,
+ 'path0': sys.path[0],
}
-print repr(globals_to_check)
+
+pprint.pprint(globals_to_check)