Skip to content

optimize compact_path() in pytest discovery for the common case#26016

Open
vaclavHala wants to merge 1 commit into
microsoft:mainfrom
vaclavHala:compact-discovery-payload-shortcut
Open

optimize compact_path() in pytest discovery for the common case#26016
vaclavHala wants to merge 1 commit into
microsoft:mainfrom
vaclavHala:compact-discovery-payload-shortcut

Conversation

@vaclavHala

Copy link
Copy Markdown

I tested the latest changes to discovery where absolute paths get compacted to global prefix + relative path in each test node, and this change has made the discovery 2 to 3 times slower for us.

The problem is mainly the pathlib.Path.relative_to operation which is relatively expensive:

profile_compact

This PR adds optimization which first checks using simple str.startswith if the test is in some subfolder of the root, in which case creating of the relative path is done as trivial (and cheap) substring:

profile_shortcut

I pass the base paths as both str and pathlib.Path.relative_to so each operation can use whichever form of the base path is needed without having to convert to the other, i.e. str(pathlibPath), which in my testing also adds noticeable overhead.

…ommon case before using expensive relative_to
@vaclavHala

Copy link
Copy Markdown
Author

Hello @eleanorjboyd should I create a separate issue for this?

Copilot AI left a comment

Copy link
Copy Markdown

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Pull request overview

This PR optimizes pytest discovery payload compaction by avoiding the relatively expensive pathlib.Path.relative_to() for the common case where a node’s absolute path is nested under the discovery root, using a cheaper string-prefix check and substring instead.

Changes:

  • Adds a fast-path in compact_path() using str.startswith() + slicing, and passes precomputed base-path strings through the compaction pipeline to avoid repeated conversions.
  • Updates compact_test_id() / compact_test_node() / create_compact_discovery_payload() signatures and call sites to propagate the precomputed base strings.
  • Updates an existing discovery compaction test to match the new function signatures.
Show a summary per file
File Description
python_files/vscode_pytest/__init__.py Introduces the startswith/substring optimization and threads precomputed base-path strings through the compacting functions.
python_files/tests/pytestadapter/test_discovery.py Updates existing assertions to call the new signatures for compact_path() / compact_test_id().

Review details

  • Files reviewed: 2/2 changed files
  • Comments generated: 2
  • Review effort level: Low

Comment on lines +1018 to +1026
# pathlib.Path.relative_to is an expensive operation,
# for common cases where path is nested in path_base and path_base is therefore a prefix
# we skip the expensive check and just chop off the prefix to yield relative path
if path_str.startswith(path_base_str):
# +1 because pathlib.Path never ends by a trailing separator which we also need to chop off:
# path_base_str= /some/prefix
# path_str= /some/prefix/tests/mytest.py
rel_str = path_str[(len(path_base_str) + 1):]
return "." if rel_str == "" else rel_str
Comment on lines 81 to 89
def test_compact_discovery_payload_keeps_paths_outside_base_absolute(tmp_path):
base_path = tmp_path / "workspace"
external_file = tmp_path / "external" / "test_external.py"

assert vscode_pytest.compact_path(external_file, base_path) == os.fspath(external_file)
assert vscode_pytest.compact_path(external_file, base_path, str(base_path)) == os.fspath(external_file)
assert (
vscode_pytest.compact_test_id(f"{os.fspath(external_file)}::test_external", base_path)
vscode_pytest.compact_test_id(f"{os.fspath(external_file)}::test_external", base_path, str(base_path))
== f"{os.fspath(external_file)}::test_external"
)
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment

Labels

None yet

Projects

None yet

Development

Successfully merging this pull request may close these issues.

2 participants