Thanks to visit codestin.com
Credit goes to github.com

Skip to content
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
28 commits
Select commit Hold shift + click to select a range
bdd8d70
InterpreterError inherits from Exception.
ericsnowcurrently Apr 8, 2024
940f24d
Add _PyInterpreterState_GetIDObject().
ericsnowcurrently Apr 5, 2024
528c6e3
Add _PyXI_NewInterpreter() and _PyXI_EndInterpreter().
ericsnowcurrently Apr 5, 2024
0bd2e6b
Add _testinternalcapi.next_interpreter_id().
ericsnowcurrently Apr 1, 2024
76e32cd
Add _testinternalcapi.exec_interpreter().
ericsnowcurrently Apr 8, 2024
9b7bdc4
Sketch out tests.
ericsnowcurrently Mar 22, 2024
fa28f9b
Flesh out the tests.
ericsnowcurrently Mar 27, 2024
5e31f6e
Add PipeEnd.
ericsnowcurrently Mar 28, 2024
9111a83
Refactor _captured_script().
ericsnowcurrently Mar 28, 2024
0c24e5c
Finish the tests.
ericsnowcurrently Mar 28, 2024
611fa31
Add missing tests.
ericsnowcurrently Apr 1, 2024
bdc09f9
Add more capture/exec helpers.
ericsnowcurrently Apr 8, 2024
be85ff5
Fill out the tests.
ericsnowcurrently Apr 8, 2024
51f18f8
Adjust the tests.
ericsnowcurrently Apr 8, 2024
b1f96d2
Fix clean_up_interpreters().
ericsnowcurrently Apr 9, 2024
cd61643
Fix test_capi.test_misc.
ericsnowcurrently Apr 9, 2024
7de523d
external/unmanaged -> from_capi
ericsnowcurrently Apr 9, 2024
11d38ab
Add a missing decorator.
ericsnowcurrently Apr 9, 2024
1cf31b6
Fix other tests.
ericsnowcurrently Apr 9, 2024
4421169
Fix test_capi.
ericsnowcurrently Apr 9, 2024
c75a115
Add _interpreters.capture_exception().
ericsnowcurrently Apr 9, 2024
b00476d
Raise ExecutionFailed when possible.
ericsnowcurrently Apr 9, 2024
6a82a33
Handle OSError in the _running() script.
ericsnowcurrently Apr 9, 2024
11dae3d
Add PyInterpreterState._whence.
ericsnowcurrently Apr 10, 2024
7c9a2b9
Fix up _testinternalcapi.
ericsnowcurrently Apr 10, 2024
e68654c
Add PyInterpreterState.initialized.
ericsnowcurrently Apr 10, 2024
175080c
Add tests for whence.
ericsnowcurrently Apr 10, 2024
9db5a2b
Set interp-_ready on the main interpreter.
ericsnowcurrently Apr 10, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Prev Previous commit
Next Next commit
Add PipeEnd.
  • Loading branch information
ericsnowcurrently committed Apr 8, 2024
commit 5e31f6e35ff285ff599259f904fb3acd9aa2fd48
18 changes: 5 additions & 13 deletions Lib/test/test_interpreters/test_api.py
Original file line number Diff line number Diff line change
Expand Up @@ -968,7 +968,8 @@ def _run_string(self, interpid, script, maxout):
err = _interpreters.run_string(interpid, captured.script)
if err is not None:
return None, err
raw = captured.read(maxout)
raw = b''
#raw = captured.read(maxout)
return raw.decode('utf-8'), None

def run_and_capture(self, interpid, script, maxout=1000):
Expand Down Expand Up @@ -1126,25 +1127,16 @@ def test_get_current(self):
self.assertFalse(owned)

