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

NPM: Playwright does not work #16899

Closed
d4h0 opened this issue Dec 2, 2022 · 46 comments · Fixed by #27779
Closed

NPM: Playwright does not work #16899

d4h0 opened this issue Dec 2, 2022 · 46 comments · Fixed by #27779
Labels
needs investigation requires further investigation before determining if it is an issue or not node API Related to various "node:*" modules APIs node compat

Comments

@d4h0
Copy link

d4h0 commented Dec 2, 2022

Hi,

I'm trying to use Playwright via the new NPM compatibility layer, but it fails with:

Uncaught TypeError: browserType.launch: Cannot read properties of undefined (reading 'on')

The following snippet can be used to reproduce the error:

import { chromium } from 'npm:playwright';

async function main() {
  const browser = await chromium.launch({
    headless: false,
  });
  const page = await browser.newPage();
  await page.goto('http://example.com');
  await browser.close();
};

if (import.meta.main) {
  main()
}

After running the above via deno run --unstable --allow-all main.ts, the following message is displayed:

error: Uncaught Error: browserType.launch: Executable doesn't exist at /home/user/.cache/ms-playwright/chromium-1033/chrome-linux/chrome
╔═════════════════════════════════════════════════════════════════════════╗
║ Looks like Playwright Test or Playwright was just installed or updated. ║
║ Please run the following command to download new browsers:              ║
║                                                                         ║
║     npx playwright install                                              ║
║                                                                         ║
║ <3 Playwright Team                                                      ║
╚═════════════════════════════════════════════════════════════════════════╝

Executing deno run --unstable --allow-all npm:playwright install downloads the required binaries.

Executing deno run --unstable --allow-all main.ts again, leads to the above-mentioned error:

error: Uncaught TypeError: browserType.launch: Cannot read properties of undefined (reading 'on')
=========================== logs ===========================
<launching> /home/user/.cache/ms-playwright/chromium-1033/chrome-linux/chrome --disable-field-trial-config --disable-background-networking --enable-features=NetworkService,NetworkServiceInProcess --disable-background-timer-throttling --disable-backgrounding-occluded-windows --disable-back-forward-cache --disable-breakpad --disable-client-side-phishing-detection --disable-component-extensions-with-background-pages --disable-component-update --no-default-browser-check --disable-default-apps --disable-dev-shm-usage --disable-extensions --disable-features=ImprovedCookieControls,LazyFrameLoading,GlobalMediaControls,DestroyProfileOnBrowserClose,MediaRouter,DialMediaRouteProvider,AcceptCHFrame,AutoExpandDetailsElement,CertificateTransparencyComponentUpdater,AvoidUnnecessaryBeforeUnloadCheckSync,Translate --allow-pre-commit-input --disable-hang-monitor --disable-ipc-flooding-protection --disable-popup-blocking --disable-prompt-on-repost --disable-renderer-backgrounding --disable-sync --force-color-profile=srgb --metrics-recording-only --no-first-run --enable-automation --password-store=basic --use-mock-keychain --no-service-autorun --export-tagged-pdf --no-sandbox --user-data-dir=/tmp/playwright_chromiumdev_profile-4e2wKK --remote-debugging-pipe --no-startup-window
<launched> pid=835204
============================================================

Playwright/Deno was already discussed in #16298, but I thought it makes sense to open an issue that focuses only on Playwright. There was also some discussion in the Playwright repo: microsoft/playwright#3146

@uasi
Copy link

uasi commented Dec 2, 2022

I ran deno run --unstable --allow-all --inspect-brk main.ts and examined the call stack.

Screenshot
% deno --version
deno 1.28.3 (release, aarch64-apple-darwin)
v8 10.9.194.5
typescript 4.8.3

The cause is that stdio[3] and stdio[4] are undefined here. Through these pipes Playwright communicates with the launched Chromium process.

https://github.com/microsoft/playwright/blob/v1.28.1/packages/playwright-core/src/server/browserType.ts#L257-L258

      const stdio = launchedProcess.stdio as unknown as [NodeJS.ReadableStream, NodeJS.WritableStream, NodeJS.WritableStream, NodeJS.WritableStream, NodeJS.ReadableStream];
      transport = new PipeTransport(stdio[3], stdio[4]);

@bartlomieju bartlomieju added node compat needs investigation requires further investigation before determining if it is an issue or not labels Dec 2, 2022
@uasi
Copy link

uasi commented Dec 2, 2022

It seems deno's child_process shim (and ultimately Deno.Command) does not yet support to open fds other than std{in,out,err}.

https://github.com/denoland/deno_std/blob/0.167.0/node/internal/child_process.ts#L153

@tamusjroyce
Copy link

tamusjroyce commented Dec 6, 2022

http://docs.libuv.org/en/v1.x/guide/processes.html

This is correct. Deno doesn't currently support libuv's cross platform api for fds other than std{in,out,err}. via #16298

libuv's icon is a dino with a unicorn. Kind of makes you wonder. :-)

But yes. My incident is about supporting all of libuv. Not just a few additional pipelines beyond stdin/stdout/stderr.

@d4h0
Copy link
Author

d4h0 commented Dec 25, 2022

As @tamusjroyce wrote:

There are two ways for playwright and puppeteer to communicate with chrome/edge. websockets or signals & streams.

So if signals & streams are too difficult to implement (in the short-term), then Websockets might be a more feasible way to run Playwright under Deno.

I actually tried that in the past already (see here), but that attempt also failed.

After re-reading everything about Deno/Playwright, I realized that there might be different reasons for why the Websocket route doesn't work, so I tried again.

The resulting error is as follows:

