@pencil-agent/nano-pencil 2.0.1 → 2.0.3

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 (188) hide show
  1. package/README.md +267 -267
  2. package/dist/build-meta.json +3 -3
  3. package/dist/core/export-html/AGENT.md +11 -11
  4. package/dist/core/export-html/template.css +971 -971
  5. package/dist/core/export-html/template.html +54 -54
  6. package/dist/core/model/custom-providers.js +1 -1
  7. package/dist/core/model-registry.js +5 -5
  8. package/dist/extensions/builtin/AGENT.md +115 -115
  9. package/dist/extensions/builtin/browser/AGENT.md +17 -17
  10. package/dist/extensions/builtin/browser/agent-workspace/agent_helpers.py +12 -12
  11. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/amazon/product-search.md +198 -198
  12. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/archive-org/scraping.md +341 -341
  13. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/arxiv/scraping.md +311 -311
  14. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/arxiv-bulk/scraping.md +333 -333
  15. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/atlas/overview.md +70 -70
  16. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/booking-com/scraping.md +578 -578
  17. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/capterra/scraping.md +440 -440
  18. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/centilebrain/generate-estimates.md +110 -110
  19. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/coingecko/scraping.md +325 -325
  20. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/coinmarketcap/scraping.md +463 -463
  21. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/coursera/scraping.md +360 -360
  22. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/craigslist/scraping.md +390 -390
  23. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/crossref/scraping.md +568 -568
  24. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/dev-to/scraping.md +323 -323
  25. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/duckduckgo/scraping.md +349 -349
  26. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/ebay/scraping.md +435 -435
  27. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/etsy/scraping.md +506 -506
  28. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/eventbrite/scraping.md +363 -363
  29. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/expedia/automation.md +168 -168
  30. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/facebook/groups.md +236 -236
  31. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/facebook/pages.md +295 -295
  32. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/framer/editor.md +108 -108
  33. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/fred/scraping.md +493 -493
  34. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/g2/scraping.md +580 -580
  35. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/genius/scraping.md +511 -511
  36. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/github/repo-actions.md +65 -65
  37. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/github/scraping.md +184 -184
  38. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/glassdoor/scraping.md +543 -543
  39. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/gmail/compose.md +122 -122
  40. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/goodreads/scraping.md +461 -461
  41. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/gutenberg/scraping.md +383 -383
  42. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/hackernews/scraping.md +243 -243
  43. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/howlongtobeat/scraping.md +473 -473
  44. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/imdb/scraping.md +271 -271
  45. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/itch-io/scraping.md +436 -436
  46. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/job-boards/indeed-glassdoor.md +1021 -1021
  47. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/letterboxd/scraping.md +349 -349
  48. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/linkedin/invitation-manager.md +109 -109
  49. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/loom/folder-enumeration.md +170 -170
  50. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/macrotrends/scraping.md +537 -537
  51. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/medium/article-hydration.md +120 -120
  52. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/medium/scraping.md +414 -414
  53. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/metacritic/scraping.md +477 -477
  54. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/musicbrainz/scraping.md +478 -478
  55. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/nasa/scraping.md +339 -339
  56. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/news-aggregation/multi-source.md +205 -205
  57. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/open-library/scraping.md +472 -472
  58. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/openalex/scraping.md +470 -470
  59. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/openstreetmap/scraping.md +490 -490
  60. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/package-registries/npm-pypi.md +478 -478
  61. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/polymarket/scraping.md +234 -234
  62. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/producthunt/scraping.md +307 -307
  63. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/pubmed/scraping.md +421 -421
  64. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/quora/scraping.md +364 -364
  65. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/rawg/scraping.md +352 -352
  66. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/reddit/scraping.md +124 -124
  67. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/rest-countries/scraping.md +233 -233
  68. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/sec-edgar/scraping.md +361 -361
  69. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/shopify-admin/README.md +36 -36
  70. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/shopify-admin/embedded-apps.md +72 -72
  71. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/shopify-admin/knowledge-base.md +109 -109
  72. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/shopify-admin/polaris-inputs.md +137 -137
  73. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/soundcloud/scraping.md +362 -362
  74. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/spotify/scraping.md +339 -339
  75. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/stackoverflow/scraping.md +435 -435
  76. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/steam/scraping.md +575 -575
  77. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/substack/scraping.md +338 -338
  78. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/thetechgeeks/pricing.md +52 -52
  79. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/tiktok/upload.md +107 -107
  80. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/tradingview/scraping.md +309 -309
  81. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/trello/boards-and-lists.md +88 -88
  82. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/trustpilot/scraping.md +375 -375
  83. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/walmart/scraping.md +444 -444
  84. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/wayback-machine/scraping.md +306 -306
  85. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/weather/scraping.md +398 -398
  86. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/wellfound/scraping.md +596 -596
  87. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/world-bank/scraping.md +356 -356
  88. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/xiaohongshu/scraping.md +84 -84
  89. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/youtube/scraping.md +418 -418
  90. package/dist/extensions/builtin/browser/agent-workspace/domain-skills/zillow/scraping.md +433 -433
  91. package/dist/extensions/builtin/browser/browser.md +73 -73
  92. package/dist/extensions/builtin/browser/install.md +142 -142
  93. package/dist/extensions/builtin/browser/interaction-skills/connection.md +48 -48
  94. package/dist/extensions/builtin/browser/interaction-skills/cookies.md +3 -3
  95. package/dist/extensions/builtin/browser/interaction-skills/cross-origin-iframes.md +3 -3
  96. package/dist/extensions/builtin/browser/interaction-skills/dialogs.md +64 -64
  97. package/dist/extensions/builtin/browser/interaction-skills/downloads.md +3 -3
  98. package/dist/extensions/builtin/browser/interaction-skills/drag-and-drop.md +3 -3
  99. package/dist/extensions/builtin/browser/interaction-skills/dropdowns.md +3 -3
  100. package/dist/extensions/builtin/browser/interaction-skills/iframes.md +3 -3
  101. package/dist/extensions/builtin/browser/interaction-skills/network-requests.md +3 -3
  102. package/dist/extensions/builtin/browser/interaction-skills/print-as-pdf.md +3 -3
  103. package/dist/extensions/builtin/browser/interaction-skills/profile-sync.md +90 -90
  104. package/dist/extensions/builtin/browser/interaction-skills/screenshots.md +17 -17
  105. package/dist/extensions/builtin/browser/interaction-skills/scrolling.md +3 -3
  106. package/dist/extensions/builtin/browser/interaction-skills/shadow-dom.md +3 -3
  107. package/dist/extensions/builtin/browser/interaction-skills/tabs.md +69 -69
  108. package/dist/extensions/builtin/browser/interaction-skills/uploads.md +1 -1
  109. package/dist/extensions/builtin/browser/interaction-skills/viewport.md +3 -3
  110. package/dist/extensions/builtin/browser/src/browser_harness/AGENT.md +15 -15
  111. package/dist/extensions/builtin/browser/src/browser_harness/__init__.py +8 -8
  112. package/dist/extensions/builtin/browser/src/browser_harness/_ipc.py +90 -90
  113. package/dist/extensions/builtin/browser/src/browser_harness/admin.py +722 -722
  114. package/dist/extensions/builtin/browser/src/browser_harness/daemon.py +328 -328
  115. package/dist/extensions/builtin/browser/src/browser_harness/helpers.py +396 -396
  116. package/dist/extensions/builtin/browser/src/browser_harness/run.py +103 -103
  117. package/dist/extensions/builtin/debug/index.js +9 -9
  118. package/dist/extensions/builtin/discipline/skills/brainstorming/SKILL.md +33 -33
  119. package/dist/extensions/builtin/discipline/skills/executing-plans/SKILL.md +25 -25
  120. package/dist/extensions/builtin/discipline/skills/finishing-development-branch/SKILL.md +25 -25
  121. package/dist/extensions/builtin/discipline/skills/receiving-code-review/SKILL.md +22 -22
  122. package/dist/extensions/builtin/discipline/skills/requesting-code-review/SKILL.md +31 -31
  123. package/dist/extensions/builtin/discipline/skills/systematic-debugging/SKILL.md +28 -28
  124. package/dist/extensions/builtin/discipline/skills/test-driven-development/SKILL.md +32 -32
  125. package/dist/extensions/builtin/discipline/skills/using-git-worktrees/SKILL.md +25 -25
  126. package/dist/extensions/builtin/discipline/skills/verification-before-completion/SKILL.md +27 -27
  127. package/dist/extensions/builtin/discipline/skills/writing-plans/SKILL.md +26 -26
  128. package/dist/extensions/builtin/goal/README.md +67 -67
  129. package/dist/extensions/builtin/goal/index.js +6 -6
  130. package/dist/extensions/builtin/grub/README.md +112 -112
  131. package/dist/extensions/builtin/link-world/agent-workspace/README.md +16 -16
  132. package/dist/extensions/builtin/link-world/internet-search/internet-search.md +65 -65
  133. package/dist/extensions/builtin/link-world/link-world-agent.md +82 -82
  134. package/dist/extensions/builtin/link-world/linkworld.md +313 -313
  135. package/dist/extensions/builtin/link-world/network-routing/network-routing.md +67 -67
  136. package/dist/extensions/builtin/loop/README.md +92 -92
  137. package/dist/extensions/builtin/mcp/figma-design.md +68 -68
  138. package/dist/extensions/builtin/mcp/mcp-management.md +85 -85
  139. package/dist/extensions/builtin/recap/AGENT.md +15 -15
  140. package/dist/extensions/builtin/sal/README.md +72 -72
  141. package/dist/extensions/builtin/security-audit/README.md +289 -289
  142. package/dist/extensions/builtin/team/AGENT.md +112 -112
  143. package/dist/extensions/builtin/team/TESTING.md +299 -299
  144. package/dist/extensions/builtin/token-save/README.md +56 -56
  145. package/dist/extensions/optional/AGENT.md +10 -10
  146. package/dist/modes/interactive/controllers/input-submit-controller.js +2 -2
  147. package/dist/modes/interactive/controllers/stream-render-controller.js +2 -2
  148. package/dist/modes/interactive/interactive-mode.js +19 -19
  149. package/dist/modes/interactive/theme/dark.json +85 -85
  150. package/dist/modes/interactive/theme/light.json +84 -84
  151. package/dist/modes/interactive/theme/theme-schema.json +335 -335
  152. package/dist/modes/interactive/theme/warm.json +81 -81
  153. package/dist/node_modules/@pencil-agent/ai/dist/cli.js +0 -0
  154. package/dist/node_modules/@pencil-agent/ai/dist/models.generated.js +1 -1
  155. package/docs/ACP/345/215/217/350/256/256/351/233/206/346/210/220/345/274/200/345/217/221/346/226/207/346/241/243.md +851 -0
  156. package/docs/SDK-TESTING.md +364 -0
  157. package/docs/codex-goal-command-impl.md +1055 -1055
  158. package/docs/codex-goal-vs-grub.md +500 -500
  159. package/docs/custom-provider.md +27 -27
  160. package/docs/extensions.md +27 -27
  161. package/docs/keybindings.md +27 -27
  162. package/docs/loop /351/207/215/346/236/204/345/256/214/346/210/220/346/200/273/347/273/223.md" +250 -250
  163. package/docs/loop /351/207/215/346/236/204/345/256/214/346/210/220/346/212/245/345/221/212.md" +122 -122
  164. package/docs/loop /351/207/215/346/236/204/346/226/271/346/241/210.md" +1222 -1222
  165. package/docs/loop /351/207/215/346/236/204/346/226/271/346/241/210/345/256/236/347/216/260/346/212/245/345/221/212.md" +158 -158
  166. package/docs/loop /351/207/215/346/236/204/346/226/271/346/241/210/345/257/271/346/257/224/345/210/206/346/236/220.md" +128 -128
  167. package/docs/loop /351/207/215/346/236/204/350/256/241/345/210/222.md" +320 -320
  168. package/docs/loop-usage-examples.md +214 -214
  169. package/docs/mem-core/346/212/200/346/234/257/346/226/207/346/241/243.md +593 -0
  170. package/docs/models.md +27 -27
  171. package/docs/packages.md +27 -27
  172. package/docs/pi-design-philosophy.md +457 -457
  173. package/docs/planmode.md +1987 -1987
  174. package/docs/prompt-templates.md +27 -27
  175. package/docs/providers.md +27 -27
  176. package/docs/sdk.md +27 -27
  177. package/docs/skills.md +27 -27
  178. package/docs/startup-performance-optimization.md +301 -0
  179. package/docs/themes.md +27 -27
  180. package/docs/tui.md +27 -27
  181. package/docs//350/256/244/347/237/245/345/234/260/345/233/276.md +47 -0
  182. package/package.json +190 -190
  183. package/docs/cc-agent-design.md +0 -1297
  184. package/docs/cc-tui-design.md +0 -1333
  185. package/docs/nanoPencil-/345/255/246/344/271/240/350/256/241/345/210/222.md +0 -170
  186. package/docs/scan-report.md +0 -3820
  187. package/docs//345/257/271/346/240/207Claude-Code.md +0 -1775
  188. package/docs//351/230/277/351/207/214/345/267/264/345/267/264/350/264/242/346/212/245/345/210/206/346/236/220/344/271/246.md +0 -261
