Skip to content

Commit d6a8b11

Browse files
authored
Merge pull request #1093 from carmenbianca/refactor-exceptions
Refactor exceptions
2 parents d942142 + 0af6fb6 commit d6a8b11

12 files changed

+115
-93
lines changed

src/reuse/__init__.py

-8
Original file line numberDiff line numberDiff line change
@@ -182,11 +182,3 @@ def __bool__(self) -> bool:
182182

183183
def __or__(self, value: "ReuseInfo") -> "ReuseInfo":
184184
return self.union(value)
185-
186-
187-
class ReuseException(Exception):
188-
"""Base exception."""
189-
190-
191-
class IdentifierNotFound(ReuseException):
192-
"""Could not find SPDX identifier for license file."""

src/reuse/_annotate.py

+3-3
Original file line numberDiff line numberDiff line change
@@ -30,12 +30,12 @@
3030
)
3131
from .comment import (
3232
NAME_STYLE_MAP,
33-
CommentCreateError,
3433
CommentStyle,
3534
EmptyCommentStyle,
3635
get_comment_style,
3736
)
38-
from .header import MissingReuseInfo, add_new_header, find_and_replace_header
37+
from .exceptions import CommentCreateError, MissingReuseInfoError
38+
from .header import add_new_header, find_and_replace_header
3939
from .i18n import _
4040
from .project import Project
4141
from .types import StrPath
@@ -152,7 +152,7 @@ def add_header_to_file(
152152
)
153153
out.write("\n")
154154
result = 1
155-
except MissingReuseInfo:
155+
except MissingReuseInfoError:
156156
out.write(
157157
_(
158158
"Error: Generated comment header for '{path}' is missing"

src/reuse/cli/common.py

+3-3
Original file line numberDiff line numberDiff line change
@@ -13,9 +13,9 @@
1313
from license_expression import ExpressionError
1414

1515
from .._util import _LICENSING
16-
from ..global_licensing import GlobalLicensingParseError
16+
from ..exceptions import GlobalLicensingConflictError, GlobalLicensingParseError
1717
from ..i18n import _
18-
from ..project import GlobalLicensingConflict, Project
18+
from ..project import Project
1919
from ..vcs import find_root
2020

2121

@@ -60,7 +60,7 @@ def project(self) -> Project:
6060
).format(path=error.source, message=str(error))
6161
) from error
6262

63-
except (GlobalLicensingConflict, OSError) as error:
63+
except (GlobalLicensingConflictError, OSError) as error:
6464
raise click.UsageError(str(error)) from error
6565

6666
self._project = project

src/reuse/comment.py

+1-8
Original file line numberDiff line numberDiff line change
@@ -32,19 +32,12 @@
3232
from textwrap import dedent
3333
from typing import NamedTuple, Optional, Type, cast
3434

35+
from .exceptions import CommentCreateError, CommentParseError
3536
from .types import StrPath
3637

3738
_LOGGER = logging.getLogger(__name__)
3839

3940

