@browserbasehq/orca 3.1.0-patch.1 → 3.1.0-patch.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (76) hide show
  1. package/dist/esm/index.d.ts +2 -2
  2. package/dist/esm/lib/modelUtils.d.ts +3 -0
  3. package/dist/esm/lib/modelUtils.js +7 -2
  4. package/dist/esm/lib/modelUtils.js.map +1 -1
  5. package/dist/esm/lib/v3/agent/tools/act.d.ts +2 -1
  6. package/dist/esm/lib/v3/agent/tools/act.js.map +1 -1
  7. package/dist/esm/lib/v3/agent/tools/extract.d.ts +2 -1
  8. package/dist/esm/lib/v3/agent/tools/extract.js.map +1 -1
  9. package/dist/esm/lib/v3/agent/tools/fillform.d.ts +2 -1
  10. package/dist/esm/lib/v3/agent/tools/fillform.js.map +1 -1
  11. package/dist/esm/lib/v3/agent/tools/index.d.ts +2 -2
  12. package/dist/esm/lib/v3/agent/tools/index.js.map +1 -1
  13. package/dist/esm/lib/v3/api.d.ts +16 -1
  14. package/dist/esm/lib/v3/api.js +41 -5
  15. package/dist/esm/lib/v3/api.js.map +1 -1
  16. package/dist/esm/lib/v3/handlers/handlerUtils/actHandlerUtils.d.ts +0 -3
  17. package/dist/esm/lib/v3/handlers/handlerUtils/actHandlerUtils.js +22 -20
  18. package/dist/esm/lib/v3/handlers/handlerUtils/actHandlerUtils.js.map +1 -1
  19. package/dist/esm/lib/v3/handlers/v3AgentHandler.d.ts +2 -2
  20. package/dist/esm/lib/v3/handlers/v3AgentHandler.js.map +1 -1
  21. package/dist/esm/lib/v3/index.d.ts +0 -1
  22. package/dist/esm/lib/v3/index.js +0 -1
  23. package/dist/esm/lib/v3/index.js.map +1 -1
  24. package/dist/esm/lib/v3/llm/aisdk.js +5 -2
  25. package/dist/esm/lib/v3/llm/aisdk.js.map +1 -1
  26. package/dist/esm/lib/v3/shutdown/supervisor.d.ts +5 -7
  27. package/dist/esm/lib/v3/shutdown/supervisor.js +52 -62
  28. package/dist/esm/lib/v3/shutdown/supervisor.js.map +1 -1
  29. package/dist/esm/lib/v3/shutdown/supervisorClient.js +52 -48
  30. package/dist/esm/lib/v3/shutdown/supervisorClient.js.map +1 -1
  31. package/dist/esm/lib/v3/tests/click-count.spec.js +12 -47
  32. package/dist/esm/lib/v3/tests/click-count.spec.js.map +2 -2
  33. package/dist/esm/lib/v3/tests/envReporter.js +57 -0
  34. package/dist/esm/lib/v3/tests/envReporter.js.map +7 -0
  35. package/dist/esm/lib/v3/tests/iframe-ctx-addInitScript.spec.js +21 -67
  36. package/dist/esm/lib/v3/tests/iframe-ctx-addInitScript.spec.js.map +2 -2
  37. package/dist/esm/lib/v3/tests/v3.playwright.config.js +60 -3
  38. package/dist/esm/lib/v3/tests/v3.playwright.config.js.map +2 -2
  39. package/dist/esm/lib/v3/types/private/shutdown.d.ts +13 -1
  40. package/dist/esm/lib/v3/types/private/shutdown.js.map +1 -1
  41. package/dist/esm/lib/v3/types/public/api.d.ts +8 -0
  42. package/dist/esm/lib/v3/types/public/api.js +5 -3
  43. package/dist/esm/lib/v3/types/public/api.js.map +1 -1
  44. package/dist/esm/lib/v3/types/public/index.d.ts +1 -0
  45. package/dist/esm/lib/v3/types/public/index.js.map +1 -1
  46. package/dist/esm/lib/v3/types/public/sdkErrors.d.ts +3 -0
  47. package/dist/esm/lib/v3/types/public/sdkErrors.js +12 -6
  48. package/dist/esm/lib/v3/types/public/sdkErrors.js.map +1 -1
  49. package/dist/esm/lib/v3/understudy/context.js +1 -10
  50. package/dist/esm/lib/v3/understudy/context.js.map +1 -1
  51. package/dist/esm/lib/v3/understudy/locator.js +2 -2
  52. package/dist/esm/lib/v3/understudy/locator.js.map +1 -1
  53. package/dist/esm/lib/v3/understudy/page.js +1 -2
  54. package/dist/esm/lib/v3/understudy/page.js.map +1 -1
  55. package/dist/esm/lib/v3/v3.js +13 -10
  56. package/dist/esm/lib/v3/v3.js.map +1 -1
  57. package/dist/esm/tests/agent-execution-model.test.js +139 -0
  58. package/dist/esm/tests/agent-execution-model.test.js.map +7 -0
  59. package/dist/esm/tests/api-multiregion.test.js +73 -0
  60. package/dist/esm/tests/api-multiregion.test.js.map +7 -0
  61. package/dist/esm/tests/model-utils.test.js +43 -0
  62. package/dist/esm/tests/model-utils.test.js.map +7 -0
  63. package/dist/esm/tests/public-api/export-surface.test.js +0 -1
  64. package/dist/esm/tests/public-api/export-surface.test.js.map +2 -2
  65. package/dist/esm/tests/public-api/public-error-types.test.js +2 -1
  66. package/dist/esm/tests/public-api/public-error-types.test.js.map +2 -2
  67. package/dist/esm/tests/understudy-command-exception.test.js +55 -0
  68. package/dist/esm/tests/understudy-command-exception.test.js.map +7 -0
  69. package/package.json +9 -13
  70. package/dist/esm/lib/v3/cli.d.ts +0 -2
  71. package/dist/esm/lib/v3/cli.js +0 -10
  72. package/dist/esm/lib/v3/cli.js.map +0 -1
  73. package/dist/esm/lib/v3/dom/build/rerender-index.d.ts +0 -0
  74. package/dist/esm/lib/v3/dom/build/rerender-index.js.map +0 -1
  75. package/dist/esm/lib/v3/dom/build/v3-index.d.ts +0 -0
  76. package/dist/esm/lib/v3/dom/build/v3-index.js.map +0 -1
@@ -1,37 +1,6 @@
1
1
  import { test, expect } from "@playwright/test";
2
2
  import { V3 } from "../v3.js";
3
3
  import { v3TestConfig } from "./v3.config.js";