@@ -1,36 +1,36 @@
1
- # shopify-admin
2
-
3
- Browser-harness patterns for `admin.shopify.com` and embedded Shopify apps.
4
-
5
- ## Files in this folder
6
-
7
- - `embedded-apps.md` — every Shopify app runs in an iframe; how to target it
8
- - `polaris-inputs.md` — Polaris React inputs reject synthetic value setters; use CDP type_text
9
- - `knowledge-base.md` — automating the Shopify Knowledge Base App for FAQ entries
10
-
11
- ## When to use these
12
-
13
- You're driving Shopify admin and need to add / edit / configure something. The Shopify admin UI is large and many surfaces are embedded apps — first check whether what you need is in an embedded app (most apps under `admin.shopify.com/store/<store>/apps/<app-slug>/...` are).
14
-
15
- ## When to skip
16
-
17
- - If the operation is read-only product / inventory data → use the **Storefront API** (HTTP) instead, much faster
18
- - If the store has a custom admin app with API token provisioned → use the **Admin API** (GraphQL or REST) instead, no UI scraping
19
- - If you're editing theme code → use the **Shopify CLI** (`shopify theme push`) — don't touch the theme editor UI
20
-
21
- The browser is the right tool only when:
22
- - The setting / app exposes no API
23
- - The change is one-time or rare enough not to justify scripting
24
- - You're discovering / exploring the admin (e.g., finding selectors for a future automation)
25
-
26
- ## Authentication
27
-
28
- Mike (or the human owner) must be logged into `admin.shopify.com` in the Chrome session that browser-harness attaches to. The harness does NOT log in — it inherits the human's session.
29
-
30
- If you hit `accounts.shopify.com` redirect, stop and ask the human to log in. Don't type credentials.
31
-
32
- ## Polaris is in transition (Jan 2026 onward)
33
-
34
- Shopify is migrating its design system from React-based Polaris to Web-Components-based Polaris. Most legacy admin surfaces are still React. Newer surfaces (Catalog Mapping, parts of Settings) may be web components.
35
-
36
- Screenshot first. If you see `<s-text-field>` or `<s-button>` web component tags → use the web component pattern. If you see `[class*="Polaris-"]` React class names → use the CDP keystrokes pattern in `polaris-inputs.md`.
1
+ # shopify-admin
2
+
3
+ Browser-harness patterns for `admin.shopify.com` and embedded Shopify apps.
4
+
5
+ ## Files in this folder
6
+
7
+ - `embedded-apps.md` — every Shopify app runs in an iframe; how to target it
8
+ - `polaris-inputs.md` — Polaris React inputs reject synthetic value setters; use CDP type_text
9
+ - `knowledge-base.md` — automating the Shopify Knowledge Base App for FAQ entries
10
+
11
+ ## When to use these
12
+
13
+ You're driving Shopify admin and need to add / edit / configure something. The Shopify admin UI is large and many surfaces are embedded apps — first check whether what you need is in an embedded app (most apps under `admin.shopify.com/store/<store>/apps/<app-slug>/...` are).
14
+
15
+ ## When to skip
16
+
17
+ - If the operation is read-only product / inventory data → use the **Storefront API** (HTTP) instead, much faster
18
+ - If the store has a custom admin app with API token provisioned → use the **Admin API** (GraphQL or REST) instead, no UI scraping
19
+ - If you're editing theme code → use the **Shopify CLI** (`shopify theme push`) — don't touch the theme editor UI
20
+
21
+ The browser is the right tool only when:
22
+ - The setting / app exposes no API
23
+ - The change is one-time or rare enough not to justify scripting
24
+ - You're discovering / exploring the admin (e.g., finding selectors for a future automation)
25
+
26
+ ## Authentication
27
+
28
+ Mike (or the human owner) must be logged into `admin.shopify.com` in the Chrome session that browser-harness attaches to. The harness does NOT log in — it inherits the human's session.
29
+
30
+ If you hit `accounts.shopify.com` redirect, stop and ask the human to log in. Don't type credentials.
31
+
32
+ ## Polaris is in transition (Jan 2026 onward)
33
+
34
+ Shopify is migrating its design system from React-based Polaris to Web-Components-based Polaris. Most legacy admin surfaces are still React. Newer surfaces (Catalog Mapping, parts of Settings) may be web components.
35
+
36
+ Screenshot first. If you see `<s-text-field>` or `<s-button>` web component tags → use the web component pattern. If you see `[class*="Polaris-"]` React class names → use the CDP keystrokes pattern in `polaris-inputs.md`.
@@ -1,72 +1,72 @@
1
- # Shopify embedded apps run in iframes
2
-
3
- Every Shopify app surfaced in the admin (first-party like Knowledge Base, third-party like Okendo) renders inside a sandboxed iframe. Your top-level `document` queries find the Shopify chrome (sidebar, header, search bar) but **none of the app's UI**.
4
-
5
- ## How to target the iframe
6
-
7
- ```python
8
- from helpers import iframe_target, js, type_text
9
-
10
- # 1. Find the iframe by URL substring
11
- tid = iframe_target("qa-pairs-app") # Knowledge Base App
12
-
13
- # 2. Run JS inside the iframe by passing target_id
14
- result = js("""
15
- (() => {
16
- const button = Array.from(document.querySelectorAll('button')).find(b => b.textContent.trim() === 'Add FAQ');
17
- if (button) { button.click(); return {clicked: true}; }
18
- return {clicked: false};
19
- })()
20
- """, target_id=tid)
21
- ```
22
-
23
- ## Finding the URL substring
24
-
25
- The iframe's URL contains the app slug. Run:
26
-
27
- ```python
28
- import json
29
- for t in cdp("Target.getTargets")["targetInfos"]:
30
- if t["type"] == "iframe" and "shopify" in t.get("url", "").lower():
31
- print(t["url"])
32
- ```
33
-
34
- Then pick a substring unique to your target app.
35
-
36
- ## Known Shopify app iframe slugs
37
-
38
- | App | iframe URL substring |
39
- |---|---|
40
- | Shopify Knowledge Base (qa-pairs-app) | `qa-pairs-app` |
41
- | Shopify Online Store editor | `online-store-web.shopifyapps.com` |
42
- | Shopify Hydrogen Storefront | `hydrogen-storefronts` (or similar — verify) |
43
-
44
- Add to this table when you discover new ones.
45
-
46
- ## Why iframes
47
-
48
- Shopify uses App Bridge to embed third-party apps with isolation. Your top-level page CAN'T directly access app DOM for security reasons — you need iframe targeting (which the harness does via CDP `Target.attachToTarget`).
49
-
50
- ## Coordinate clicks vs JS clicks
51
-
52
- Coordinate clicks (`click(x, y)`) pass through iframes at the compositor level — they work. But JS clicks scoped to the iframe target are more reliable for routine button taps because:
53
-
54
- - Element text content is stable across UI redesigns
55
- - DPR scaling on retina is automatic
56
- - React event handlers are guaranteed to fire (vs. CDP mouse events which sometimes hit a transparent layer above the button)
57
-
58
- ## Gotcha — multiple iframes from same app
59
-
60
- The Online Store editor renders the storefront preview AND the editor toolbar in two separate iframes. Pick the right one by URL substring; don't assume the first match is correct.
61
-
62
- ```python
63
- # WRONG — picks first match
64
- tid = iframe_target("online-store-web")
65
-
66
- # RIGHT — disambiguate
67
- for t in cdp("Target.getTargets")["targetInfos"]:
68
- url = t.get("url", "")
69
- if "online-store-web" in url and "editor" in url:
70
- tid = t["targetId"]
71
- break
72
- ```
1
+ # Shopify embedded apps run in iframes
2
+
3
+ Every Shopify app surfaced in the admin (first-party like Knowledge Base, third-party like Okendo) renders inside a sandboxed iframe. Your top-level `document` queries find the Shopify chrome (sidebar, header, search bar) but **none of the app's UI**.
4
+
5
+ ## How to target the iframe
6
+
7
+ ```python
8
+ from helpers import iframe_target, js, type_text
9
+
10
+ # 1. Find the iframe by URL substring
11
+ tid = iframe_target("qa-pairs-app") # Knowledge Base App
12
+
13
+ # 2. Run JS inside the iframe by passing target_id
14
+ result = js("""
15
+ (() => {
16
+ const button = Array.from(document.querySelectorAll('button')).find(b => b.textContent.trim() === 'Add FAQ');
17
+ if (button) { button.click(); return {clicked: true}; }
18
+ return {clicked: false};
19
+ })()
20
+ """, target_id=tid)
21
+ ```
22
+
23
+ ## Finding the URL substring
24
+
25
+ The iframe's URL contains the app slug. Run:
26
+
27
+ ```python
28
+ import json
29
+ for t in cdp("Target.getTargets")["targetInfos"]:
30
+ if t["type"] == "iframe" and "shopify" in t.get("url", "").lower():
31
+ print(t["url"])
32
+ ```
33
+
34
+ Then pick a substring unique to your target app.
35
+
36
+ ## Known Shopify app iframe slugs
37
+
38
+ | App | iframe URL substring |
39
+ |---|---|
40
+ | Shopify Knowledge Base (qa-pairs-app) | `qa-pairs-app` |
41
+ | Shopify Online Store editor | `online-store-web.shopifyapps.com` |
42
+ | Shopify Hydrogen Storefront | `hydrogen-storefronts` (or similar — verify) |
43
+
44
+ Add to this table when you discover new ones.
45
+
46
+ ## Why iframes
47
+
48
+ Shopify uses App Bridge to embed third-party apps with isolation. Your top-level page CAN'T directly access app DOM for security reasons — you need iframe targeting (which the harness does via CDP `Target.attachToTarget`).
49
+
50
+ ## Coordinate clicks vs JS clicks
51
+
52
+ Coordinate clicks (`click(x, y)`) pass through iframes at the compositor level — they work. But JS clicks scoped to the iframe target are more reliable for routine button taps because:
53
+
54
+ - Element text content is stable across UI redesigns
55
+ - DPR scaling on retina is automatic
56
+ - React event handlers are guaranteed to fire (vs. CDP mouse events which sometimes hit a transparent layer above the button)
57
+
58
+ ## Gotcha — multiple iframes from same app
59
+
60
+ The Online Store editor renders the storefront preview AND the editor toolbar in two separate iframes. Pick the right one by URL substring; don't assume the first match is correct.
61
+
62
+ ```python
63
+ # WRONG — picks first match
64
+ tid = iframe_target("online-store-web")
65
+
66
+ # RIGHT — disambiguate
67
+ for t in cdp("Target.getTargets")["targetInfos"]:
68
+ url = t.get("url", "")
69
+ if "online-store-web" in url and "editor" in url:
70
+ tid = t["targetId"]
71
+ break
72
+ ```
@@ -1,109 +1,109 @@
1
- # Shopify Knowledge Base App — automating FAQ entries
2
-
3
- The Knowledge Base App (Shopify Winter '26 Edition) lets merchants control how AI agents (ChatGPT, Perplexity, Claude, Copilot, Gemini) answer questions about their brand. Each entry is a Question / Answer pair. The app currently has no public API and is English-only as of Winter '26 — browser automation is the canonical path.
4
-
5
- ## URL pattern
6
-
7
- ```
8
- https://admin.shopify.com/store/<store-handle>/apps/shopify-knowledge-base/app
9
- ```
10
-
11
- Sub-routes:
12
- - `/app` — overview (FAQ list, top unanswered questions, query log)
13
- - `/app/new` — Add FAQ form
14
- - `/app/pairs/<id>` — entry detail / edit
15
-
16
- ## Iframe slug
17
-
18
- The app runs at iframe URL containing `qa-pairs-app`:
19
-
20
- ```python
21
- tid = iframe_target("qa-pairs-app")
22
- ```
23
-
24
- ## Adding a single FAQ
25
-
26
- See `polaris-inputs.md` for the full canonical pattern. Quick version:
27
-
28
- ```python
29
- def add_faq(question, answer):
30
- tid = iframe_target("qa-pairs-app")
31
- # focus question input via JS, type via CDP, focus answer, type, click Save
32
- # poll URL for /pairs/<id> success signal
33
- ```
34
-
35
- ## Batching multiple FAQs
36
-
37
- After saving an entry, the success page shows "FAQ created. Add another FAQ" link. Click it via JS to skip navigating back to overview:
38
-
39
- ```python
40
- def click_add_another():
41
- tid = iframe_target("qa-pairs-app")
42
- js("""
43
- (() => {
44
- const link = Array.from(document.querySelectorAll('a, button'))
45
- .find(x => x.textContent.trim() === 'Add another FAQ');
46
- if (link) link.click();
47
- })()
48
- """, target_id=tid)
49
- ```
50
-
51
- Loop:
52
-
53
- ```python
54
- ENTRIES = [(q1, a1), (q2, a2), ...]
55
- for q, a in ENTRIES:
56
- click_add_another()
57
- time.sleep(1.5) # wait for form to render
58
- ok, info = add_faq(q, a)
59
- print(f"{q[:40]} -> {ok} ({info})")
60
- if not ok: break
61
- ```
62
-
63
- ## Brand voice — what to put in answers
64
-
65
- This is application-specific (depends on the merchant). For JING the rule was Aesop founder-letter tone — sentence case, no exclamation points, "JING" not "we", specific over generic.
66
-
67
- The Shopify guidance "Provide a brief answer in 1 or 2 sentences" is a soft hint. The textarea accepts longer text and AI agents prefer specific multi-sentence answers. Aim for 2-4 short sentences with concrete details.
68
-
69
- ## What to put in the Knowledge Base
70
-
71
- Categories that materially shape AI agent answers about your brand:
72
-
73
- 1. **Brand voice / DNA** — "What is your brand?" / "What's your tone?"
74
- 2. **Specs** — exact materials, dimensions, weights, sizes (NOT marketing prose)
75
- 3. **Comparisons** — "How does X compare to <competitor>?" with concrete differences
76
- 4. **Policies** — returns, shipping, care, warranty, contact (in brand voice)
77
- 5. **Origin** — founder, where made, why brand exists
78
- 6. **Limitations** — what you DON'T do (V1 scope, US-only, etc.) — agents that hallucinate availability hurt conversion
79
-
80
- Skip: anything marketing-speak. The Knowledge Base is for **truth, in voice**, not pitch copy.
81
-
82
- ## Top unanswered questions
83
-
84
- The overview shows up to 7 "Top unanswered questions" Shopify auto-detected from query logs. **Answer these first** — they're real shopper queries hitting your store right now. Once answered, the section empties.
85
-
86
- ## Query log
87
-
88
- `/admin/apps/shopify-knowledge-base/app/queries` (or "Query log" in app sidebar) shows what shoppers actually asked AI agents about your brand. Read weekly. New patterns become new FAQ entries.
89
-
90
- ## Verifying entries surface in AI
91
-
92
- After adding an entry, allow 24 hours for AI provider indexing, then test:
93
-
94
- - ChatGPT: "Tell me about <your brand>'s return policy" → check if your exact wording surfaces
95
- - Perplexity: same
96
- - Claude: "Compare <your brand> vs <competitor>" → see if your comparison framing appears
97
-
98
- If the answer doesn't surface, the entry might be too long, too vague, or contradicted by another source (your homepage, an outdated blog post). Tighten the answer.
99
-
100
- ## Limits
101
-
102
- As of Winter '26 Edition:
103
- - English-only
104
- - No bulk import / CSV upload
105
- - No API for read or write
106
- - Each entry maximum ~500 words (soft cap; UI shows guidance "1 or 2 sentences")
107
- - No version history visible to the merchant
108
-
109
- Watch Shopify changelogs for API exposure — likely in Spring '26 or Summer '26 Edition. When it ships, switch to API-driven population.
1
+ # Shopify Knowledge Base App — automating FAQ entries
2
+
3
+ The Knowledge Base App (Shopify Winter '26 Edition) lets merchants control how AI agents (ChatGPT, Perplexity, Claude, Copilot, Gemini) answer questions about their brand. Each entry is a Question / Answer pair. The app currently has no public API and is English-only as of Winter '26 — browser automation is the canonical path.
4
+
5
+ ## URL pattern
6
+
7
+ ```
8
+ https://admin.shopify.com/store/<store-handle>/apps/shopify-knowledge-base/app
9
+ ```
10
+
11
+ Sub-routes:
12
+ - `/app` — overview (FAQ list, top unanswered questions, query log)
13
+ - `/app/new` — Add FAQ form
14
+ - `/app/pairs/<id>` — entry detail / edit
15
+
16
+ ## Iframe slug
17
+
18
+ The app runs at iframe URL containing `qa-pairs-app`:
19
+
20
+ ```python
21
+ tid = iframe_target("qa-pairs-app")
22
+ ```
23
+
24
+ ## Adding a single FAQ
25
+
26
+ See `polaris-inputs.md` for the full canonical pattern. Quick version:
27
+
28
+ ```python
29
+ def add_faq(question, answer):
30
+ tid = iframe_target("qa-pairs-app")
31
+ # focus question input via JS, type via CDP, focus answer, type, click Save
32
+ # poll URL for /pairs/<id> success signal
33
+ ```
34
+
35
+ ## Batching multiple FAQs
36
+
37
+ After saving an entry, the success page shows "FAQ created. Add another FAQ" link. Click it via JS to skip navigating back to overview:
38
+
39
+ ```python
40
+ def click_add_another():
41
+ tid = iframe_target("qa-pairs-app")
42
+ js("""
43
+ (() => {
44
+ const link = Array.from(document.querySelectorAll('a, button'))
45
+ .find(x => x.textContent.trim() === 'Add another FAQ');
46
+ if (link) link.click();
47
+ })()
48
+ """, target_id=tid)
49
+ ```
50
+
51
+ Loop:
52
+
53
+ ```python
54
+ ENTRIES = [(q1, a1), (q2, a2), ...]
55
+ for q, a in ENTRIES:
56
+ click_add_another()
57
+ time.sleep(1.5) # wait for form to render
58
+ ok, info = add_faq(q, a)
59
+ print(f"{q[:40]} -> {ok} ({info})")
60
+ if not ok: break
61
+ ```
62
+
63
+ ## Brand voice — what to put in answers
64
+
65
+ This is application-specific (depends on the merchant). For JING the rule was Aesop founder-letter tone — sentence case, no exclamation points, "JING" not "we", specific over generic.
66
+
67
+ The Shopify guidance "Provide a brief answer in 1 or 2 sentences" is a soft hint. The textarea accepts longer text and AI agents prefer specific multi-sentence answers. Aim for 2-4 short sentences with concrete details.
68
+
69
+ ## What to put in the Knowledge Base
70
+
71
+ Categories that materially shape AI agent answers about your brand:
72
+
73
+ 1. **Brand voice / DNA** — "What is your brand?" / "What's your tone?"
74
+ 2. **Specs** — exact materials, dimensions, weights, sizes (NOT marketing prose)
75
+ 3. **Comparisons** — "How does X compare to <competitor>?" with concrete differences
76
+ 4. **Policies** — returns, shipping, care, warranty, contact (in brand voice)
77
+ 5. **Origin** — founder, where made, why brand exists
78
+ 6. **Limitations** — what you DON'T do (V1 scope, US-only, etc.) — agents that hallucinate availability hurt conversion
79
+
80
+ Skip: anything marketing-speak. The Knowledge Base is for **truth, in voice**, not pitch copy.
81
+
82
+ ## Top unanswered questions
83
+
84
+ The overview shows up to 7 "Top unanswered questions" Shopify auto-detected from query logs. **Answer these first** — they're real shopper queries hitting your store right now. Once answered, the section empties.
85
+
86
+ ## Query log
87
+
88
+ `/admin/apps/shopify-knowledge-base/app/queries` (or "Query log" in app sidebar) shows what shoppers actually asked AI agents about your brand. Read weekly. New patterns become new FAQ entries.
89
+
90
+ ## Verifying entries surface in AI
91
+
92
+ After adding an entry, allow 24 hours for AI provider indexing, then test:
93
+
94
+ - ChatGPT: "Tell me about <your brand>'s return policy" → check if your exact wording surfaces
95
+ - Perplexity: same
96
+ - Claude: "Compare <your brand> vs <competitor>" → see if your comparison framing appears
97
+
98
+ If the answer doesn't surface, the entry might be too long, too vague, or contradicted by another source (your homepage, an outdated blog post). Tighten the answer.
99
+
100
+ ## Limits
101
+
102
+ As of Winter '26 Edition:
103
+ - English-only
104
+ - No bulk import / CSV upload
105
+ - No API for read or write
106
+ - Each entry maximum ~500 words (soft cap; UI shows guidance "1 or 2 sentences")
107
+ - No version history visible to the merchant
108
+
109
+ Watch Shopify changelogs for API exposure — likely in Spring '26 or Summer '26 Edition. When it ships, switch to API-driven population.