error: Uncaught Error: browserType.connectOverCDP: WebSocket error: ws://127.0.0.1:39959/devtools/browser/dc49d18d-00b4-4a28-9de9-92208e57363d 101 Switching Protocols
=========================== logs ===========================
<ws connecting> ws://127.0.0.1:39959/devtools/browser/dc49d18d-00b4-4a28-9de9-92208e57363d
<ws unexpected response> ws://127.0.0.1:39959/devtools/browser/dc49d18d-00b4-4a28-9de9-92208e57363d 101 Switching Protocols
<ws error> error WebSocket was closed before the connection was established
<ws connect error> ws://127.0.0.1:39959/devtools/browser/dc49d18d-00b4-4a28-9de9-92208e57363d WebSocket was closed before the connection was established
<ws disconnected> ws://127.0.0.1:39959/devtools/browser/dc49d18d-00b4-4a28-9de9-92208e57363d code=1006 reason=
<ws error> error WebSocket was closed before the connection was established
<ws connect error> ws://127.0.0.1:39959/devtools/browser/dc49d18d-00b4-4a28-9de9-92208e57363d WebSocket was closed before the connection was established
<ws disconnected> ws://127.0.0.1:39959/devtools/browser/dc49d18d-00b4-4a28-9de9-92208e57363d code=1006 reason=
============================================================

If I run my test script (see below) with Node.js, everything works as expected.

So somehow Websockets are not behaving as expected under Deno.

If someone wants to give this a try, here is how to do it:

  1. Create the file main.ts with the following content:
import { chromium } from 'npm:playwright';
// import { chromium } from 'playwright';

// To run this script with Node.js (e.g. for testing),
// replace the above lines (don't forget to install
// Playwright via `npm i playwright`)

async function main() {
  // If this line is uncommented, the script fails.
  // This will print the executed command, which we'll
  // need soon:
  const browser = await chromium.launch({ headless: false });
  
  // If this line is uncommented (the above line should be commented out),
  // then Playwright will connect to the browser via Websocket.
  // `$WEBSOCKET_ADDRESS` needs to be replaced with a valid Websocke
  // address (see later step)
  // const browser = await chromium.connectOverCDP("$WEBSOCKET_ADDRESS");

  const page = await browser.newPage();
  await page.goto('http://example.com');
  const title = await page.title();
  console.log(`=> Page title: ${title}`)
  await browser.close();
};

main()
  1. Run the script via deno run --unstable --allow-all main.ts
  2. Copy the command that was executed from the printed log (the part after <launching>)
  3. Replace --remote-debugging-pipe with --remote-debugging-port=0
  4. Execute the resulting command
  5. This will start a browser, which will print a line starting with DevTools listening on ws://
  6. Copy the displayed Websocket URL (the text after DevTools listening on till the end of the line)
  7. Replace $WEBSOCKET_ADDRESS in the script with the Websocket URL you just copied
  8. Switch the script to use Websockets (see the comment above the chromium.connectOverCDP line)
  9. Execute deno run --unstable --allow-all main.ts

As mentioned above, this will result in a Websocket error.

Does anyone have any clue what the problem could be?

@d4h0
Copy link
Author

d4h0 commented Dec 25, 2022

I've set up a "man-in-the-middle" proxy to see what Websocket requests Node.js and Deno are sending, to figure out what's going on.

Here is the request and response when Node.js executes the above script:

Request:

GET /devtools/browser/dc49d18d-00b4-4a28-9de9-92208e57363d HTTP/1.1
Sec-WebSocket-Version: 13
Sec-WebSocket-Key: V9zFa5WtpjHUIIAfif2+LQ==
Connection: Upgrade
Upgrade: websocket
User-Agent: Playwright/1.29.0 (x64; arch unknown) node/19.2
Host: 127.0.0.1:39959
content-length: 0

Response:

HTTP/1.1 101 WebSocket Protocol Handshake
Upgrade: WebSocket
Connection: Upgrade
Sec-WebSocket-Accept: ZG7YduOgTTPTF8X+QAb3wvI3tiY=
content-length: 0

The WebSocket tab shows that 81 Websocket messages where sent (I don't see an option to export them, unfortunately. But this doesn't seem to be relevant, see below).

The last displayed message was Closed by client with code 1005.

Here is the request and response when Deno executes the above script:

Request:

