Skip to content

Commit

Permalink
Merge pull request #181 from dhellmann/import-filter-error-handling
Browse files Browse the repository at this point in the history
filters: trap exits from imported modules
  • Loading branch information
mergify[bot] authored Jun 19, 2022
2 parents fcbaee9 + 36e3658 commit 2973d51
Show file tree
Hide file tree
Showing 4 changed files with 70 additions and 9 deletions.
8 changes: 8 additions & 0 deletions docs/source/history.rst
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,14 @@
Next
====

Bug Fixes
---------

- `#180 <https://github.com/sphinx-contrib/spelling/issues/180>`__
Suppress `SystemExit` errors in `ImportableModuleFilter` caused by
importing modules that run code on import and exit when that code
sees an error. Bug report and reproducer provided by Trevor Gross.

Features
--------

Expand Down
4 changes: 2 additions & 2 deletions sphinxcontrib/spelling/filters.py
Original file line number Diff line number Diff line change
Expand Up @@ -209,8 +209,8 @@ def _skip(self, word):
self.sought_modules.add(word)
try:
mod = importlib.util.find_spec(word)
except Exception as err:
# This could be an ImportError, some more detailed
except BaseException as err:
# This could be an ImportError, SystemExit, some more detailed
# error out of distutils, or something else triggered
# by failing to be able to import a parent package to
# use the metadata to search for a subpackage.
Expand Down
62 changes: 57 additions & 5 deletions tests/test_filter.py
Original file line number Diff line number Diff line change
Expand Up @@ -4,13 +4,20 @@
"""Tests for filters.
"""

import contextlib
import logging
import os
import sys
from pathlib import Path

import pytest
from enchant.tokenize import get_tokenizer

from sphinxcontrib.spelling import filters
from sphinxcontrib.spelling import filters # isort:skip

# Replace the sphinx logger with a normal one so pytest can collect
# the output.
filters.logger = logging.getLogger('test.filters')


def test_builtin_unicode():
Expand Down Expand Up @@ -91,9 +98,54 @@ def test_importable_module_skip(word, expected):
assert f._skip(word) is expected


@contextlib.contextmanager
def import_path(new_path):
"Temporarily change sys.path for imports."
before = sys.path
try:
sys.path = new_path
yield
finally:
sys.path = before


def test_importable_module_with_side_effets(tmpdir):
with tmpdir.as_cwd():
path = tmpdir.join('setup.py')
path.write('raise SystemExit\n')
logging.debug('tmpdir %r', tmpdir)
logging.debug('cwd %r', os.getcwd())

parentdir = tmpdir.join('parent')
parentdir.mkdir()

parentdir.join('__init__.py').write(
'raise SystemExit("exit as side-effect")\n'
)
parentdir.join('child.py').write('')

with import_path([str(tmpdir)] + sys.path):
f = filters.ImportableModuleFilter(None)
assert f._skip('setup.cfg') is False
skip_parent = f._skip('parent')
skip_both = f._skip('parent.child')

# The parent module name is valid because it is not imported, only
# discovered.
assert skip_parent is True
assert 'parent' in f.found_modules

# The child module name is not valid because the parent is
# imported to find the child and that triggers the side-effect.
assert skip_both is False
assert 'parent.child' not in f.found_modules


def test_importable_module_with_system_exit(tmpdir):
path = tmpdir.join('mytestmodule.py')
path.write('raise SystemExit("exit as side-effect")\n')

with import_path([str(tmpdir)] + sys.path):
f = filters.ImportableModuleFilter(None)
skip = f._skip('mytestmodule')

# The filter does not actually import the module in this case, so
# it shows up as a valid word.
assert skip is True
assert 'mytestmodule' in f.found_modules
5 changes: 3 additions & 2 deletions tox.ini
Original file line number Diff line number Diff line change
Expand Up @@ -6,9 +6,10 @@ envlist=py,linter,docs
extras =
test
commands=
pytest \
python -m pytest \
--cov=sphinxcontrib.spelling \
--cov-report term-missing
--cov-report term-missing \
--log-level DEBUG
passenv=
# Let the caller point to where libenchant is on their system, in
# case it's in an unusual place. For example, on macOS with
Expand Down

0 comments on commit 2973d51

Please sign in to comment.