A Playwright wrapper for the Effect ecosystem. This library provides a set of services and layers to interact with Playwright in a type-safe way using Effect.
Note
This library is currently focused on using Playwright for automation and scraping. It does not provide a wrapper for @playwright/test (the test runner).
pnpm add effect-playwright playwright-coreor
npm install effect-playwright playwright-coreYou can also install playwright instead of playwright-core if you want the post-build auto install of the browsers.
import { Playwright } from "effect-playwright";
import { Effect } from "effect";
import { chromium } from "playwright-core";
const program = Effect.gen(function* () {
const playwright = yield* Playwright;
const browser = yield* playwright.launchScoped(chromium);
const page = yield* browser.newPage();
yield* page.goto("https://example.com");
const title = yield* page.title;
console.log(`Page title: ${title}`);
}).pipe(Effect.scoped, Effect.provide(Playwright.layer));
await Effect.runPromise(program);Using launchScoped is the recommended way to manage the browser lifecycle. It ensures that the browser is closed automatically when the effect's scope ends, preventing resource leaks.
const program = Effect.gen(function* () {
const playwright = yield* Playwright;
const browser = yield* playwright.launchScoped(chromium);
// Browser will be closed automatically after this block
}).pipe(Effect.scoped);You can connect to an existing browser instance using the Chrome DevTools Protocol (CDP).
const program = Effect.gen(function* () {
const playwright = yield* Playwright;
// Use connectCDPScoped to automatically close the CONNECTION when the scope ends
// Note: This does NOT close the browser process itself, only the CDP connection.
const browser = yield* playwright.connectCDPScoped("http://localhost:9222");
const page = yield* browser.newPage();
// ...
}).pipe(Effect.scoped);If you need to manage the connection lifecycle manually, use connectCDP:
const program = Effect.gen(function* () {
const playwright = yield* Playwright;
const browser = yield* playwright.connectCDP("http://localhost:9222");
// ... use browser ...
yield* browser.close;
});The PlaywrightEnvironment simplifies setup by allowing you to configure the browser type and launch options once and reuse them across your application.
import { PlaywrightBrowser } from "effect-playwright";
import { PlaywrightEnvironment } from "effect-playwright/experimental";
import { Effect } from "effect";
import { chromium } from "playwright-core";
const liveLayer = PlaywrightEnvironment.layer(chromium, {
headless: true /** any other launch options */,
});
const program = Effect.gen(function* () {
const browser = yield* PlaywrightBrowser;
const page = yield* browser.newPage();
yield* page.goto("https://example.com");
}).pipe(PlaywrightEnvironment.withBrowser);
await Effect.runPromise(program.pipe(Effect.provide(liveLayer)));The withBrowser utility provides the PlaywrightBrowser service to your effect. It internally manages a Scope, which means the browser will be launched when the effect starts and closed automatically when the effect finishes (including on failure or interruption).
const program = Effect.gen(function* () {
const browser = yield* PlaywrightBrowser; // Now available in context
const page = yield* browser.newPage();
// ...
// Browser close is ensured
}).pipe(PlaywrightEnvironment.withBrowser);You can listen to Playwright events using the eventStream method. This returns an Effect Stream that emits events as they occur.
Note
eventStream emits "Effectified" wrappers (e.g., PlaywrightRequest, PlaywrightResponse, PlaywrightPage) for most events. This allows you to continue using the Effect ecosystem seamlessly within your event handlers.
The stream is automatically managed and will close when the underlying resource (like the Page or Browser) is closed.
Since event streams run indefinitely until the resource closes, you often need to fork the resulting effect so it runs in the background without blocking your main program flow.
const program = Effect.gen(function* () {
const browser = yield* PlaywrightBrowser;
const page = yield* browser.newPage();
// Create a stream of request events
yield* page.eventStream("request").pipe(
Stream.runForEach((request) =>
Effect.gen(function* () {
const url = yield* request.url;
yield* Effect.log(`Request: ${url}`);
}),
),
// Fork to run it in the background
Effect.fork,
);
yield* page.goto("https://example.com");
}).pipe(PlaywrightEnvironment.withBrowser);If you need to access functionality from the underlying Playwright objects that isn't directly exposed, you can use the use method available on most services/objects (browsers, pages, locators).
import { Playwright } from "effect-playwright";
import { Effect } from "effect";
import { chromium } from "playwright-core";
const program = Effect.gen(function* () {
const playwright = yield* Playwright;
const browser = yield* playwright.launchScoped(chromium);
const page = yield* browser.newPage();
// Use the native Playwright Page object
const screenshot = yield* page.use((p) => p.screenshot());
});All methods return effects that can fail with a PlaywrightError. This error wraps the original error from Playwright.
Note that Playwright does not support interruption, so Effect.timeout or similar code does not behave like you
might expect. Playwright provides its own timeout option for almost every method.