40-
class CommentParseError(Exception):
41-
"""An error occurred during the parsing of a comment."""
42-
43-
44-
class CommentCreateError(Exception):
45-
"""An error occurred during the creation of a comment."""
46-
47-
4841
class MultiLineSegments(NamedTuple):
4942
"""Components that make up a multi-line comment style, e.g. '/*', '*', and
5043
'*/'.

src/reuse/exceptions.py

+61
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,61 @@
1+
# SPDX-FileCopyrightText: 2017 Free Software Foundation Europe e.V. <https://fsfe.org>
2+
#
3+
# SPDX-License-Identifier: GPL-3.0-or-later
4+
5+
"""All exceptions owned by :mod:`reuse`. These exceptions all inherit
6+
:class:`ReuseError`.
7+
"""
8+
9+
from typing import Any, Optional
10+
11+
12+
class ReuseError(Exception):
13+
"""Base exception."""
14+
15+
16+
class SpdxIdentifierNotFoundError(ReuseError):
17+
"""Could not find SPDX identifier for license file."""
18+
19+
20+
class GlobalLicensingParseError(ReuseError):
21+
"""An exception representing any kind of error that occurs when trying to
22+
parse a :class:`reuse.global_licensing.GlobalLicensing` file.
23+
"""
24+
25+
def __init__(self, *args: Any, source: Optional[str] = None):
26+
super().__init__(*args)
27+
self.source = source
28+
29+
30+
class GlobalLicensingParseTypeError(GlobalLicensingParseError, TypeError):
31+
"""An exception representing a type error while trying to parse a
32+
:class:`reuse.global_licensing.GlobalLicensing` file.
33+
"""
34+
35+
36+
class GlobalLicensingParseValueError(GlobalLicensingParseError, ValueError):
37+
"""An exception representing a value error while trying to parse a
38+
:class:`reuse.global_licensing.GlobalLicensing` file.
39+
"""
40+
41+
42+
class GlobalLicensingConflictError(ReuseError):
43+
"""There are two global licensing files in the project that are not
44+
compatible.
45+
"""
46+
47+
48+
class MissingReuseInfoError(ReuseError):
49+
"""Some REUSE information is missing from the result."""
50+
51+
52+
class CommentError(ReuseError):
53+
"""An error occurred during an interaction with a comment."""
54+
55+
56+
class CommentCreateError(Exception):
57+
"""An error occurred during the creation of a comment."""
58+
59+
60+
class CommentParseError(Exception):
61+
"""An error occurred during the parsing of a comment."""

src/reuse/global_licensing.py

+6-23
Original file line numberDiff line numberDiff line change
@@ -33,9 +33,14 @@
3333
from debian.copyright import Error as DebianError
3434
from license_expression import ExpressionError
3535

36-
from . import ReuseException, ReuseInfo, SourceType
36+
from . import ReuseInfo, SourceType
3737
from ._util import _LICENSING
3838
from .covered_files import iter_files
39+
from .exceptions import (
40+
GlobalLicensingParseError,
41+
GlobalLicensingParseTypeError,
42+
GlobalLicensingParseValueError,
43+
)
3944
from .i18n import _
4045
from .types import StrPath
4146
from .vcs import VCSStrategy
@@ -72,28 +77,6 @@ class PrecedenceType(Enum):
7277
OVERRIDE = "override"
7378

7479

75-
class GlobalLicensingParseError(ReuseException):
76-
"""An exception representing any kind of error that occurs when trying to
77-
parse a :class:`GlobalLicensing` file.
78-
"""
79-
80-
def __init__(self, *args: Any, source: Optional[str] = None):
81-
super().__init__(*args)
82-
self.source = source
83-
84-
85-
class GlobalLicensingParseTypeError(GlobalLicensingParseError, TypeError):
86-
"""An exception representing a type error while trying to parse a
87-
:class:`GlobalLicensing` file.
88-
"""
89-
90-
91-
class GlobalLicensingParseValueError(GlobalLicensingParseError, ValueError):
92-
"""An exception representing a value error while trying to parse a
93-
:class:`GlobalLicensing` file.
94-
"""
95-
96-
9780
@attrs.define
9881
class _CollectionOfValidator:
9982
collection_type: Type[Collection] = attrs.field()

src/reuse/header.py

+13-15
Original file line numberDiff line numberDiff line change
@@ -28,12 +28,11 @@
2828
extract_reuse_info,
2929
merge_copyright_lines,
3030
)
31-
from .comment import (
31+
from .comment import CommentStyle, EmptyCommentStyle, PythonCommentStyle
32+
from .exceptions import (
3233
CommentCreateError,
3334
CommentParseError,
34-
CommentStyle,
35-
EmptyCommentStyle,
36-
PythonCommentStyle,
35+
MissingReuseInfoError,
3736
)
3837
from .i18n import _
3938

@@ -53,10 +52,6 @@ class _TextSections(NamedTuple):
5352
after: str
5453

5554

56-
class MissingReuseInfo(Exception):
57-
"""Some REUSE information is missing from the result."""
58-
59-
6055
def _create_new_header(
6156
reuse_info: ReuseInfo,
6257
template: Optional[Template] = None,
@@ -68,7 +63,8 @@ def _create_new_header(
6863
6964
Raises:
7065
CommentCreateError: if a comment could not be created.
71-
MissingReuseInfo: if the generated comment is missing SPDX information.
66+
MissingReuseInfoError: if the generated comment is missing SPDX
67+
information.
7268
"""
7369
if template is None:
7470
template = DEFAULT_TEMPLATE
@@ -101,7 +97,7 @@ def _create_new_header(
10197
)
10298
)
10399
_LOGGER.debug(result)
104-
raise MissingReuseInfo()
100+
raise MissingReuseInfoError()
105101

106102
return result
107103