with self.subTest('owned'):
# r, w = self.pipe()
# orig = _interpreters.create()
# _interpreters.run_string(orig, dedent(f"""
# import os
# import {_interpreters.__name__} as _interpreters
# interpid = _interpreters.get_current()
# os.write({w}, interpid.as_bytes(8, 'big'))
# """))
# raw = os.read(r, 8)
# interpid = int.from_bytes(raw, 'big')
# self.assertEqual(interpid, orig)
orig = _interpreters.create()
text = self.run_and_capture(orig, f"""
import {_interpreters.__name__} as _interpreters
interpid, owned = _interpreters.get_current()
print(interpid)
print(owned)
""")
interpid, owned = text.split()
parts = text.split()
assert len(parts) == 2, parts
interpid, owned = parts
interpid = int(interpid)
owned = eval(owned)
self.assertEqual(interpid, orig)
Expand Down
237 changes: 206 additions & 31 deletions Lib/test/test_interpreters/utils.py
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,8 @@
import os
import os.path
import pickle
import select
import queue
#import select
import subprocess
import sys
import tempfile
Expand Down Expand Up @@ -79,6 +80,156 @@ def run():
t.join()


@contextlib.contextmanager
def fd_maybe_closed():
try:
yield
except OSError as exc:
if exc.errno != 9:
raise # re-raise
# The file descriptor has closed.


class PipeEnd:

def __init__(self, fd):
self._fd = fd
self._closed = False
self._lock = threading.Lock()

def __del__(self):
self.close()

def __str__(self):
return str(self._fd)

def __repr__(self):
return f'{type_self.__name__}({self._fd!r})'

def __index__(self):
return self._fd

def close(self):
with self._lock:
if self._closed:
return
self._closed = True
with fd_maybe_closed():
self._close()

@contextlib.contextmanager
def _maybe_closed(self):
assert self._lock.locked()
with fd_maybe_closed():
yield
return
# It was closed already.
if not self._closed:
self._closed = True
with fd_maybe_closed():
self._close()

def _close(self):
os.close(self._fd)


class ReadPipe(PipeEnd):

def __init__(self, fd):
super().__init__(fd)

self._requests = queue.Queue(1)
self._looplock = threading.Lock()
self._thread = threading.Thread(target=self._handle_read_requests)
self._thread.start()
self._buffer = bytearray()
self._bufferlock = threading.Lock()

def _handle_read_requests(self):
while True:
try:
req = self._requests.get()
except queue.ShutDown:
# The pipe is closed.
break
finally:
# It was locked in either self.close() or self.read().
assert self._lock.locked()

# The loop lock was taken for us in self._read().
try:
if req is None:
# Read all available bytes.
buf = bytearray()
data = os.read(self._fd, 8147) # a prime number
while data:
buf += data
data = os.read(self._fd, 8147)
with self._bufferlock:
self._buffer += buf
else:
assert req >= 0, repr(req)
with self._bufferlock:
missing = req - len(self._buffer)
if missing > 0:
with self._maybe_closed():
data = os.read(self._fd, missing)
# The rest is skipped if its already closed.
with self._bufferlock:
self._buffer += data
finally:
self._looplock.release()

def read(self, n=None, timeout=-1):
if n == 0:
return b''
elif n is not None and n < 0:
# This will fail.
os.read(self._fd, n)
raise OSError('invalid argument')

with self._lock:
# The looo will release it after handling the request.
self._looplock.acquire()
try:
self._requests.put_nowait(n)
except BaseException:
self._looplock.release()
raise # re-raise
# Wait for the request to finish.
if self._looplock.acquire(timeout=timeout):
self._looplock.release()
# Return (up to) the requested bytes.
with self._bufferlock:
data = self._buffer[:n]
self._buffer = self._buffer[n:]
return bytes(data)

def _close(self):
# Ideally the write end is already closed.
self._requests.shutdown()
with self._bufferlock:
# Throw away any leftover bytes.
self._buffer = b''
super()._close()


class WritePipe(PipeEnd):

def write(self, n=None):
try:
with self._maybe_closed():
return os.write(self._fd, n)
except BrokenPipeError:
# The read end was already closed.
self.close()