GET /devtools/browser/dc49d18d-00b4-4a28-9de9-92208e57363d HTTP/1.1
sec-websocket-version: 13
sec-websocket-key: HGN6NYPMLjBIFVkwSfW69g==
connection: Upgrade
upgrade: websocket
user-agent: Playwright/1.28.1 (x64; arch unknown) node/16.17
accept: */*
accept-language: *
accept-encoding: gzip, br
host: 127.0.0.1:39959
content-length: 0

Response:

HTTP/1.1 101 WebSocket Protocol Handshake
Upgrade: WebSocket
Connection: Upgrade
Sec-WebSocket-Accept: sza7yPet20xGLEbspLeZ8ZQxGDg=
content-length: 0

The WebSocket tab shows that 0 Websocket messages where sent.

The last displayed message was Closed by client with code 1006.

There are a few differences I see:

  1. The user agent is different

With Node.js the user agent is Playwright/1.29.0 (x64; arch unknown) node/19.2.

With Deno the user agent is Playwright/1.28.1 (x64; arch unknown) node/16.17.

The question is, why does Deno not use the latest version of Playwright? And why is the Node.js version so old with Deno?

  1. The Websocket exit code:

With Node.js the exit code is 1005.

Here is a list of Websocket status codes, and their meaning. For 1005 it says:

# 1005 CLOSED_NO_STATUS

1005 is a reserved value and MUST NOT be set as a status code in a Close control frame by an endpoint. It is designated for use in applications expecting a status code to indicate that no status code was actually present.

With Deno the exit code is 1006, which means (also from the above link):

# 1006 CLOSE_ABNORMAL

1006 is a reserved value and MUST NOT be set as a status code in a Close control frame by an endpoint. It is designated for use in applications expecting a status code to indicate that the connection was closed abnormally, e.g., without sending or receiving a Close control frame.

So something doesn't seem to work properly.

Does anybody have any clue what the reason for 1) and 2) could be?

PS: If anybody wants to see themselves what Websocket messages are sent, here is how:

  1. Install mitmproxy. This is a proxy server that can be put between Playwright and the HTTP server of the browser, to see what requests are sent
  2. Execute mitmweb --mode reverse:http://127.0.0.1:$WEBSOCKET_PORT ($WEBSOCKET_PORT needs to be replaced with the port of $WEBSOCKET_ADDRESS)
  3. Change the host and port part of $WEBSOCKET_ADDRESS in the script to 127.0.0.1:8080
  4. Go to http://http://127.0.0.1:8082/, to open the mitmproxy web UI
  5. Execute the script
  6. Now you can see the request that was sent

@d4h0
Copy link
Author

d4h0 commented Dec 25, 2022

I've tried running my test script with the --inspect-brk option (thanks @uasi, for mentioning that option!), which lead to some odd discoveries.

First, I'll list the steps to reproduce what I did:

  1. Execute deno run --unstable --allow-all --inspect-brk main.ts
  2. Open chrome://inspect in Chromium to connect the debugger
  3. Under Targets click on inspect near the Deno target
  4. Click on the stop sign icon (right side panel, top icon bar), and then activate Pause on caught exceptions
  5. Press the play button ("Resume script execution")

Now script execution stops on any caught exceptions.

The following two exceptions seem odd to me:

  • Cannot find module 'bufferutil'
  • Cannot find module 'utf-8-validate'

Could these lead to problems?

Besides that, there are some breakpoints/exceptions where I have no clue what they are about.

And there is (of course) the exception that is printed to the terminal.

With the above approach it should be not too difficult to find the location where things break.

@d4h0
Copy link
Author

d4h0 commented Dec 26, 2022

Today I continued to debug "Playwright via Websockets".

(Btw., I'm describing what I do in so much detail partly so I myself can reproduce it, e.g., if I have to stop and come back a few weeks/months later).


The following two exceptions seem odd to me:

  • Cannot find module 'bufferutil'
  • Cannot find module 'utf-8-validate'

Node.js also has a --inspect-brk option. The above exceptions are also raised with Node.js, so don't seem to be a problem.


While debugging, it makes sense to disable the timeout of the chromium.connectOverCDP call. The endpoint can also be simplified:

const browser = await chromium.connectOverCDP("http://127.0.0.1:8080/", { timeout: 0 });

(Defining the endpoint as the address of the HTTP server breaks support for the "man-in-the-middle" proxy. If the proxy is needed, then the Websocket address needs to be supplied directly to Playwright)


With Deno the user agent is Playwright/1.28.1 (x64; arch unknown) node/16.17.
The question is, why does Deno not use the latest version of Playwright? And why is the Node.js version so old with Deno?

Importing Playwright via import { chromium } from 'npm:playwright@^1.29.1'; and adding the --reload option to Deno updates Playwright to the last version (maybe --reload would have been enough).

However, that doesn't seem to change anything

Request & response of the upgraded version
GET /devtools/browser/dc49d18d-00b4-4a28-9de9-92208e57363d HTTP/1.1
sec-websocket-version: 13
sec-websocket-key: kzNBDIAF44Oa9LDirK8y9A==
connection: Upgrade
upgrade: websocket
user-agent: Playwright/1.29.1 (x64; arch unknown) node/16.17
accept: */*
accept-language: *
accept-encoding: gzip, br
host: 127.0.0.1:39959
content-length: 0


HTTP/1.1 101 WebSocket Protocol Handshake
Upgrade: WebSocket
Connection: Upgrade
Sec-WebSocket-Accept: uv5MBqXMM5q6pLKwzmwr1X/pBsA=
content-length: 0

Here is where the exception is triggered (playwright-core/src/server/transport.ts#L82):

      transport._ws.on('unexpected-response', (request: ClientRequest, response: IncomingMessage) => {
        const chunks: Buffer[] = [];
        const errorPrefix = `${url} ${response.statusCode} ${response.statusMessage}`;
        response.on('data', chunk => chunks.push(chunk));
        response.on('close', () => {
          const error = chunks.length ? `${errorPrefix}\n${Buffer.concat(chunks)}` : errorPrefix;
          progress?.log(`<ws unexpected response> ${error}`);
          reject(new Error('WebSocket error: ' + error));
          transport._ws.close();
        });
      });

The value of the error that this code triggers is:

"ws://127.0.0.1:8080/devtools/browser/dc49d18d-00b4-4a28-9de9-92208e57363d 101 Switching Protocols"

transport._ws is an instance of WebSocket defined in the ws package.

Here are the API docs for the unexpected-response event (not really anything useful).

The request argument is https://nodejs.org/api/http.html#class-httpclientrequest, and the response argument is https://nodejs.org/api/http.html#class-httpincomingmessage.


Here is the location where the unexpected-response event is emitted.

This code is within an event handler that handles responses for the WebSocket client (which are expected to be redirects).

The odd thing is, that the response has the status code 101 Switching Protocols (see error above, or via debugger).

This is odd, because below the response handler is a upgrade event handler (see here).

It seems, the response is somehow wrongly categories as regular response, instead of as an upgrade...

So I guess, the 'NPM compat' code does not implement https://nodejs.org/api/http.html#event-upgrade correctly.

This can be reproduced via the example of Event: 'upgrade' (replace the import at the top with import * as http from "https://deno.land/[email protected]/node/http.ts";, to make it compatible with Deno).

This example is definitely a much better target to debug, than Playwright directly.

I tried to figure out why the upgrade event isn't emitted, but I'm not familiar with the code base of Deno/Node.js, and I'm not really a TypeScript programmer. It seems, that the upgrade event isn't implemented at all for the HTTP client (it is, however, for the HTTP server).

I think it makes sense to open a separate issue for this (to eliminate all the unnecessary information). It should be linked below this comment.

@bartlomieju
Copy link
Member

bartlomieju commented Dec 26, 2022

Hey @d4h0 thanks for detailed examination, this should be enough to debug the problem on our side.

I can answer some questions outright:

With Deno the user agent is Playwright/1.28.1 (x64; arch unknown) node/16.17.

The question is, why does Deno not use the latest version of Playwright? And why is the Node.js version so old with Deno?

That's probably because you used unversioned import and you had older version of Playwright cached, as you noticed --reload flag fixed the problem (ie. Deno pulls the latest Playwrigth package available). The node/16.17 is because that's what we were targeting in our Node compat layer, but it should most likely be updated as we target latest LTS version these days - opened denoland/std#3057 to fix it.

This can be reproduced via the example of Event: 'upgrade' (replace the import at the top with import * as http from "https://deno.land/[email protected]/node/http.ts";, to make it compatible with Deno).

This seems like main crux of the problem and we received reports that Vite is not working properly with Deno too; it uses HTTP upgrade as well. We are currently working on a rewrite of our HTTP server that should help us fix this problem. Obviously if we could fix up the problem before rewrite of HTTP server lands that would be preferable. @kt3k could you take a look at this problem?

EDIT: Ooops, it seems the problem is with the HTTP client, not HTTP server. So the Vite problem is not related to this one.

@d4h0
Copy link
Author

d4h0 commented Jan 2, 2023

@bartlomieju: Thank you for the feedback!

Ooops, it seems the problem is with the HTTP client, not HTTP server. So the Vite problem is not related to this one.

Is there also a rewrite of the HTTP client planned?

In theory, it shouldn't be too difficult to add HTTP upgrade support to the client. In practice, I gave up because I'm too unfamiliar with the code base and TypeScript itself (also, because I already spent three full days debugging this).

I think, Deno is in almost every way better than Node.js. Unfortunately, Playwright is essential for me, and it is too much work to maintain systems written for two different JavaScript runtimes. So, at least for me, the lack of Playwright on Deno is blocker for switching to Deno (besides Playwright, everything I need can be replaced with something else that works on Deno).

@tamusjroyce
Copy link

tamusjroyce commented Jan 3, 2023

@d4h0 if that is your only blocker, try https://github.com/kt3k/deno-bin

Then package.json can be upgraded to use deno where it makes sense. And e2e test can run via node. At least until deno is fully compatible.

My biggest issue with deno is my fingers will still type node and npm. Should be easy to have deno compile bin to run package.json scripts & alias deno & translate command line params when running node & npm from folder :)