4
- const doubleClickFixtureUrl = `data:text/html,${encodeURIComponent(`<!DOCTYPE html>
5
- <html>
6
- <body>
7
- <div id="target" style="width: 240px; height: 120px; border: 1px solid #000;">target</div>
8
- <input id="clickCount" value="0" readonly />
9
- <input id="dblClickCount" value="0" readonly />
10
- <input id="lastClickDetail" value="0" readonly />
11
- <input id="lastDblClickDetail" value="0" readonly />
12
- <script>
13
- const target = document.getElementById("target");
14
- const clickCount = document.getElementById("clickCount");
15
- const dblClickCount = document.getElementById("dblClickCount");
16
- const lastClickDetail = document.getElementById("lastClickDetail");
17
- const lastDblClickDetail = document.getElementById("lastDblClickDetail");
18
- let clicks = 0;
19
- let dblClicks = 0;
20
-
21
- target.addEventListener("click", (event) => {
22
- clicks += 1;
23
- clickCount.value = String(clicks);
24
- lastClickDetail.value = String(event.detail);
25
- });
26
-
27
- target.addEventListener("dblclick", (event) => {
28
- dblClicks += 1;
29
- dblClickCount.value = String(dblClicks);
30
- lastDblClickDetail.value = String(event.detail);
31
- });
32
- </script>
33
- </body>
34
- </html>`)}`;
35
4
  test.describe("Locator and Page click methods", () => {
36
5
  let v3;
37
6
  test.beforeEach(async () => {
@@ -58,24 +27,22 @@ test.describe("Locator and Page click methods", () => {
58
27
  });
59
28
  test("locator.click() with clickCount: 2 performs double-click", async () => {
60
29
  const page = v3.context.pages()[0];
61
- await page.goto(doubleClickFixtureUrl);
30
+ await page.goto(
31
+ "https://browserbase.github.io/stagehand-eval-sites/sites/click-test/"
32
+ );
62
33
  await page.waitForLoadState("domcontentloaded");
63
- const countDisplay = page.locator("#clickCount");
64
- const dcCountDisplay = page.locator("#dblClickCount");
65
- const clickDetailDisplay = page.locator("#lastClickDetail");
66
- const dblClickDetailDisplay = page.locator("#lastDblClickDetail");
34
+ const countDisplay = page.locator("#count");
35
+ const dcCountDisplay = page.locator("#dcCount");
67
36
  const initialCount = await countDisplay.inputValue();
68
37
  const initialDcCount = await dcCountDisplay.inputValue();
69
38
  expect(initialCount).toBe("0");
70
39
  expect(initialDcCount).toBe("0");
71
- const clickArea = page.locator("#target");
40
+ const clickArea = page.locator("#textarea");
72
41
  await clickArea.click({ clickCount: 2 });
73
42
  const newCount = await countDisplay.inputValue();
74
43
  expect(newCount).toBe("2");
75
44
  const newDcCount = await dcCountDisplay.inputValue();
76
45
  expect(newDcCount).toBe("1");
77
- expect(await clickDetailDisplay.inputValue()).toBe("2");
78
- expect(await dblClickDetailDisplay.inputValue()).toBe("2");
79
46
  });
80
47
  test("locator.click() with clickCount: 3 performs triple-click", async () => {
81
48
  const page = v3.context.pages()[0];
@@ -108,25 +75,23 @@ test.describe("Locator and Page click methods", () => {
108
75
  });
109
76
  test("page.click() with clickCount: 2 performs double-click", async () => {
110
77
  const page = v3.context.pages()[0];
111
- await page.goto(doubleClickFixtureUrl);
78
+ await page.goto(
79
+ "https://browserbase.github.io/stagehand-eval-sites/sites/click-test/"
80
+ );
112
81
  await page.waitForLoadState("domcontentloaded");
113
- const countDisplay = page.locator("#clickCount");
114
- const dcCountDisplay = page.locator("#dblClickCount");
115
- const clickDetailDisplay = page.locator("#lastClickDetail");
116
- const dblClickDetailDisplay = page.locator("#lastDblClickDetail");
82
+ const countDisplay = page.locator("#count");
83
+ const dcCountDisplay = page.locator("#dcCount");
117
84
  const initialCount = await countDisplay.inputValue();
118
85
  const initialDcCount = await dcCountDisplay.inputValue();
119
86
  expect(initialCount).toBe("0");
120
87
  expect(initialDcCount).toBe("0");
121
- const clickArea = page.locator("#target");
88
+ const clickArea = page.locator("#textarea");
122
89
  const { x, y } = await clickArea.centroid();
123
90
  await page.click(x, y, { clickCount: 2 });
124
91
  const newCount = await countDisplay.inputValue();
125
92
  expect(newCount).toBe("2");
126
93
  const newDcCount = await dcCountDisplay.inputValue();
127
94
  expect(newDcCount).toBe("1");
128
- expect(await clickDetailDisplay.inputValue()).toBe("2");
129
- expect(await dblClickDetailDisplay.inputValue()).toBe("2");
130
95
  });
131
96
  test("page.click() with clickCount: 3 performs triple-click", async () => {
132
97
  const page = v3.context.pages()[0];
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../../../lib/v3/tests/click-count.spec.ts"],
4
- "sourcesContent": ["import { test, expect } from \"@playwright/test\";\nimport { V3 } from \"../v3.js\";\nimport { v3TestConfig } from \"./v3.config.js\";\n\n// Keep double-click verification event-based and deterministic.\n// Time-delta counters (Date.now() between mousedowns) are flaky at ms boundaries\n// and can miss valid double-clicks when synthetic input lands in the same millisecond.\nconst doubleClickFixtureUrl = `data:text/html,${encodeURIComponent(`<!DOCTYPE html>\n<html>\n <body>\n <div id=\"target\" style=\"width: 240px; height: 120px; border: 1px solid #000;\">target</div>\n <input id=\"clickCount\" value=\"0\" readonly />\n <input id=\"dblClickCount\" value=\"0\" readonly />\n <input id=\"lastClickDetail\" value=\"0\" readonly />\n <input id=\"lastDblClickDetail\" value=\"0\" readonly />\n <script>\n const target = document.getElementById(\"target\");\n const clickCount = document.getElementById(\"clickCount\");\n const dblClickCount = document.getElementById(\"dblClickCount\");\n const lastClickDetail = document.getElementById(\"lastClickDetail\");\n const lastDblClickDetail = document.getElementById(\"lastDblClickDetail\");\n let clicks = 0;\n let dblClicks = 0;\n\n target.addEventListener(\"click\", (event) => {\n clicks += 1;\n clickCount.value = String(clicks);\n lastClickDetail.value = String(event.detail);\n });\n\n target.addEventListener(\"dblclick\", (event) => {\n dblClicks += 1;\n dblClickCount.value = String(dblClicks);\n lastDblClickDetail.value = String(event.detail);\n });\n </script>\n </body>\n</html>`)}`;\n\ntest.describe(\"Locator and Page click methods\", () => {\n let v3: V3;\n\n test.beforeEach(async () => {\n v3 = new V3(v3TestConfig);\n await v3.init();\n });\n\n test.afterEach(async () => {\n await v3?.close?.().catch(() => {});\n });\n\n test(\"locator.click() performs single click by default\", async () => {\n const page = v3.context.pages()[0];\n await page.goto(\n \"https://browserbase.github.io/stagehand-eval-sites/sites/click-test/\",\n );\n\n // Wait for page to be fully loaded\n await page.waitForLoadState(\"domcontentloaded\");\n\n // Get initial count\n const countDisplay = page.locator(\"#count\");\n const initialCount = await countDisplay.inputValue();\n expect(initialCount).toBe(\"0\");\n\n // Perform single click on the textarea (the clickable area)\n const clickArea = page.locator(\"#textarea\");\n await clickArea.click();\n\n // Verify count incremented by 1\n const newCount = await countDisplay.inputValue();\n expect(newCount).toBe(\"1\");\n });\n\n test(\"locator.click() with clickCount: 2 performs double-click\", async () => {\n const page = v3.context.pages()[0];\n await page.goto(doubleClickFixtureUrl);\n await page.waitForLoadState(\"domcontentloaded\");\n\n const countDisplay = page.locator(\"#clickCount\");\n const dcCountDisplay = page.locator(\"#dblClickCount\");\n const clickDetailDisplay = page.locator(\"#lastClickDetail\");\n const dblClickDetailDisplay = page.locator(\"#lastDblClickDetail\");\n\n const initialCount = await countDisplay.inputValue();\n const initialDcCount = await dcCountDisplay.inputValue();\n expect(initialCount).toBe(\"0\");\n expect(initialDcCount).toBe(\"0\");\n\n const clickArea = page.locator(\"#target\");\n await clickArea.click({ clickCount: 2 });\n\n const newCount = await countDisplay.inputValue();\n expect(newCount).toBe(\"2\");\n\n const newDcCount = await dcCountDisplay.inputValue();\n expect(newDcCount).toBe(\"1\");\n // `dblclick` is the browser-level contract for double-click behavior.\n // Verifying `detail=2` ensures the click sequence is recognized as a true multi-click.\n expect(await clickDetailDisplay.inputValue()).toBe(\"2\");\n expect(await dblClickDetailDisplay.inputValue()).toBe(\"2\");\n });\n\n test(\"locator.click() with clickCount: 3 performs triple-click\", async () => {\n const page = v3.context.pages()[0];\n await page.goto(\n \"https://browserbase.github.io/stagehand-eval-sites/sites/click-test/\",\n );\n\n // Wait for page to be fully loaded\n await page.waitForLoadState(\"domcontentloaded\");\n\n const countDisplay = page.locator(\"#count\");\n const initialCount = await countDisplay.inputValue();\n expect(initialCount).toBe(\"0\");\n\n // Perform triple-click on the textarea\n const clickArea = page.locator(\"#textarea\");\n await clickArea.click({ clickCount: 3 });\n\n // Verify count incremented by 3\n const newCount = await countDisplay.inputValue();\n expect(newCount).toBe(\"3\");\n });\n\n test(\"page.click() performs single click with coordinates\", async () => {\n const page = v3.context.pages()[0];\n await page.goto(\n \"https://browserbase.github.io/stagehand-eval-sites/sites/click-test/\",\n );\n\n // Wait for page to be fully loaded\n await page.waitForLoadState(\"domcontentloaded\");\n\n // Get initial count\n const countDisplay = page.locator(\"#count\");\n const initialCount = await countDisplay.inputValue();\n expect(initialCount).toBe(\"0\");\n\n // Get the centroid of the textarea to click\n const clickArea = page.locator(\"#textarea\");\n const { x, y } = await clickArea.centroid();\n\n // Perform single click using page.click() with coordinates\n await page.click(x, y);\n\n // Verify count incremented by 1\n const newCount = await countDisplay.inputValue();\n expect(newCount).toBe(\"1\");\n });\n\n test(\"page.click() with clickCount: 2 performs double-click\", async () => {\n const page = v3.context.pages()[0];\n await page.goto(doubleClickFixtureUrl);\n await page.waitForLoadState(\"domcontentloaded\");\n\n const countDisplay = page.locator(\"#clickCount\");\n const dcCountDisplay = page.locator(\"#dblClickCount\");\n const clickDetailDisplay = page.locator(\"#lastClickDetail\");\n const dblClickDetailDisplay = page.locator(\"#lastDblClickDetail\");\n\n const initialCount = await countDisplay.inputValue();\n const initialDcCount = await dcCountDisplay.inputValue();\n expect(initialCount).toBe(\"0\");\n expect(initialDcCount).toBe(\"0\");\n\n const clickArea = page.locator(\"#target\");\n const { x, y } = await clickArea.centroid();\n\n await page.click(x, y, { clickCount: 2 });\n\n const newCount = await countDisplay.inputValue();\n expect(newCount).toBe(\"2\");\n\n const newDcCount = await dcCountDisplay.inputValue();\n expect(newDcCount).toBe(\"1\");\n // `dblclick` is the browser-level contract for double-click behavior.\n // Verifying `detail=2` ensures the click sequence is recognized as a true multi-click.\n expect(await clickDetailDisplay.inputValue()).toBe(\"2\");\n expect(await dblClickDetailDisplay.inputValue()).toBe(\"2\");\n });\n\n test(\"page.click() with clickCount: 3 performs triple-click\", async () => {\n const page = v3.context.pages()[0];\n await page.goto(\n \"https://browserbase.github.io/stagehand-eval-sites/sites/click-test/\",\n );\n\n // Wait for page to be fully loaded\n await page.waitForLoadState(\"domcontentloaded\");\n\n const countDisplay = page.locator(\"#count\");\n const initialCount = await countDisplay.inputValue();\n expect(initialCount).toBe(\"0\");\n\n // Get the centroid of the textarea to click\n const clickArea = page.locator(\"#textarea\");\n const { x, y } = await clickArea.centroid();\n\n // Perform triple-click using page.click() with coordinates\n await page.click(x, y, { clickCount: 3 });\n\n // Verify count incremented by 3\n const newCount = await countDisplay.inputValue();\n expect(newCount).toBe(\"3\");\n });\n});\n"],
5
- "mappings": "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",
4
+ "sourcesContent": ["import { test, expect } from \"@playwright/test\";\nimport { V3 } from \"../v3.js\";\nimport { v3TestConfig } from \"./v3.config.js\";\n\ntest.describe(\"Locator and Page click methods\", () => {\n let v3: V3;\n\n test.beforeEach(async () => {\n v3 = new V3(v3TestConfig);\n await v3.init();\n });\n\n test.afterEach(async () => {\n await v3?.close?.().catch(() => {});\n });\n\n test(\"locator.click() performs single click by default\", async () => {\n const page = v3.context.pages()[0];\n await page.goto(\n \"https://browserbase.github.io/stagehand-eval-sites/sites/click-test/\",\n );\n\n // Wait for page to be fully loaded\n await page.waitForLoadState(\"domcontentloaded\");\n\n // Get initial count\n const countDisplay = page.locator(\"#count\");\n const initialCount = await countDisplay.inputValue();\n expect(initialCount).toBe(\"0\");\n\n // Perform single click on the textarea (the clickable area)\n const clickArea = page.locator(\"#textarea\");\n await clickArea.click();\n\n // Verify count incremented by 1\n const newCount = await countDisplay.inputValue();\n expect(newCount).toBe(\"1\");\n });\n\n test(\"locator.click() with clickCount: 2 performs double-click\", async () => {\n const page = v3.context.pages()[0];\n await page.goto(\n \"https://browserbase.github.io/stagehand-eval-sites/sites/click-test/\",\n );\n\n // Wait for page to be fully loaded\n await page.waitForLoadState(\"domcontentloaded\");\n\n // Get initial counts\n const countDisplay = page.locator(\"#count\");\n const dcCountDisplay = page.locator(\"#dcCount\");\n\n const initialCount = await countDisplay.inputValue();\n const initialDcCount = await dcCountDisplay.inputValue();\n expect(initialCount).toBe(\"0\");\n expect(initialDcCount).toBe(\"0\");\n\n // Perform double-click on the textarea\n const clickArea = page.locator(\"#textarea\");\n await clickArea.click({ clickCount: 2 });\n\n // Verify both counters incremented\n // Regular count should be 2 (one for each click in the double-click)\n const newCount = await countDisplay.inputValue();\n expect(newCount).toBe(\"2\");\n\n // Double-click count should be 1 (one double-click event detected)\n const newDcCount = await dcCountDisplay.inputValue();\n expect(newDcCount).toBe(\"1\");\n });\n\n test(\"locator.click() with clickCount: 3 performs triple-click\", async () => {\n const page = v3.context.pages()[0];\n await page.goto(\n \"https://browserbase.github.io/stagehand-eval-sites/sites/click-test/\",\n );\n\n // Wait for page to be fully loaded\n await page.waitForLoadState(\"domcontentloaded\");\n\n const countDisplay = page.locator(\"#count\");\n const initialCount = await countDisplay.inputValue();\n expect(initialCount).toBe(\"0\");\n\n // Perform triple-click on the textarea\n const clickArea = page.locator(\"#textarea\");\n await clickArea.click({ clickCount: 3 });\n\n // Verify count incremented by 3\n const newCount = await countDisplay.inputValue();\n expect(newCount).toBe(\"3\");\n });\n\n test(\"page.click() performs single click with coordinates\", async () => {\n const page = v3.context.pages()[0];\n await page.goto(\n \"https://browserbase.github.io/stagehand-eval-sites/sites/click-test/\",\n );\n\n // Wait for page to be fully loaded\n await page.waitForLoadState(\"domcontentloaded\");\n\n // Get initial count\n const countDisplay = page.locator(\"#count\");\n const initialCount = await countDisplay.inputValue();\n expect(initialCount).toBe(\"0\");\n\n // Get the centroid of the textarea to click\n const clickArea = page.locator(\"#textarea\");\n const { x, y } = await clickArea.centroid();\n\n // Perform single click using page.click() with coordinates\n await page.click(x, y);\n\n // Verify count incremented by 1\n const newCount = await countDisplay.inputValue();\n expect(newCount).toBe(\"1\");\n });\n\n test(\"page.click() with clickCount: 2 performs double-click\", async () => {\n const page = v3.context.pages()[0];\n await page.goto(\n \"https://browserbase.github.io/stagehand-eval-sites/sites/click-test/\",\n );\n\n // Wait for page to be fully loaded\n await page.waitForLoadState(\"domcontentloaded\");\n\n // Get initial counts\n const countDisplay = page.locator(\"#count\");\n const dcCountDisplay = page.locator(\"#dcCount\");\n\n const initialCount = await countDisplay.inputValue();\n const initialDcCount = await dcCountDisplay.inputValue();\n expect(initialCount).toBe(\"0\");\n expect(initialDcCount).toBe(\"0\");\n\n // Get the centroid of the textarea to click\n const clickArea = page.locator(\"#textarea\");\n const { x, y } = await clickArea.centroid();\n\n // Perform double-click using page.click() with coordinates\n await page.click(x, y, { clickCount: 2 });\n\n // Verify both counters incremented\n const newCount = await countDisplay.inputValue();\n expect(newCount).toBe(\"2\");\n\n // Double-click count should be 1\n const newDcCount = await dcCountDisplay.inputValue();\n expect(newDcCount).toBe(\"1\");\n });\n\n test(\"page.click() with clickCount: 3 performs triple-click\", async () => {\n const page = v3.context.pages()[0];\n await page.goto(\n \"https://browserbase.github.io/stagehand-eval-sites/sites/click-test/\",\n );\n\n // Wait for page to be fully loaded\n await page.waitForLoadState(\"domcontentloaded\");\n\n const countDisplay = page.locator(\"#count\");\n const initialCount = await countDisplay.inputValue();\n expect(initialCount).toBe(\"0\");\n\n // Get the centroid of the textarea to click\n const clickArea = page.locator(\"#textarea\");\n const { x, y } = await clickArea.centroid();\n\n // Perform triple-click using page.click() with coordinates\n await page.click(x, y, { clickCount: 3 });\n\n // Verify count incremented by 3\n const newCount = await countDisplay.inputValue();\n expect(newCount).toBe(\"3\");\n });\n});\n"],
5
+ "mappings": "AAAA,SAAS,MAAM,cAAc;AAC7B,SAAS,UAAU;AACnB,SAAS,oBAAoB;AAE7B,KAAK,SAAS,kCAAkC,MAAM;AACpD,MAAI;AAEJ,OAAK,WAAW,YAAY;AAC1B,SAAK,IAAI,GAAG,YAAY;AACxB,UAAM,GAAG,KAAK;AAAA,EAChB,CAAC;AAED,OAAK,UAAU,YAAY;AACzB,UAAM,IAAI,QAAQ,EAAE,MAAM,MAAM;AAAA,IAAC,CAAC;AAAA,EACpC,CAAC;AAED,OAAK,oDAAoD,YAAY;AACnE,UAAM,OAAO,GAAG,QAAQ,MAAM,EAAE,CAAC;AACjC,UAAM,KAAK;AAAA,MACT;AAAA,IACF;AAGA,UAAM,KAAK,iBAAiB,kBAAkB;AAG9C,UAAM,eAAe,KAAK,QAAQ,QAAQ;AAC1C,UAAM,eAAe,MAAM,aAAa,WAAW;AACnD,WAAO,YAAY,EAAE,KAAK,GAAG;AAG7B,UAAM,YAAY,KAAK,QAAQ,WAAW;AAC1C,UAAM,UAAU,MAAM;AAGtB,UAAM,WAAW,MAAM,aAAa,WAAW;AAC/C,WAAO,QAAQ,EAAE,KAAK,GAAG;AAAA,EAC3B,CAAC;AAED,OAAK,4DAA4D,YAAY;AAC3E,UAAM,OAAO,GAAG,QAAQ,MAAM,EAAE,CAAC;AACjC,UAAM,KAAK;AAAA,MACT;AAAA,IACF;AAGA,UAAM,KAAK,iBAAiB,kBAAkB;AAG9C,UAAM,eAAe,KAAK,QAAQ,QAAQ;AAC1C,UAAM,iBAAiB,KAAK,QAAQ,UAAU;AAE9C,UAAM,eAAe,MAAM,aAAa,WAAW;AACnD,UAAM,iBAAiB,MAAM,eAAe,WAAW;AACvD,WAAO,YAAY,EAAE,KAAK,GAAG;AAC7B,WAAO,cAAc,EAAE,KAAK,GAAG;AAG/B,UAAM,YAAY,KAAK,QAAQ,WAAW;AAC1C,UAAM,UAAU,MAAM,EAAE,YAAY,EAAE,CAAC;AAIvC,UAAM,WAAW,MAAM,aAAa,WAAW;AAC/C,WAAO,QAAQ,EAAE,KAAK,GAAG;AAGzB,UAAM,aAAa,MAAM,eAAe,WAAW;AACnD,WAAO,UAAU,EAAE,KAAK,GAAG;AAAA,EAC7B,CAAC;AAED,OAAK,4DAA4D,YAAY;AAC3E,UAAM,OAAO,GAAG,QAAQ,MAAM,EAAE,CAAC;AACjC,UAAM,KAAK;AAAA,MACT;AAAA,IACF;AAGA,UAAM,KAAK,iBAAiB,kBAAkB;AAE9C,UAAM,eAAe,KAAK,QAAQ,QAAQ;AAC1C,UAAM,eAAe,MAAM,aAAa,WAAW;AACnD,WAAO,YAAY,EAAE,KAAK,GAAG;AAG7B,UAAM,YAAY,KAAK,QAAQ,WAAW;AAC1C,UAAM,UAAU,MAAM,EAAE,YAAY,EAAE,CAAC;AAGvC,UAAM,WAAW,MAAM,aAAa,WAAW;AAC/C,WAAO,QAAQ,EAAE,KAAK,GAAG;AAAA,EAC3B,CAAC;AAED,OAAK,uDAAuD,YAAY;AACtE,UAAM,OAAO,GAAG,QAAQ,MAAM,EAAE,CAAC;AACjC,UAAM,KAAK;AAAA,MACT;AAAA,IACF;AAGA,UAAM,KAAK,iBAAiB,kBAAkB;AAG9C,UAAM,eAAe,KAAK,QAAQ,QAAQ;AAC1C,UAAM,eAAe,MAAM,aAAa,WAAW;AACnD,WAAO,YAAY,EAAE,KAAK,GAAG;AAG7B,UAAM,YAAY,KAAK,QAAQ,WAAW;AAC1C,UAAM,EAAE,GAAG,EAAE,IAAI,MAAM,UAAU,SAAS;AAG1C,UAAM,KAAK,MAAM,GAAG,CAAC;AAGrB,UAAM,WAAW,MAAM,aAAa,WAAW;AAC/C,WAAO,QAAQ,EAAE,KAAK,GAAG;AAAA,EAC3B,CAAC;AAED,OAAK,yDAAyD,YAAY;AACxE,UAAM,OAAO,GAAG,QAAQ,MAAM,EAAE,CAAC;AACjC,UAAM,KAAK;AAAA,MACT;AAAA,IACF;AAGA,UAAM,KAAK,iBAAiB,kBAAkB;AAG9C,UAAM,eAAe,KAAK,QAAQ,QAAQ;AAC1C,UAAM,iBAAiB,KAAK,QAAQ,UAAU;AAE9C,UAAM,eAAe,MAAM,aAAa,WAAW;AACnD,UAAM,iBAAiB,MAAM,eAAe,WAAW;AACvD,WAAO,YAAY,EAAE,KAAK,GAAG;AAC7B,WAAO,cAAc,EAAE,KAAK,GAAG;AAG/B,UAAM,YAAY,KAAK,QAAQ,WAAW;AAC1C,UAAM,EAAE,GAAG,EAAE,IAAI,MAAM,UAAU,SAAS;AAG1C,UAAM,KAAK,MAAM,GAAG,GAAG,EAAE,YAAY,EAAE,CAAC;AAGxC,UAAM,WAAW,MAAM,aAAa,WAAW;AAC/C,WAAO,QAAQ,EAAE,KAAK,GAAG;AAGzB,UAAM,aAAa,MAAM,eAAe,WAAW;AACnD,WAAO,UAAU,EAAE,KAAK,GAAG;AAAA,EAC7B,CAAC;AAED,OAAK,yDAAyD,YAAY;AACxE,UAAM,OAAO,GAAG,QAAQ,MAAM,EAAE,CAAC;AACjC,UAAM,KAAK;AAAA,MACT;AAAA,IACF;AAGA,UAAM,KAAK,iBAAiB,kBAAkB;AAE9C,UAAM,eAAe,KAAK,QAAQ,QAAQ;AAC1C,UAAM,eAAe,MAAM,aAAa,WAAW;AACnD,WAAO,YAAY,EAAE,KAAK,GAAG;AAG7B,UAAM,YAAY,KAAK,QAAQ,WAAW;AAC1C,UAAM,EAAE,GAAG,EAAE,IAAI,MAAM,UAAU,SAAS;AAG1C,UAAM,KAAK,MAAM,GAAG,GAAG,EAAE,YAAY,EAAE,CAAC;AAGxC,UAAM,WAAW,MAAM,aAAa,WAAW;AAC/C,WAAO,QAAQ,EAAE,KAAK,GAAG;AAAA,EAC3B,CAAC;AACH,CAAC;",
6
6
  "names": []
7
7
  }
@@ -0,0 +1,57 @@
1
+ import { getV3DynamicTestConfig } from "./v3.dynamic.config.js";
2
+ function summarizeV3Config() {
3
+ try {
4
+ const cfg = getV3DynamicTestConfig();
5
+ return {
6
+ env: cfg.env,
7
+ disableAPI: cfg.disableAPI,
8
+ selfHeal: cfg.selfHeal,
9
+ experimental: cfg.experimental,
10
+ localBrowserLaunchOptions: cfg.localBrowserLaunchOptions ? {
11
+ headless: cfg.localBrowserLaunchOptions.headless,
12
+ viewport: cfg.localBrowserLaunchOptions.viewport,
13
+ hasExecutablePath: Boolean(
14
+ cfg.localBrowserLaunchOptions.executablePath
15
+ ),
16
+ argsCount: cfg.localBrowserLaunchOptions.args?.length ?? 0
17
+ } : void 0,
18
+ browserbaseSessionCreateParams: cfg.browserbaseSessionCreateParams ? {
19
+ region: cfg.browserbaseSessionCreateParams.region,
20
+ hasViewport: Boolean(
21
+ cfg.browserbaseSessionCreateParams.browserSettings?.viewport
22
+ )
23
+ } : void 0
24
+ };
25
+ } catch (error) {
26
+ return { error: String(error) };
27
+ }
28
+ }
29
+ function summarizeEnv() {
30
+ return {
31
+ STAGEHAND_BROWSER_TARGET: process.env.STAGEHAND_BROWSER_TARGET,
32
+ BB_ENV: process.env.BB_ENV,
33
+ // BB_ENV = 'local' | 'dev' | 'prod' (hosting environment the stagehand-api server is running in)
34
+ NODE_ENV: process.env.NODE_ENV,
35
+ // NODE_ENV = 'development' | 'test' | 'production' | 'staging' (used only to control logging)
36
+ CI: process.env.CI,
37
+ // CI = 'true' | 'false' (used only to control test parallelism and pnpm prepare script)
38
+ STAGEHAND_API_URL: process.env.STAGEHAND_API_URL,
39
+ BROWSERBASE_REGION: process.env.BROWSERBASE_REGION,
40
+ BROWSERBASE_API_KEY: process.env.BROWSERBASE_API_KEY ? "[redacted]" : "missing!",
41
+ BROWSERBASE_PROJECT_ID: process.env.BROWSERBASE_PROJECT_ID ? "[redacted]" : "missing!"
42
+ };
43
+ }
44
+ class EnvReporter {
45
+ onTestBegin(test) {
46
+ const payload = {
47
+ test: test.titlePath().join(" > "),
48
+ env: summarizeEnv(),
49
+ config: summarizeV3Config()
50
+ };
51
+ console.log(`[e2e-env] ${JSON.stringify(payload)}`);
52
+ }
53
+ }
54
+ export {
55
+ EnvReporter as default
56
+ };
57
+ //# sourceMappingURL=envReporter.js.map
@@ -0,0 +1,7 @@
1
+ {
2
+ "version": 3,
3
+ "sources": ["../../../../../lib/v3/tests/envReporter.ts"],
4
+ "sourcesContent": ["import type { Reporter, TestCase } from \"@playwright/test/reporter\";\nimport { getV3DynamicTestConfig } from \"./v3.dynamic.config.js\";\n\ntype ConfigSummary = {\n env?: string;\n disableAPI?: boolean;\n selfHeal?: boolean;\n experimental?: boolean;\n localBrowserLaunchOptions?: {\n headless?: boolean;\n viewport?: { width?: number; height?: number };\n hasExecutablePath?: boolean;\n argsCount?: number;\n };\n browserbaseSessionCreateParams?: {\n region?: string;\n hasViewport?: boolean;\n };\n error?: string;\n};\n\n// Keep this small and log-safe; never emit secrets in CI logs.\nfunction summarizeV3Config(): ConfigSummary {\n try {\n const cfg = getV3DynamicTestConfig();\n return {\n env: cfg.env,\n disableAPI: cfg.disableAPI,\n selfHeal: cfg.selfHeal,\n experimental: cfg.experimental,\n localBrowserLaunchOptions: cfg.localBrowserLaunchOptions\n ? {\n headless: cfg.localBrowserLaunchOptions.headless,\n viewport: cfg.localBrowserLaunchOptions.viewport,\n hasExecutablePath: Boolean(\n cfg.localBrowserLaunchOptions.executablePath,\n ),\n argsCount: cfg.localBrowserLaunchOptions.args?.length ?? 0,\n }\n : undefined,\n browserbaseSessionCreateParams: cfg.browserbaseSessionCreateParams\n ? {\n region: cfg.browserbaseSessionCreateParams.region,\n hasViewport: Boolean(\n cfg.browserbaseSessionCreateParams.browserSettings?.viewport,\n ),\n }\n : undefined,\n };\n } catch (error) {\n return { error: String(error) };\n }\n}\n\nfunction summarizeEnv() {\n return {\n STAGEHAND_BROWSER_TARGET: process.env.STAGEHAND_BROWSER_TARGET,\n BB_ENV: process.env.BB_ENV, // BB_ENV = 'local' | 'dev' | 'prod' (hosting environment the stagehand-api server is running in)\n NODE_ENV: process.env.NODE_ENV, // NODE_ENV = 'development' | 'test' | 'production' | 'staging' (used only to control logging)\n CI: process.env.CI, // CI = 'true' | 'false' (used only to control test parallelism and pnpm prepare script)\n STAGEHAND_API_URL: process.env.STAGEHAND_API_URL,\n BROWSERBASE_REGION: process.env.BROWSERBASE_REGION,\n BROWSERBASE_API_KEY: process.env.BROWSERBASE_API_KEY\n ? \"[redacted]\"\n : \"missing!\",\n BROWSERBASE_PROJECT_ID: process.env.BROWSERBASE_PROJECT_ID\n ? \"[redacted]\"\n : \"missing!\",\n };\n}\n\nexport default class EnvReporter implements Reporter {\n onTestBegin(test: TestCase): void {\n const payload = {\n test: test.titlePath().join(\" > \"),\n env: summarizeEnv(),\n config: summarizeV3Config(),\n };\n console.log(`[e2e-env] ${JSON.stringify(payload)}`);\n }\n}\n"],
5
+ "mappings": "AACA,SAAS,8BAA8B;AAqBvC,SAAS,oBAAmC;AAC1C,MAAI;AACF,UAAM,MAAM,uBAAuB;AACnC,WAAO;AAAA,MACL,KAAK,IAAI;AAAA,MACT,YAAY,IAAI;AAAA,MAChB,UAAU,IAAI;AAAA,MACd,cAAc,IAAI;AAAA,MAClB,2BAA2B,IAAI,4BAC3B;AAAA,QACE,UAAU,IAAI,0BAA0B;AAAA,QACxC,UAAU,IAAI,0BAA0B;AAAA,QACxC,mBAAmB;AAAA,UACjB,IAAI,0BAA0B;AAAA,QAChC;AAAA,QACA,WAAW,IAAI,0BAA0B,MAAM,UAAU;AAAA,MAC3D,IACA;AAAA,MACJ,gCAAgC,IAAI,iCAChC;AAAA,QACE,QAAQ,IAAI,+BAA+B;AAAA,QAC3C,aAAa;AAAA,UACX,IAAI,+BAA+B,iBAAiB;AAAA,QACtD;AAAA,MACF,IACA;AAAA,IACN;AAAA,EACF,SAAS,OAAO;AACd,WAAO,EAAE,OAAO,OAAO,KAAK,EAAE;AAAA,EAChC;AACF;AAEA,SAAS,eAAe;AACtB,SAAO;AAAA,IACL,0BAA0B,QAAQ,IAAI;AAAA,IACtC,QAAQ,QAAQ,IAAI;AAAA;AAAA,IACpB,UAAU,QAAQ,IAAI;AAAA;AAAA,IACtB,IAAI,QAAQ,IAAI;AAAA;AAAA,IAChB,mBAAmB,QAAQ,IAAI;AAAA,IAC/B,oBAAoB,QAAQ,IAAI;AAAA,IAChC,qBAAqB,QAAQ,IAAI,sBAC7B,eACA;AAAA,IACJ,wBAAwB,QAAQ,IAAI,yBAChC,eACA;AAAA,EACN;AACF;AAEA,MAAO,YAA8C;AAAA,EACnD,YAAY,MAAsB;AAChC,UAAM,UAAU;AAAA,MACd,MAAM,KAAK,UAAU,EAAE,KAAK,KAAK;AAAA,MACjC,KAAK,aAAa;AAAA,MAClB,QAAQ,kBAAkB;AAAA,IAC5B;AACA,YAAQ,IAAI,aAAa,KAAK,UAAU,OAAO,CAAC,EAAE;AAAA,EACpD;AACF;",
6
+ "names": []
7
+ }
@@ -132,11 +132,10 @@ async function closeAllPages(ctx) {
132
132
  const pages = ctx.pages();
133
133
  await Promise.allSettled(pages.map((page) => page.close()));
134
134
  }
135
- async function waitForChildFrame(page, expectedChildUrl, timeoutMs = CHILD_FRAME_TIMEOUT_MS) {
135
+ async function waitForChildFrame(page, timeoutMs = CHILD_FRAME_TIMEOUT_MS) {
136
136
  const mainFrameId = page.mainFrame().frameId;
137
137
  const deadline = Date.now() + timeoutMs;
138
138
  let observedFrameCount = 0;
139
- const observedChildFrameIds = /* @__PURE__ */ new Set();
140
139
  let lastUrl = "";
141
140
  let lastLogAt = Date.now();
142
141
  while (Date.now() < deadline) {
@@ -149,71 +148,29 @@ async function waitForChildFrame(page, expectedChildUrl, timeoutMs = CHILD_FRAME
149
148
  url: lastUrl,
150
149
  mainFrameId,
151
150
  observedFrameCount,
152
- childIds,
153
- expectedChildUrl
151
+ childIds
154
152
  });
155
153
  lastLogAt = Date.now();
156
154
  }
157
- for (const childId of childIds) observedChildFrameIds.add(childId);
158
- const childFrames = frames.filter((f) => f.frameId !== mainFrameId).reverse();
159
- if (childFrames.length) {
160
- const probes = await Promise.all(
161
- childFrames.map(async (child) => {
162
- try {
163
- const state = await child.evaluate(() => ({
164
- href: location.href,
165
- readyState: document.readyState
166
- }));
167
- return {
168
- child,
169
- href: state.href,
170
- readyState: state.readyState,
171
- error: void 0
172
- };
173
- } catch (error) {
174
- return {
175
- child,
176
- href: void 0,
177
- readyState: void 0,
178
- error: formatError(error)
179
- };
180
- }
181
- })
182
- );
183
- const ready = probes.find(
184
- (probe) => probe.readyState === "complete" && probe.href === expectedChildUrl
185
- );
186
- if (ready) {
187
- debugLog("waitForChildFrame:ready", {
188
- childFrameId: ready.child.frameId,
189
- childSessionId: ready.child.sessionId ?? "root",
190
- childUrl: ready.href ?? "<unknown>",
191
- expectedChildUrl,
192
- url: lastUrl
193
- });
194
- return ready.child;
195
- }
196
- if (iframeDebugEnabled && Date.now() - lastLogAt >= DEBUG_INTERVAL_MS) {
197
- debugLog("waitForChildFrame:not-ready", {
198
- url: lastUrl,
199
- mainFrameId,
200
- expectedChildUrl,
201
- probes: probes.map((probe) => ({
202
- frameId: probe.child.frameId,
203
- sessionId: probe.child.sessionId ?? "root",
204
- readyState: probe.readyState ?? "<unknown>",
205
- href: probe.href ?? "<unknown>",
206
- error: probe.error ?? "<none>"
207
- }))
208
- });
209
- lastLogAt = Date.now();
155
+ const child = frames.find((f) => f.frameId !== mainFrameId);
156
+ if (child) {
157
+ try {
158
+ const ready = await child.evaluate(() => document.readyState);
159
+ if (ready === "complete") {
160
+ debugLog("waitForChildFrame:ready", {
161
+ childFrameId: child.frameId,
162
+ url: lastUrl
163
+ });
164
+ return child;
165
+ }
166
+ } catch {
210
167
  }
211
168
  }
212
169
  await new Promise((r) => setTimeout(r, 100));
213
170
  }
214
171
  await logPageDiagnostics(page, "waitForChildFrame timeout");
215
172
  throw new Error(
216
- `Timed out waiting for child frame to load (timeout=${timeoutMs}ms, mainFrameId=${mainFrameId}, expectedChildUrl=${expectedChildUrl}, maxObservedFrames=${observedFrameCount}, observedChildFrameIds=[${[...observedChildFrameIds].join(",")}], url=${lastUrl || "<unknown>"})`
173
+ `Timed out waiting for child frame to load (timeout=${timeoutMs}ms, mainFrameId=${mainFrameId}, maxObservedFrames=${observedFrameCount}, url=${lastUrl || "<unknown>"})`
217
174
  );
218
175
  }
219
176
  async function waitForPageUrl(page, expectedUrlSubstring, timeoutMs = POPUP_URL_TIMEOUT_MS) {
@@ -321,9 +278,6 @@ async function waitForPopupPage(ctx, opener, timeoutMs = POPUP_TIMEOUT_MS) {
321
278
  );
322
279
  }
323
280
  test.describe("context.addInitScript with iframes", () => {
324
- const OOPIF_CHILD_URL = "https://seanmcguire12.github.io/stagehand-oopif-sites/sites/form-filling/";
325
- const SPIF_CHILD_URL = "https://browserbase.github.io/stagehand-eval-sites/sites/spif-in-closed-shadow-dom/iframe.html";
326
- const POPUP_SPIF_CHILD_URL = "https://browserbase.github.io/stagehand-eval-sites/sites/closed-shadow-dom-in-spif/embedded.html";
327
281
  if (isBrowserbase) {
328
282
  test.describe.configure({ mode: "serial" });
329
283
  }
@@ -364,7 +318,7 @@ test.describe("context.addInitScript with iframes", () => {
364
318
  "https://browserbase.github.io/stagehand-eval-sites/sites/oopif-in-closed-shadow-dom/",
365
319
  { waitUntil: "networkidle" }
366
320
  );
367
- const iframe = await waitForChildFrame(page, OOPIF_CHILD_URL);
321
+ const iframe = await waitForChildFrame(page);
368
322
  const mainBgColor = await page.mainFrame().evaluate(() => {
369
323
  return getComputedStyle(document.documentElement).backgroundColor;
370
324
  });
@@ -380,7 +334,7 @@ test.describe("context.addInitScript with iframes", () => {
380
334
  "https://browserbase.github.io/stagehand-eval-sites/sites/spif-in-closed-shadow-dom/",
381
335
  { waitUntil: "networkidle" }
382
336
  );
383
- const iframe = await waitForChildFrame(page, SPIF_CHILD_URL);
337
+ const iframe = await waitForChildFrame(page);
384
338
  const mainBgColor = await page.mainFrame().evaluate(() => {
385
339
  return getComputedStyle(document.documentElement).backgroundColor;
386
340
  });
@@ -398,7 +352,7 @@ test.describe("context.addInitScript with iframes", () => {
398
352
  "https://browserbase.github.io/stagehand-eval-sites/sites/oopif-in-closed-shadow-dom/",
399
353
  { waitUntil: "networkidle" }
400
354
  );
401
- const iframe = await waitForChildFrame(page, OOPIF_CHILD_URL);
355
+ const iframe = await waitForChildFrame(page);
402
356
  const mainBgColor = await page.mainFrame().evaluate(() => {
403
357
  return getComputedStyle(document.documentElement).backgroundColor;
404
358
  });
@@ -414,7 +368,7 @@ test.describe("context.addInitScript with iframes", () => {
414
368
  "https://browserbase.github.io/stagehand-eval-sites/sites/spif-in-closed-shadow-dom/",
415
369
  { waitUntil: "networkidle" }
416
370
  );
417
- const iframe = await waitForChildFrame(page, SPIF_CHILD_URL);
371
+ const iframe = await waitForChildFrame(page);
418
372
  const mainBgColor = await page.mainFrame().evaluate(() => {
419
373
  return getComputedStyle(document.documentElement).backgroundColor;
420
374
  });
@@ -452,7 +406,7 @@ test.describe("context.addInitScript with iframes", () => {
452
406
  "shadow-host"
453
407
  );
454
408
  await preparePopupForFrameAttach(popup, "shadow-host");
455
- const iframe = await waitForChildFrame(popup, OOPIF_CHILD_URL);
409
+ const iframe = await waitForChildFrame(popup);
456
410
  const mainBgColor = await popup.mainFrame().evaluate(() => {
457
411
  return getComputedStyle(document.documentElement).backgroundColor;
458
412
  });
@@ -478,7 +432,7 @@ test.describe("context.addInitScript with iframes", () => {
478
432
  "/stagehand-eval-sites/sites/closed-shadow-dom-in-spif/"
479
433
  );
480
434
  await preparePopupForFrameAttach(popup, "iframe");
481
- const iframe = await waitForChildFrame(popup, POPUP_SPIF_CHILD_URL);
435
+ const iframe = await waitForChildFrame(popup);
482
436
  const mainBgColor = await popup.mainFrame().evaluate(() => {
483
437
  return getComputedStyle(document.documentElement).backgroundColor;
484
438
  });
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../../../lib/v3/tests/iframe-ctx-addInitScript.spec.ts"],
4
- "sourcesContent": ["import { test, expect } from \"@playwright/test\";\nimport { V3 } from \"../v3.js\";\nimport { v3TestConfig } from \"./v3.config.js\";\nimport { V3Context } from \"../understudy/context.js\";\nimport type { Page } from \"../understudy/page.js\";\n\nconst isBrowserbase =\n (process.env.STAGEHAND_BROWSER_TARGET ?? \"local\").toLowerCase() ===\n \"browserbase\";\nconst MIN_TIMEOUT_MS = 3_000;\nconst MAX_TIMEOUT_MS = 120_000;\n\nconst parseBoundedTimeoutMs = (\n value: string | undefined,\n fallbackMs: number,\n): number => {\n const parsed = Number(value ?? fallbackMs);\n if (!Number.isFinite(parsed)) return fallbackMs;\n return Math.max(MIN_TIMEOUT_MS, Math.min(MAX_TIMEOUT_MS, parsed));\n};\n\nconst CHILD_FRAME_TIMEOUT_MS = parseBoundedTimeoutMs(\n process.env.IFRAME_CHILD_FRAME_TIMEOUT_MS,\n isBrowserbase ? 80_000 : 40_000,\n);\nconst POPUP_TIMEOUT_MS = parseBoundedTimeoutMs(\n process.env.IFRAME_POPUP_TIMEOUT_MS,\n isBrowserbase ? 60_000 : 40_000,\n);\nconst POPUP_URL_TIMEOUT_MS = parseBoundedTimeoutMs(\n process.env.IFRAME_POPUP_URL_TIMEOUT_MS,\n isBrowserbase ? 80_000 : 40_000,\n);\nconst DEBUG_INTERVAL_MS = 5_000;\nconst iframeDebugEnabled = isBrowserbase || process.env.IFRAME_DEBUG === \"1\";\nconst TEST_VIEWPORT = { width: 1288, height: 711 };\n\ntype FrameTreeNode = {\n frame: { id: string; parentId?: string; url?: string };\n childFrames?: FrameTreeNode[];\n};\n\nconst formatError = (error: unknown): string => {\n if (error instanceof Error) return error.message;\n return String(error);\n};\n\nconst flattenFrameTree = (\n node: FrameTreeNode,\n out: Array<{ id: string; parentId: string | null; url: string }> = [],\n): Array<{ id: string; parentId: string | null; url: string }> => {\n out.push({\n id: node.frame.id,\n parentId: node.frame.parentId ?? null,\n url: node.frame.url ?? \"\",\n });\n for (const child of node.childFrames ?? []) {\n flattenFrameTree(child, out);\n }\n return out;\n};\n\nfunction debugLog(\n step: string,\n payload?: Record<string, unknown> | string,\n): void {\n if (!iframeDebugEnabled) return;\n if (payload === undefined) {\n console.log(`[iframe-debug] ${step}`);\n return;\n }\n if (typeof payload === \"string\") {\n console.log(`[iframe-debug] ${step}: ${payload}`);\n return;\n }\n try {\n console.log(`[iframe-debug] ${step}: ${JSON.stringify(payload)}`);\n } catch {\n console.log(`[iframe-debug] ${step}: <unserializable payload>`);\n }\n}\n\nasync function collectFrameSnapshot(\n page: Page,\n): Promise<Array<Record<string, unknown>>> {\n const known = new Map<string, ReturnType<Page[\"frames\"]>[number]>();\n known.set(page.mainFrame().frameId, page.mainFrame());\n for (const frame of page.frames()) known.set(frame.frameId, frame);\n\n return Promise.all(\n [...known.values()].map(async (frame) => {\n try {\n const state = await frame.evaluate(() => {\n return {\n href: location.href,\n readyState: document.readyState,\n visibilityState: document.visibilityState,\n iframeCount: document.querySelectorAll(\"iframe\").length,\n hasShadowHost: Boolean(document.querySelector(\"shadow-host\")),\n };\n });\n return {\n frameId: frame.frameId,\n sessionId: frame.sessionId ?? \"root\",\n ...state,\n };\n } catch (error) {\n return {\n frameId: frame.frameId,\n sessionId: frame.sessionId ?? \"root\",\n error: formatError(error),\n };\n }\n }),\n );\n}\n\nasync function logPageDiagnostics(\n page: Page,\n reason: string,\n markerSelector?: string,\n): Promise<void> {\n if (!iframeDebugEnabled) return;\n const diagnostics: Record<string, unknown> = {\n reason,\n pageUrl: page.url(),\n mainFrameId: page.mainFrame().frameId,\n knownFrameCount: page.frames().length,\n };\n\n try {\n const domState = await page.mainFrame().evaluate((marker) => {\n const el = marker ? document.querySelector(marker) : null;\n const rect =\n el instanceof Element ? el.getBoundingClientRect().toJSON() : null;\n return {\n href: location.href,\n readyState: document.readyState,\n visibilityState: document.visibilityState,\n hidden: document.hidden,\n hasFocus: document.hasFocus(),\n innerWidth: window.innerWidth,\n innerHeight: window.innerHeight,\n devicePixelRatio: window.devicePixelRatio,\n markerSelector: marker,\n markerPresent: Boolean(el),\n markerRect: rect,\n iframeCount: document.querySelectorAll(\"iframe\").length,\n };\n }, markerSelector);\n diagnostics.domState = domState;\n } catch (error) {\n diagnostics.domStateError = formatError(error);\n }\n\n try {\n const frameTreeResponse = (await page.sendCDP(\"Page.getFrameTree\")) as {\n frameTree?: FrameTreeNode;\n };\n if (frameTreeResponse.frameTree) {\n diagnostics.cdpFrameTree = flattenFrameTree(frameTreeResponse.frameTree);\n }\n } catch (error) {\n diagnostics.cdpFrameTreeError = formatError(error);\n }\n\n diagnostics.frameSnapshot = await collectFrameSnapshot(page);\n debugLog(\"page-diagnostics\", diagnostics);\n}\n\nasync function closeAllPages(ctx: V3Context): Promise<void> {\n const pages = ctx.pages();\n await Promise.allSettled(pages.map((page) => page.close()));\n}\n\n/**\n * Poll until a child frame (non-main) appears on `page` and its document\n * has finished loading. Returns the child frame.\n */\nasync function waitForChildFrame(\n page: Page,\n expectedChildUrl: string,\n timeoutMs = CHILD_FRAME_TIMEOUT_MS,\n): Promise<ReturnType<Page[\"frames\"]>[number]> {\n const mainFrameId = page.mainFrame().frameId;\n const deadline = Date.now() + timeoutMs;\n let observedFrameCount = 0;\n const observedChildFrameIds = new Set<string>();\n let lastUrl = \"\";\n let lastLogAt = Date.now();\n\n while (Date.now() < deadline) {\n const frames = page.frames();\n observedFrameCount = Math.max(observedFrameCount, frames.length);\n lastUrl = page.url();\n const childIds = frames\n .filter((f) => f.frameId !== mainFrameId)\n .map((f) => f.frameId);\n if (iframeDebugEnabled && Date.now() - lastLogAt >= DEBUG_INTERVAL_MS) {\n debugLog(\"waitForChildFrame:progress\", {\n url: lastUrl,\n mainFrameId,\n observedFrameCount,\n childIds,\n expectedChildUrl,\n });\n lastLogAt = Date.now();\n }\n for (const childId of childIds) observedChildFrameIds.add(childId);\n\n const childFrames = frames\n .filter((f) => f.frameId !== mainFrameId)\n // Prefer recently-discovered frames first; stale swapped frame ids\n // can remain visible in the registry while the live OOPIF is ready.\n .reverse();\n\n if (childFrames.length) {\n const probes = await Promise.all(\n childFrames.map(async (child) => {\n try {\n const state = await child.evaluate(() => ({\n href: location.href,\n readyState: document.readyState,\n }));\n return {\n child,\n href: state.href,\n readyState: state.readyState,\n error: undefined,\n };\n } catch (error) {\n return {\n child,\n href: undefined,\n readyState: undefined,\n error: formatError(error),\n };\n }\n }),\n );\n\n const ready = probes.find(\n (probe) =>\n probe.readyState === \"complete\" && probe.href === expectedChildUrl,\n );\n if (ready) {\n debugLog(\"waitForChildFrame:ready\", {\n childFrameId: ready.child.frameId,\n childSessionId: ready.child.sessionId ?? \"root\",\n childUrl: ready.href ?? \"<unknown>\",\n expectedChildUrl,\n url: lastUrl,\n });\n return ready.child;\n }\n\n if (iframeDebugEnabled && Date.now() - lastLogAt >= DEBUG_INTERVAL_MS) {\n debugLog(\"waitForChildFrame:not-ready\", {\n url: lastUrl,\n mainFrameId,\n expectedChildUrl,\n probes: probes.map((probe) => ({\n frameId: probe.child.frameId,\n sessionId: probe.child.sessionId ?? \"root\",\n readyState: probe.readyState ?? \"<unknown>\",\n href: probe.href ?? \"<unknown>\",\n error: probe.error ?? \"<none>\",\n })),\n });\n lastLogAt = Date.now();\n }\n }\n await new Promise((r) => setTimeout(r, 100));\n }\n await logPageDiagnostics(page, \"waitForChildFrame timeout\");\n throw new Error(\n `Timed out waiting for child frame to load (timeout=${timeoutMs}ms, mainFrameId=${mainFrameId}, expectedChildUrl=${expectedChildUrl}, maxObservedFrames=${observedFrameCount}, observedChildFrameIds=[${[...observedChildFrameIds].join(\",\")}], url=${lastUrl || \"<unknown>\"})`,\n );\n}\n\nasync function waitForPageUrl(\n page: Page,\n expectedUrlSubstring: string,\n timeoutMs = POPUP_URL_TIMEOUT_MS,\n): Promise<void> {\n const deadline = Date.now() + timeoutMs;\n let lastUrl = \"\";\n let lastLogAt = Date.now();\n while (Date.now() < deadline) {\n lastUrl = page.url();\n if (iframeDebugEnabled && Date.now() - lastLogAt >= DEBUG_INTERVAL_MS) {\n debugLog(\"waitForPageUrl:progress\", {\n expectedUrlSubstring,\n lastUrl,\n });\n lastLogAt = Date.now();\n }\n if (lastUrl.includes(expectedUrlSubstring)) {\n debugLog(\"waitForPageUrl:ready\", {\n expectedUrlSubstring,\n lastUrl,\n });\n return;\n }\n await new Promise((r) => setTimeout(r, 100));\n }\n await logPageDiagnostics(\n page,\n `waitForPageUrl timeout for ${expectedUrlSubstring}`,\n );\n throw new Error(\n `Timed out waiting for popup URL to include \"${expectedUrlSubstring}\" (timeout=${timeoutMs}ms, lastUrl=${lastUrl || \"<unknown>\"})`,\n );\n}\n\nasync function preparePopupForFrameAttach(\n page: Page,\n markerSelector: string,\n timeoutMs = CHILD_FRAME_TIMEOUT_MS,\n): Promise<void> {\n debugLog(\"preparePopupForFrameAttach:start\", {\n markerSelector,\n timeoutMs,\n url: page.url(),\n });\n await page.waitForLoadState(\"domcontentloaded\", timeoutMs);\n await page.waitForSelector(markerSelector, {\n state: \"attached\",\n timeout: timeoutMs,\n });\n await page.mainFrame().evaluate(() => {\n const host = document.querySelector(\"shadow-host\");\n if (host instanceof HTMLElement) {\n host.scrollIntoView({ block: \"center\", inline: \"center\" });\n } else {\n window.scrollTo(0, document.body.scrollHeight);\n window.scrollTo(0, 0);\n }\n window.dispatchEvent(new Event(\"scroll\"));\n });\n await logPageDiagnostics(\n page,\n \"preparePopupForFrameAttach:ready\",\n markerSelector,\n );\n}\n\nasync function ensurePopupViewport(page: Page): Promise<void> {\n await page.setViewportSize(TEST_VIEWPORT.width, TEST_VIEWPORT.height);\n await logPageDiagnostics(page, \"ensurePopupViewport\");\n}\n\nasync function waitForPopupPage(\n ctx: V3Context,\n opener: Page,\n timeoutMs = POPUP_TIMEOUT_MS,\n): Promise<Page> {\n const openerMainFrameId = opener.mainFrame().frameId;\n const deadline = Date.now() + timeoutMs;\n let lastLogAt = Date.now();\n\n while (Date.now() < deadline) {\n const pages = ctx.pages();\n const popup = pages.find((candidate) => {\n return candidate.mainFrame().frameId !== openerMainFrameId;\n });\n if (popup) {\n debugLog(\"waitForPopupPage:found\", {\n openerMainFrameId,\n popupMainFrameId: popup.mainFrame().frameId,\n popupUrl: popup.url(),\n });\n return popup;\n }\n\n if (iframeDebugEnabled && Date.now() - lastLogAt >= DEBUG_INTERVAL_MS) {\n debugLog(\"waitForPopupPage:progress\", {\n openerMainFrameId,\n observedPageIds: pages.map((p) => p.mainFrame().frameId),\n });\n lastLogAt = Date.now();\n }\n\n try {\n const active = await ctx.awaitActivePage(500);\n if (active.mainFrame().frameId !== openerMainFrameId) {\n debugLog(\"waitForPopupPage:active-non-opener\", {\n openerMainFrameId,\n activeMainFrameId: active.mainFrame().frameId,\n activeUrl: active.url(),\n });\n return active;\n }\n } catch {\n // keep polling until timeout\n }\n\n await new Promise((r) => setTimeout(r, 100));\n }\n\n const pageIds = ctx\n .pages()\n .map((p) => p.mainFrame().frameId)\n .join(\", \");\n throw new Error(\n `Timed out waiting for popup page (timeout=${timeoutMs}ms, openerMainFrameId=${openerMainFrameId}, observedPages=[${pageIds}])`,\n );\n}\n\ntest.describe(\"context.addInitScript with iframes\", () => {\n const OOPIF_CHILD_URL =\n \"https://seanmcguire12.github.io/stagehand-oopif-sites/sites/form-filling/\";\n const SPIF_CHILD_URL =\n \"https://browserbase.github.io/stagehand-eval-sites/sites/spif-in-closed-shadow-dom/iframe.html\";\n const POPUP_SPIF_CHILD_URL =\n \"https://browserbase.github.io/stagehand-eval-sites/sites/closed-shadow-dom-in-spif/embedded.html\";\n\n if (isBrowserbase) {\n test.describe.configure({ mode: \"serial\" });\n }\n\n let v3: V3;\n let ctx: V3Context;\n\n test.beforeAll(async () => {\n debugLog(\"beforeAll:config\", {\n browserTarget: process.env.STAGEHAND_BROWSER_TARGET ?? \"local\",\n childFrameTimeoutMs: CHILD_FRAME_TIMEOUT_MS,\n popupTimeoutMs: POPUP_TIMEOUT_MS,\n popupUrlTimeoutMs: POPUP_URL_TIMEOUT_MS,\n });\n v3 = new V3(v3TestConfig);\n await v3.init();\n ctx = v3.context;\n\n // Add init script that sets background to red\n await ctx.addInitScript(`\n (() => {\n document.addEventListener('DOMContentLoaded', () => {\n document.documentElement.style.backgroundColor = 'red';\n });\n })();\n `);\n });\n\n test.beforeEach(async () => {\n await closeAllPages(ctx);\n });\n\n test.afterEach(async () => {\n await closeAllPages(ctx);\n });\n\n test.afterAll(async () => {\n await v3?.close?.().catch(() => {});\n });\n\n test.describe(\"direct navigation\", () => {\n test(\"with OOPIF - sets background red in main page and iframe\", async () => {\n const page = await ctx.newPage();\n\n await page.goto(\n \"https://browserbase.github.io/stagehand-eval-sites/sites/oopif-in-closed-shadow-dom/\",\n { waitUntil: \"networkidle\" },\n );\n\n const iframe = await waitForChildFrame(page, OOPIF_CHILD_URL);\n\n // Check main page background\n const mainBgColor = await page.mainFrame().evaluate(() => {\n return getComputedStyle(document.documentElement).backgroundColor;\n });\n expect(mainBgColor).toBe(\"rgb(255, 0, 0)\");\n\n const iframeBgColor = await iframe.evaluate(() => {\n return getComputedStyle(document.documentElement).backgroundColor;\n });\n expect(iframeBgColor).toBe(\"rgb(255, 0, 0)\");\n });\n\n test(\"with SPIF - sets background red in main page and iframe\", async () => {\n const page = await ctx.newPage();\n\n await page.goto(\n \"https://browserbase.github.io/stagehand-eval-sites/sites/spif-in-closed-shadow-dom/\",\n { waitUntil: \"networkidle\" },\n );\n\n const iframe = await waitForChildFrame(page, SPIF_CHILD_URL);\n\n // Check main page background\n const mainBgColor = await page.mainFrame().evaluate(() => {\n return getComputedStyle(document.documentElement).backgroundColor;\n });\n expect(mainBgColor).toBe(\"rgb(255, 0, 0)\");\n\n const iframeBgColor = await iframe.evaluate(() => {\n return getComputedStyle(document.documentElement).backgroundColor;\n });\n expect(iframeBgColor).toBe(\"rgb(255, 0, 0)\");\n });\n });\n\n test.describe(\"via newPage\", () => {\n test(\"with OOPIF - sets background red in main page and iframe\", async () => {\n const page = await ctx.newPage();\n\n await page.goto(\n \"https://browserbase.github.io/stagehand-eval-sites/sites/oopif-in-closed-shadow-dom/\",\n { waitUntil: \"networkidle\" },\n );\n\n const iframe = await waitForChildFrame(page, OOPIF_CHILD_URL);\n\n // Check main page background\n const mainBgColor = await page.mainFrame().evaluate(() => {\n return getComputedStyle(document.documentElement).backgroundColor;\n });\n expect(mainBgColor).toBe(\"rgb(255, 0, 0)\");\n\n const iframeBgColor = await iframe.evaluate(() => {\n return getComputedStyle(document.documentElement).backgroundColor;\n });\n expect(iframeBgColor).toBe(\"rgb(255, 0, 0)\");\n });\n\n test(\"with SPIF - sets background red in main page and iframe\", async () => {\n const page = await ctx.newPage();\n\n await page.goto(\n \"https://browserbase.github.io/stagehand-eval-sites/sites/spif-in-closed-shadow-dom/\",\n { waitUntil: \"networkidle\" },\n );\n\n const iframe = await waitForChildFrame(page, SPIF_CHILD_URL);\n\n // Check main page background\n const mainBgColor = await page.mainFrame().evaluate(() => {\n return getComputedStyle(document.documentElement).backgroundColor;\n });\n expect(mainBgColor).toBe(\"rgb(255, 0, 0)\");\n\n const iframeBgColor = await iframe.evaluate(() => {\n return getComputedStyle(document.documentElement).backgroundColor;\n });\n expect(iframeBgColor).toBe(\"rgb(255, 0, 0)\");\n });\n });\n\n test.describe(\"via popup\", () => {\n test(\"with OOPIF - sets background red in main page and iframe\", async () => {\n const page = await ctx.newPage();\n\n await page.goto(\n \"https://browserbase.github.io/stagehand-eval-sites/sites/ctx-add-init-script-oopif/\",\n { waitUntil: \"networkidle\" },\n );\n\n // Click link to open popup\n await page.locator(\"a\").click();\n debugLog(\"popup-oopif:clicked-link\", { openerUrl: page.url() });\n\n // Wait for popup to open and become active\n const popup = await waitForPopupPage(ctx, page);\n ctx.setActivePage(popup);\n await ensurePopupViewport(popup);\n await waitForPageUrl(\n popup,\n \"/stagehand-eval-sites/sites/oopif-in-closed-shadow-dom/\",\n );\n debugLog(\"popup-oopif:refresh-navigation\", { url: popup.url() });\n await popup.goto(\n \"https://browserbase.github.io/stagehand-eval-sites/sites/oopif-in-closed-shadow-dom/\",\n { waitUntil: \"networkidle\" },\n );\n await logPageDiagnostics(\n popup,\n \"popup-oopif:after-refresh\",\n \"shadow-host\",\n );\n await preparePopupForFrameAttach(popup, \"shadow-host\");\n const iframe = await waitForChildFrame(popup, OOPIF_CHILD_URL);\n\n // Check popup main page background\n const mainBgColor = await popup.mainFrame().evaluate(() => {\n return getComputedStyle(document.documentElement).backgroundColor;\n });\n expect(mainBgColor).toBe(\"rgb(255, 0, 0)\");\n\n const iframeBgColor = await iframe.evaluate(() => {\n return getComputedStyle(document.documentElement).backgroundColor;\n });\n expect(iframeBgColor).toBe(\"rgb(255, 0, 0)\");\n });\n\n test(\"with SPIF - sets background red in main page and iframe\", async () => {\n const page = await ctx.newPage();\n\n await page.goto(\n \"https://browserbase.github.io/stagehand-eval-sites/sites/ctx-add-init-script-spif/\",\n { waitUntil: \"networkidle\" },\n );\n\n // Click link to open popup\n await page.locator(\"a\").click();\n debugLog(\"popup-spif:clicked-link\", { openerUrl: page.url() });\n\n // Wait for popup to open and become active\n const popup = await waitForPopupPage(ctx, page);\n ctx.setActivePage(popup);\n await ensurePopupViewport(popup);\n await waitForPageUrl(\n popup,\n \"/stagehand-eval-sites/sites/closed-shadow-dom-in-spif/\",\n );\n await preparePopupForFrameAttach(popup, \"iframe\");\n const iframe = await waitForChildFrame(popup, POPUP_SPIF_CHILD_URL);\n\n // Check popup main page background\n const mainBgColor = await popup.mainFrame().evaluate(() => {\n return getComputedStyle(document.documentElement).backgroundColor;\n });\n expect(mainBgColor).toBe(\"rgb(255, 0, 0)\");\n\n const iframeBgColor = await iframe.evaluate(() => {\n return getComputedStyle(document.documentElement).backgroundColor;\n });\n expect(iframeBgColor).toBe(\"rgb(255, 0, 0)\");\n });\n });\n});\n"],
5
- "mappings": "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",
4
+ "sourcesContent": ["import { test, expect } from \"@playwright/test\";\nimport { V3 } from \"../v3.js\";\nimport { v3TestConfig } from \"./v3.config.js\";\nimport { V3Context } from \"../understudy/context.js\";\nimport type { Page } from \"../understudy/page.js\";\n\nconst isBrowserbase =\n (process.env.STAGEHAND_BROWSER_TARGET ?? \"local\").toLowerCase() ===\n \"browserbase\";\nconst MIN_TIMEOUT_MS = 3_000;\nconst MAX_TIMEOUT_MS = 120_000;\n\nconst parseBoundedTimeoutMs = (\n value: string | undefined,\n fallbackMs: number,\n): number => {\n const parsed = Number(value ?? fallbackMs);\n if (!Number.isFinite(parsed)) return fallbackMs;\n return Math.max(MIN_TIMEOUT_MS, Math.min(MAX_TIMEOUT_MS, parsed));\n};\n\nconst CHILD_FRAME_TIMEOUT_MS = parseBoundedTimeoutMs(\n process.env.IFRAME_CHILD_FRAME_TIMEOUT_MS,\n isBrowserbase ? 80_000 : 40_000,\n);\nconst POPUP_TIMEOUT_MS = parseBoundedTimeoutMs(\n process.env.IFRAME_POPUP_TIMEOUT_MS,\n isBrowserbase ? 60_000 : 40_000,\n);\nconst POPUP_URL_TIMEOUT_MS = parseBoundedTimeoutMs(\n process.env.IFRAME_POPUP_URL_TIMEOUT_MS,\n isBrowserbase ? 80_000 : 40_000,\n);\nconst DEBUG_INTERVAL_MS = 5_000;\nconst iframeDebugEnabled = isBrowserbase || process.env.IFRAME_DEBUG === \"1\";\nconst TEST_VIEWPORT = { width: 1288, height: 711 };\n\ntype FrameTreeNode = {\n frame: { id: string; parentId?: string; url?: string };\n childFrames?: FrameTreeNode[];\n};\n\nconst formatError = (error: unknown): string => {\n if (error instanceof Error) return error.message;\n return String(error);\n};\n\nconst flattenFrameTree = (\n node: FrameTreeNode,\n out: Array<{ id: string; parentId: string | null; url: string }> = [],\n): Array<{ id: string; parentId: string | null; url: string }> => {\n out.push({\n id: node.frame.id,\n parentId: node.frame.parentId ?? null,\n url: node.frame.url ?? \"\",\n });\n for (const child of node.childFrames ?? []) {\n flattenFrameTree(child, out);\n }\n return out;\n};\n\nfunction debugLog(\n step: string,\n payload?: Record<string, unknown> | string,\n): void {\n if (!iframeDebugEnabled) return;\n if (payload === undefined) {\n console.log(`[iframe-debug] ${step}`);\n return;\n }\n if (typeof payload === \"string\") {\n console.log(`[iframe-debug] ${step}: ${payload}`);\n return;\n }\n try {\n console.log(`[iframe-debug] ${step}: ${JSON.stringify(payload)}`);\n } catch {\n console.log(`[iframe-debug] ${step}: <unserializable payload>`);\n }\n}\n\nasync function collectFrameSnapshot(\n page: Page,\n): Promise<Array<Record<string, unknown>>> {\n const known = new Map<string, ReturnType<Page[\"frames\"]>[number]>();\n known.set(page.mainFrame().frameId, page.mainFrame());\n for (const frame of page.frames()) known.set(frame.frameId, frame);\n\n return Promise.all(\n [...known.values()].map(async (frame) => {\n try {\n const state = await frame.evaluate(() => {\n return {\n href: location.href,\n readyState: document.readyState,\n visibilityState: document.visibilityState,\n iframeCount: document.querySelectorAll(\"iframe\").length,\n hasShadowHost: Boolean(document.querySelector(\"shadow-host\")),\n };\n });\n return {\n frameId: frame.frameId,\n sessionId: frame.sessionId ?? \"root\",\n ...state,\n };\n } catch (error) {\n return {\n frameId: frame.frameId,\n sessionId: frame.sessionId ?? \"root\",\n error: formatError(error),\n };\n }\n }),\n );\n}\n\nasync function logPageDiagnostics(\n page: Page,\n reason: string,\n markerSelector?: string,\n): Promise<void> {\n if (!iframeDebugEnabled) return;\n const diagnostics: Record<string, unknown> = {\n reason,\n pageUrl: page.url(),\n mainFrameId: page.mainFrame().frameId,\n knownFrameCount: page.frames().length,\n };\n\n try {\n const domState = await page.mainFrame().evaluate((marker) => {\n const el = marker ? document.querySelector(marker) : null;\n const rect =\n el instanceof Element ? el.getBoundingClientRect().toJSON() : null;\n return {\n href: location.href,\n readyState: document.readyState,\n visibilityState: document.visibilityState,\n hidden: document.hidden,\n hasFocus: document.hasFocus(),\n innerWidth: window.innerWidth,\n innerHeight: window.innerHeight,\n devicePixelRatio: window.devicePixelRatio,\n markerSelector: marker,\n markerPresent: Boolean(el),\n markerRect: rect,\n iframeCount: document.querySelectorAll(\"iframe\").length,\n };\n }, markerSelector);\n diagnostics.domState = domState;\n } catch (error) {\n diagnostics.domStateError = formatError(error);\n }\n\n try {\n const frameTreeResponse = (await page.sendCDP(\"Page.getFrameTree\")) as {\n frameTree?: FrameTreeNode;\n };\n if (frameTreeResponse.frameTree) {\n diagnostics.cdpFrameTree = flattenFrameTree(frameTreeResponse.frameTree);\n }\n } catch (error) {\n diagnostics.cdpFrameTreeError = formatError(error);\n }\n\n diagnostics.frameSnapshot = await collectFrameSnapshot(page);\n debugLog(\"page-diagnostics\", diagnostics);\n}\n\nasync function closeAllPages(ctx: V3Context): Promise<void> {\n const pages = ctx.pages();\n await Promise.allSettled(pages.map((page) => page.close()));\n}\n\n/**\n * Poll until a child frame (non-main) appears on `page` and its document\n * has finished loading. Returns the child frame.\n */\nasync function waitForChildFrame(\n page: Page,\n timeoutMs = CHILD_FRAME_TIMEOUT_MS,\n): Promise<ReturnType<Page[\"frames\"]>[number]> {\n const mainFrameId = page.mainFrame().frameId;\n const deadline = Date.now() + timeoutMs;\n let observedFrameCount = 0;\n let lastUrl = \"\";\n let lastLogAt = Date.now();\n\n while (Date.now() < deadline) {\n const frames = page.frames();\n observedFrameCount = Math.max(observedFrameCount, frames.length);\n lastUrl = page.url();\n const childIds = frames\n .filter((f) => f.frameId !== mainFrameId)\n .map((f) => f.frameId);\n if (iframeDebugEnabled && Date.now() - lastLogAt >= DEBUG_INTERVAL_MS) {\n debugLog(\"waitForChildFrame:progress\", {\n url: lastUrl,\n mainFrameId,\n observedFrameCount,\n childIds,\n });\n lastLogAt = Date.now();\n }\n const child = frames.find((f) => f.frameId !== mainFrameId);\n if (child) {\n try {\n const ready = await child.evaluate(() => document.readyState);\n if (ready === \"complete\") {\n debugLog(\"waitForChildFrame:ready\", {\n childFrameId: child.frameId,\n url: lastUrl,\n });\n return child;\n }\n } catch {\n // frame not ready yet\n }\n }\n await new Promise((r) => setTimeout(r, 100));\n }\n await logPageDiagnostics(page, \"waitForChildFrame timeout\");\n throw new Error(\n `Timed out waiting for child frame to load (timeout=${timeoutMs}ms, mainFrameId=${mainFrameId}, maxObservedFrames=${observedFrameCount}, url=${lastUrl || \"<unknown>\"})`,\n );\n}\n\nasync function waitForPageUrl(\n page: Page,\n expectedUrlSubstring: string,\n timeoutMs = POPUP_URL_TIMEOUT_MS,\n): Promise<void> {\n const deadline = Date.now() + timeoutMs;\n let lastUrl = \"\";\n let lastLogAt = Date.now();\n while (Date.now() < deadline) {\n lastUrl = page.url();\n if (iframeDebugEnabled && Date.now() - lastLogAt >= DEBUG_INTERVAL_MS) {\n debugLog(\"waitForPageUrl:progress\", {\n expectedUrlSubstring,\n lastUrl,\n });\n lastLogAt = Date.now();\n }\n if (lastUrl.includes(expectedUrlSubstring)) {\n debugLog(\"waitForPageUrl:ready\", {\n expectedUrlSubstring,\n lastUrl,\n });\n return;\n }\n await new Promise((r) => setTimeout(r, 100));\n }\n await logPageDiagnostics(\n page,\n `waitForPageUrl timeout for ${expectedUrlSubstring}`,\n );\n throw new Error(\n `Timed out waiting for popup URL to include \"${expectedUrlSubstring}\" (timeout=${timeoutMs}ms, lastUrl=${lastUrl || \"<unknown>\"})`,\n );\n}\n\nasync function preparePopupForFrameAttach(\n page: Page,\n markerSelector: string,\n timeoutMs = CHILD_FRAME_TIMEOUT_MS,\n): Promise<void> {\n debugLog(\"preparePopupForFrameAttach:start\", {\n markerSelector,\n timeoutMs,\n url: page.url(),\n });\n await page.waitForLoadState(\"domcontentloaded\", timeoutMs);\n await page.waitForSelector(markerSelector, {\n state: \"attached\",\n timeout: timeoutMs,\n });\n await page.mainFrame().evaluate(() => {\n const host = document.querySelector(\"shadow-host\");\n if (host instanceof HTMLElement) {\n host.scrollIntoView({ block: \"center\", inline: \"center\" });\n } else {\n window.scrollTo(0, document.body.scrollHeight);\n window.scrollTo(0, 0);\n }\n window.dispatchEvent(new Event(\"scroll\"));\n });\n await logPageDiagnostics(\n page,\n \"preparePopupForFrameAttach:ready\",\n markerSelector,\n );\n}\n\nasync function ensurePopupViewport(page: Page): Promise<void> {\n await page.setViewportSize(TEST_VIEWPORT.width, TEST_VIEWPORT.height);\n await logPageDiagnostics(page, \"ensurePopupViewport\");\n}\n\nasync function waitForPopupPage(\n ctx: V3Context,\n opener: Page,\n timeoutMs = POPUP_TIMEOUT_MS,\n): Promise<Page> {\n const openerMainFrameId = opener.mainFrame().frameId;\n const deadline = Date.now() + timeoutMs;\n let lastLogAt = Date.now();\n\n while (Date.now() < deadline) {\n const pages = ctx.pages();\n const popup = pages.find((candidate) => {\n return candidate.mainFrame().frameId !== openerMainFrameId;\n });\n if (popup) {\n debugLog(\"waitForPopupPage:found\", {\n openerMainFrameId,\n popupMainFrameId: popup.mainFrame().frameId,\n popupUrl: popup.url(),\n });\n return popup;\n }\n\n if (iframeDebugEnabled && Date.now() - lastLogAt >= DEBUG_INTERVAL_MS) {\n debugLog(\"waitForPopupPage:progress\", {\n openerMainFrameId,\n observedPageIds: pages.map((p) => p.mainFrame().frameId),\n });\n lastLogAt = Date.now();\n }\n\n try {\n const active = await ctx.awaitActivePage(500);\n if (active.mainFrame().frameId !== openerMainFrameId) {\n debugLog(\"waitForPopupPage:active-non-opener\", {\n openerMainFrameId,\n activeMainFrameId: active.mainFrame().frameId,\n activeUrl: active.url(),\n });\n return active;\n }\n } catch {\n // keep polling until timeout\n }\n\n await new Promise((r) => setTimeout(r, 100));\n }\n\n const pageIds = ctx\n .pages()\n .map((p) => p.mainFrame().frameId)\n .join(\", \");\n throw new Error(\n `Timed out waiting for popup page (timeout=${timeoutMs}ms, openerMainFrameId=${openerMainFrameId}, observedPages=[${pageIds}])`,\n );\n}\n\ntest.describe(\"context.addInitScript with iframes\", () => {\n if (isBrowserbase) {\n test.describe.configure({ mode: \"serial\" });\n }\n\n let v3: V3;\n let ctx: V3Context;\n\n test.beforeAll(async () => {\n debugLog(\"beforeAll:config\", {\n browserTarget: process.env.STAGEHAND_BROWSER_TARGET ?? \"local\",\n childFrameTimeoutMs: CHILD_FRAME_TIMEOUT_MS,\n popupTimeoutMs: POPUP_TIMEOUT_MS,\n popupUrlTimeoutMs: POPUP_URL_TIMEOUT_MS,\n });\n v3 = new V3(v3TestConfig);\n await v3.init();\n ctx = v3.context;\n\n // Add init script that sets background to red\n await ctx.addInitScript(`\n (() => {\n document.addEventListener('DOMContentLoaded', () => {\n document.documentElement.style.backgroundColor = 'red';\n });\n })();\n `);\n });\n\n test.beforeEach(async () => {\n await closeAllPages(ctx);\n });\n\n test.afterEach(async () => {\n await closeAllPages(ctx);\n });\n\n test.afterAll(async () => {\n await v3?.close?.().catch(() => {});\n });\n\n test.describe(\"direct navigation\", () => {\n test(\"with OOPIF - sets background red in main page and iframe\", async () => {\n const page = await ctx.newPage();\n\n await page.goto(\n \"https://browserbase.github.io/stagehand-eval-sites/sites/oopif-in-closed-shadow-dom/\",\n { waitUntil: \"networkidle\" },\n );\n\n const iframe = await waitForChildFrame(page);\n\n // Check main page background\n const mainBgColor = await page.mainFrame().evaluate(() => {\n return getComputedStyle(document.documentElement).backgroundColor;\n });\n expect(mainBgColor).toBe(\"rgb(255, 0, 0)\");\n\n const iframeBgColor = await iframe.evaluate(() => {\n return getComputedStyle(document.documentElement).backgroundColor;\n });\n expect(iframeBgColor).toBe(\"rgb(255, 0, 0)\");\n });\n\n test(\"with SPIF - sets background red in main page and iframe\", async () => {\n const page = await ctx.newPage();\n\n await page.goto(\n \"https://browserbase.github.io/stagehand-eval-sites/sites/spif-in-closed-shadow-dom/\",\n { waitUntil: \"networkidle\" },\n );\n\n const iframe = await waitForChildFrame(page);\n\n // Check main page background\n const mainBgColor = await page.mainFrame().evaluate(() => {\n return getComputedStyle(document.documentElement).backgroundColor;\n });\n expect(mainBgColor).toBe(\"rgb(255, 0, 0)\");\n\n const iframeBgColor = await iframe.evaluate(() => {\n return getComputedStyle(document.documentElement).backgroundColor;\n });\n expect(iframeBgColor).toBe(\"rgb(255, 0, 0)\");\n });\n });\n\n test.describe(\"via newPage\", () => {\n test(\"with OOPIF - sets background red in main page and iframe\", async () => {\n const page = await ctx.newPage();\n\n await page.goto(\n \"https://browserbase.github.io/stagehand-eval-sites/sites/oopif-in-closed-shadow-dom/\",\n { waitUntil: \"networkidle\" },\n );\n\n const iframe = await waitForChildFrame(page);\n\n // Check main page background\n const mainBgColor = await page.mainFrame().evaluate(() => {\n return getComputedStyle(document.documentElement).backgroundColor;\n });\n expect(mainBgColor).toBe(\"rgb(255, 0, 0)\");\n\n const iframeBgColor = await iframe.evaluate(() => {\n return getComputedStyle(document.documentElement).backgroundColor;\n });\n expect(iframeBgColor).toBe(\"rgb(255, 0, 0)\");\n });\n\n test(\"with SPIF - sets background red in main page and iframe\", async () => {\n const page = await ctx.newPage();\n\n await page.goto(\n \"https://browserbase.github.io/stagehand-eval-sites/sites/spif-in-closed-shadow-dom/\",\n { waitUntil: \"networkidle\" },\n );\n\n const iframe = await waitForChildFrame(page);\n\n // Check main page background\n const mainBgColor = await page.mainFrame().evaluate(() => {\n return getComputedStyle(document.documentElement).backgroundColor;\n });\n expect(mainBgColor).toBe(\"rgb(255, 0, 0)\");\n\n const iframeBgColor = await iframe.evaluate(() => {\n return getComputedStyle(document.documentElement).backgroundColor;\n });\n expect(iframeBgColor).toBe(\"rgb(255, 0, 0)\");\n });\n });\n\n test.describe(\"via popup\", () => {\n test(\"with OOPIF - sets background red in main page and iframe\", async () => {\n const page = await ctx.newPage();\n\n await page.goto(\n \"https://browserbase.github.io/stagehand-eval-sites/sites/ctx-add-init-script-oopif/\",\n { waitUntil: \"networkidle\" },\n );\n\n // Click link to open popup\n await page.locator(\"a\").click();\n debugLog(\"popup-oopif:clicked-link\", { openerUrl: page.url() });\n\n // Wait for popup to open and become active\n const popup = await waitForPopupPage(ctx, page);\n ctx.setActivePage(popup);\n await ensurePopupViewport(popup);\n await waitForPageUrl(\n popup,\n \"/stagehand-eval-sites/sites/oopif-in-closed-shadow-dom/\",\n );\n debugLog(\"popup-oopif:refresh-navigation\", { url: popup.url() });\n await popup.goto(\n \"https://browserbase.github.io/stagehand-eval-sites/sites/oopif-in-closed-shadow-dom/\",\n { waitUntil: \"networkidle\" },\n );\n await logPageDiagnostics(\n popup,\n \"popup-oopif:after-refresh\",\n \"shadow-host\",\n );\n await preparePopupForFrameAttach(popup, \"shadow-host\");\n const iframe = await waitForChildFrame(popup);\n\n // Check popup main page background\n const mainBgColor = await popup.mainFrame().evaluate(() => {\n return getComputedStyle(document.documentElement).backgroundColor;\n });\n expect(mainBgColor).toBe(\"rgb(255, 0, 0)\");\n\n const iframeBgColor = await iframe.evaluate(() => {\n return getComputedStyle(document.documentElement).backgroundColor;\n });\n expect(iframeBgColor).toBe(\"rgb(255, 0, 0)\");\n });\n\n test(\"with SPIF - sets background red in main page and iframe\", async () => {\n const page = await ctx.newPage();\n\n await page.goto(\n \"https://browserbase.github.io/stagehand-eval-sites/sites/ctx-add-init-script-spif/\",\n { waitUntil: \"networkidle\" },\n );\n\n // Click link to open popup\n await page.locator(\"a\").click();\n debugLog(\"popup-spif:clicked-link\", { openerUrl: page.url() });\n\n // Wait for popup to open and become active\n const popup = await waitForPopupPage(ctx, page);\n ctx.setActivePage(popup);\n await ensurePopupViewport(popup);\n await waitForPageUrl(\n popup,\n \"/stagehand-eval-sites/sites/closed-shadow-dom-in-spif/\",\n );\n await preparePopupForFrameAttach(popup, \"iframe\");\n const iframe = await waitForChildFrame(popup);\n\n // Check popup main page background\n const mainBgColor = await popup.mainFrame().evaluate(() => {\n return getComputedStyle(document.documentElement).backgroundColor;\n });\n expect(mainBgColor).toBe(\"rgb(255, 0, 0)\");\n\n const iframeBgColor = await iframe.evaluate(() => {\n return getComputedStyle(document.documentElement).backgroundColor;\n });\n expect(iframeBgColor).toBe(\"rgb(255, 0, 0)\");\n });\n });\n});\n"],
5
+ "mappings": "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",
6
6
  "names": []
7
7
  }
@@ -1,6 +1,39 @@
1
1
  import { defineConfig } from "@playwright/test";
2
- import { fileURLToPath } from "node:url";
3
- const testDir = fileURLToPath(new URL(".", import.meta.url));
2
+ import fs from "node:fs";
3
+ import path from "node:path";
4
+ const resolveRepoRoot = (startDir) => {
5
+ let current = path.resolve(startDir);
6
+ while (true) {
7
+ if (fs.existsSync(path.join(current, "pnpm-workspace.yaml"))) {
8
+ return current;
9
+ }
10
+ const parent = path.dirname(current);
11
+ if (parent === current) {
12
+ return startDir;
13
+ }
14
+ current = parent;
15
+ }
16
+ };
17
+ const repoRoot = resolveRepoRoot(process.cwd());
18
+ const distTestDir = path.join(
19
+ repoRoot,
20
+ "packages",
21
+ "core",
22
+ "dist",
23
+ "esm",
24
+ "lib",
25
+ "v3",
26
+ "tests"
27
+ );
28
+ const srcTestDir = path.join(
29
+ repoRoot,
30
+ "packages",
31
+ "core",
32
+ "lib",
33
+ "v3",
34
+ "tests"
35
+ );
36
+ const testDir = fs.existsSync(distTestDir) ? distTestDir : srcTestDir;
4
37
  const browserTarget = (process.env.STAGEHAND_BROWSER_TARGET ?? "local").toLowerCase();
5
38
  const isBrowserbase = browserTarget === "browserbase";
6
39
  const consoleReporter = process.env.PLAYWRIGHT_CONSOLE_REPORTER ?? "list";
@@ -13,10 +46,34 @@ const ciWorkerOverride = Number(
13
46
  );
14
47
  const bbWorkers = process.env.CI && Number.isFinite(ciWorkerOverride) && ciWorkerOverride > 0 ? ciWorkerOverride : 3;
15
48
  const ctrfJunitPath = process.env.CTRF_JUNIT_PATH;
49
+ const envReporterPath = (() => {
50
+ const distPath = path.join(
51
+ repoRoot,
52
+ "packages",
53
+ "core",
54
+ "dist",
55
+ "esm",
56
+ "lib",
57
+ "v3",
58
+ "tests",
59
+ "envReporter.js"
60
+ );
61
+ if (fs.existsSync(distPath)) return distPath;
62
+ return path.join(
63
+ repoRoot,
64
+ "packages",
65
+ "core",
66
+ "lib",
67
+ "v3",
68
+ "tests",
69
+ "envReporter.ts"
70
+ );
71
+ })();
16
72
  const reporter = ctrfJunitPath ? [
17
73
  [consoleReporter],
74
+ [envReporterPath],
18
75
  ["junit", { outputFile: ctrfJunitPath, includeProjectInTestName: true }]
19
- ] : [[consoleReporter]];
76
+ ] : [[consoleReporter], [envReporterPath]];
20
77
  var v3_playwright_config_default = defineConfig({
21
78
  testDir,
22
79
  timeout: 9e4,