def create_pipe():
r, w = os.pipe()
return ReadPipe(r), WritePipe(w)


class CapturingScript:
"""
Embeds a script in a new script that captures stdout/stderr
Expand All @@ -94,13 +245,13 @@ class CapturingScript:
if w_err == w_out:
stdout = stderr = open(w_out, 'w', encoding='utf-8')
elif w_err is not None:
stdout = open(w_out, 'w', encoding='utf-8')
stderr = open(w_err, 'w', encoding='utf-8')
stdout = open(w_out, 'w', encoding='utf-8', closefd=False)
stderr = open(w_err, 'w', encoding='utf-8', closefd=False)
else:
stdout = open(w_out, 'w', encoding='utf-8')
stdout = open(w_out, 'w', encoding='utf-8', closefd=False)
else:
assert w_err is not None
stderr = open(w_err, 'w', encoding='utf-8')
stderr = open(w_err, 'w', encoding='utf-8', closefd=False)

sys.stdout = stdout
sys.stderr = stderr
Expand All @@ -117,12 +268,12 @@ class CapturingScript:
""")

def __init__(self, script, *, combined=True):
self._r_out, self._w_out = os.pipe()
self._r_out, self._w_out = create_pipe()
if combined:
self._r_err = self._w_err = None
w_err = self._w_out
else:
self._r_err, self._w_err = os.pipe()
self._r_err, self._w_err = create_pipe()
w_err = self._w_err
self._combined = combined

Expand All @@ -132,6 +283,9 @@ def __init__(self, script, *, combined=True):
wrapped=indent(script, ' '),
)

self._buf_stdout = None
self._buf_stderr = None

def __del__(self):
self.close()

Expand All @@ -146,38 +300,59 @@ def script(self):
return self._script

def close(self):
for fd in [self._r_out, self._w_out, self._r_err, self._w_err]:
if fd is not None:
try:
os.close(fd)
except OSError:
if exc.errno != 9:
raise # re-raise
# It was closed already.
self._r_out = self._w_out = self._r_err = self._w_err = None
if self._w_out is not None:
assert self._r_out is not None
self._w_out.close()
self._w_out = None
self._buf_stdout = self._r_out.read()
self._r_out.close()
self._r_out = None
else:
assert self._r_out is None

if self._combined:
assert self._w_err is None
assert self._r_err is None
elif self._w_err is not None:
assert self._r_err is not None
self._w_err.close()
self._w_err = None
self._buf_stderr = self._r_err.read()
self._r_err.close()
self._r_err = None
else:
assert self._r_err is None

def read(self, n=None):
return self.read_stdout(n)

def read_stdout(self, n=None):
try:
return os.read(self._r_out, n)
except OSError as exc:
if exc.errno != 9:
raise # re-raise
# It was closed already.
return b''
if self._r_out is not None:
data = self._r_out.read(n)
elif self._buf_stdout is None:
data = b''
elif n is None or n == len(self._buf_stdout):
data = self._buf_stdout
self._buf_stdout = None
else:
data = self._buf_stdout[:n]
self._buf_stdout = self._buf_stdout[n:]
return data

def read_stderr(self, n=None):
if self._combined:
return b''
try:
return os.read(self._r_err, n)
except OSError as exc:
if exc.errno != 9:
raise # re-raise
# It was closed already.
return b''
if self._r_err is not None:
data = self._r_err.read(n)
elif self._buf_stderr is None:
data = b''
elif n is None or n == len(self._buf_stderr):
data = self._buf_stderr
self._buf_stderr = None
else:
data = self._buf_stderr[:n]
self._buf_stderr = self._buf_stderr[n:]
return data


class WatchedScript:
Expand Down Expand Up @@ -435,7 +610,7 @@ def run_external(self, script, config='legacy'):
rc = run_in_interpreter(watched.script, config)
assert rc == 0, rc
text = watched.read(100).decode('utf-8')
err = captured.finished
err = watched.finished
if err is True:
err = None
return err, text