@d4h0
Copy link
Author

d4h0 commented Jan 3, 2023

@tamusjroyce: Thanks for the tip, this looks pretty interesting.

The biggest advantage of Deno over Node.js (for me) is the fact that you can embed Deno easily into Rust applications (via the deno_core Rust library).

Currently, I'm using Node.js via a homegrown RPC utility. Deno would allow me to integrate much more ergonomically with my applications (also because Deno natively runs Wasm, so I even can run Rust directly in Deno, which makes many things easier).

My main goal for Playwright on Deno is to somehow make it possible to ergonomically use Playwright from Rust (there is already playwright-rust, but it doesn't support the latest Playwright version, and embeds a Zip binary of a Node.js application, which is unzipped at runtime, which I don't really like).

That being said, I probably will still start using kt3k/deno-bin with my current RPC-setup to get used to and benefit from the better tooling that Deno offers. Thanks again!

@progrium
Copy link

With #19412 is this closable?

@kt3k
Copy link
Member

kt3k commented Jul 26, 2023

The steps in #16899 (comment) seem working now! But it looks too hacky to me (involving a lot of manual copy pasting during test run). Is there any reasonable option to start playwright with websocket configuration? cc @d4h0

@d4h0
Copy link
Author

d4h0 commented Jul 26, 2023

The steps in #16899 (comment) seem working now!

That sounds fantastic!

Unfortunately, I didn't have any time yet, to play with Playwright on Deno after the recent changes related to this issue.

But it looks too hacky to me (involving a lot of manual copy pasting during test run). Is there any reasonable option to start playwright with websocket configuration?

It should be possible to run Playwright similarly to how you'd run Playwright on Node.js.

For example, BrowserType::launchServer should be usable to start the browser and to get a WebSocket address via BrowserServer::wsEndpoint.

After that, we should be able to use BrowserType::connect to connect to the WebSocket endpoint (instead of using BrowserType::connectOverCDP, as the last test script does).

For example:

import { chromium } from 'npm:playwright';

async function main() {
  const browserServer = await chromium.launchServer();
  const wsEndpoint = browserServer.wsEndpoint();
  const browser = await chromium.connect(wsEndpoint);
  const page = await browser.newPage();
  await page.goto('http://example.com');
  const title = await page.title();
  console.log(`=> Page title: ${title}`)
  await browser.close();
};

main()

