diff options
author | Antoine Pitrou <solipsis@pitrou.net> | 2012-01-17 23:21:11 (GMT) |
---|---|---|
committer | Antoine Pitrou <solipsis@pitrou.net> | 2012-01-17 23:21:11 (GMT) |
commit | 2f828f2c8859b331bf9519bdc660eb8a2288779a (patch) | |
tree | d137b7be2783abe51bd5f15913ef73283b25b962 | |
parent | 03757ec4a571df0e871bd9136cae9d08841f8a6a (diff) | |
download | cpython-2f828f2c8859b331bf9519bdc660eb8a2288779a.zip cpython-2f828f2c8859b331bf9519bdc660eb8a2288779a.tar.gz cpython-2f828f2c8859b331bf9519bdc660eb8a2288779a.tar.bz2 |
Test running of code in a sub-interpreter
(prelude to issue #6531).
-rw-r--r-- | Lib/test/test_capi.py | 17 | ||||
-rw-r--r-- | Modules/_testcapimodule.c | 27 |
2 files changed, 44 insertions, 0 deletions
diff --git a/Lib/test/test_capi.py b/Lib/test/test_capi.py index ada393d..c2c633f 100644 --- a/Lib/test/test_capi.py +++ b/Lib/test/test_capi.py @@ -3,6 +3,7 @@ from __future__ import with_statement import os +import pickle import random import subprocess import sys @@ -137,6 +138,22 @@ class TestPendingCalls(unittest.TestCase): self.pendingcalls_submit(l, n) self.pendingcalls_wait(l, n) + def test_subinterps(self): + # XXX this test leaks in refleak runs + import builtins + r, w = os.pipe() + code = """if 1: + import sys, builtins, pickle + with open({:d}, "wb") as f: + pickle.dump(id(sys.modules), f) + pickle.dump(id(builtins), f) + """.format(w) + with open(r, "rb") as f: + ret = _testcapi.run_in_subinterp(code) + self.assertEqual(ret, 0) + self.assertNotEqual(pickle.load(f), id(sys.modules)) + self.assertNotEqual(pickle.load(f), id(builtins)) + # Bug #6012 class Test6012(unittest.TestCase): def test(self): diff --git a/Modules/_testcapimodule.c b/Modules/_testcapimodule.c index 6c61f7d..94b6e67 100644 --- a/Modules/_testcapimodule.c +++ b/Modules/_testcapimodule.c @@ -2300,6 +2300,32 @@ crash_no_current_thread(PyObject *self) return NULL; } +/* To run some code in a sub-interpreter. */ +static PyObject * +run_in_subinterp(PyObject *self, PyObject *args) +{ + const char *code; + int r; + PyThreadState *substate, *mainstate; + + if (!PyArg_ParseTuple(args, "s:run_in_subinterp", + &code)) + return NULL; + + mainstate = PyThreadState_Get(); + + PyThreadState_Swap(NULL); + + substate = Py_NewInterpreter(); + r = PyRun_SimpleString(code); + Py_EndInterpreter(substate); + + PyThreadState_Swap(mainstate); + + return PyLong_FromLong(r); +} + + static PyMethodDef TestMethods[] = { {"raise_exception", raise_exception, METH_VARARGS}, {"raise_memoryerror", (PyCFunction)raise_memoryerror, METH_NOARGS}, @@ -2385,6 +2411,7 @@ static PyMethodDef TestMethods[] = { {"make_memoryview_from_NULL_pointer", (PyCFunction)make_memoryview_from_NULL_pointer, METH_NOARGS}, {"crash_no_current_thread", (PyCFunction)crash_no_current_thread, METH_NOARGS}, + {"run_in_subinterp", run_in_subinterp, METH_VARARGS}, {NULL, NULL} /* sentinel */ }; |