@@ -125,7 +121,8 @@ def create_header(
125121
126122
Raises:
127123
CommentCreateError: if a comment could not be created.
128-
MissingReuseInfo: if the generated comment is missing SPDX information.
124+
MissingReuseInfoError: if the generated comment is missing SPDX
125+
information.
129126
"""
130127
if template is None:
131128
template = DEFAULT_TEMPLATE
@@ -186,7 +183,7 @@ def _find_first_spdx_comment(
186183
preceding the comment, the comment itself, and everything following it.
187184
188185
Raises:
189-
MissingReuseInfo: if no REUSE info can be found in any comment
186+
MissingReuseInfoError: if no REUSE info can be found in any comment.
190187
"""
191188
if style is None:
192189
style = PythonCommentStyle
@@ -203,7 +200,7 @@ def _find_first_spdx_comment(
203200
text[:index], comment + "\n", text[index + len(comment) + 1 :]
204201
)
205202

206-
raise MissingReuseInfo()
203+
raise MissingReuseInfoError()
207204

208205

209206
def _extract_shebang(prefix: str, text: str) -> tuple[str, str]:
@@ -248,14 +245,15 @@ def find_and_replace_header(
248245
249246
Raises:
250247
CommentCreateError: if a comment could not be created.
251-
MissingReuseInfo: if the generated comment is missing SPDX information.
248+
MissingReuseInfoError: if the generated comment is missing SPDX
249+
information.
252250
"""
253251
if style is None:
254252
style = PythonCommentStyle
255253

256254
try:
257255
before, header, after = _find_first_spdx_comment(text, style=style)
258-
except MissingReuseInfo:
256+
except MissingReuseInfoError:
259257
before, header, after = "", "", text
260258

261259
# Workaround. EmptyCommentStyle should always be completely replaced.

src/reuse/project.py

+13-15
Original file line numberDiff line numberDiff line change
@@ -21,7 +21,7 @@
2121
import attrs
2222
from binaryornot.check import is_binary
2323

24-
from . import IdentifierNotFound, ReuseInfo
24+
from . import ReuseInfo
2525
from ._licenses import EXCEPTION_MAP, LICENSE_MAP
2626
from ._util import (
2727
_LICENSEREF_PATTERN,
@@ -30,6 +30,10 @@
3030
reuse_info_of_file,
3131
)
3232
from .covered_files import iter_files
33+
from .exceptions import (
34+
GlobalLicensingConflictError,
35+
SpdxIdentifierNotFoundError,
36+
)
3337
from .global_licensing import (
3438
GlobalLicensing,
3539
NestedReuseTOML,
@@ -44,12 +48,6 @@
4448
_LOGGER = logging.getLogger(__name__)
4549

4650

47-
class GlobalLicensingConflict(Exception):
48-
"""There are two global licensing files in the project that are not
49-
compatible.
50-
"""
51-
52-
5351
class GlobalLicensingFound(NamedTuple):
5452
path: Path
5553
cls: Type[GlobalLicensing]
@@ -111,8 +109,8 @@ def from_directory(
111109
decoded.
112110
GlobalLicensingParseError: if the global licensing config file could
113111
not be parsed.
114-
GlobalLicensingConflict: if more than one global licensing config
115-
file is present.
112+
GlobalLicensingConflictError: if more than one global licensing
113+
config file is present.
116114
"""
117115
root = Path(root)
118116
if not root.exists():
@@ -323,8 +321,8 @@ def find_global_licensing(
323321
:class:`GlobalLicensing`.
324322
325323
Raises:
326-
GlobalLicensingConflict: if more than one global licensing config
327-
file is present.
324+
GlobalLicensingConflictError: if more than one global licensing
325+
config file is present.
328326
"""
329327
candidates: list[GlobalLicensingFound] = []
330328
dep5_path = root / ".reuse/dep5"
@@ -352,7 +350,7 @@ def find_global_licensing(
352350
]
353351
if reuse_toml_candidates:
354352
if candidates:
355-
raise GlobalLicensingConflict(
353+
raise GlobalLicensingConflictError(
356354
_(
357355
"Found both '{new_path}' and '{old_path}'. You"
358356
" cannot keep both files simultaneously; they are"
@@ -384,13 +382,13 @@ def _identifier_of_license(self, path: Path) -> str:
384382
License Identifier.
385383
"""
386384
if not path.suffix:
387-
raise IdentifierNotFound(f"{path} has no file extension")
385+
raise SpdxIdentifierNotFoundError(f"{path} has no file extension")
388386
if path.stem in self.license_map:
389387
return path.stem
390388
if _LICENSEREF_PATTERN.match(path.stem):
391389
return path.stem
392390

393-
raise IdentifierNotFound(
391+
raise SpdxIdentifierNotFoundError(
394392
f"Could not find SPDX License Identifier for {path}"
395393
)
396394

@@ -418,7 +416,7 @@ def _find_licenses(self) -> dict[str, Path]:
418416

419417
try:
420418
identifier = self._identifier_of_license(path)
421-
except IdentifierNotFound:
419+
except SpdxIdentifierNotFoundError:
422420
if path.name in self.license_map:
423421
_LOGGER.info(
424422
_("{path} does not have a file extension").format(

tests/test_comment.py

+1-2
Original file line numberDiff line numberDiff line change
@@ -13,15 +13,14 @@
1313
import pytest
1414

1515
from reuse.comment import (
16-
CommentCreateError,
17-
CommentParseError,
1816
CommentStyle,
1917
CppCommentStyle,
2018
HtmlCommentStyle,
2119
LispCommentStyle,
2220
PythonCommentStyle,
2321
_all_style_classes,
2422
)
23+
from reuse.exceptions import CommentCreateError, CommentParseError
2524

2625

2726
@pytest.fixture(

tests/test_global_licensing.py

+4-2
Original file line numberDiff line numberDiff line change
@@ -15,11 +15,13 @@
1515

1616
from reuse import ReuseInfo, SourceType
1717
from reuse._util import _LICENSING
18-
from reuse.global_licensing import (
19-
AnnotationsItem,
18+
from reuse.exceptions import (
2019
GlobalLicensingParseError,
2120
GlobalLicensingParseTypeError,
2221
GlobalLicensingParseValueError,
22+
)
23+
from reuse.global_licensing import (
24+
AnnotationsItem,
2325
NestedReuseTOML,
2426
PrecedenceType,
2527
ReuseDep5,

0 commit comments

Comments
 (0)