mirror of
https://github.com/progval/irctest.git
synced 2025-04-05 23:09:48 +00:00
Fix lock on the set of used ports (#235)
pytest-xdist (well, execnet) re-loads modules after forking, so each process had its own lock, making the lock useless. Co-authored-by: Shivaram Lingamneni <slingamn@cs.stanford.edu>
This commit is contained in:
@ -1,7 +1,8 @@
|
|||||||
from __future__ import annotations
|
from __future__ import annotations
|
||||||
|
|
||||||
|
import contextlib
|
||||||
import dataclasses
|
import dataclasses
|
||||||
import multiprocessing
|
import json
|
||||||
import os
|
import os
|
||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
import shutil
|
import shutil
|
||||||
@ -10,23 +11,13 @@ import subprocess
|
|||||||
import tempfile
|
import tempfile
|
||||||
import textwrap
|
import textwrap
|
||||||
import time
|
import time
|
||||||
from typing import (
|
from typing import IO, Any, Callable, Dict, Iterator, List, Optional, Set, Tuple, Type
|
||||||
IO,
|
|
||||||
Any,
|
|
||||||
Callable,
|
|
||||||
Dict,
|
|
||||||
List,
|
|
||||||
MutableMapping,
|
|
||||||
Optional,
|
|
||||||
Set,
|
|
||||||
Tuple,
|
|
||||||
Type,
|
|
||||||
)
|
|
||||||
|
|
||||||
import irctest
|
import irctest
|
||||||
|
|
||||||
from . import authentication, tls
|
from . import authentication, tls
|
||||||
from .client_mock import ClientMock
|
from .client_mock import ClientMock
|
||||||
|
from .irc_utils.filelock import FileLock
|
||||||
from .irc_utils.junkdrawer import find_hostname_and_port
|
from .irc_utils.junkdrawer import find_hostname_and_port
|
||||||
from .irc_utils.message_parser import Message
|
from .irc_utils.message_parser import Message
|
||||||
from .runner import NotImplementedByController
|
from .runner import NotImplementedByController
|
||||||
@ -71,41 +62,36 @@ class _BaseController:
|
|||||||
|
|
||||||
proc: Optional[subprocess.Popen]
|
proc: Optional[subprocess.Popen]
|
||||||
|
|
||||||
_used_ports: Set[Tuple[str, int]]
|
_used_ports_path = Path(tempfile.gettempdir()) / "irctest_ports.json"
|
||||||
"""``(hostname, port))`` used by this controller."""
|
_port_lock = FileLock(Path(tempfile.gettempdir()) / "irctest_ports.json.lock")
|
||||||
# the following need to be shared between processes in case we are running in
|
|
||||||
# parallel (with pytest-xdist)
|
|
||||||
# The dicts are used as a set of (hostname, port), because _manager.set() doesn't
|
|
||||||
# exist.
|
|
||||||
_manager = multiprocessing.Manager()
|
|
||||||
_port_lock = _manager.Lock()
|
|
||||||
"""Lock for access to ``_all_used_ports`` and ``_available_ports``."""
|
|
||||||
_all_used_ports: MutableMapping[Tuple[str, int], None] = _manager.dict()
|
|
||||||
"""``(hostname, port)`` used by all controllers."""
|
|
||||||
_available_ports: MutableMapping[Tuple[str, int], None] = _manager.dict()
|
|
||||||
"""``(hostname, port)`` available to any controller."""
|
|
||||||
|
|
||||||
def __init__(self, test_config: TestCaseControllerConfig):
|
def __init__(self, test_config: TestCaseControllerConfig):
|
||||||
self.test_config = test_config
|
self.test_config = test_config
|
||||||
self.proc = None
|
self.proc = None
|
||||||
self._used_ports = set()
|
self._own_ports: Set[Tuple[str, int]] = set()
|
||||||
|
|
||||||
|
@contextlib.contextmanager
|
||||||
|
def _used_ports(self) -> Iterator[Set[Tuple[str, int]]]:
|
||||||
|
with self._port_lock:
|
||||||
|
if not self._used_ports_path.exists():
|
||||||
|
self._used_ports_path.write_text("[]")
|
||||||
|
used_ports = {
|
||||||
|
(h, p) for (h, p) in json.loads(self._used_ports_path.read_text())
|
||||||
|
}
|
||||||
|
yield used_ports
|
||||||
|
self._used_ports_path.write_text(json.dumps(list(used_ports)))
|
||||||
|
|
||||||
def get_hostname_and_port(self) -> Tuple[str, int]:
|
def get_hostname_and_port(self) -> Tuple[str, int]:
|
||||||
with self._port_lock:
|
with self._used_ports() as used_ports:
|
||||||
try:
|
while True:
|
||||||
# try to get a known available port
|
(hostname, port) = find_hostname_and_port()
|
||||||
((hostname, port), _) = self._available_ports.popitem()
|
if (hostname, port) not in used_ports:
|
||||||
except KeyError:
|
# double-checking in self._used_ports to prevent collisions
|
||||||
# if there aren't any, iterate while we get a fresh one.
|
# between controllers starting at the same time.
|
||||||
while True:
|
break
|
||||||
(hostname, port) = find_hostname_and_port()
|
|
||||||
if (hostname, port) not in self._all_used_ports:
|
|
||||||
# double-checking in self._used_ports to prevent collisions
|
|
||||||
# between controllers starting at the same time.
|
|
||||||
break
|
|
||||||
|
|
||||||
# Make this port unavailable to other processes
|
used_ports.add((hostname, port))
|
||||||
self._all_used_ports[(hostname, port)] = None
|
self._own_ports.add((hostname, port))
|
||||||
|
|
||||||
return (hostname, port)
|
return (hostname, port)
|
||||||
|
|
||||||
@ -131,10 +117,10 @@ class _BaseController:
|
|||||||
if self.proc:
|
if self.proc:
|
||||||
self.kill_proc()
|
self.kill_proc()
|
||||||
|
|
||||||
# move this controller's ports from _all_used_ports to _available_ports
|
with self._used_ports() as used_ports:
|
||||||
for hostname, port in self._used_ports:
|
for hostname, port in list(self._own_ports):
|
||||||
del self._all_used_ports[(hostname, port)]
|
used_ports.remove((hostname, port))
|
||||||
self._available_ports[(hostname, port)] = None
|
self._own_ports.remove((hostname, port))
|
||||||
|
|
||||||
|
|
||||||
class DirectoryBasedController(_BaseController):
|
class DirectoryBasedController(_BaseController):
|
||||||
|
19
irctest/irc_utils/filelock.py
Normal file
19
irctest/irc_utils/filelock.py
Normal file
@ -0,0 +1,19 @@
|
|||||||
|
"""
|
||||||
|
Compatibility layer for filelock ( https://pypi.org/project/filelock/ );
|
||||||
|
commonly packaged by Linux distributions but might not be available
|
||||||
|
in some environments.
|
||||||
|
"""
|
||||||
|
|
||||||
|
import os
|
||||||
|
from typing import ContextManager
|
||||||
|
|
||||||
|
if os.getenv("PYTEST_XDIST_WORKER"):
|
||||||
|
# running under pytest-xdist; filelock is required for reliability
|
||||||
|
from filelock import FileLock
|
||||||
|
else:
|
||||||
|
# normal test execution, no port races
|
||||||
|
import contextlib
|
||||||
|
from typing import Any
|
||||||
|
|
||||||
|
def FileLock(*args: Any, **kwargs: Any) -> ContextManager[None]:
|
||||||
|
return contextlib.nullcontext()
|
@ -1,3 +1,5 @@
|
|||||||
|
pytest
|
||||||
|
|
||||||
# The following dependencies are actually optional:
|
# The following dependencies are actually optional:
|
||||||
ecdsa
|
ecdsa
|
||||||
pytest
|
filelock
|
||||||
|
Reference in New Issue
Block a user