web-agent-bridge 3.0.0 → 3.3.0
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.
- package/LICENSE +72 -21
- package/README.ar.md +1286 -1073
- package/README.md +1764 -1535
- package/bin/agent-runner.js +474 -474
- package/bin/cli.js +237 -138
- package/bin/wab.js +80 -80
- package/examples/bidi-agent.js +119 -119
- package/examples/cross-site-agent.js +91 -91
- package/examples/mcp-agent.js +94 -94
- package/examples/next-app-router/README.md +44 -44
- package/examples/puppeteer-agent.js +108 -108
- package/examples/saas-dashboard/README.md +55 -55
- package/examples/shopify-hydrogen/README.md +74 -74
- package/examples/vision-agent.js +171 -171
- package/examples/wordpress-elementor/README.md +77 -77
- package/package.json +17 -3
- package/public/.well-known/agent-tools.json +180 -180
- package/public/.well-known/ai-assets.json +59 -59
- package/public/.well-known/ai-plugin.json +28 -0
- package/public/.well-known/security.txt +8 -0
- package/public/agent-workspace.html +349 -347
- package/public/ai.html +198 -196
- package/public/api.html +413 -0
- package/public/browser.html +486 -484
- package/public/commander-dashboard.html +243 -243
- package/public/cookies.html +210 -208
- package/public/css/agent-workspace.css +1713 -1713
- package/public/css/premium.css +317 -317
- package/public/css/styles.css +1235 -1235
- package/public/dashboard.html +706 -704
- package/public/demo.html +1770 -1
- package/public/dns.html +507 -0
- package/public/docs.html +587 -585
- package/public/feed.xml +89 -89
- package/public/growth.html +463 -0
- package/public/index.html +341 -9
- package/public/integrations.html +556 -0
- package/public/js/agent-workspace.js +1740 -1740
- package/public/js/auth-nav.js +31 -31
- package/public/js/auth-redirect.js +12 -12
- package/public/js/cookie-consent.js +56 -56
- package/public/js/wab-demo-page.js +721 -721
- package/public/js/ws-client.js +74 -74
- package/public/llms-full.txt +360 -309
- package/public/llms.txt +125 -86
- package/public/login.html +85 -83
- package/public/mesh-dashboard.html +328 -328
- package/public/openapi.json +580 -580
- package/public/phone-shield.html +281 -0
- package/public/premium-dashboard.html +2489 -2487
- package/public/premium.html +793 -791
- package/public/privacy.html +297 -295
- package/public/register.html +105 -103
- package/public/robots.txt +87 -87
- package/public/script/wab-consent.d.ts +36 -36
- package/public/script/wab-consent.js +104 -104
- package/public/script/wab-schema.js +131 -131
- package/public/script/wab.d.ts +108 -108
- package/public/script/wab.min.js +580 -580
- package/public/security.txt +8 -0
- package/public/terms.html +256 -254
- package/script/ai-agent-bridge.js +1754 -1754
- package/sdk/README.md +99 -99
- package/sdk/agent-mesh.js +449 -449
- package/sdk/commander.js +262 -262
- package/sdk/index.d.ts +464 -464
- package/sdk/index.js +18 -1
- package/sdk/multi-agent.js +318 -318
- package/sdk/package.json +12 -1
- package/sdk/safety-shield.js +219 -0
- package/sdk/schema-discovery.js +83 -83
- package/server/adapters/index.js +520 -520
- package/server/config/plans.js +367 -367
- package/server/config/secrets.js +102 -102
- package/server/control-plane/index.js +301 -301
- package/server/data-plane/index.js +354 -354
- package/server/index.js +175 -19
- package/server/llm/index.js +404 -404
- package/server/middleware/adminAuth.js +35 -35
- package/server/middleware/auth.js +50 -50
- package/server/middleware/featureGate.js +88 -88
- package/server/middleware/rateLimits.js +100 -100
- package/server/middleware/sensitiveAction.js +157 -0
- package/server/migrations/001_add_analytics_indexes.sql +7 -7
- package/server/migrations/002_premium_features.sql +418 -418
- package/server/migrations/003_ads_integer_cents.sql +33 -33
- package/server/migrations/004_agent_os.sql +158 -158
- package/server/migrations/005_marketplace_metering.sql +126 -126
- package/server/models/adapters/index.js +33 -33
- package/server/models/adapters/mysql.js +183 -183
- package/server/models/adapters/postgresql.js +172 -172
- package/server/models/adapters/sqlite.js +7 -7
- package/server/models/db.js +681 -681
- package/server/observability/failure-analysis.js +337 -337
- package/server/observability/index.js +394 -394
- package/server/protocol/capabilities.js +223 -223
- package/server/protocol/index.js +243 -243
- package/server/protocol/schema.js +584 -584
- package/server/registry/certification.js +271 -271
- package/server/registry/index.js +326 -326
- package/server/routes/admin-premium.js +671 -671
- package/server/routes/admin.js +261 -261
- package/server/routes/ads.js +130 -130
- package/server/routes/agent-workspace.js +540 -378
- package/server/routes/api.js +150 -150
- package/server/routes/auth.js +71 -71
- package/server/routes/billing.js +45 -45
- package/server/routes/commander.js +316 -316
- package/server/routes/demo-showcase.js +332 -0
- package/server/routes/demo-store.js +154 -0
- package/server/routes/discovery.js +417 -406
- package/server/routes/gateway.js +173 -0
- package/server/routes/license.js +251 -240
- package/server/routes/mesh.js +469 -469
- package/server/routes/noscript.js +543 -543
- package/server/routes/premium-v2.js +686 -686
- package/server/routes/premium.js +724 -724
- package/server/routes/runtime.js +2148 -2147
- package/server/routes/sovereign.js +465 -385
- package/server/routes/universal.js +200 -177
- package/server/routes/wab-api.js +850 -491
- package/server/runtime/container-worker.js +111 -111
- package/server/runtime/container.js +448 -448
- package/server/runtime/distributed-worker.js +362 -362
- package/server/runtime/event-bus.js +210 -210
- package/server/runtime/index.js +253 -253
- package/server/runtime/queue.js +599 -599
- package/server/runtime/replay.js +666 -666
- package/server/runtime/sandbox.js +266 -266
- package/server/runtime/scheduler.js +534 -534
- package/server/runtime/session-engine.js +293 -293
- package/server/runtime/state-manager.js +188 -188
- package/server/security/cross-site-redactor.js +196 -0
- package/server/security/dry-run.js +180 -0
- package/server/security/human-gate-rate-limit.js +147 -0
- package/server/security/human-gate-transports.js +178 -0
- package/server/security/human-gate.js +281 -0
- package/server/security/index.js +368 -368
- package/server/security/intent-engine.js +245 -0
- package/server/security/reward-guard.js +171 -0
- package/server/security/rollback-store.js +239 -0
- package/server/security/token-scope.js +404 -0
- package/server/security/url-policy.js +139 -0
- package/server/services/agent-chat.js +506 -506
- package/server/services/agent-learning.js +601 -575
- package/server/services/agent-memory.js +625 -625
- package/server/services/agent-mesh.js +555 -539
- package/server/services/agent-symphony.js +717 -717
- package/server/services/agent-tasks.js +1807 -1807
- package/server/services/api-key-engine.js +292 -0
- package/server/services/cluster.js +894 -894
- package/server/services/commander.js +738 -738
- package/server/services/edge-compute.js +440 -440
- package/server/services/email.js +204 -204
- package/server/services/hosted-runtime.js +205 -205
- package/server/services/lfd.js +635 -616
- package/server/services/local-ai.js +389 -389
- package/server/services/marketplace.js +270 -270
- package/server/services/metering.js +182 -182
- package/server/services/modules/affiliate-intelligence.js +93 -0
- package/server/services/modules/agent-firewall.js +90 -0
- package/server/services/modules/bounty.js +89 -0
- package/server/services/modules/collective-bargaining.js +92 -0
- package/server/services/modules/dark-pattern.js +66 -0
- package/server/services/modules/gov-intelligence.js +45 -0
- package/server/services/modules/neural.js +55 -0
- package/server/services/modules/notary.js +49 -0
- package/server/services/modules/price-time-machine.js +86 -0
- package/server/services/modules/protocol.js +104 -0
- package/server/services/negotiation.js +439 -439
- package/server/services/plugins.js +771 -771
- package/server/services/premium.js +1 -1
- package/server/services/price-intelligence.js +566 -565
- package/server/services/price-shield.js +1137 -1137
- package/server/services/reputation.js +465 -465
- package/server/services/search-engine.js +357 -357
- package/server/services/security.js +513 -513
- package/server/services/self-healing.js +843 -843
- package/server/services/sovereign-shield.js +542 -0
- package/server/services/stripe.js +192 -192
- package/server/services/swarm.js +788 -788
- package/server/services/universal-scraper.js +662 -661
- package/server/services/verification.js +481 -481
- package/server/services/vision.js +1163 -1163
- package/server/utils/cache.js +125 -125
- package/server/utils/migrate.js +81 -81
- package/server/utils/safe-fetch.js +228 -0
- package/server/utils/secureFields.js +50 -50
- package/server/ws.js +161 -161
- package/templates/artisan-marketplace.yaml +104 -104
- package/templates/book-price-scout.yaml +98 -98
- package/templates/electronics-price-tracker.yaml +108 -108
- package/templates/flight-deal-hunter.yaml +113 -113
- package/templates/freelancer-direct.yaml +116 -116
- package/templates/grocery-price-compare.yaml +93 -93
- package/templates/hotel-direct-booking.yaml +113 -113
- package/templates/local-services.yaml +98 -98
- package/templates/olive-oil-tunisia.yaml +88 -88
- package/templates/organic-farm-fresh.yaml +101 -101
- package/templates/restaurant-direct.yaml +97 -97
- package/server/services/fairness-engine.js +0 -409
- package/server/services/fairness.js +0 -420
|
@@ -1,104 +1,104 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* WAB Consent — optional GDPR/CCPA-style prompt before agents run actions.
|
|
3
|
-
* Load after wab.min.js: <script src="/script/wab-consent.js"></script>
|
|
4
|
-
*
|
|
5
|
-
* WABConsent.showBanner({
|
|
6
|
-
* policyUrl: '/privacy',
|
|
7
|
-
* onAccept: () => WAB.init({ ... }),
|
|
8
|
-
* onDecline: () => { ... }
|
|
9
|
-
* });
|
|
10
|
-
*
|
|
11
|
-
* Before execute, check WABConsent.hasConsent() or gate your own flows.
|
|
12
|
-
*/
|
|
13
|
-
(function (global) {
|
|
14
|
-
'use strict';
|
|
15
|
-
|
|
16
|
-
var STORAGE_KEY = 'wab_agent_consent_v1';
|
|
17
|
-
|
|
18
|
-
function hasConsent() {
|
|
19
|
-
try {
|
|
20
|
-
return global.localStorage.getItem(STORAGE_KEY) === 'granted';
|
|
21
|
-
} catch (e) {
|
|
22
|
-
return false;
|
|
23
|
-
}
|
|
24
|
-
}
|
|
25
|
-
|
|
26
|
-
function setConsent(value) {
|
|
27
|
-
try {
|
|
28
|
-
if (value === null) global.localStorage.removeItem(STORAGE_KEY);
|
|
29
|
-
else global.localStorage.setItem(STORAGE_KEY, value);
|
|
30
|
-
} catch (e) {}
|
|
31
|
-
}
|
|
32
|
-
|
|
33
|
-
function injectStyles() {
|
|
34
|
-
if (global.document.getElementById('wab-consent-styles')) return;
|
|
35
|
-
var s = global.document.createElement('style');
|
|
36
|
-
s.id = 'wab-consent-styles';
|
|
37
|
-
s.textContent =
|
|
38
|
-
'#wab-consent-bar{position:fixed;left:16px;right:16px;bottom:16px;max-width:520px;margin:0 auto;' +
|
|
39
|
-
'background:#0f172a;color:#e2e8f0;padding:16px 18px;border-radius:12px;border:1px solid #334155;' +
|
|
40
|
-
'box-shadow:0 12px 40px rgba(0,0,0,0.35);z-index:99999;font-family:system-ui,sans-serif;font-size:14px;line-height:1.45}' +
|
|
41
|
-
'#wab-consent-bar p{margin:0 0 12px}' +
|
|
42
|
-
'#wab-consent-bar .wab-c-actions{display:flex;flex-wrap:wrap;gap:10px;align-items:center}' +
|
|
43
|
-
'#wab-consent-bar button{padding:8px 16px;border-radius:8px;border:none;font-weight:600;cursor:pointer;font-size:13px}' +
|
|
44
|
-
'#wab-consent-bar .wab-c-accept{background:#3b82f6;color:#fff}' +
|
|
45
|
-
'#wab-consent-bar .wab-c-decline{background:#1e293b;color:#94a3b8;border:1px solid #334155}' +
|
|
46
|
-
'#wab-consent-bar a{color:#93c5fd}';
|
|
47
|
-
global.document.head.appendChild(s);
|
|
48
|
-
}
|
|
49
|
-
|
|
50
|
-
function showBanner(options) {
|
|
51
|
-
options = options || {};
|
|
52
|
-
if (hasConsent() && options.skipIfGranted !== false) {
|
|
53
|
-
if (typeof options.onAccept === 'function') options.onAccept();
|
|
54
|
-
return;
|
|
55
|
-
}
|
|
56
|
-
injectStyles();
|
|
57
|
-
var existing = global.document.getElementById('wab-consent-bar');
|
|
58
|
-
if (existing) existing.remove();
|
|
59
|
-
|
|
60
|
-
var bar = global.document.createElement('div');
|
|
61
|
-
bar.id = 'wab-consent-bar';
|
|
62
|
-
bar.setAttribute('role', 'dialog');
|
|
63
|
-
bar.setAttribute('aria-label', 'AI agent consent');
|
|
64
|
-
|
|
65
|
-
var policy = options.policyUrl
|
|
66
|
-
? '<a href="' + options.policyUrl + '" target="_blank" rel="noopener">Privacy policy</a>'
|
|
67
|
-
: '';
|
|
68
|
-
|
|
69
|
-
bar.innerHTML =
|
|
70
|
-
'<p><strong>AI assistance on this site</strong><br>' +
|
|
71
|
-
(options.message ||
|
|
72
|
-
'This site can expose actions to AI agents (forms, cart, navigation). Allow this site to register agent-ready actions in your browser?') +
|
|
73
|
-
(policy ? ' ' + policy + '.' : '') +
|
|
74
|
-
'</p>' +
|
|
75
|
-
'<div class="wab-c-actions">' +
|
|
76
|
-
'<button type="button" class="wab-c-accept" id="wab-c-accept">Allow</button>' +
|
|
77
|
-
'<button type="button" class="wab-c-decline" id="wab-c-decline">Decline</button>' +
|
|
78
|
-
'</div>';
|
|
79
|
-
|
|
80
|
-
global.document.body.appendChild(bar);
|
|
81
|
-
|
|
82
|
-
bar.querySelector('#wab-c-accept').addEventListener('click', function () {
|
|
83
|
-
setConsent('granted');
|
|
84
|
-
bar.remove();
|
|
85
|
-
if (typeof options.onAccept === 'function') options.onAccept();
|
|
86
|
-
});
|
|
87
|
-
bar.querySelector('#wab-c-decline').addEventListener('click', function () {
|
|
88
|
-
setConsent('denied');
|
|
89
|
-
bar.remove();
|
|
90
|
-
if (typeof options.onDecline === 'function') options.onDecline();
|
|
91
|
-
});
|
|
92
|
-
}
|
|
93
|
-
|
|
94
|
-
function clear() {
|
|
95
|
-
setConsent(null);
|
|
96
|
-
}
|
|
97
|
-
|
|
98
|
-
global.WABConsent = {
|
|
99
|
-
showBanner: showBanner,
|
|
100
|
-
hasConsent: hasConsent,
|
|
101
|
-
clear: clear,
|
|
102
|
-
STORAGE_KEY: STORAGE_KEY
|
|
103
|
-
};
|
|
104
|
-
})(typeof window !== 'undefined' ? window : globalThis);
|
|
1
|
+
/**
|
|
2
|
+
* WAB Consent — optional GDPR/CCPA-style prompt before agents run actions.
|
|
3
|
+
* Load after wab.min.js: <script src="/script/wab-consent.js"></script>
|
|
4
|
+
*
|
|
5
|
+
* WABConsent.showBanner({
|
|
6
|
+
* policyUrl: '/privacy',
|
|
7
|
+
* onAccept: () => WAB.init({ ... }),
|
|
8
|
+
* onDecline: () => { ... }
|
|
9
|
+
* });
|
|
10
|
+
*
|
|
11
|
+
* Before execute, check WABConsent.hasConsent() or gate your own flows.
|
|
12
|
+
*/
|
|
13
|
+
(function (global) {
|
|
14
|
+
'use strict';
|
|
15
|
+
|
|
16
|
+
var STORAGE_KEY = 'wab_agent_consent_v1';
|
|
17
|
+
|
|
18
|
+
function hasConsent() {
|
|
19
|
+
try {
|
|
20
|
+
return global.localStorage.getItem(STORAGE_KEY) === 'granted';
|
|
21
|
+
} catch (e) {
|
|
22
|
+
return false;
|
|
23
|
+
}
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
function setConsent(value) {
|
|
27
|
+
try {
|
|
28
|
+
if (value === null) global.localStorage.removeItem(STORAGE_KEY);
|
|
29
|
+
else global.localStorage.setItem(STORAGE_KEY, value);
|
|
30
|
+
} catch (e) {}
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
function injectStyles() {
|
|
34
|
+
if (global.document.getElementById('wab-consent-styles')) return;
|
|
35
|
+
var s = global.document.createElement('style');
|
|
36
|
+
s.id = 'wab-consent-styles';
|
|
37
|
+
s.textContent =
|
|
38
|
+
'#wab-consent-bar{position:fixed;left:16px;right:16px;bottom:16px;max-width:520px;margin:0 auto;' +
|
|
39
|
+
'background:#0f172a;color:#e2e8f0;padding:16px 18px;border-radius:12px;border:1px solid #334155;' +
|
|
40
|
+
'box-shadow:0 12px 40px rgba(0,0,0,0.35);z-index:99999;font-family:system-ui,sans-serif;font-size:14px;line-height:1.45}' +
|
|
41
|
+
'#wab-consent-bar p{margin:0 0 12px}' +
|
|
42
|
+
'#wab-consent-bar .wab-c-actions{display:flex;flex-wrap:wrap;gap:10px;align-items:center}' +
|
|
43
|
+
'#wab-consent-bar button{padding:8px 16px;border-radius:8px;border:none;font-weight:600;cursor:pointer;font-size:13px}' +
|
|
44
|
+
'#wab-consent-bar .wab-c-accept{background:#3b82f6;color:#fff}' +
|
|
45
|
+
'#wab-consent-bar .wab-c-decline{background:#1e293b;color:#94a3b8;border:1px solid #334155}' +
|
|
46
|
+
'#wab-consent-bar a{color:#93c5fd}';
|
|
47
|
+
global.document.head.appendChild(s);
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
function showBanner(options) {
|
|
51
|
+
options = options || {};
|
|
52
|
+
if (hasConsent() && options.skipIfGranted !== false) {
|
|
53
|
+
if (typeof options.onAccept === 'function') options.onAccept();
|
|
54
|
+
return;
|
|
55
|
+
}
|
|
56
|
+
injectStyles();
|
|
57
|
+
var existing = global.document.getElementById('wab-consent-bar');
|
|
58
|
+
if (existing) existing.remove();
|
|
59
|
+
|
|
60
|
+
var bar = global.document.createElement('div');
|
|
61
|
+
bar.id = 'wab-consent-bar';
|
|
62
|
+
bar.setAttribute('role', 'dialog');
|
|
63
|
+
bar.setAttribute('aria-label', 'AI agent consent');
|
|
64
|
+
|
|
65
|
+
var policy = options.policyUrl
|
|
66
|
+
? '<a href="' + options.policyUrl + '" target="_blank" rel="noopener">Privacy policy</a>'
|
|
67
|
+
: '';
|
|
68
|
+
|
|
69
|
+
bar.innerHTML =
|
|
70
|
+
'<p><strong>AI assistance on this site</strong><br>' +
|
|
71
|
+
(options.message ||
|
|
72
|
+
'This site can expose actions to AI agents (forms, cart, navigation). Allow this site to register agent-ready actions in your browser?') +
|
|
73
|
+
(policy ? ' ' + policy + '.' : '') +
|
|
74
|
+
'</p>' +
|
|
75
|
+
'<div class="wab-c-actions">' +
|
|
76
|
+
'<button type="button" class="wab-c-accept" id="wab-c-accept">Allow</button>' +
|
|
77
|
+
'<button type="button" class="wab-c-decline" id="wab-c-decline">Decline</button>' +
|
|
78
|
+
'</div>';
|
|
79
|
+
|
|
80
|
+
global.document.body.appendChild(bar);
|
|
81
|
+
|
|
82
|
+
bar.querySelector('#wab-c-accept').addEventListener('click', function () {
|
|
83
|
+
setConsent('granted');
|
|
84
|
+
bar.remove();
|
|
85
|
+
if (typeof options.onAccept === 'function') options.onAccept();
|
|
86
|
+
});
|
|
87
|
+
bar.querySelector('#wab-c-decline').addEventListener('click', function () {
|
|
88
|
+
setConsent('denied');
|
|
89
|
+
bar.remove();
|
|
90
|
+
if (typeof options.onDecline === 'function') options.onDecline();
|
|
91
|
+
});
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
function clear() {
|
|
95
|
+
setConsent(null);
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
global.WABConsent = {
|
|
99
|
+
showBanner: showBanner,
|
|
100
|
+
hasConsent: hasConsent,
|
|
101
|
+
clear: clear,
|
|
102
|
+
STORAGE_KEY: STORAGE_KEY
|
|
103
|
+
};
|
|
104
|
+
})(typeof window !== 'undefined' ? window : globalThis);
|
|
@@ -1,131 +1,131 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* WAB Schema hints — scan JSON-LD (schema.org Product, Offer) to suggest WAB actions.
|
|
3
|
-
* <script src="/script/wab-schema.js"></script>
|
|
4
|
-
*
|
|
5
|
-
* var products = WABSchema.scanJsonLd();
|
|
6
|
-
* var suggested = WABSchema.suggestActions(products);
|
|
7
|
-
* WAB.init(WABSchema.mergeWithManual({ buy: { ... } }, suggested));
|
|
8
|
-
*/
|
|
9
|
-
(function (global) {
|
|
10
|
-
'use strict';
|
|
11
|
-
|
|
12
|
-
function normalizeNode(node) {
|
|
13
|
-
if (!node || typeof node !== 'object') return null;
|
|
14
|
-
var types = node['@type'];
|
|
15
|
-
if (typeof types === 'string') types = [types];
|
|
16
|
-
if (!Array.isArray(types)) types = [];
|
|
17
|
-
return { raw: node, types: types };
|
|
18
|
-
}
|
|
19
|
-
|
|
20
|
-
function flattenGraph(data) {
|
|
21
|
-
if (Array.isArray(data)) return data;
|
|
22
|
-
if (data && Array.isArray(data['@graph'])) return data['@graph'];
|
|
23
|
-
return [data];
|
|
24
|
-
}
|
|
25
|
-
|
|
26
|
-
/**
|
|
27
|
-
* @returns {Array<{ type: string, name?: string, sku?: string, offers?: unknown, source: string }>}
|
|
28
|
-
*/
|
|
29
|
-
function scanJsonLd() {
|
|
30
|
-
var out = [];
|
|
31
|
-
var scripts = global.document.querySelectorAll('script[type="application/ld+json"]');
|
|
32
|
-
for (var i = 0; i < scripts.length; i++) {
|
|
33
|
-
var text = scripts[i].textContent;
|
|
34
|
-
if (!text || !text.trim()) continue;
|
|
35
|
-
try {
|
|
36
|
-
var data = JSON.parse(text);
|
|
37
|
-
var items = flattenGraph(data);
|
|
38
|
-
for (var j = 0; j < items.length; j++) {
|
|
39
|
-
var n = normalizeNode(items[j]);
|
|
40
|
-
if (!n) continue;
|
|
41
|
-
var isProduct = n.types.indexOf('Product') !== -1;
|
|
42
|
-
if (!isProduct) continue;
|
|
43
|
-
out.push({
|
|
44
|
-
type: 'Product',
|
|
45
|
-
name: n.raw.name,
|
|
46
|
-
sku: n.raw.sku,
|
|
47
|
-
offers: n.raw.offers,
|
|
48
|
-
source: 'schema.org'
|
|
49
|
-
});
|
|
50
|
-
}
|
|
51
|
-
} catch (e) {}
|
|
52
|
-
}
|
|
53
|
-
return out;
|
|
54
|
-
}
|
|
55
|
-
|
|
56
|
-
/**
|
|
57
|
-
* @param {ReturnType<typeof scanJsonLd>} products
|
|
58
|
-
*/
|
|
59
|
-
function suggestActions(products) {
|
|
60
|
-
var actions = [];
|
|
61
|
-
if (products && products.length) {
|
|
62
|
-
actions.push({
|
|
63
|
-
name: 'getProductFromSchema',
|
|
64
|
-
description: 'Return structured product data detected from schema.org JSON-LD on this page',
|
|
65
|
-
source: 'schema.org',
|
|
66
|
-
auto: true
|
|
67
|
-
});
|
|
68
|
-
}
|
|
69
|
-
var hasOffer = products && products.some(function (p) { return p.offers; });
|
|
70
|
-
if (hasOffer) {
|
|
71
|
-
actions.push({
|
|
72
|
-
name: 'getOfferPrice',
|
|
73
|
-
description: 'Read price/availability from schema.org Offer if present',
|
|
74
|
-
source: 'schema.org',
|
|
75
|
-
auto: true
|
|
76
|
-
});
|
|
77
|
-
}
|
|
78
|
-
return actions;
|
|
79
|
-
}
|
|
80
|
-
|
|
81
|
-
/**
|
|
82
|
-
* Merge manual WAB.init actions with schema-derived runners (read-only helpers).
|
|
83
|
-
* @param {Record<string, object>} manual — same shape as WAB.init({ actions })
|
|
84
|
-
* @param {ReturnType<typeof suggestActions>} suggestions
|
|
85
|
-
*/
|
|
86
|
-
function mergeWithManual(manual, suggestions) {
|
|
87
|
-
manual = manual || {};
|
|
88
|
-
var products = scanJsonLd();
|
|
89
|
-
var extra = {};
|
|
90
|
-
if (suggestions && suggestions.some(function (s) { return s.name === 'getProductFromSchema'; })) {
|
|
91
|
-
extra.getProductFromSchema = {
|
|
92
|
-
description: 'List products from schema.org JSON-LD embedded in the page',
|
|
93
|
-
run: function () {
|
|
94
|
-
return { products: products, count: products.length };
|
|
95
|
-
}
|
|
96
|
-
};
|
|
97
|
-
}
|
|
98
|
-
if (suggestions && suggestions.some(function (s) { return s.name === 'getOfferPrice'; })) {
|
|
99
|
-
extra.getOfferPrice = {
|
|
100
|
-
description: 'Extract offer price strings from detected Product nodes',
|
|
101
|
-
run: function () {
|
|
102
|
-
var prices = [];
|
|
103
|
-
products.forEach(function (p) {
|
|
104
|
-
var o = p.offers;
|
|
105
|
-
if (!o) return;
|
|
106
|
-
var list = Array.isArray(o) ? o : [o];
|
|
107
|
-
list.forEach(function (off) {
|
|
108
|
-
if (off && typeof off === 'object') {
|
|
109
|
-
prices.push({
|
|
110
|
-
product: p.name,
|
|
111
|
-
price: off.price,
|
|
112
|
-
priceCurrency: off.priceCurrency,
|
|
113
|
-
availability: off.availability
|
|
114
|
-
});
|
|
115
|
-
}
|
|
116
|
-
});
|
|
117
|
-
});
|
|
118
|
-
return { offers: prices };
|
|
119
|
-
}
|
|
120
|
-
};
|
|
121
|
-
}
|
|
122
|
-
var merged = Object.assign({}, extra, manual);
|
|
123
|
-
return { actions: merged };
|
|
124
|
-
}
|
|
125
|
-
|
|
126
|
-
global.WABSchema = {
|
|
127
|
-
scanJsonLd: scanJsonLd,
|
|
128
|
-
suggestActions: suggestActions,
|
|
129
|
-
mergeWithManual: mergeWithManual
|
|
130
|
-
};
|
|
131
|
-
})(typeof window !== 'undefined' ? window : globalThis);
|
|
1
|
+
/**
|
|
2
|
+
* WAB Schema hints — scan JSON-LD (schema.org Product, Offer) to suggest WAB actions.
|
|
3
|
+
* <script src="/script/wab-schema.js"></script>
|
|
4
|
+
*
|
|
5
|
+
* var products = WABSchema.scanJsonLd();
|
|
6
|
+
* var suggested = WABSchema.suggestActions(products);
|
|
7
|
+
* WAB.init(WABSchema.mergeWithManual({ buy: { ... } }, suggested));
|
|
8
|
+
*/
|
|
9
|
+
(function (global) {
|
|
10
|
+
'use strict';
|
|
11
|
+
|
|
12
|
+
function normalizeNode(node) {
|
|
13
|
+
if (!node || typeof node !== 'object') return null;
|
|
14
|
+
var types = node['@type'];
|
|
15
|
+
if (typeof types === 'string') types = [types];
|
|
16
|
+
if (!Array.isArray(types)) types = [];
|
|
17
|
+
return { raw: node, types: types };
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
function flattenGraph(data) {
|
|
21
|
+
if (Array.isArray(data)) return data;
|
|
22
|
+
if (data && Array.isArray(data['@graph'])) return data['@graph'];
|
|
23
|
+
return [data];
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
/**
|
|
27
|
+
* @returns {Array<{ type: string, name?: string, sku?: string, offers?: unknown, source: string }>}
|
|
28
|
+
*/
|
|
29
|
+
function scanJsonLd() {
|
|
30
|
+
var out = [];
|
|
31
|
+
var scripts = global.document.querySelectorAll('script[type="application/ld+json"]');
|
|
32
|
+
for (var i = 0; i < scripts.length; i++) {
|
|
33
|
+
var text = scripts[i].textContent;
|
|
34
|
+
if (!text || !text.trim()) continue;
|
|
35
|
+
try {
|
|
36
|
+
var data = JSON.parse(text);
|
|
37
|
+
var items = flattenGraph(data);
|
|
38
|
+
for (var j = 0; j < items.length; j++) {
|
|
39
|
+
var n = normalizeNode(items[j]);
|
|
40
|
+
if (!n) continue;
|
|
41
|
+
var isProduct = n.types.indexOf('Product') !== -1;
|
|
42
|
+
if (!isProduct) continue;
|
|
43
|
+
out.push({
|
|
44
|
+
type: 'Product',
|
|
45
|
+
name: n.raw.name,
|
|
46
|
+
sku: n.raw.sku,
|
|
47
|
+
offers: n.raw.offers,
|
|
48
|
+
source: 'schema.org'
|
|
49
|
+
});
|
|
50
|
+
}
|
|
51
|
+
} catch (e) {}
|
|
52
|
+
}
|
|
53
|
+
return out;
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
/**
|
|
57
|
+
* @param {ReturnType<typeof scanJsonLd>} products
|
|
58
|
+
*/
|
|
59
|
+
function suggestActions(products) {
|
|
60
|
+
var actions = [];
|
|
61
|
+
if (products && products.length) {
|
|
62
|
+
actions.push({
|
|
63
|
+
name: 'getProductFromSchema',
|
|
64
|
+
description: 'Return structured product data detected from schema.org JSON-LD on this page',
|
|
65
|
+
source: 'schema.org',
|
|
66
|
+
auto: true
|
|
67
|
+
});
|
|
68
|
+
}
|
|
69
|
+
var hasOffer = products && products.some(function (p) { return p.offers; });
|
|
70
|
+
if (hasOffer) {
|
|
71
|
+
actions.push({
|
|
72
|
+
name: 'getOfferPrice',
|
|
73
|
+
description: 'Read price/availability from schema.org Offer if present',
|
|
74
|
+
source: 'schema.org',
|
|
75
|
+
auto: true
|
|
76
|
+
});
|
|
77
|
+
}
|
|
78
|
+
return actions;
|
|
79
|
+
}
|
|
80
|
+
|
|
81
|
+
/**
|
|
82
|
+
* Merge manual WAB.init actions with schema-derived runners (read-only helpers).
|
|
83
|
+
* @param {Record<string, object>} manual — same shape as WAB.init({ actions })
|
|
84
|
+
* @param {ReturnType<typeof suggestActions>} suggestions
|
|
85
|
+
*/
|
|
86
|
+
function mergeWithManual(manual, suggestions) {
|
|
87
|
+
manual = manual || {};
|
|
88
|
+
var products = scanJsonLd();
|
|
89
|
+
var extra = {};
|
|
90
|
+
if (suggestions && suggestions.some(function (s) { return s.name === 'getProductFromSchema'; })) {
|
|
91
|
+
extra.getProductFromSchema = {
|
|
92
|
+
description: 'List products from schema.org JSON-LD embedded in the page',
|
|
93
|
+
run: function () {
|
|
94
|
+
return { products: products, count: products.length };
|
|
95
|
+
}
|
|
96
|
+
};
|
|
97
|
+
}
|
|
98
|
+
if (suggestions && suggestions.some(function (s) { return s.name === 'getOfferPrice'; })) {
|
|
99
|
+
extra.getOfferPrice = {
|
|
100
|
+
description: 'Extract offer price strings from detected Product nodes',
|
|
101
|
+
run: function () {
|
|
102
|
+
var prices = [];
|
|
103
|
+
products.forEach(function (p) {
|
|
104
|
+
var o = p.offers;
|
|
105
|
+
if (!o) return;
|
|
106
|
+
var list = Array.isArray(o) ? o : [o];
|
|
107
|
+
list.forEach(function (off) {
|
|
108
|
+
if (off && typeof off === 'object') {
|
|
109
|
+
prices.push({
|
|
110
|
+
product: p.name,
|
|
111
|
+
price: off.price,
|
|
112
|
+
priceCurrency: off.priceCurrency,
|
|
113
|
+
availability: off.availability
|
|
114
|
+
});
|
|
115
|
+
}
|
|
116
|
+
});
|
|
117
|
+
});
|
|
118
|
+
return { offers: prices };
|
|
119
|
+
}
|
|
120
|
+
};
|
|
121
|
+
}
|
|
122
|
+
var merged = Object.assign({}, extra, manual);
|
|
123
|
+
return { actions: merged };
|
|
124
|
+
}
|
|
125
|
+
|
|
126
|
+
global.WABSchema = {
|
|
127
|
+
scanJsonLd: scanJsonLd,
|
|
128
|
+
suggestActions: suggestActions,
|
|
129
|
+
mergeWithManual: mergeWithManual
|
|
130
|
+
};
|
|
131
|
+
})(typeof window !== 'undefined' ? window : globalThis);
|