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

fix(runner): Properly attribute error messages to locator handlers #34074

Closed
wants to merge 2 commits into from

Conversation

agg23
Copy link
Contributor

@agg23 agg23 commented Dec 18, 2024

Fix #34070. Provide a mechanism for overriding apiName in situations where the stack trace doesn't indicate the actual source of the test error.

@agg23 agg23 requested review from dgozman and Skn0tt December 18, 2024 19:17
@agg23
Copy link
Contributor Author

agg23 commented Dec 18, 2024

Please let me know if you have a better idea of how to plumb the overridden API name through the error reporting system. I'm not a fan of adding a new Error type, but it's along the lines of what @Skn0tt recommended.

await page.locator('ul > li').first().boundingBox();

// Unreachable
expect(false).toBeTruthy();
Copy link
Contributor Author

Choose a reason for hiding this comment

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

If there's a nicer way to handle testing for an exception, please let me know. I couldn't find a way for expect.toThrow() to be async.

This comment has been minimized.

Copy link
Contributor

Test results for "tests 1"

5 flaky ⚠️ [firefox-page] › page/page-evaluate.spec.ts:403:3 › should throw for too deep reference chain @firefox-ubuntu-22.04-node18
⚠️ [playwright-test] › ui-mode-test-output.spec.ts:80:5 › should show console messages for test @ubuntu-latest-node20-1
⚠️ [webkit-library] › library/logger.spec.ts:34:3 › should log context-level @webkit-ubuntu-22.04-node18
⚠️ [webkit-page] › page/page-leaks.spec.ts:161:5 › waitFor should not leak @webkit-ubuntu-22.04-node18
⚠️ [webkit-page] › page/page-set-input-files.spec.ts:205:3 › should upload multiple large files @webkit-ubuntu-22.04-node18

37344 passed, 650 skipped
✔️✔️✔️

Merge workflow run.

@@ -203,15 +204,18 @@ export abstract class ChannelOwner<T extends channels.Channel = channels.Channel
return result;
} catch (e) {
const innerError = ((process.env.PWDEBUGIMPL || isUnderTest()) && e.stack) ? '\n<inner error>\n' + e.stack : '';
if (apiName && !apiName.includes('<anonymous>'))
e.message = apiName + ': ' + e.message;

Copy link
Member

Choose a reason for hiding this comment

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

We should have a very good reason to change the fundamentals (channelOwner) for a niche DevX like the one in the bug. This isn't even an end-user reported feature, so I'd rather minimize the maintenance cost.

public apiNameOverride: string;

constructor(error: Exclude<SerializedError['error'], undefined>, apiNameOverride: string) {
super(error.message);
Copy link
Member

Choose a reason for hiding this comment

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

We already have 4 layers of processing of the api names and errors, adding another one for this use case will add to the confusion.

Copy link
Member

@Skn0tt Skn0tt left a comment

Choose a reason for hiding this comment

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

Oof, wasn't expecting this fix to become this big of a change! I see how that happens though. Here's a sketch for an alternative solution that doesn't require any protocol changes:

diff --git a/packages/playwright-core/src/client/channelOwner.ts b/packages/playwright-core/src/client/channelOwner.ts
index a5d753507..6615c0031 100644
--- a/packages/playwright-core/src/client/channelOwner.ts
+++ b/packages/playwright-core/src/client/channelOwner.ts
@@ -203,7 +203,10 @@ export abstract class ChannelOwner<T extends channels.Channel = channels.Channel
       return result;
     } catch (e) {
       const innerError = ((process.env.PWDEBUGIMPL || isUnderTest()) && e.stack) ? '\n<inner error>\n' + e.stack : '';
-      if (apiName && !apiName.includes('<anonymous>'))
+      // some APIs have checkpoints that are executed within other APIs
+      if (e.message.startsWith('locatorHandler'))
+        e.message = 'locatorHandler: ' + e.message.substring(...);
+      else if (apiName && !apiName.includes('<anonymous>'))
         e.message = apiName + ': ' + e.message;
       const stackFrames = '\n' + stringifyStackFrames(stackTrace.frames).join('\n') + innerError;
       if (stackFrames.trim())
diff --git a/packages/playwright-core/src/server/page.ts b/packages/playwright-core/src/server/page.ts
index fe483b834..9c7b75179 100644
--- a/packages/playwright-core/src/server/page.ts
+++ b/packages/playwright-core/src/server/page.ts
@@ -466,7 +466,12 @@ export class Page extends SdkObject {
   async performActionPreChecks(progress: Progress) {
     await this._performWaitForNavigationCheck(progress);
     progress.throwIfAborted();
+    try {
     await this._performLocatorHandlersCheckpoint(progress);
+    } catch (e) {
+      e.message = 'locatorHandler: ' + e.message;
+      throw e;
+    }
     progress.throwIfAborted();
     // Wait once again, just in case a locator handler caused a navigation.
     await this._performWaitForNavigationCheck(progress);

This would do the job and be much smaller of a change. It also doesn't feel like much added complexity. What do y'all think?

@agg23 agg23 closed this Jan 22, 2025
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.

[Bug]: addLocatorHandler displays incorrect error message
3 participants