mirror of
				https://github.com/python/cpython.git
				synced 2025-10-30 21:21:22 +00:00 
			
		
		
		
	 08bea299bf
			
		
	
	
		08bea299bf
		
			
		
	
	
	
	
		
			
			gh-112729: Correctly fail when the process is out of memory during interpreter creation (GH-139164)
(cherry picked from commit d06113c7a7)
Co-authored-by: Peter Bierma <zintensitydev@gmail.com>
		
	
			
		
			
				
	
	
		
			89 lines
		
	
	
	
		
			3 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			89 lines
		
	
	
	
		
			3 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| import threading
 | |
| import unittest
 | |
| 
 | |
| from test import support
 | |
| from test.support import import_helper
 | |
| from test.support import threading_helper
 | |
| # Raise SkipTest if subinterpreters not supported.
 | |
| import_helper.import_module('_interpreters')
 | |
| from concurrent import interpreters
 | |
| from concurrent.interpreters import InterpreterError
 | |
| from .utils import TestBase
 | |
| 
 | |
| 
 | |
| class StressTests(TestBase):
 | |
| 
 | |
|     # In these tests we generally want a lot of interpreters,
 | |
|     # but not so many that any test takes too long.
 | |
| 
 | |
|     @support.requires_resource('cpu')
 | |
|     def test_create_many_sequential(self):
 | |
|         alive = []
 | |
|         for _ in range(100):
 | |
|             interp = interpreters.create()
 | |
|             alive.append(interp)
 | |
|         del alive
 | |
|         support.gc_collect()
 | |
| 
 | |
|     @support.bigmemtest(size=200, memuse=32*2**20, dry_run=False)
 | |
|     def test_create_many_threaded(self, size):
 | |
|         alive = []
 | |
|         start = threading.Event()
 | |
|         def task():
 | |
|             # try to create all interpreters simultaneously
 | |
|             if not start.wait(support.SHORT_TIMEOUT):
 | |
|                 raise TimeoutError
 | |
|             interp = interpreters.create()
 | |
|             alive.append(interp)
 | |
|         threads = [threading.Thread(target=task) for _ in range(size)]
 | |
|         with threading_helper.start_threads(threads):
 | |
|             start.set()
 | |
|         del alive
 | |
|         support.gc_collect()
 | |
| 
 | |
|     @threading_helper.requires_working_threading()
 | |
|     @support.bigmemtest(size=200, memuse=34*2**20, dry_run=False)
 | |
|     def test_many_threads_running_interp_in_other_interp(self, size):
 | |
|         start = threading.Event()
 | |
|         interp = interpreters.create()
 | |
| 
 | |
|         script = f"""if True:
 | |
|             import _interpreters
 | |
|             _interpreters.run_string({interp.id}, '1')
 | |
|             """
 | |
| 
 | |
|         def run():
 | |
|             interp = interpreters.create()
 | |
|             alreadyrunning = (f'{interpreters.InterpreterError}: '
 | |
|                               'interpreter already running')
 | |
|             # try to run all interpreters simultaneously
 | |
|             if not start.wait(support.SHORT_TIMEOUT):
 | |
|                 raise TimeoutError
 | |
|             success = False
 | |
|             while not success:
 | |
|                 try:
 | |
|                     interp.exec(script)
 | |
|                 except interpreters.ExecutionFailed as exc:
 | |
|                     if exc.excinfo.msg != 'interpreter already running':
 | |
|                         raise  # re-raise
 | |
|                     assert exc.excinfo.type.__name__ == 'InterpreterError'
 | |
|                 else:
 | |
|                     success = True
 | |
| 
 | |
|         threads = [threading.Thread(target=run) for _ in range(size)]
 | |
|         with threading_helper.start_threads(threads):
 | |
|             start.set()
 | |
|         support.gc_collect()
 | |
| 
 | |
|     def test_create_interpreter_no_memory(self):
 | |
|         import _interpreters
 | |
|         _testcapi = import_helper.import_module("_testcapi")
 | |
| 
 | |
|         with self.assertRaises(InterpreterError):
 | |
|             _testcapi.set_nomemory(0, 1)
 | |
|             _interpreters.create()
 | |
| 
 | |
| 
 | |
| if __name__ == '__main__':
 | |
|     # Test needs to be a package, so we can do relative imports.
 | |
|     unittest.main()
 |