blob: f9d2b65dd25f3704926f8c5a30920399029b0370 [file] [log] [blame]
Nick Coghlan260bd3e2009-11-16 06:49:25 +00001# Common utility functions used by various script execution tests
2# e.g. test_cmd_line, test_cmd_line_script and test_runpy
3
Brett Cannonc8287ef2012-04-27 13:52:03 -04004import importlib
Nick Coghlan260bd3e2009-11-16 06:49:25 +00005import sys
6import os
7import os.path
8import tempfile
9import subprocess
10import py_compile
11import contextlib
12import shutil
13import zipfile
14
Brett Cannon9529fbf2013-06-15 17:11:25 -040015from importlib.util import source_from_cache
Nick Coghlan55175962013-07-28 22:11:50 +100016from test.support import make_legacy_pyc, strip_python_stderr, temp_dir
Barry Warsaw28a691b2010-04-17 00:19:56 +000017
Nick Coghlan260bd3e2009-11-16 06:49:25 +000018# Executing the interpreter in a subprocess
Antoine Pitrou9bc35682010-11-09 21:33:55 +000019def _assert_python(expected_success, *args, **env_vars):
Victor Stinner383a8202013-06-25 21:24:36 +020020 cmd_line = [sys.executable, '-X', 'faulthandler']
Antoine Pitrouadffced2010-11-09 22:04:44 +000021 if not env_vars:
Antoine Pitrou9bc35682010-11-09 21:33:55 +000022 cmd_line.append('-E')
Antoine Pitrouadffced2010-11-09 22:04:44 +000023 # Need to preserve the original environment, for in-place testing of
24 # shared library builds.
25 env = os.environ.copy()
Georg Brandl2daf6ae2012-02-20 19:54:16 +010026 # But a special flag that can be set to override -- in this case, the
27 # caller is responsible to pass the full environment.
28 if env_vars.pop('__cleanenv', None):
29 env = {}
Antoine Pitrouadffced2010-11-09 22:04:44 +000030 env.update(env_vars)
Georg Brandl2daf6ae2012-02-20 19:54:16 +010031 cmd_line.extend(args)
Antoine Pitrouf51d8d32010-10-08 18:05:42 +000032 p = subprocess.Popen(cmd_line, stdin=subprocess.PIPE,
Antoine Pitrou9bc35682010-11-09 21:33:55 +000033 stdout=subprocess.PIPE, stderr=subprocess.PIPE,
34 env=env)
Antoine Pitrouf51d8d32010-10-08 18:05:42 +000035 try:
36 out, err = p.communicate()
37 finally:
38 subprocess._cleanup()
Brian Curtinc4ac8872010-11-01 14:00:33 +000039 p.stdout.close()
40 p.stderr.close()
Antoine Pitrouf51d8d32010-10-08 18:05:42 +000041 rc = p.returncode
Eli Bendersky8f2c2bc2013-08-11 16:48:44 -070042 err = strip_python_stderr(err)
Antoine Pitrouf51d8d32010-10-08 18:05:42 +000043 if (rc and expected_success) or (not rc and not expected_success):
44 raise AssertionError(
45 "Process return code is %d, "
46 "stderr follows:\n%s" % (rc, err.decode('ascii', 'ignore')))
47 return rc, out, err
48
Antoine Pitrou9bc35682010-11-09 21:33:55 +000049def assert_python_ok(*args, **env_vars):
50 """
51 Assert that running the interpreter with `args` and optional environment
Eli Bendersky8f2c2bc2013-08-11 16:48:44 -070052 variables `env_vars` succeeds (rc == 0) and return a (return code, stdout,
53 stderr) tuple.
Antoine Pitrou9bc35682010-11-09 21:33:55 +000054 """
55 return _assert_python(True, *args, **env_vars)
Antoine Pitrouf51d8d32010-10-08 18:05:42 +000056
Antoine Pitrou9bc35682010-11-09 21:33:55 +000057def assert_python_failure(*args, **env_vars):
58 """
59 Assert that running the interpreter with `args` and optional environment
Eli Bendersky8f2c2bc2013-08-11 16:48:44 -070060 variables `env_vars` fails (rc != 0) and return a (return code, stdout,
61 stderr) tuple.
Antoine Pitrou9bc35682010-11-09 21:33:55 +000062 """
63 return _assert_python(False, *args, **env_vars)
Nick Coghlan260bd3e2009-11-16 06:49:25 +000064
Victor Stinner024e37a2011-03-31 01:31:06 +020065def spawn_python(*args, **kw):
Eli Bendersky8f2c2bc2013-08-11 16:48:44 -070066 """Run a Python subprocess with the given arguments.
67
68 kw is extra keyword args to pass to subprocess.Popen. Returns a Popen
69 object.
70 """
Nick Coghlan260bd3e2009-11-16 06:49:25 +000071 cmd_line = [sys.executable, '-E']
72 cmd_line.extend(args)
73 return subprocess.Popen(cmd_line, stdin=subprocess.PIPE,
Victor Stinner024e37a2011-03-31 01:31:06 +020074 stdout=subprocess.PIPE, stderr=subprocess.STDOUT,
75 **kw)
Nick Coghlan260bd3e2009-11-16 06:49:25 +000076
77def kill_python(p):
Eli Bendersky8f2c2bc2013-08-11 16:48:44 -070078 """Run the given Popen process until completion and return stdout."""
Nick Coghlan260bd3e2009-11-16 06:49:25 +000079 p.stdin.close()
80 data = p.stdout.read()
81 p.stdout.close()
82 # try to cleanup the child so we don't appear to leak when running
Antoine Pitrou4e7dc5f2009-12-08 19:27:24 +000083 # with regrtest -R.
84 p.wait()
Nick Coghlan260bd3e2009-11-16 06:49:25 +000085 subprocess._cleanup()
86 return data
87
Nick Coghlan260bd3e2009-11-16 06:49:25 +000088def make_script(script_dir, script_basename, source):
89 script_filename = script_basename+os.extsep+'py'
90 script_name = os.path.join(script_dir, script_filename)
Florent Xicluna8de42e22010-02-27 16:12:22 +000091 # The script should be encoded to UTF-8, the default string encoding
92 script_file = open(script_name, 'w', encoding='utf-8')
Nick Coghlan260bd3e2009-11-16 06:49:25 +000093 script_file.write(source)
94 script_file.close()
Brett Cannonc8287ef2012-04-27 13:52:03 -040095 importlib.invalidate_caches()
Nick Coghlan260bd3e2009-11-16 06:49:25 +000096 return script_name
97
Nick Coghlan260bd3e2009-11-16 06:49:25 +000098def make_zip_script(zip_dir, zip_basename, script_name, name_in_zip=None):
99 zip_filename = zip_basename+os.extsep+'zip'
100 zip_name = os.path.join(zip_dir, zip_filename)
101 zip_file = zipfile.ZipFile(zip_name, 'w')
102 if name_in_zip is None:
Barry Warsaw28a691b2010-04-17 00:19:56 +0000103 parts = script_name.split(os.sep)
104 if len(parts) >= 2 and parts[-2] == '__pycache__':
105 legacy_pyc = make_legacy_pyc(source_from_cache(script_name))
106 name_in_zip = os.path.basename(legacy_pyc)
107 script_name = legacy_pyc
108 else:
109 name_in_zip = os.path.basename(script_name)
Nick Coghlan260bd3e2009-11-16 06:49:25 +0000110 zip_file.write(script_name, name_in_zip)
111 zip_file.close()
Florent Xicluna02ea12b22010-07-28 16:39:41 +0000112 #if test.support.verbose:
Nick Coghlan260bd3e2009-11-16 06:49:25 +0000113 # zip_file = zipfile.ZipFile(zip_name, 'r')
114 # print 'Contents of %r:' % zip_name
115 # zip_file.printdir()
116 # zip_file.close()
117 return zip_name, os.path.join(zip_name, name_in_zip)
118
Nick Coghland26c18a2010-08-17 13:06:11 +0000119def make_pkg(pkg_dir, init_source=''):
Nick Coghlan260bd3e2009-11-16 06:49:25 +0000120 os.mkdir(pkg_dir)
Nick Coghland26c18a2010-08-17 13:06:11 +0000121 make_script(pkg_dir, '__init__', init_source)
Nick Coghlan260bd3e2009-11-16 06:49:25 +0000122
123def make_zip_pkg(zip_dir, zip_basename, pkg_name, script_basename,
124 source, depth=1, compiled=False):
125 unlink = []
126 init_name = make_script(zip_dir, '__init__', '')
127 unlink.append(init_name)
128 init_basename = os.path.basename(init_name)
129 script_name = make_script(zip_dir, script_basename, source)
130 unlink.append(script_name)
131 if compiled:
Barry Warsaw28a691b2010-04-17 00:19:56 +0000132 init_name = py_compile(init_name, doraise=True)
133 script_name = py_compile(script_name, doraise=True)
Nick Coghlan260bd3e2009-11-16 06:49:25 +0000134 unlink.extend((init_name, script_name))
135 pkg_names = [os.sep.join([pkg_name]*i) for i in range(1, depth+1)]
136 script_name_in_zip = os.path.join(pkg_names[-1], os.path.basename(script_name))
137 zip_filename = zip_basename+os.extsep+'zip'
138 zip_name = os.path.join(zip_dir, zip_filename)
139 zip_file = zipfile.ZipFile(zip_name, 'w')
140 for name in pkg_names:
141 init_name_in_zip = os.path.join(name, init_basename)
142 zip_file.write(init_name, init_name_in_zip)
143 zip_file.write(script_name, script_name_in_zip)
144 zip_file.close()
145 for name in unlink:
146 os.unlink(name)
Florent Xicluna02ea12b22010-07-28 16:39:41 +0000147 #if test.support.verbose:
Nick Coghlan260bd3e2009-11-16 06:49:25 +0000148 # zip_file = zipfile.ZipFile(zip_name, 'r')
149 # print 'Contents of %r:' % zip_name
150 # zip_file.printdir()
151 # zip_file.close()
152 return zip_name, os.path.join(zip_name, script_name_in_zip)