Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Improve state of the art for screenshot testing #555

Merged
merged 6 commits into from
Jan 3, 2025
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
13 changes: 13 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -178,3 +178,16 @@ differences.

If you want to update the reference screenshot for a given example, you can grab
those from the build artifacts as well and commit them to your branch.

### Testing Locally

Testing locally is possible, however pixel perfect results will differ from
those on the CIs due to discrepencies in hardware, and driver (we use llvmpipe)
versions.

If you want to force the usage of LLVMPIPE to speed up local testing you
may do so with the WGPUPY_WGPU_ADAPTER_NAME environment variable
almarklein marked this conversation as resolved.
Show resolved Hide resolved

```
WGPUPY_WGPU_ADAPTER_NAME=llvmpipe pytest -v examples/
```
71 changes: 50 additions & 21 deletions examples/tests/test_examples.py
Original file line number Diff line number Diff line change
Expand Up @@ -94,7 +94,10 @@ def unload_module():
# the first part of the test everywhere else; ensuring that examples
# can at least import, run and render something
if not is_lavapipe:
pytest.skip("screenshot comparisons are only done when using lavapipe")
pytest.skip(
"screenshot comparisons are only done when using lavapipe. "
"Rerun your tests with WGPUPY_WGPU_ADAPTER_NAME=llvmpipe"
)

# regenerate screenshot if requested
screenshots_dir.mkdir(exist_ok=True)
Expand All @@ -108,36 +111,62 @@ def unload_module():
), "found # test_example = true but no reference screenshot available"
stored_img = imageio.imread(screenshot_path)
# assert similarity
is_similar = np.allclose(img, stored_img, atol=1)
update_diffs(module, is_similar, img, stored_img)
assert is_similar, (
f"rendered image for example {module} changed, see "
f"the {diffs_dir.relative_to(ROOT).as_posix()} folder"
" for visual diffs (you can download this folder from"
" CI build artifacts as well)"
)
atol = 1
try:
np.testing.assert_allclose(img, stored_img, atol=atol)
is_similar = True
except Exception as e:
is_similar = False
raise AssertionError(
f"rendered image for example {module_name} changed, see "
f"the {diffs_dir.relative_to(ROOT).as_posix()} folder"
" for visual diffs (you can download this folder from"
" CI build artifacts as well)"
) from e
finally:
update_diffs(module_name, is_similar, img, stored_img, atol=atol)


def update_diffs(module, is_similar, img, stored_img):
def update_diffs(module, is_similar, img, stored_img, *, atol):
diffs_dir.mkdir(exist_ok=True)

if is_similar:
for path in [
# Keep filename in sync with the ones generated below
diffs_dir / f"{module}-rgb.png",
diffs_dir / f"{module}-alpha.png",
diffs_dir / f"{module}-rgb-above_atol.png",
diffs_dir / f"{module}-alpha-above_atol.png",
diffs_dir / f"{module}.png",
]:
if path.exists():
path.unlink()
return

# cast to float32 to avoid overflow
# compute absolute per-pixel difference
diffs_rgba = np.abs(stored_img.astype("f4") - img)

diffs_rgba_above_atol = diffs_rgba.copy()
diffs_rgba_above_atol[diffs_rgba <= atol] = 0

# magnify small values, making it easier to spot small errors
diffs_rgba = ((diffs_rgba / 255) ** 0.25) * 255
# cast back to uint8
diffs_rgba = diffs_rgba.astype("u1")
# split into an rgb and an alpha diff
diffs = {
diffs_dir / f"{module}-rgb.png": diffs_rgba[..., :3],
diffs_dir / f"{module}-alpha.png": diffs_rgba[..., 3],
}

for path, diff in diffs.items():
if not is_similar:
imageio.imwrite(path, diff)
elif path.exists():
path.unlink()

diffs_rgba_above_atol = ((diffs_rgba_above_atol / 255) ** 0.25) * 255
diffs_rgba_above_atol = diffs_rgba_above_atol.astype("u1")
# And highlight differences that are above the atol
imageio.imwrite(diffs_dir / f"{module}-rgb.png", diffs_rgba[..., :3])
imageio.imwrite(diffs_dir / f"{module}-alpha.png", diffs_rgba[..., 3])
imageio.imwrite(
diffs_dir / f"{module}-rgb-above_atol.png", diffs_rgba_above_atol[..., :3]
)
imageio.imwrite(
diffs_dir / f"{module}-alpha-above_atol.png", diffs_rgba_above_atol[..., 3]
)
imageio.imwrite(diffs_dir / f"{module}.png", img)


@pytest.mark.parametrize("module", examples_to_run)
Expand Down
32 changes: 32 additions & 0 deletions wgpu/backends/wgpu_native/_api.py
Original file line number Diff line number Diff line change
Expand Up @@ -367,6 +367,21 @@ def request_adapter_sync(
This is the implementation based on wgpu-native.
"""
check_can_use_sync_variants()
# Similar to https://github.com/gfx-rs/wgpu?tab=readme-ov-file#environment-variables
# It seems that the environment variables are only respected in their
# testing environments maybe????
# In Dec 2024 we couldn't get the use of their environment variables to work
# This should only be used in testing environments and API users
# should beware
# We chose the variable name WGPUPY_WGPU_ADAPTER_NAME instead WGPU_ADAPTER_NAME
# to avoid a clash
if adapter_name := os.getenv(("WGPUPY_WGPU_ADAPTER_NAME")):
adapters = self.enumerate_adapters_sync()
adapters_llvm = [a for a in adapters if adapter_name in a.summary]
if not adapters_llvm:
raise ValueError(f"Adapter with name '{adapter_name}' not found.")
return adapters_llvm[0]
almarklein marked this conversation as resolved.
Show resolved Hide resolved

awaitable = self._request_adapter(
power_preference=power_preference,
force_fallback_adapter=force_fallback_adapter,
Expand Down Expand Up @@ -394,6 +409,23 @@ async def request_adapter_async(
canvas : The canvas that the adapter should be able to render to. This can typically
be left to None. If given, the object must implement ``WgpuCanvasInterface``.
"""
# Similar to https://github.com/gfx-rs/wgpu?tab=readme-ov-file#environment-variables
# It seems that the environment variables are only respected in their
# testing environments maybe????
# In Dec 2024 we couldn't get the use of their environment variables to work
# This should only be used in testing environments and API users
# should beware
# We chose the variable name WGPUPY_WGPU_ADAPTER_NAME instead WGPU_ADAPTER_NAME
# to avoid a clash
if adapter_name := os.getenv(("WGPUPY_WGPU_ADAPTER_NAME")):
# Is this correct for async??? I know nothing of async...
almarklein marked this conversation as resolved.
Show resolved Hide resolved
awaitable = self.enumerate_adapters_async()
adapters = await awaitable
adapters_llvm = [a for a in adapters if adapter_name in a.summary]
if not adapters_llvm:
raise ValueError(f"Adapter with name '{adapter_name}' not found.")
return adapters_llvm[0]

awaitable = self._request_adapter(
power_preference=power_preference,
force_fallback_adapter=force_fallback_adapter,
Expand Down
Loading