1
0

Filtering, to avoid changing all tests in the repository
Some checks failed
Run Python tests (through Pytest) / Test (push) Failing after 23s
Verify Python project can be installed, loaded and have version checked / Test (push) Successful in 22s

This commit is contained in:
Jon Michael Aanes 2025-03-05 09:52:00 +01:00
parent 151978cb1f
commit 44458b3ba9
2 changed files with 63 additions and 32 deletions

View File

@ -4,15 +4,31 @@ Tool for standardizing Partisia's Java tests.
Is capable of migrating from Javadoc based property specification to the Is capable of migrating from Javadoc based property specification to the
`@DisplayName` specification and back. `@DisplayName` specification and back.
## Usage
Convert tests with `@DisplayName` to Javadoc:
```bash
python -m standardize_java_text --javadoc --only-displayname FILE...
```
It it recommended that you do this with no unstaged changes, to more easily
verify the result.
You may need to call your formatting tool afterwards, as the converter doesn't
take formatting into account.
""" """
import enum import enum
import hashlib import hashlib
import re import re
import dataclasses
from ._version import __version__ from ._version import __version__
__all__ = ['__version__', 'standardize_java_text', 'NamingScheme'] __all__ = ['__version__', 'standardize_java_text', 'NamingScheme', 'Config']
class NamingScheme(enum.Enum): class NamingScheme(enum.Enum):
@ -21,10 +37,19 @@ class NamingScheme(enum.Enum):
FROM_DESC = 'desc' FROM_DESC = 'desc'
@dataclasses.dataclass
class Config:
with_javadoc: bool
with_display_name: bool
only_for_javadoc: bool
only_for_display_name: bool
naming_scheme: NamingScheme
PATTERN_JAVADOC = r'/\*\*(?P<javadoc>.*(\n\s*\*.*)*?)\s*\*/' PATTERN_JAVADOC = r'/\*\*(?P<javadoc>.*(\n\s*\*.*)*?)\s*\*/'
PATTERN_DISPLAY_NAME_EXPRESSION = r'\s*(?:"[^"]*?")(?:\s*\+\s*"[^"]*?")*\s*' PATTERN_DISPLAY_NAME_EXPRESSION = r'\s*(?:"[^"]*?")(?:\s*\+\s*"[^"]*?")*\s*'
PATTERN_DISPLAY_NAME = ( PATTERN_DISPLAY_NAME = (
r'@DisplayName\((?P<display>' + PATTERN_DISPLAY_NAME_EXPRESSION + r')\)' r'@DisplayName\((?P<displayname>' + PATTERN_DISPLAY_NAME_EXPRESSION + r')\)'
) )
TEST_PATTERN: re.Pattern = re.compile( TEST_PATTERN: re.Pattern = re.compile(
@ -142,25 +167,30 @@ def parse_display_name_to_description(text: str | None):
def replace_test_pattern( def replace_test_pattern(
match: re.Match, match: re.Match,
with_javadoc: bool, config: Config,
with_display_name: bool,
naming_scheme: NamingScheme,
) -> str: ) -> str:
javadoc = parse_javadoc_to_description(match.group('javadoc')) javadoc = parse_javadoc_to_description(match.group('javadoc'))
annotation = match.group('annotation').strip() annotation = match.group('annotation').strip()
visibility = (match.group('visibility') or '').strip() visibility = (match.group('visibility') or '').strip()
name = match.group('name').strip() name = match.group('name').strip()
display = parse_display_name_to_description(match.group('display')) displayname = parse_display_name_to_description(match.group('displayname'))
description = display or javadoc or from_camel_case(name) # Filtering
if config.only_for_javadoc and not javadoc:
return match.group(0)
if config.only_for_display_name and not displayname:
return match.group(0)
# Formatting
description = displayname or javadoc or from_camel_case(name)
description = re.sub(r'[ \t]+', ' ', description) description = re.sub(r'[ \t]+', ' ', description)
if naming_scheme == NamingScheme.PRESERVE: if config.naming_scheme == NamingScheme.PRESERVE:
pass pass
elif naming_scheme == NamingScheme.FROM_DESC: elif config.naming_scheme == NamingScheme.FROM_DESC:
name = to_camel_case(description) name = to_camel_case(description)
elif naming_scheme == NamingScheme.HASH_OF_DESC: elif config.naming_scheme == NamingScheme.HASH_OF_DESC:
h = hashlib.sha256() h = hashlib.sha256()
h.update(description.encode('utf8')) h.update(description.encode('utf8'))
name = 'test' + h.hexdigest() name = 'test' + h.hexdigest()
@ -171,8 +201,8 @@ def replace_test_pattern(
description, description,
annotation, annotation,
visibility, visibility,
with_javadoc=with_javadoc, with_javadoc=config.with_javadoc,
with_display_name=with_display_name, with_display_name=config.with_display_name,
) )
@ -181,16 +211,12 @@ IMPORT_DISPLAY_NAME = 'import org.junit.jupiter.api.DisplayName;'
def standardize_java_text( def standardize_java_text(
text: str, text: str,
with_javadoc: bool, config: Config,
with_display_name: bool,
naming_scheme: NamingScheme,
): ):
text = TEST_PATTERN.sub( text = TEST_PATTERN.sub(
lambda m: replace_test_pattern( lambda m: replace_test_pattern(
m, m,
with_javadoc, config,
with_display_name,
naming_scheme,
), ),
text, text,
) )

View File

@ -1,7 +1,7 @@
import argparse import argparse
import pathlib import pathlib
from . import NamingScheme, standardize_java_text from . import NamingScheme, standardize_java_text, Config
def test_files(repo: pathlib.Path) -> list[pathlib.Path]: def test_files(repo: pathlib.Path) -> list[pathlib.Path]:
@ -11,18 +11,14 @@ def test_files(repo: pathlib.Path) -> list[pathlib.Path]:
def standardize_in_file( def standardize_in_file(
path: pathlib.Path, path: pathlib.Path,
inline: bool, inline: bool,
with_javadoc: bool, config: Config,
with_display_name: bool,
naming_scheme: NamingScheme,
): ):
with open(path) as f: with open(path) as f:
text = f.read() text = f.read()
text_updated = standardize_java_text( text_updated = standardize_java_text(
text, text,
with_javadoc, config,
with_display_name,
naming_scheme=naming_scheme,
) )
if text_updated == text: if text_updated == text:
return return
@ -36,15 +32,18 @@ def standardize_in_file(
def argument_parser(): def argument_parser():
argparser = argparse.ArgumentParser() argparser = argparse.ArgumentParser()
argparser.add_argument('repo', type=pathlib.Path) argparser.add_argument('repo', type=pathlib.Path, help='Repository')
argparser.add_argument('-i', action='store_true') argparser.add_argument('-i', action='store_true', help='Write to files')
argparser.add_argument('--javadoc', action='store_true') argparser.add_argument('--javadoc', action='store_true', help='Generate Javadoc for tests')
argparser.add_argument('--displayname', action='store_true') argparser.add_argument('--displayname', action='store_true', help='Generate DisplayName for tests')
argparser.add_argument('--only-javadoc', action='store_true', help='Only change tests with an existing Javadoc')
argparser.add_argument('--only-displayname', action='store_true', help='Only change test with an existing DisplayName')
argparser.add_argument( argparser.add_argument(
'--naming', '--naming',
type=NamingScheme, type=NamingScheme,
choices=list(NamingScheme), choices=list(NamingScheme),
default=NamingScheme.FROM_DESC, default=NamingScheme.FROM_DESC,
help='How to derive the name of the test method',
) )
return argparser return argparser
@ -52,13 +51,19 @@ def argument_parser():
def main(): def main():
args = argument_parser().parse_args() args = argument_parser().parse_args()
config = Config(
with_javadoc=args.javadoc,
with_display_name=args.displayname,
only_for_javadoc=args.only_javadoc,
only_for_display_name=args.only_displayname,
naming_scheme=args.naming,
)
for path in test_files(args.repo): for path in test_files(args.repo):
standardize_in_file( standardize_in_file(
path, path,
inline=args.i, inline=args.i,
with_javadoc=args.javadoc, config=config,
with_display_name=args.displayname,
naming_scheme=args.naming,
) )