(Save to main.js and run via deno run --unstable --allow-all main.js. --unstable might not be required anymore, I'm not sure).

Unfortunate, that fails with the following error:

Uncaught TypeError: Cannot read properties of undefined (reading 'on') Failed to launch browser

Full error
error: Uncaught TypeError: Cannot read properties of undefined (reading 'on') Failed to launch browser.
==================== Browser output: ====================
<launching> /home/user/.cache/ms-playwright/chromium-1041/chrome-linux/chrome --disable-field-trial-config --disable-background-networking --enable-features=NetworkService,NetworkServiceInProcess --disable-background-timer-throttling --disable-backgrounding-occluded-windows --disable-back-forward-cache --disable-breakpad --disable-client-side-phishing-detection --disable-component-extensions-with-background-pages --disable-component-update --no-default-browser-check --disable-default-apps --disable-dev-shm-usage --disable-extensions --disable-features=ImprovedCookieControls,LazyFrameLoading,GlobalMediaControls,DestroyProfileOnBrowserClose,MediaRouter,DialMediaRouteProvider,AcceptCHFrame,AutoExpandDetailsElement,CertificateTransparencyComponentUpdater,AvoidUnnecessaryBeforeUnloadCheckSync,Translate --allow-pre-commit-input --disable-hang-monitor --disable-ipc-flooding-protection --disable-popup-blocking --disable-prompt-on-repost --disable-renderer-backgrounding --disable-sync --force-color-profile=srgb --metrics-recording-only --no-first-run --enable-automation --password-store=basic --use-mock-keychain --no-service-autorun --export-tagged-pdf --headless --hide-scrollbars --mute-audio --blink-settings=primaryHoverType=2,availableHoverTypes=2,primaryPointerType=4,availablePointerTypes=4 --no-sandbox --proxy-server=socks5://127.0.0.1:34541 --proxy-bypass-list=<-loopback> --user-data-dir=/tmp/playwright_chromiumdev_profile-iGNzNf --remote-debugging-pipe --no-startup-window
<launched> pid=448716
    at new PipeTransport (file:///home/user/.cache/deno/npm/registry.npmjs.org/playwright-core/1.29.1/lib/server/pipeTransport.js:37:14)
    at Chromium._launchProcess (file:///home/user/.cache/deno/npm/registry.npmjs.org/playwright-core/1.29.1/lib/server/browserType.js:241:19)
    at Object.runMicrotasks (ext:core/01_core.js:836:30)
    at processTicksAndRejections (ext:deno_node/_next_tick.ts:53:10)
    at runNextTicks (ext:deno_node/_next_tick.ts:71:3)
    at eventLoopTick (ext:core/01_core.js:189:21)
    at async Chromium._innerLaunch (file:///home/user/.cache/deno/npm/registry.npmjs.org/playwright-core/1.29.1/lib/server/browserType.js:105:9)
    at async Chromium._innerLaunchWithRetries (file:///home/user/.cache/deno/npm/registry.npmjs.org/playwright-core/1.29.1/lib/server/browserType.js:86:14)
    at async ProgressController.run (file:///home/user/.cache/deno/npm/registry.npmjs.org/playwright-core/1.29.1/lib/server/progress.js:99:22)
    at async Chromium.launch (file:///home/user/.cache/deno/npm/registry.npmjs.org/playwright-core/1.29.1/lib/server/browserType.js:63:21)

The pipe in the error message made me think, that the issue might be the same as with the default way to run Playwright (Playwright on Node.js normally connects to the browser via pipes, not via a WebSocket), so I tried to disable this "pipe mode" via:

  const browserServer = await chromium.launchServer({ignoreDefaultArgs: ["--remote-debugging-pipe"]});

...but this doesn't seem to change anything, unfortunately.

The error is triggered by the first line (that contains chromium.launchServer) – so a workaround might be:

  1. Somehow automate the manual way to start the browser
  2. Write a small Node.js script that starts the browser, to which Deno-Playwright can connect.

Both options are not great, however, so a fix for the above error would be the ideal solution.

Unfortunately, I don't have more time to play with this, right now.

@Ravenstine
Copy link

The code in @d4h0 's code actually can partially work today when the option useWebSocket: true is passed to chromium.launchServer. This option doesn't appear documented, but it almost gets things working out of the box.

The problem is that, when the browser is navigated, such as in await page.goto('http://example.com');, the page will almost immediately close and the following error presents itself in the console:

error: Uncaught (in promise) Error: page.goto: Browser has been closed
    ==== Closed by ====
    at Proxy.emit (ext:deno_node/_events.mjs:382:28)
    at Object.action (ext:deno_web/02_timers.js:153:11)
    at handleTimerMacrotask (ext:deno_web/02_timers.js:67:10)
    at eventLoopTick (ext:core/01_core.js:189:21)

This is effectively covering up the actual error which is somehow leading to the page getting closed by Playwright:

Uncaught RangeError: Invalid WebSocket frame: RSV2 and RSV3 must be clear

Unfortunately, I didn't copy the whole stack trace to my notes, and I lost track of where it originally occurred. Might have been from the "Connection" class. Not sure now.

In any case, I'm pretty sure that this seems indicative that there's something either incorrect with Deno's WebSocket implementation or with Playwright's handling of its Node shim for WebSocket. Though it's possible to successfully use some methods like evaluate on the initial blank page, any navigation causes this issue.

In the meantime, I came up with a way to automate the connectOverCDP approach that seems to get Playwright to work with Chromium under Deno with seemingly all basic functionality working properly.

import { chromium } from 'playwright';

// Since there's no good way for us to get the
// websocket that Chromium randomly generates,
// we must generate our own and keep track of it.
const port = getRandomPortNum();
const browserServer = await chromium.launchServer({
  // This option isn't documented, but sets up the server
  // to listen to a debugging port.  Unfortunately, this doesn't
  // seem to work out of the box, at least in Deno.  By itself,
  // a window can be launched, but encounters an error:
  // `Uncaught RangeError: Invalid WebSocket frame: RSV2 and RSV3 must be clear`.
  // So instead, we're going to connect to a devtools websocket
  // later.  However, I'm still including this because it
  // allows us to await the browserServer object.
  //
  // If this option isn't available to you with your
  // version of Playwright, you can remove it if you
  // also remove the `await` keyword before `chromium.launchServer`.
  // However, you won't have access to a BrowserServer
  // object if you go down that route.
  useWebSocket: true,
  // Pass in our randomly generated port.
  args: [`--remote-debugging-port=${port}`],
  // Prevent Playwright from overriding our custom port number.
  // The "--remote-debugging-pipe" flag is already removed by
  // `useWebSocket`, but I left it here in case you want to
  // remove that option and not await `chromium.launchServer`.
  ignoreDefaultArgs: ["--remote-debugging-pipe", "--remote-debugging-port=0"],
  // For demonstration purposes
  headless: false,
});

// Even though we've awaited the browser server, sometimes
// the endpoint for information on the devtools websocket
// still isn't totally available.  This seems to happen
// 1/10 attempts.  I wrapped our call to connect in a function
// that will retry until the socket is available.  There's
// probably a better way, but I wanted to make sure this
// was more reliable.
const browser = await tryWithBackoff({
  // I've tried the approach in the following article and it
  // simply isn't usable (yet) in Deno:
  // https://playwright.dev/docs/api/class-browsertype#browser-type-launch-server
  //
  // Instead, we're going to let Playwright ask Chromium for the
  // devtools websocket endpoint.
  fn: () => chromium.connectOverCDP(`http://localhost:${port}`)
});
const page = await browser.newPage();

await page.goto('https://duck.com');

const title = await page.title();

console.log(`=> Page title: ${title}`)

await new Promise(resolve => setTimeout(resolve, 5000));
await browserServer.close();

Deno.exit();

/**
 * This is finds a random port number that is not being used by something else.
 **/
function getRandomPortNum (): number {
  const MIN_PORT_NUM = 1024;
  const MAX_PORT_NUM = 65535;
  const portNum = Math.ceil(Math.random() * ((MAX_PORT_NUM - 1) - MIN_PORT_NUM + 1) + MIN_PORT_NUM + 1);

  try {
    const server = Deno.listen({ port: portNum });

    server.close();

    return portNum;
  } catch (e) {
    if (e.name !== 'AddrInUse') throw e;

    return getRandomPortNum();
  }
}

/**
 * Tries to execute a function and retries (with backoff and timeout) if an error occurs.
 **/
async function tryWithBackoff (args: {
  fn: () => any;
  delay?: number;
  timeout?: number;
  startedAt?: number;
  error?: Error;
}): Promise<any> {
  const { fn, delay, timeout, startedAt, error } = {
    delay: 0,
    timeout: 30000,
    startedAt: Date.now(),
    ...args,
  };

  await new Promise(resolve => setTimeout(resolve, delay));

  if ((Date.now() - startedAt) > timeout) {
    throw (error || new Error('Function call timed out'));
  }

  try {
    return await fn();
  } catch (error) {
    console.error(error);
    return tryWithBackoff({
      fn,
      delay: delay + 1000,
      timeout,
      startedAt,
      error,
    });
  }
}

@jollytoad
Copy link
Contributor

jollytoad commented Nov 6, 2023

Just trying playwright out myself, in Deno 1.38.0, when running:

deno run -A npm:[email protected] install

I get the follow error: process.geteuid is not a function, full error:

error: Uncaught TypeError: process.geteuid is not a function
    at file:///Users/mgibson/Library/Caches/deno/npm/registry.npmjs.org/playwright/1.39.0/lib/transform/compilationCache.js:46:91
    at Object.<anonymous> (file:///Users/mgibson/Library/Caches/deno/npm/registry.npmjs.org/playwright/1.39.0/lib/transform/compilationCache.js:47:3)
    at Object.<anonymous> (file:///Users/mgibson/Library/Caches/deno/npm/registry.npmjs.org/playwright/1.39.0/lib/transform/compilationCache.js:223:4)
    at Module._compile (node:module:733:34)
    at Object.Module._extensions..js (node:module:747:10)
    at Module.load (node:module:658:32)
    at Function.Module._load (node:module:539:12)
    at Module.require (node:module:677:19)
    at require (node:module:791:16)
    at Object.<anonymous> (file:///Users/mgibson/Library/Caches/deno/npm/registry.npmjs.org/playwright/1.39.0/lib/transform/transform.js:20:25)

Running on MacOS btw, so this won't affect Windows, but will for Linux users too, looks like this is the cause...

https://github.com/microsoft/playwright/blob/ffd2e02aa3ba2cbaee8e9de01540b6ab66f1ce3b/packages/playwright/src/transform/compilationCache.ts#L49

@jollytoad
Copy link
Contributor

jollytoad commented Nov 21, 2023

Deno 1.38.2 now implements process.geteuid() so no longer get the above, but the following...

deno run -A npm:[email protected] install
Downloading Chromium 120.0.6099.28 (playwright build v1091) from https://playwright.azureedge.net/builds/chromium/1091/chromium-mac.zip
Failed to install browsers
Error: Failed to download Chromium 120.0.6099.28 (playwright build v1091), caused by
TypeError: cp.send is not a function
    at downloadBrowserWithProgressBarOutOfProcess (file:///Users/mgibson/ahx/ahx_lib/node_modules/.deno/[email protected]/node_modules/playwright-core/lib/server/registry/browserFetcher.js:118:6)
    at downloadBrowserWithProgressBar (file:///Users/mgibson/ahx/ahx_lib/node_modules/.deno/[email protected]/node_modules/playwright-core/lib/server/registry/browserFetcher.js:51:17)
    at eventLoopTick (ext:core/01_core.js:178:11)
    at async Registry._downloadExecutable (file:///Users/mgibson/ahx/ahx_lib/node_modules/.deno/[email protected]/node_modules/playwright-core/lib/server/registry/index.js:738:5)
    at async Registry.install (file:///Users/mgibson/ahx/ahx_lib/node_modules/.deno/[email protected]/node_modules/playwright-core/lib/server/registry/index.js:687:9)
    at async t.<anonymous> (file:///Users/mgibson/ahx/ahx_lib/node_modules/.deno/[email protected]/node_modules/playwright-core/lib/cli/program.js:113:7)

seems ChildProcess is missing send(). (#12879)

But also, after installing browsers via node and manually caching a package, then attempting to run tests...

yarn playwright install
deno cache npm:@playwright/test

deno run -A deno run -A npm:[email protected] test

it did attempt to run the entire test suite, but with lots of warnings...

Warning: Not implemented: process.on("message")
Warning: Not implemented: process.on("disconnect")
Error: worker process exited unexpectedly (code=0, signal=null)

and then successfully started the test report server and opened the report in my browser.

So, I wonder if it's actually quite close to working once message and disconnect events are supported. Do we know if there a fundamental issue preventing those, or simply a case of not got round to it yet?
(EDIT: is this related to #16298 ?)

nathanwhit added a commit that referenced this issue Aug 15, 2024
Linux/macos only currently.

Part of #23524 (fixes it on
platforms other than windows).
Part of #16899  (fixes it on platforms other than windows).

After this PR, playwright is functional on mac/linux.
@nathanwhit
Copy link
Member

This should now be working on canary, and will be in the next release (only on macOS and linux, for now).

import { chromium } from "npm:playwright";

const sleep = (ms) => new Promise((r) => setTimeout(r, ms));

async function main() {
  const browser = await chromium.launch({
    headless: false,
  });
  const page = await browser.newPage();
  await page.goto("http://deno.com");
  await sleep(2000);

  await browser.close();
}

if (import.meta.main) {
  await main();
}
Screen.Recording.2024-08-15.at.12.56.47.PM.mov

@cowboyd
Copy link

cowboyd commented Aug 15, 2024

@nathanwhit You are a hero! 🦸🏻

💪🏻 💪🏻 💪🏻

@jollytoad
Copy link
Contributor

Just trying out Playwright 1.46.1 on Deno 1.46.1, which is a strange coincidence in itself!

Anyway, I can confirm that playwright install now works perfectly.

But I can't get my test suite to run...

Task test deno run --allow-all npm:playwright test
Error: Cannot find module '@playwright/test'
Require stack:
- /Users/mgibson/ahx/ahx_lib/playwright.config.js
- /Users/mgibson/Library/Caches/deno/npm/registry.npmjs.org/playwright/1.46.1/lib/transform/transform.js
- /Users/mgibson/Library/Caches/deno/npm/registry.npmjs.org/playwright/1.46.1/lib/common/config.js
- /Users/mgibson/Library/Caches/deno/npm/registry.npmjs.org/playwright/1.46.1/lib/reporters/json.js
- /Users/mgibson/Library/Caches/deno/npm/registry.npmjs.org/playwright/1.46.1/lib/runner/reporters.js
- /Users/mgibson/Library/Caches/deno/npm/registry.npmjs.org/playwright/1.46.1/lib/runner/runner.js
- /Users/mgibson/Library/Caches/deno/npm/registry.npmjs.org/playwright/1.46.1/lib/program.js
- /Users/mgibson/Library/Caches/deno/npm/registry.npmjs.org/playwright/1.46.1/cli.js
    at Function.Module._resolveFilename (node:module:608:15)
    at Function.resolveFilename (/Users/mgibson/Library/Caches/deno/npm/registry.npmjs.org/playwright/1.46.1/lib/transform/transform.js:212:36)
    at Function.Module._load (node:module:486:27)
    at Module.require (node:module:674:19)
    at require (node:module:798:16)
    at Object.<anonymous> (/Users/mgibson/ahx/ahx_lib/playwright.config.js:1:1)
    at Object.<anonymous> (/Users/mgibson/ahx/ahx_lib/playwright.config.js:86:4)
    at Module._compile (node:module:736:34)
    at Module.f._compile (/Users/mgibson/Library/Caches/deno/npm/registry.npmjs.org/playwright/1.46.1/lib/utilsBundleImpl.js:16:994)
    at Module._extensions..js (node:module:757:11)
    at Object.i.<computed>.ut._extensions.<computed> (/Users/mgibson/Library/Caches/deno/npm/registry.npmjs.org/playwright/1.46.1/lib/utilsBundleImpl.js:16:1010)
    at Module.load (node:module:655:32)
    at Function.Module._load (node:module:523:13)
    at Module.require (node:module:674:19)
    at require (node:module:798:16)
    at requireOrImport (/Users/mgibson/Library/Caches/deno/npm/registry.npmjs.org/playwright/1.46.1/lib/transform/transform.js:192:20)
    at loadUserConfig (/Users/mgibson/Library/Caches/deno/npm/registry.npmjs.org/playwright/1.46.1/lib/common/configLoader.js:96:83)
    at loadConfig (/Users/mgibson/Library/Caches/deno/npm/registry.npmjs.org/playwright/1.46.1/lib/common/configLoader.js:101:28)
    at loadConfigFromFileRestartIfNeeded (/Users/mgibson/Library/Caches/deno/npm/registry.npmjs.org/playwright/1.46.1/lib/common/configLoader.js:258:16)
    at runTests (/Users/mgibson/Library/Caches/deno/npm/registry.npmjs.org/playwright/1.46.1/lib/program.js:194:76)
    at t.<anonymous> (/Users/mgibson/Library/Caches/deno/npm/registry.npmjs.org/playwright/1.46.1/lib/program.js:54:7) {
  code: "MODULE_NOT_FOUND",
  requireStack: [
    "/Users/mgibson/ahx/ahx_lib/playwright.config.js",
    "/Users/mgibson/Library/Caches/deno/npm/registry.npmjs.org/playwright/1.46.1/lib/transform/transform.js",
    "/Users/mgibson/Library/Caches/deno/npm/registry.npmjs.org/playwright/1.46.1/lib/common/config.js",
    "/Users/mgibson/Library/Caches/deno/npm/registry.npmjs.org/playwright/1.46.1/lib/reporters/json.js",
    "/Users/mgibson/Library/Caches/deno/npm/registry.npmjs.org/playwright/1.46.1/lib/runner/reporters.js",
    "/Users/mgibson/Library/Caches/deno/npm/registry.npmjs.org/playwright/1.46.1/lib/runner/runner.js",
    "/Users/mgibson/Library/Caches/deno/npm/registry.npmjs.org/playwright/1.46.1/lib/program.js",
    "/Users/mgibson/Library/Caches/deno/npm/registry.npmjs.org/playwright/1.46.1/cli.js"
  ]
}

Here is my playwright.config.js... https://github.com/jollytoad/ahx/blob/playwright-deno/playwright.config.js
@playwright/test is mapped in my deno.json... https://github.com/jollytoad/ahx/blob/playwright-deno/deno.json#L31

I've deliberated removed node and related tools, so it should be running pure deno.

Any ideas? Anyone else got a playwright test suite to run yet on the recent releases?

@nathanwhit
Copy link
Member

nathanwhit commented Aug 23, 2024

@jollytoad Hello! I dug into this and it seems we have a bug in this case, specifically when you aren't using a node modules directory (#25189).

To work around this for now, you can add

"nodeModulesDir": true

to your deno.json to opt into a node_modules folder.

You'll also need to make sure that the dependencies get cached for playwright.config.js, before you run the tests.
You can do this with just deno cache playwright.config.js (I would put it in your task probably, just before running playwright test)


Testing this, it does seem while the tests do run correctly, the process doesn't always exit cleanly, and I sometimes have to exit with ctrl-c – I haven't looked into what's happening there yet.

@cowboyd
Copy link

cowboyd commented Aug 28, 2024

@nathanwhit Thank you so much for pushing this forward! Is there a write up anywhere of what it will take to get playwright working on Windows? Unfortunately, many of the testers on our team use it, and so I can't fully migrate us until it lands.

@jollytoad
Copy link
Contributor

@nathanwhit thanks for the advice, I can confirm my tests run after setting nodeModulesDir and also running deno cache to ensure the node_modules is populated as expected. And I also get the problem that the test suite appears to hang at the end, I have to manually hit ctrl+c, after which it launches the results web app. I've found that hitting ctrl+c again to then kill the server seems to return me to the shell prompt, but the web server appears to continue to run in the background!

@nathanwhit
Copy link
Member

@nathanwhit Thank you so much for pushing this forward! Is there a write up anywhere of what it will take to get playwright working on Windows? Unfortunately, many of the testers on our team use it, and so I can't fully migrate us until it lands.

Not a full writeup, but the gist of it is that we need to either vendor and modify or handroll std::command::Command from the rust stdlib. Supporting this requires lower level control over process spawning than what the standard library provides, so we have to do it ourselves. It's a fairly substantial task.

@mardukbp
Copy link

The Trait std::os::windows::process::CommandExt provides Windows-specific extensions to the process::Command builder. Which is a process builder, providing fine-grained control over how a new process should be spawned. Maybe this can be used directly or at least as a starting point.

@nathanwhit
Copy link
Member

nathanwhit commented Sep 12, 2024

The Trait std::os::windows::process::CommandExt provides Windows-specific extensions to the process::Command builder. Which is a process builder, providing fine-grained control over how a new process should be spawned. Maybe this can be used directly or at least as a starting point.

Unfortunately that still doesn't give us enough control. We need to set one of the arguments to the underlying CreateProcessW call (the windows api for creating a subprocess), specifically the lpReserved2 field on the StartupInfoW struct passed to CreateProcessW. Additionally, node also relies on setting this field in an undocumented way – the official microsoft documentation states that it must always be null, but node uses this to pass down file descriptors to be inherited by the child process. As a result, the likelihood of upstreaming a change to the rust standard library to give us access to this is incredibly low (not to mention the fact that stabilizing new std APIs typically takes >1 year).

As a result we would need to modify the internals of std::process::Command to support this (likely by vendoring the code into deno), or write our own process spawning logic directly on windows APIs with just the features we need, in order to reduce the maintenance burden.

@mardukbp
Copy link

Thank you for the detailed explanation. I found the code you are referring to: sys::pal::windows::process. I do not know if it can be reused somehow. Maybe it has to be reimplemented using the windows crate from Microsoft.

@magurotuna
Copy link
Member

I tried to run the following script both on macOS and Linux (on GitHub Actions). On macOS it works, while on Linux it doesn't.

import { chromium } from "npm:[email protected]";

async function main() {
  await using _browser = await chromium.launch({
    headless: true,
  });
}

if (import.meta.main) {
  await main();
}

Firefox doesn't work either on Linux (but again, does work on macOS)

import { firefox } from "npm:[email protected]";

async function main() {
  await using _browser = await firefox.launch({
    headless: true,
  });
}

if (import.meta.main) {
  await main();
}

Checked deno 1.46.3, 2.0.0-rc.6, and canary (eff6423), all of these yield the error for the code above.

The actual code and error logs can be seen at magurotuna/playwright-deno-ci#1

@Jess182
Copy link

Jess182 commented Oct 2, 2024

I tried to run the following script both on macOS and Linux (on GitHub Actions). On macOS it works, while on Linux it doesn't.

Same problem and same exception here, on macos works but not in linux

nathanwhit added a commit that referenced this issue Oct 24, 2024
…mode (#26495)

Fixes playwright on linux, as reported in
#16899 (comment).

The issue was that we were opening the socket in nonblocking mode, which
meant that subprocesses trying to use it would get a `EWOULDBLOCK` error
(unexpectedly). The fix here is to only set nonblocking mode on our end
(which we need to use asynchronously)
@nathanwhit
Copy link
Member

@magurotuna @Jess182 playwright on linux should be fixed in the latest canary (deno upgrade canary) if you want to give it a go.

bartlomieju pushed a commit that referenced this issue Oct 25, 2024
…mode (#26495)

Fixes playwright on linux, as reported in
#16899 (comment).

The issue was that we were opening the socket in nonblocking mode, which
meant that subprocesses trying to use it would get a `EWOULDBLOCK` error
(unexpectedly). The fix here is to only set nonblocking mode on our end
(which we need to use asynchronously)
@Jess182
Copy link

Jess182 commented Oct 25, 2024

@magurotuna @Jess182 playwright on linux should be fixed in the latest canary (deno upgrade canary) if you want to give it a go.

Thanks so much @nathanwhit , now works on linux, deno v2.0.3!

@jollytoad
Copy link
Contributor

Awesome work @nathanwhit, thank you very much.

Just to clarify the config and commands to run for anyone else stumbling across this:

Ensure "nodeModulesDir": "auto" is set in deno.json.

  • deno install
  • deno run --allow-all npm:playwright install
  • deno run --allow-all npm:playwright test

I now get results consistent with node, and the process now exits cleanly.

deno run --allow-all npm:playwright test --ui doesn't seem to work yet though:

error: unexpected argument '--experimental-loader' found

tip: a similar argument exists: '--ext'
tip: to pass '--experimental-loader' as a value, use '-- --experimental-loader'

Usage: deno run --quiet --ext [SCRIPT_ARG]...

Looks like that is relying on node specific module loader features!

@nathanwhit
Copy link
Member

Awesome work @nathanwhit, thank you very much.

Just to clarify the config and commands to run for anyone else stumbling across this:

Ensure "nodeModulesDir": "auto" is set in deno.json.

  • deno install
  • deno run --allow-all npm:playwright install
  • deno run --allow-all npm:playwright test

I now get results consistent with node, and the process now exits cleanly.

deno run --allow-all npm:playwright test --ui doesn't seem to work yet though:

error: unexpected argument '--experimental-loader' found
tip: a similar argument exists: '--ext'
tip: to pass '--experimental-loader' as a value, use '-- --experimental-loader'
Usage: deno run --quiet --ext [SCRIPT_ARG]...

Looks like that is relying on node specific module loader features!

Ah, yes that is a known issue. Playwright primarily uses loaders to transpile typescript ESM at runtime. That isn't actually needed in deno, since we support typescript natively. You can set the PW_DISABLE_TS_ESM environment variable to work around it.

@magurotuna
Copy link
Member

Thanks so much @nathanwhit, I've confirmed that my very simple test suite passes with the canary version on Linux: https://github.com/magurotuna/playwright-deno-ci/actions/runs/11568994515/job/32201894429?pr=2

@cwegener
Copy link

Thanks so much @nathanwhit, I've confirmed that my very simple test suite passes with the canary version on Linux: https://github.com/magurotuna/playwright-deno-ci/actions/runs/11568994515/job/32201894429?pr=2

The fix is also in the sable release starting with v2.0.3

@acrodrig
Copy link

acrodrig commented Jan 8, 2025

Can this be closed?

@cowboyd
Copy link

cowboyd commented Jan 8, 2025

@acrodrig Dunno if it should be considered a blocker, but there is still the issue of windows support

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
needs investigation requires further investigation before determining if it is an issue or not node API Related to various "node:*" modules APIs node compat
Projects
None yet
Development

Successfully merging a pull request may close this issue.