web-agent-bridge 3.3.0 → 3.8.1
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 +84 -72
- package/README.ar.md +1563 -1286
- package/README.md +137 -1764
- package/bin/agent-runner.js +474 -474
- package/bin/cli.js +237 -237
- package/bin/wab-init.js +244 -0
- package/bin/wab.js +80 -80
- package/examples/azure-dns-wab.js +83 -0
- package/examples/bidi-agent.js +119 -119
- package/examples/cloudflare-wab-dns.js +121 -0
- package/examples/cpanel-wab-dns.js +114 -0
- package/examples/cross-site-agent.js +91 -91
- package/examples/dns-discovery-agent.js +166 -0
- package/examples/gcp-dns-wab.js +76 -0
- package/examples/governance-agent.js +169 -0
- package/examples/mcp-agent.js +94 -94
- package/examples/next-app-router/README.md +44 -44
- package/examples/plesk-wab-dns.js +103 -0
- package/examples/puppeteer-agent.js +108 -108
- package/examples/route53-wab-dns.js +144 -0
- package/examples/saas-dashboard/README.md +55 -55
- package/examples/safe-mode-agent.js +96 -0
- package/examples/self-discovery.js +106 -0
- package/examples/shopify-hydrogen/README.md +74 -74
- package/examples/vision-agent.js +171 -171
- package/examples/wab-sign.js +74 -0
- package/examples/wab-verify.js +60 -0
- package/examples/wordpress-elementor/README.md +77 -77
- package/package.json +93 -93
- package/public/.well-known/agent-tools.json +180 -180
- package/public/.well-known/ai-assets.json +59 -59
- package/public/.well-known/security.txt +8 -8
- package/public/.well-known/wab.json +28 -0
- package/public/activate.html +448 -0
- package/public/adopt.html +236 -0
- package/public/adoption-metrics.html +188 -0
- package/public/agent-workspace.html +359 -349
- package/public/ai.html +198 -198
- package/public/api.html +397 -413
- package/public/azure-dns-integration.html +289 -0
- package/public/browser.html +486 -486
- package/public/cloudflare-integration.html +380 -0
- package/public/commander-dashboard.html +243 -243
- package/public/cookies.html +210 -210
- package/public/cpanel-integration.html +398 -0
- package/public/css/agent-workspace.css +1713 -1713
- package/public/css/premium.css +317 -317
- package/public/css/styles.css +1401 -1235
- package/public/dashboard-shieldlink.html +295 -0
- package/public/dashboard.html +711 -706
- package/public/dns.html +436 -507
- package/public/docs.html +588 -587
- package/public/enterprise-mesh.ar.html +80 -0
- package/public/enterprise-mesh.html +81 -0
- package/public/feed.xml +89 -89
- package/public/gcp-dns-integration.html +318 -0
- package/public/governance.ar.html +70 -0
- package/public/governance.html +69 -0
- package/public/growth.html +465 -463
- package/public/index.html +1372 -1070
- package/public/integrations.html +556 -556
- package/public/js/activate.js +449 -0
- package/public/js/agent-workspace.js +1740 -1740
- package/public/js/auth-nav.js +117 -31
- package/public/js/auth-redirect.js +12 -12
- package/public/js/cookie-consent.js +56 -56
- package/public/js/dns.js +438 -0
- package/public/js/wab-demo-page.js +721 -721
- package/public/js/ws-client.js +74 -74
- package/public/l-preview.html +242 -0
- package/public/llms-full.txt +360 -360
- package/public/llms.txt +125 -125
- package/public/login.html +85 -85
- package/public/mesh-dashboard.html +328 -328
- package/public/milestones.html +346 -0
- package/public/one-click.html +779 -0
- package/public/openapi.json +669 -580
- package/public/partners.ar.html +145 -0
- package/public/partners.html +143 -0
- package/public/phone-shield.html +281 -281
- package/public/plesk-integration.html +375 -0
- package/public/premium-dashboard.html +2489 -2489
- package/public/premium.html +793 -793
- package/public/privacy.html +297 -297
- package/public/provider-onboarding.html +172 -0
- package/public/provider-sandbox.html +134 -0
- package/public/providers.html +359 -0
- package/public/refusals.html +172 -0
- package/public/register.html +105 -105
- package/public/registrar-integrations.html +141 -0
- package/public/ring4.html +292 -0
- package/public/robots.txt +99 -87
- package/public/route53-integration.html +531 -0
- package/public/score.html +263 -0
- 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 -8
- package/public/shieldlink.html +244 -0
- package/public/shieldqr.html +231 -0
- package/public/sitemap.xml +19 -1
- package/public/terms.html +256 -256
- package/public/trust-graph-api.ar.html +92 -0
- package/public/trust-graph-api.html +91 -0
- package/public/wab-features.html +560 -0
- package/public/wab-trust.html +200 -0
- package/public/wab-truth.html +375 -0
- package/public/wab-vs-protocols.html +210 -0
- package/public/whitepaper.html +449 -0
- package/script/ai-agent-bridge.js +1754 -1754
- package/sdk/README.md +99 -99
- package/sdk/agent-mesh.js +449 -449
- package/sdk/auto-discovery.js +301 -0
- package/sdk/commander.js +262 -262
- package/sdk/governance.js +262 -0
- package/sdk/index.d.ts +464 -464
- package/sdk/index.js +649 -636
- package/sdk/multi-agent.js +318 -318
- package/sdk/package.json +2 -2
- package/sdk/safe-mode.js +221 -0
- package/sdk/safety-shield.js +219 -219
- package/sdk/schema-discovery.js +83 -83
- package/server/adapters/index.js +520 -520
- package/server/config/plans.js +412 -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 +790 -531
- package/server/llm/index.js +404 -404
- package/server/middleware/adminAuth.js +35 -35
- package/server/middleware/api-tier.js +170 -0
- 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 -157
- package/server/middleware/wab-trust.js +141 -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/migrations/006_growth_suite.sql +138 -0
- package/server/migrations/007_governance.sql +106 -0
- package/server/migrations/008_plans.sql +144 -0
- package/server/migrations/009_shieldqr.sql +30 -0
- package/server/migrations/010_extended_trust.sql +33 -0
- package/server/migrations/011_outreach.sql +47 -0
- package/server/migrations/012_shieldlink.sql +116 -0
- package/server/migrations/013_ct_monitor.sql +13 -0
- package/server/migrations/014_wab_advanced_features.sql +128 -0
- package/server/migrations/015_wab_truth_layer.sql +101 -0
- package/server/migrations/016_ring4_external_trust.sql +84 -0
- package/server/migrations/017_ring4_extensions.sql +69 -0
- package/server/migrations/018_commercial_foundations.sql +167 -0
- package/server/migrations/019_unify_tier_constraints.sql +133 -0
- 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 +740 -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/activate.js +478 -0
- package/server/routes/admin-outreach.js +239 -0
- package/server/routes/admin-plans.js +76 -0
- package/server/routes/admin-premium.js +674 -671
- package/server/routes/admin-shieldlink.js +137 -0
- package/server/routes/admin-shieldqr.js +90 -0
- package/server/routes/admin-trust-monitor.js +139 -0
- package/server/routes/admin.js +550 -261
- package/server/routes/adopt.js +61 -0
- package/server/routes/ads.js +130 -130
- package/server/routes/agent-workspace.js +540 -540
- package/server/routes/api-keys.js +127 -0
- package/server/routes/api.js +150 -150
- package/server/routes/auth.js +71 -71
- package/server/routes/billing.js +57 -45
- package/server/routes/commander.js +316 -316
- package/server/routes/customer-shieldlink.js +133 -0
- package/server/routes/demo-showcase.js +332 -332
- package/server/routes/demo-store.js +154 -154
- package/server/routes/diagnose.js +373 -0
- package/server/routes/discovery.js +2348 -417
- package/server/routes/enterprise-mesh.js +170 -0
- package/server/routes/gateway.js +173 -173
- package/server/routes/governance-saas.js +203 -0
- package/server/routes/governance.js +208 -0
- package/server/routes/growth.js +1048 -0
- package/server/routes/intent.js +328 -0
- package/server/routes/license.js +251 -251
- package/server/routes/mesh.js +469 -469
- package/server/routes/noscript.js +543 -543
- package/server/routes/partners.js +201 -0
- package/server/routes/plans.js +33 -0
- package/server/routes/premium-v2.js +686 -686
- package/server/routes/premium.js +724 -724
- package/server/routes/providers.js +650 -0
- package/server/routes/reputation.js +411 -0
- package/server/routes/ring4.js +885 -0
- package/server/routes/runtime.js +2148 -2148
- package/server/routes/shieldlink.js +70 -0
- package/server/routes/shieldqr.js +88 -0
- package/server/routes/sovereign.js +465 -465
- package/server/routes/truth-layer.js +670 -0
- package/server/routes/universal.js +200 -200
- package/server/routes/unsubscribe.js +51 -0
- package/server/routes/wab-api.js +850 -850
- package/server/routes/wab-cache.js +282 -0
- 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/secrets/wab-signing-key.pem +3 -0
- package/server/secrets/wab-signing-pub.pem +3 -0
- package/server/security/cross-site-redactor.js +196 -196
- package/server/security/dry-run.js +180 -180
- package/server/security/human-gate-rate-limit.js +147 -147
- package/server/security/human-gate-transports.js +178 -178
- package/server/security/human-gate.js +281 -281
- package/server/security/index.js +368 -368
- package/server/security/intent-engine.js +245 -245
- package/server/security/reward-guard.js +171 -171
- package/server/security/rollback-store.js +239 -239
- package/server/security/token-scope.js +404 -404
- package/server/security/url-policy.js +139 -139
- package/server/services/adoption-agent.js +182 -0
- package/server/services/agent-chat.js +506 -506
- package/server/services/agent-learning.js +601 -601
- package/server/services/agent-memory.js +625 -625
- package/server/services/agent-mesh.js +555 -555
- package/server/services/agent-symphony.js +717 -717
- package/server/services/agent-tasks.js +1807 -1807
- package/server/services/api-key-engine.js +292 -292
- 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 +233 -204
- package/server/services/fairness-engine.js +409 -0
- package/server/services/fairness.js +420 -0
- package/server/services/governance.js +466 -0
- package/server/services/hosted-runtime.js +205 -205
- package/server/services/lfd.js +635 -635
- 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 -93
- package/server/services/modules/agent-firewall.js +90 -90
- package/server/services/modules/bounty.js +89 -89
- package/server/services/modules/collective-bargaining.js +92 -92
- package/server/services/modules/dark-pattern.js +66 -66
- package/server/services/modules/gov-intelligence.js +45 -45
- package/server/services/modules/neural.js +55 -55
- package/server/services/modules/notary.js +49 -49
- package/server/services/modules/price-time-machine.js +86 -86
- package/server/services/modules/protocol.js +104 -104
- package/server/services/negotiation.js +439 -439
- package/server/services/outreach-agent.js +312 -0
- package/server/services/plans.js +214 -0
- package/server/services/plugins.js +771 -771
- package/server/services/premium.js +1 -1
- package/server/services/price-intelligence.js +566 -566
- package/server/services/price-shield.js +1137 -1137
- package/server/services/provider-clients.js +740 -0
- 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/shieldlink.js +492 -0
- package/server/services/shieldqr.js +322 -0
- package/server/services/sovereign-shield.js +542 -542
- package/server/services/ssl-ct-monitor.js +224 -0
- package/server/services/ssl-inspector.js +42 -0
- package/server/services/ssl-monitor.js +167 -0
- package/server/services/stripe.js +206 -192
- package/server/services/swarm.js +788 -788
- package/server/services/universal-scraper.js +662 -662
- package/server/services/verification.js +481 -481
- package/server/services/vision.js +1163 -1163
- package/server/services/wab-crypto.js +178 -0
- package/server/utils/cache.js +125 -125
- package/server/utils/migrate.js +81 -81
- package/server/utils/safe-fetch.js +228 -228
- 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/templates/ring4/banking-sovereign.yaml +55 -0
- package/templates/ring4/ecommerce-sovereign.yaml +58 -0
- package/templates/ring4/healthcare-sovereign.yaml +60 -0
package/public/security.txt
CHANGED
|
@@ -1,8 +1,8 @@
|
|
|
1
|
-
Contact: mailto:security@webagentbridge.com
|
|
2
|
-
Expires: 2027-04-26T00:00:00.000Z
|
|
3
|
-
Encryption: https://www.webagentbridge.com/.well-known/pgp-key.txt
|
|
4
|
-
Acknowledgments: https://www.webagentbridge.com/security#hall-of-fame
|
|
5
|
-
Preferred-Languages: en, ar
|
|
6
|
-
Canonical: https://www.webagentbridge.com/.well-known/security.txt
|
|
7
|
-
Policy: https://www.webagentbridge.com/security
|
|
8
|
-
Hiring: https://www.webagentbridge.com/security#bug-bounty
|
|
1
|
+
Contact: mailto:security@webagentbridge.com
|
|
2
|
+
Expires: 2027-04-26T00:00:00.000Z
|
|
3
|
+
Encryption: https://www.webagentbridge.com/.well-known/pgp-key.txt
|
|
4
|
+
Acknowledgments: https://www.webagentbridge.com/security#hall-of-fame
|
|
5
|
+
Preferred-Languages: en, ar
|
|
6
|
+
Canonical: https://www.webagentbridge.com/.well-known/security.txt
|
|
7
|
+
Policy: https://www.webagentbridge.com/security
|
|
8
|
+
Hiring: https://www.webagentbridge.com/security#bug-bounty
|
|
@@ -0,0 +1,244 @@
|
|
|
1
|
+
<!doctype html>
|
|
2
|
+
<html lang="en">
|
|
3
|
+
<head>
|
|
4
|
+
<meta charset="utf-8" />
|
|
5
|
+
<meta name="viewport" content="width=device-width,initial-scale=1" />
|
|
6
|
+
<title>WAB ShieldLink — Verified Links for Banks & Brands</title>
|
|
7
|
+
<meta name="description" content="Sender-signed, anti-phishing links from verified brands. Cryptographically signed, DNS-anchored, no install required for end users." />
|
|
8
|
+
<meta property="og:title" content="WAB ShieldLink — Verified Links" />
|
|
9
|
+
<meta property="og:description" content="Anti-phishing verified links for banks, payment processors, and businesses. Anyone can verify; no app install required." />
|
|
10
|
+
<meta property="og:url" content="https://www.webagentbridge.com/shieldlink" />
|
|
11
|
+
<link rel="canonical" href="https://www.webagentbridge.com/shieldlink" />
|
|
12
|
+
<style>
|
|
13
|
+
:root{--bg:#0b1220;--fg:#fff;--muted:#94a3b8;--card:#111a2e;--border:#1f2a44;--accent:#22d3a3;--warn:#f59e0b;--bad:#ef4444}
|
|
14
|
+
*{box-sizing:border-box}
|
|
15
|
+
body{margin:0;font-family:-apple-system,BlinkMacSystemFont,'Segoe UI',Roboto,Arial,sans-serif;background:var(--bg);color:var(--fg);line-height:1.6}
|
|
16
|
+
.wrap{max-width:1100px;margin:0 auto;padding:24px}
|
|
17
|
+
header{display:flex;justify-content:space-between;align-items:center;padding:18px 0}
|
|
18
|
+
header a.logo{color:#fff;text-decoration:none;font-weight:700;font-size:18px}
|
|
19
|
+
nav a{color:var(--muted);margin-left:20px;text-decoration:none;font-size:14px}
|
|
20
|
+
nav a:hover{color:#fff}
|
|
21
|
+
[dir="rtl"] nav a{margin-left:0;margin-right:20px}
|
|
22
|
+
.hero{padding:60px 0 40px;text-align:center}
|
|
23
|
+
.pill{display:inline-block;background:rgba(34,211,163,.15);color:var(--accent);padding:6px 14px;border-radius:99px;font-size:12px;font-weight:600;letter-spacing:.5px;text-transform:uppercase}
|
|
24
|
+
h1{font-size:44px;line-height:1.15;margin:18px 0 16px}
|
|
25
|
+
.lede{color:var(--muted);font-size:18px;max-width:720px;margin:0 auto 26px}
|
|
26
|
+
.cta{display:inline-block;background:var(--accent);color:#0b1220;font-weight:700;padding:14px 28px;border-radius:8px;text-decoration:none;margin:6px}
|
|
27
|
+
.cta.ghost{background:transparent;color:#fff;border:1px solid var(--border)}
|
|
28
|
+
.grid{display:grid;grid-template-columns:repeat(auto-fit,minmax(260px,1fr));gap:18px;margin:42px 0}
|
|
29
|
+
.card{background:var(--card);border:1px solid var(--border);border-radius:12px;padding:20px}
|
|
30
|
+
.card h3{margin:0 0 8px;font-size:18px}
|
|
31
|
+
.card p{color:var(--muted);font-size:14px;margin:0}
|
|
32
|
+
.demo{background:var(--card);border:1px solid var(--border);border-radius:12px;padding:24px;margin:28px 0}
|
|
33
|
+
.demo h2{margin:0 0 14px}
|
|
34
|
+
.preview{display:flex;gap:16px;flex-wrap:wrap}
|
|
35
|
+
.preview .col{flex:1;min-width:220px;background:#0f1730;border:1px solid var(--border);border-radius:10px;overflow:hidden}
|
|
36
|
+
.preview .head{padding:12px;color:#fff;font-weight:600;font-size:14px}
|
|
37
|
+
.preview .head.ok{background:#0a9d58}
|
|
38
|
+
.preview .head.warn{background:#f59e0b;color:#1f2933}
|
|
39
|
+
.preview .head.bad{background:#dc2626}
|
|
40
|
+
.preview ul{margin:0;padding:14px 18px;color:var(--muted);font-size:13px}
|
|
41
|
+
.preview li{margin:4px 0}
|
|
42
|
+
section{padding:36px 0;border-top:1px solid var(--border)}
|
|
43
|
+
h2{font-size:28px;margin:0 0 14px}
|
|
44
|
+
.steps{counter-reset:step;list-style:none;padding:0}
|
|
45
|
+
.steps li{padding:14px 0 14px 56px;position:relative;color:var(--muted)}
|
|
46
|
+
.steps li::before{counter-increment:step;content:counter(step);position:absolute;left:0;top:14px;width:36px;height:36px;border-radius:50%;background:var(--accent);color:#0b1220;font-weight:700;display:flex;align-items:center;justify-content:center}
|
|
47
|
+
[dir="rtl"] .steps li{padding:14px 56px 14px 0}
|
|
48
|
+
[dir="rtl"] .steps li::before{left:auto;right:0}
|
|
49
|
+
.steps li b{color:#fff;display:block;margin-bottom:2px}
|
|
50
|
+
table{width:100%;border-collapse:collapse;margin:20px 0;font-size:14px}
|
|
51
|
+
th,td{text-align:left;padding:12px;border-bottom:1px solid var(--border)}
|
|
52
|
+
th{color:var(--accent);font-size:13px;text-transform:uppercase;letter-spacing:.4px}
|
|
53
|
+
[dir="rtl"] th,[dir="rtl"] td{text-align:right}
|
|
54
|
+
.price{display:grid;grid-template-columns:repeat(auto-fit,minmax(220px,1fr));gap:18px}
|
|
55
|
+
.plan{background:var(--card);border:1px solid var(--border);border-radius:12px;padding:24px}
|
|
56
|
+
.plan.pro{border-color:var(--accent)}
|
|
57
|
+
.plan h3{margin:0 0 4px}
|
|
58
|
+
.plan .p{font-size:32px;font-weight:700;margin:6px 0 12px}
|
|
59
|
+
.plan ul{padding-left:18px;color:var(--muted);font-size:14px;margin:0}
|
|
60
|
+
[dir="rtl"] .plan ul{padding-left:0;padding-right:18px}
|
|
61
|
+
footer{text-align:center;color:var(--muted);font-size:13px;padding:36px 0;border-top:1px solid var(--border)}
|
|
62
|
+
.lang{cursor:pointer;color:var(--muted);text-decoration:none}
|
|
63
|
+
.lang:hover{color:#fff}
|
|
64
|
+
@media(max-width:640px){h1{font-size:32px}.hero{padding:36px 0 24px}}
|
|
65
|
+
</style>
|
|
66
|
+
</head>
|
|
67
|
+
<body>
|
|
68
|
+
<div class="wrap">
|
|
69
|
+
<header>
|
|
70
|
+
<a class="logo" href="/">Web Agent Bridge</a>
|
|
71
|
+
<nav>
|
|
72
|
+
<a href="/">Home</a>
|
|
73
|
+
<a href="/pricing.html" data-en="Pricing" data-ar="الأسعار">Pricing</a>
|
|
74
|
+
<a href="/dashboard.html" data-en="Dashboard" data-ar="لوحة التحكم">Dashboard</a>
|
|
75
|
+
<a class="lang" id="lang-toggle">عربي</a>
|
|
76
|
+
</nav>
|
|
77
|
+
</header>
|
|
78
|
+
|
|
79
|
+
<div class="hero">
|
|
80
|
+
<span class="pill" data-en="WAB ShieldLink · New" data-ar="شيلد لينك · جديد">WAB ShieldLink · New</span>
|
|
81
|
+
<h1 data-en="Verified links from real banks & brands."
|
|
82
|
+
data-ar="روابط موثّقة من البنوك والعلامات الحقيقية.">
|
|
83
|
+
Verified links from real banks & brands.
|
|
84
|
+
</h1>
|
|
85
|
+
<p class="lede" data-en="Stop phishing at the link layer. Banks, payment processors, telcos, and businesses sign every link they send. Anyone who clicks sees a Trust Preview before reaching the destination. No install. No browser extension. Just better trust on the open web."
|
|
86
|
+
data-ar="أوقف التصيّد عند طبقة الرابط. البنوك ومزوّدو الدفع والشركات يوقّعون كل رابط يرسلونه، وعند فتحه يظهر لأي مستخدم معاينة موثوقة قبل الوصول إلى الوجهة. بدون أي تطبيق أو إضافة متصفح.">
|
|
87
|
+
Stop phishing at the link layer. Banks, payment processors, telcos, and businesses sign every link they send. Anyone who clicks sees a Trust Preview before reaching the destination. No install. No browser extension.
|
|
88
|
+
</p>
|
|
89
|
+
<a class="cta" href="/register.html" data-en="Get verified — start free" data-ar="ابدأ التوثيق — مجاناً">Get verified — start free</a>
|
|
90
|
+
<a class="cta ghost" href="#how" data-en="How it works" data-ar="كيف يعمل">How it works</a>
|
|
91
|
+
</div>
|
|
92
|
+
|
|
93
|
+
<div class="demo">
|
|
94
|
+
<h2 data-en="What end-users see" data-ar="ما يراه المستخدم">What end-users see</h2>
|
|
95
|
+
<p style="color:var(--muted);margin:0 0 16px" data-en="Every signed link opens a Trust Preview before redirecting. Recipients see the real brand, the real amount, and the real expiry — verified by Ed25519 signature."
|
|
96
|
+
data-ar="كل رابط موقّع يفتح معاينة موثوقة قبل إعادة التوجيه. يرى المستخدم اسم الجهة الحقيقية والمبلغ ومدة الصلاحية، مُتحقَّق منها بتوقيع Ed25519.">
|
|
97
|
+
</p>
|
|
98
|
+
<div class="preview">
|
|
99
|
+
<div class="col"><div class="head ok">✓ <span data-en="Verified — Bank Example" data-ar="موثّق — بنك مثال">Verified — Bank Example</span></div>
|
|
100
|
+
<ul>
|
|
101
|
+
<li><b data-en="Brand:" data-ar="الجهة:">Brand:</b> Bank Example <span style="color:var(--accent)">●</span></li>
|
|
102
|
+
<li><b data-en="Payee:" data-ar="المستفيد:">Payee:</b> ACME LLC</li>
|
|
103
|
+
<li><b data-en="Amount:" data-ar="المبلغ:">Amount:</b> 240.00 SAR</li>
|
|
104
|
+
<li><b data-en="Domain match:" data-ar="مطابقة النطاق:">Domain match:</b> ✓</li>
|
|
105
|
+
<li><b data-en="Signature:" data-ar="التوقيع:">Signature:</b> ✓ Ed25519</li>
|
|
106
|
+
</ul>
|
|
107
|
+
</div>
|
|
108
|
+
<div class="col"><div class="head warn">⚠ <span data-en="Caution — pending review" data-ar="تنبيه — قيد المراجعة">Caution — pending review</span></div>
|
|
109
|
+
<ul>
|
|
110
|
+
<li><b data-en="Brand:" data-ar="الجهة:">Brand:</b> NewMerchant</li>
|
|
111
|
+
<li><b data-en="Status:" data-ar="الحالة:">Status:</b> brand_unverified</li>
|
|
112
|
+
<li><b data-en="Signature:" data-ar="التوقيع:">Signature:</b> ✓</li>
|
|
113
|
+
</ul>
|
|
114
|
+
</div>
|
|
115
|
+
<div class="col"><div class="head bad">✕ <span data-en="Blocked — phishing" data-ar="محظور — تصيّد">Blocked — phishing</span></div>
|
|
116
|
+
<ul>
|
|
117
|
+
<li><b data-en="Reason:" data-ar="السبب:">Reason:</b> revoked · many_reports</li>
|
|
118
|
+
<li><b data-en="Signature:" data-ar="التوقيع:">Signature:</b> ✕</li>
|
|
119
|
+
</ul>
|
|
120
|
+
</div>
|
|
121
|
+
</div>
|
|
122
|
+
</div>
|
|
123
|
+
|
|
124
|
+
<section id="how">
|
|
125
|
+
<h2 data-en="How it works" data-ar="كيف يعمل">How it works</h2>
|
|
126
|
+
<ol class="steps">
|
|
127
|
+
<li><b data-en="Verify your domain (DNS TXT)" data-ar="وثّق نطاقك عبر سجل DNS TXT">Verify your domain (DNS TXT)</b>
|
|
128
|
+
<span data-en="Add the WAB trust record to your DNS — the same root every modern protocol uses (no CA required)."
|
|
129
|
+
data-ar="أضف سجل ثقة WAB إلى DNS الخاص بك — نفس الأساس الذي تستخدمه البروتوكولات الحديثة (بدون مرجع تصديق).">
|
|
130
|
+
</span>
|
|
131
|
+
</li>
|
|
132
|
+
<li><b data-en="Apply for a Verified Brand badge" data-ar="قدّم طلب شارة الجهة الموثّقة">Apply for a Verified Brand badge</b>
|
|
133
|
+
<span data-en="Submit your display name. Lookalikes (Levenshtein distance ≤ 2) are auto-rejected, and reserved names like 'mada' or 'paypal' require admin review."
|
|
134
|
+
data-ar="أدخل اسم الجهة. الأسماء المتشابهة (مسافة Levenshtein ≤ 2) تُرفض تلقائياً، والأسماء المحجوزة مثل 'مدى' أو 'PayPal' تتطلب مراجعة الإدارة.">
|
|
135
|
+
</span>
|
|
136
|
+
</li>
|
|
137
|
+
<li><b data-en="Sign every link you send" data-ar="وقّع كل رابط ترسله">Sign every link you send</b>
|
|
138
|
+
<span data-en="Call POST /api/customer/shieldlink/sites/:siteId/sign with the destination, amount, payee, and expiry. Get back a short link like https://www.webagentbridge.com/l/abc123."
|
|
139
|
+
data-ar="استدعِ POST /api/customer/shieldlink/sites/:siteId/sign بالوجهة والمبلغ والمستفيد ومدة الصلاحية، لتحصل على رابط قصير مثل https://www.webagentbridge.com/l/abc123.">
|
|
140
|
+
</span>
|
|
141
|
+
</li>
|
|
142
|
+
<li><b data-en="Anyone verifies — no app needed" data-ar="أي شخص يتحقق — بدون تطبيق">Anyone verifies — no app needed</b>
|
|
143
|
+
<span data-en="When the recipient opens the link, the Trust Preview verifies the Ed25519 signature, fetches your DNS-anchored public key, and shows a green / yellow / red verdict before redirect."
|
|
144
|
+
data-ar="عند فتح المستلم للرابط، تتحقق المعاينة الموثوقة من التوقيع، وتجلب المفتاح العام من DNS، ثم تعرض حكماً أخضر/أصفر/أحمر قبل التحويل.">
|
|
145
|
+
</span>
|
|
146
|
+
</li>
|
|
147
|
+
</ol>
|
|
148
|
+
</section>
|
|
149
|
+
|
|
150
|
+
<section>
|
|
151
|
+
<h2 data-en="Why it stops phishing" data-ar="لماذا يوقف التصيّد">Why it stops phishing</h2>
|
|
152
|
+
<div class="grid">
|
|
153
|
+
<div class="card"><h3 data-en="Identity = domain ownership" data-ar="الهوية = ملكية النطاق">Identity = domain ownership</h3>
|
|
154
|
+
<p data-en="Only the proven owner of bank.example can sign links carrying the brand 'Bank Example'. No CA, no manual paperwork — DNS + DNSSEC are the trust root."
|
|
155
|
+
data-ar="فقط المالك الموثّق لـ bank.example يستطيع توقيع روابط تحمل اسم 'Bank Example'. لا حاجة لمرجع تصديق ولا أوراق — DNS وDNSSEC هما جذر الثقة.">
|
|
156
|
+
</p></div>
|
|
157
|
+
<div class="card"><h3 data-en="Lookalike protection" data-ar="حماية من الأسماء المتشابهة">Lookalike protection</h3>
|
|
158
|
+
<p data-en="The registry rejects any display name within Levenshtein distance 2 of an existing verified brand, plus a reserved-name list for high-value targets (banks, payment networks)."
|
|
159
|
+
data-ar="السجل يرفض أي اسم بمسافة Levenshtein ≤ 2 من جهة موثّقة قائمة، إضافة لقائمة أسماء محجوزة للأهداف عالية القيمة (البنوك وشبكات الدفع).">
|
|
160
|
+
</p></div>
|
|
161
|
+
<div class="card"><h3 data-en="Cryptographic, not cosmetic" data-ar="تشفير، وليس مجرد شكل">Cryptographic, not cosmetic</h3>
|
|
162
|
+
<p data-en="Every link is signed with Ed25519. Tampering invalidates the signature. The Trust Preview won't show green unless the math checks out."
|
|
163
|
+
data-ar="كل رابط موقّع بـ Ed25519. أي تعديل يُبطل التوقيع، ولن تظهر المعاينة باللون الأخضر إلا إذا تحققت الرياضيات.">
|
|
164
|
+
</p></div>
|
|
165
|
+
<div class="card"><h3 data-en="Community reports" data-ar="بلاغات المجتمع">Community reports</h3>
|
|
166
|
+
<p data-en="Recipients can report a link in one tap. Multiple open reports flip the verdict to red and trigger admin review automatically."
|
|
167
|
+
data-ar="يمكن للمستلم الإبلاغ عن أي رابط بضغطة واحدة. تعدد البلاغات يحوّل الحكم إلى الأحمر ويفعّل المراجعة تلقائياً.">
|
|
168
|
+
</p></div>
|
|
169
|
+
<div class="card"><h3 data-en="Revocable in real time" data-ar="قابل للإلغاء فوراً">Revocable in real time</h3>
|
|
170
|
+
<p data-en="If a campaign goes wrong, revoke a single link or rotate your signing key from the dashboard — every future verification reflects it instantly."
|
|
171
|
+
data-ar="إذا حدث خطأ في حملة ما، يمكنك إلغاء رابط واحد أو تدوير مفتاح التوقيع من لوحة التحكم — كل تحقق لاحق ينعكس فوراً.">
|
|
172
|
+
</p></div>
|
|
173
|
+
<div class="card"><h3 data-en="Zero install for end users" data-ar="بدون تثبيت لدى المستخدم">Zero install for end users</h3>
|
|
174
|
+
<p data-en="Anyone with a browser benefits. The sender pays for verification; the recipient just clicks."
|
|
175
|
+
data-ar="أي مستخدم لديه متصفح يستفيد. المرسل يدفع مقابل التوثيق، والمستلم يضغط فقط.">
|
|
176
|
+
</p></div>
|
|
177
|
+
</div>
|
|
178
|
+
</section>
|
|
179
|
+
|
|
180
|
+
<section>
|
|
181
|
+
<h2 data-en="API" data-ar="واجهة البرمجة">API</h2>
|
|
182
|
+
<table>
|
|
183
|
+
<thead><tr>
|
|
184
|
+
<th data-en="Endpoint" data-ar="نقطة النهاية">Endpoint</th>
|
|
185
|
+
<th data-en="Auth" data-ar="المصادقة">Auth</th>
|
|
186
|
+
<th data-en="Purpose" data-ar="الغرض">Purpose</th>
|
|
187
|
+
</tr></thead>
|
|
188
|
+
<tbody>
|
|
189
|
+
<tr><td><code>POST /api/customer/shieldlink/sites/:siteId/brand</code></td><td>User</td><td data-en="Apply for verified brand" data-ar="طلب جهة موثّقة">Apply for verified brand</td></tr>
|
|
190
|
+
<tr><td><code>POST /api/customer/shieldlink/sites/:siteId/sign</code></td><td>User · Pro+</td><td data-en="Sign a link" data-ar="توقيع رابط">Sign a link</td></tr>
|
|
191
|
+
<tr><td><code>GET /api/customer/shieldlink/sites/:siteId/links</code></td><td>User · Pro+</td><td data-en="List signed links" data-ar="قائمة الروابط الموقّعة">List signed links</td></tr>
|
|
192
|
+
<tr><td><code>POST /api/customer/shieldlink/sites/:siteId/links/:id/revoke</code></td><td>User · Pro+</td><td data-en="Revoke a link" data-ar="إلغاء رابط">Revoke a link</td></tr>
|
|
193
|
+
<tr><td><code>GET /api/shieldlink/verify?token=…</code></td><td data-en="Public" data-ar="عام">Public</td><td data-en="Verify a token (used by Trust Preview)" data-ar="التحقق من رمز رابط">Verify a token</td></tr>
|
|
194
|
+
<tr><td><code>POST /api/shieldlink/report</code></td><td data-en="Public" data-ar="عام">Public</td><td data-en="Report a phishing link" data-ar="الإبلاغ عن رابط تصيّد">Report a phishing link</td></tr>
|
|
195
|
+
</tbody>
|
|
196
|
+
</table>
|
|
197
|
+
</section>
|
|
198
|
+
|
|
199
|
+
<section>
|
|
200
|
+
<h2 data-en="Pricing" data-ar="الأسعار">Pricing</h2>
|
|
201
|
+
<div class="price">
|
|
202
|
+
<div class="plan"><h3>Free / Starter</h3><div class="p" data-en="Not included" data-ar="غير متاح">Not included</div>
|
|
203
|
+
<ul><li data-en="Verify links others sent" data-ar="التحقق من روابط الآخرين">Verify links others sent</li><li data-en="Submit phishing reports" data-ar="إرسال البلاغات">Submit phishing reports</li></ul>
|
|
204
|
+
</div>
|
|
205
|
+
<div class="plan pro"><h3>Pro · $99/mo</h3><div class="p">✓</div>
|
|
206
|
+
<ul>
|
|
207
|
+
<li data-en="Verified Brand badge (after review)" data-ar="شارة الجهة الموثّقة (بعد المراجعة)">Verified Brand badge (after review)</li>
|
|
208
|
+
<li data-en="Sign unlimited links" data-ar="توقيع روابط غير محدودة">Sign unlimited links</li>
|
|
209
|
+
<li data-en="Revocation + key rotation" data-ar="إلغاء + تدوير المفاتيح">Revocation + key rotation</li>
|
|
210
|
+
<li data-en="Per-link analytics" data-ar="تحليلات لكل رابط">Per-link analytics</li>
|
|
211
|
+
</ul>
|
|
212
|
+
</div>
|
|
213
|
+
<div class="plan"><h3>Enterprise</h3><div class="p" data-en="Custom" data-ar="مخصّص">Custom</div>
|
|
214
|
+
<ul>
|
|
215
|
+
<li data-en="Everything in Pro" data-ar="كل ما في Pro">Everything in Pro</li>
|
|
216
|
+
<li data-en="Custom domain for /l/ links" data-ar="نطاق مخصّص لروابط /l/">Custom domain for /l/ links</li>
|
|
217
|
+
<li data-en="BYOK signing keys, on-prem option" data-ar="مفاتيح خاصة بك أو نشر داخلي">BYOK signing keys, on-prem option</li>
|
|
218
|
+
<li data-en="Priority brand review & SLA" data-ar="مراجعة فورية واتفاقية مستوى خدمة">Priority review & SLA</li>
|
|
219
|
+
</ul>
|
|
220
|
+
</div>
|
|
221
|
+
</div>
|
|
222
|
+
<p style="text-align:center;margin-top:24px"><a class="cta" href="/register.html" data-en="Start with Pro" data-ar="ابدأ بخطة Pro">Start with Pro</a></p>
|
|
223
|
+
</section>
|
|
224
|
+
|
|
225
|
+
<footer>
|
|
226
|
+
<p>© Web Agent Bridge · <a href="/" style="color:var(--muted)">webagentbridge.com</a></p>
|
|
227
|
+
</footer>
|
|
228
|
+
</div>
|
|
229
|
+
<script>
|
|
230
|
+
(function(){
|
|
231
|
+
const html = document.documentElement;
|
|
232
|
+
const btn = document.getElementById('lang-toggle');
|
|
233
|
+
function set(lang){
|
|
234
|
+
html.setAttribute('lang', lang);
|
|
235
|
+
html.setAttribute('dir', lang === 'ar' ? 'rtl' : 'ltr');
|
|
236
|
+
document.querySelectorAll('[data-en]').forEach(el => el.textContent = el.getAttribute('data-' + lang) || el.getAttribute('data-en'));
|
|
237
|
+
btn.textContent = lang === 'ar' ? 'English' : 'عربي';
|
|
238
|
+
}
|
|
239
|
+
btn.onclick = () => set(html.getAttribute('lang') === 'ar' ? 'en' : 'ar');
|
|
240
|
+
if ((navigator.language||'').startsWith('ar')) set('ar');
|
|
241
|
+
})();
|
|
242
|
+
</script>
|
|
243
|
+
</body>
|
|
244
|
+
</html>
|
|
@@ -0,0 +1,231 @@
|
|
|
1
|
+
<!DOCTYPE html>
|
|
2
|
+
<html lang="en">
|
|
3
|
+
<head>
|
|
4
|
+
<meta charset="UTF-8">
|
|
5
|
+
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
|
6
|
+
<title>WAB ShieldQR — Scan QR codes safely</title>
|
|
7
|
+
<meta name="description" content="Scan any QR code in your browser and instantly verify the destination with WAB ShieldQR — DNS trust, Ed25519 signatures, SSL inspection.">
|
|
8
|
+
<link href="https://fonts.googleapis.com/css2?family=Inter:wght@400;500;600;700;800&display=swap" rel="stylesheet">
|
|
9
|
+
<link rel="stylesheet" href="/css/styles.css">
|
|
10
|
+
<style>
|
|
11
|
+
body { font-family: 'Inter', system-ui, sans-serif; margin:0; background:#0f172a; color:#e2e8f0; min-height:100vh; }
|
|
12
|
+
.wrap { max-width: 720px; margin: 0 auto; padding: 28px 18px 60px; }
|
|
13
|
+
h1 { font-size: 1.8rem; margin: 0 0 6px; display:flex; align-items:center; gap:10px; }
|
|
14
|
+
.sub { color:#94a3b8; margin: 0 0 22px; }
|
|
15
|
+
.card { background:#1e293b; border:1px solid #334155; border-radius:14px; padding:18px; margin-bottom:16px; }
|
|
16
|
+
#reader { width:100%; border-radius:10px; overflow:hidden; background:#000; }
|
|
17
|
+
#reader video { width:100%; max-height:60vh; }
|
|
18
|
+
.controls { display:flex; gap:8px; flex-wrap:wrap; margin-top:10px; }
|
|
19
|
+
.btn { padding:10px 16px; border-radius:8px; border:none; font-weight:600; cursor:pointer; font-size:.95rem; }
|
|
20
|
+
.btn-primary { background:#6366f1; color:#fff; }
|
|
21
|
+
.btn-primary:hover { background:#4f46e5; }
|
|
22
|
+
.btn-secondary { background:#334155; color:#e2e8f0; }
|
|
23
|
+
.btn-danger { background:#dc2626; color:#fff; }
|
|
24
|
+
.btn-success { background:#16a34a; color:#fff; }
|
|
25
|
+
input[type=text] { flex:1; min-width:200px; padding:10px 12px; border-radius:8px; border:1px solid #475569; background:#0f172a; color:#e2e8f0; font-size:.95rem; }
|
|
26
|
+
.verdict { padding:18px; border-radius:14px; border:2px solid; }
|
|
27
|
+
.verdict.green { border-color:#16a34a; background:rgba(22,163,74,.08); }
|
|
28
|
+
.verdict.yellow { border-color:#ca8a04; background:rgba(202,138,4,.08); }
|
|
29
|
+
.verdict.red { border-color:#dc2626; background:rgba(220,38,38,.08); }
|
|
30
|
+
.verdict h2 { margin:0 0 6px; font-size:1.4rem; }
|
|
31
|
+
.verdict .score { font-size:2.4rem; font-weight:800; }
|
|
32
|
+
.verdict .url { font-family: ui-monospace, Menlo, monospace; font-size:.85rem; word-break: break-all; color:#cbd5e1; margin: 6px 0 12px; }
|
|
33
|
+
.signals { margin-top:10px; }
|
|
34
|
+
.signal { padding:8px 10px; border-radius:8px; background:#0f172a; margin-bottom:6px; font-size:.85rem; display:flex; gap:8px; align-items:flex-start; }
|
|
35
|
+
.signal .sev { font-size:.7rem; font-weight:700; padding:1px 6px; border-radius:4px; text-transform:uppercase; flex-shrink:0; }
|
|
36
|
+
.sev-low { background:#0ea5e9; color:#fff; }
|
|
37
|
+
.sev-medium { background:#ca8a04; color:#fff; }
|
|
38
|
+
.sev-high { background:#dc2626; color:#fff; }
|
|
39
|
+
.meta { display:grid; grid-template-columns:repeat(auto-fit, minmax(140px,1fr)); gap:8px; margin-top:10px; font-size:.85rem; }
|
|
40
|
+
.meta div { background:#0f172a; padding:8px 10px; border-radius:8px; }
|
|
41
|
+
.meta .lbl { color:#94a3b8; font-size:.7rem; text-transform:uppercase; letter-spacing:.04em; }
|
|
42
|
+
.hidden { display:none !important; }
|
|
43
|
+
.footer-note { color:#64748b; font-size:.8rem; text-align:center; margin-top:30px; }
|
|
44
|
+
a { color:#a5b4fc; }
|
|
45
|
+
</style>
|
|
46
|
+
<!-- html5-qrcode loader (CSP-allowed origin) -->
|
|
47
|
+
<script src="https://unpkg.com/html5-qrcode@2.3.8/html5-qrcode.min.js"></script>
|
|
48
|
+
</head>
|
|
49
|
+
<body>
|
|
50
|
+
<div class="wrap">
|
|
51
|
+
<h1>🛡️ WAB ShieldQR</h1>
|
|
52
|
+
<p class="sub">Scan any QR code in your browser. We verify the destination with DNS trust records, Ed25519 signatures, and live SSL inspection — <strong>before</strong> you ever open the link.</p>
|
|
53
|
+
|
|
54
|
+
<div class="card">
|
|
55
|
+
<div id="reader"></div>
|
|
56
|
+
<div class="controls">
|
|
57
|
+
<button id="startCam" class="btn btn-primary">📷 Start camera</button>
|
|
58
|
+
<button id="stopCam" class="btn btn-secondary hidden">⏹ Stop</button>
|
|
59
|
+
</div>
|
|
60
|
+
</div>
|
|
61
|
+
|
|
62
|
+
<div class="card">
|
|
63
|
+
<strong>Or paste a URL manually:</strong>
|
|
64
|
+
<div class="controls" style="margin-top:8px;">
|
|
65
|
+
<input id="manualUrl" type="text" placeholder="https://example.com/...">
|
|
66
|
+
<button id="manualScan" class="btn btn-primary">Verify</button>
|
|
67
|
+
</div>
|
|
68
|
+
</div>
|
|
69
|
+
|
|
70
|
+
<div id="result" class="card hidden"></div>
|
|
71
|
+
|
|
72
|
+
<p class="footer-note">
|
|
73
|
+
Powered by <a href="/">Web Agent Bridge</a> · ShieldQR is open and free for everyone ·
|
|
74
|
+
<a href="/api/shieldqr/recent" target="_blank">Recent scans</a>
|
|
75
|
+
</p>
|
|
76
|
+
</div>
|
|
77
|
+
|
|
78
|
+
<script>
|
|
79
|
+
(function () {
|
|
80
|
+
const el = (id) => document.getElementById(id);
|
|
81
|
+
const H = (s) => String(s == null ? '' : s).replace(/[&<>"']/g, (c) =>
|
|
82
|
+
({'&':'&','<':'<','>':'>','"':'"',"'":'''}[c]));
|
|
83
|
+
|
|
84
|
+
let scanner = null; let lastResult = null;
|
|
85
|
+
|
|
86
|
+
function startCamera() {
|
|
87
|
+
if (typeof Html5Qrcode === 'undefined') {
|
|
88
|
+
alert('QR library failed to load. Check your network or use the manual URL field.');
|
|
89
|
+
return;
|
|
90
|
+
}
|
|
91
|
+
el('startCam').classList.add('hidden');
|
|
92
|
+
el('stopCam').classList.remove('hidden');
|
|
93
|
+
scanner = new Html5Qrcode('reader');
|
|
94
|
+
scanner.start(
|
|
95
|
+
{ facingMode: 'environment' },
|
|
96
|
+
{ fps: 10, qrbox: { width: 260, height: 260 } },
|
|
97
|
+
(decodedText) => {
|
|
98
|
+
scanner.stop().catch(() => {});
|
|
99
|
+
el('startCam').classList.remove('hidden');
|
|
100
|
+
el('stopCam').classList.add('hidden');
|
|
101
|
+
verify(decodedText);
|
|
102
|
+
},
|
|
103
|
+
() => {} // ignore per-frame errors
|
|
104
|
+
).catch((err) => {
|
|
105
|
+
alert('Camera error: ' + err);
|
|
106
|
+
el('startCam').classList.remove('hidden');
|
|
107
|
+
el('stopCam').classList.add('hidden');
|
|
108
|
+
});
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
function stopCamera() {
|
|
112
|
+
if (!scanner) return;
|
|
113
|
+
scanner.stop().catch(() => {}).finally(() => {
|
|
114
|
+
el('startCam').classList.remove('hidden');
|
|
115
|
+
el('stopCam').classList.add('hidden');
|
|
116
|
+
scanner = null;
|
|
117
|
+
});
|
|
118
|
+
}
|
|
119
|
+
|
|
120
|
+
async function verify(url) {
|
|
121
|
+
const result = el('result');
|
|
122
|
+
result.classList.remove('hidden');
|
|
123
|
+
result.innerHTML = '<div style="text-align:center;padding:20px;">⏳ Verifying <strong>' + H(url) + '</strong>…</div>';
|
|
124
|
+
try {
|
|
125
|
+
const r = await fetch('/api/shieldqr/scan', {
|
|
126
|
+
method: 'POST',
|
|
127
|
+
headers: { 'Content-Type': 'application/json' },
|
|
128
|
+
body: JSON.stringify({ url }),
|
|
129
|
+
});
|
|
130
|
+
const data = await r.json();
|
|
131
|
+
if (!r.ok) throw new Error(data.error || 'scan failed');
|
|
132
|
+
lastResult = data;
|
|
133
|
+
renderVerdict(data, url);
|
|
134
|
+
} catch (err) {
|
|
135
|
+
result.innerHTML = '<div style="color:#fca5a5;text-align:center;padding:20px;">❌ ' + H(err.message) + '</div>';
|
|
136
|
+
}
|
|
137
|
+
}
|
|
138
|
+
|
|
139
|
+
function renderVerdict(d, url) {
|
|
140
|
+
const lvl = d.level || 'yellow';
|
|
141
|
+
const icon = { green: '✅', yellow: '⚠️', red: '🛑' }[lvl] || '⚠️';
|
|
142
|
+
const headline = { green: 'Safe — verified', yellow: 'Caution — incomplete trust', red: 'Dangerous — do not open' }[lvl];
|
|
143
|
+
const sslDays = d.ssl && d.ssl.days_until_expiry != null ? d.ssl.days_until_expiry : '—';
|
|
144
|
+
const trustOk = d.trust && d.trust.ok ? '✅ Signed' : '—';
|
|
145
|
+
const dnsOk = d.dns && Array.isArray(d.dns.discovery) && d.dns.discovery.length ? '✅ Present' : '—';
|
|
146
|
+
const signals = (d.signals || []).map((s) => `
|
|
147
|
+
<div class="signal">
|
|
148
|
+
<span class="sev sev-${H(s.severity || 'low')}">${H(s.severity || 'low')}</span>
|
|
149
|
+
<div><strong>${H(s.id || '')}</strong> — ${H(s.message || '')}</div>
|
|
150
|
+
</div>
|
|
151
|
+
`).join('') || '<div class="signal"><span class="sev sev-low">ok</span><div>No signals raised.</div></div>';
|
|
152
|
+
|
|
153
|
+
const openBtn = lvl === 'green'
|
|
154
|
+
? `<a class="btn btn-success" href="${H(url)}" target="_blank" rel="noopener noreferrer">Open destination →</a>`
|
|
155
|
+
: `<button class="btn btn-danger" data-action="forceOpen">Open anyway (risky)</button>`;
|
|
156
|
+
|
|
157
|
+
el('result').innerHTML = `
|
|
158
|
+
<div class="verdict ${H(lvl)}">
|
|
159
|
+
<div style="display:flex;justify-content:space-between;align-items:flex-start;gap:14px;flex-wrap:wrap;">
|
|
160
|
+
<div>
|
|
161
|
+
<h2>${icon} ${H(headline)}</h2>
|
|
162
|
+
<div class="url">${H(url)}</div>
|
|
163
|
+
</div>
|
|
164
|
+
<div style="text-align:right;">
|
|
165
|
+
<div class="score">${d.score ?? '—'}</div>
|
|
166
|
+
<div style="font-size:.7rem;color:#94a3b8;text-transform:uppercase;letter-spacing:.05em;">score / 100</div>
|
|
167
|
+
</div>
|
|
168
|
+
</div>
|
|
169
|
+
|
|
170
|
+
<div class="meta">
|
|
171
|
+
<div><div class="lbl">Host</div>${H(d.host || '—')}</div>
|
|
172
|
+
<div><div class="lbl">DNS _wab</div>${dnsOk}</div>
|
|
173
|
+
<div><div class="lbl">Signed wab.json</div>${trustOk}</div>
|
|
174
|
+
<div><div class="lbl">SSL valid for</div>${sslDays} days</div>
|
|
175
|
+
</div>
|
|
176
|
+
|
|
177
|
+
<div class="signals">${signals}</div>
|
|
178
|
+
|
|
179
|
+
<div class="controls" style="margin-top:14px;">
|
|
180
|
+
${openBtn}
|
|
181
|
+
<button class="btn btn-secondary" data-action="rescan">Scan another</button>
|
|
182
|
+
<button class="btn btn-danger" data-action="report">Report this URL</button>
|
|
183
|
+
</div>
|
|
184
|
+
</div>
|
|
185
|
+
`;
|
|
186
|
+
|
|
187
|
+
el('result').querySelector('[data-action="rescan"]').addEventListener('click', () => {
|
|
188
|
+
el('result').classList.add('hidden');
|
|
189
|
+
el('manualUrl').value = '';
|
|
190
|
+
});
|
|
191
|
+
const forceBtn = el('result').querySelector('[data-action="forceOpen"]');
|
|
192
|
+
if (forceBtn) {
|
|
193
|
+
forceBtn.addEventListener('click', () => {
|
|
194
|
+
if (confirm('This site has not been verified as safe. Open anyway?')) {
|
|
195
|
+
window.open(url, '_blank', 'noopener,noreferrer');
|
|
196
|
+
}
|
|
197
|
+
});
|
|
198
|
+
}
|
|
199
|
+
el('result').querySelector('[data-action="report"]').addEventListener('click', () => reportUrl(url));
|
|
200
|
+
}
|
|
201
|
+
|
|
202
|
+
async function reportUrl(url) {
|
|
203
|
+
const reason = prompt('Why are you reporting this URL? (e.g. phishing, malware, scam)');
|
|
204
|
+
if (!reason || reason.trim().length < 4) return;
|
|
205
|
+
try {
|
|
206
|
+
const r = await fetch('/api/shieldqr/report', {
|
|
207
|
+
method: 'POST',
|
|
208
|
+
headers: { 'Content-Type': 'application/json' },
|
|
209
|
+
body: JSON.stringify({ url, reason: reason.trim(), scan_id: lastResult && lastResult.scan_id }),
|
|
210
|
+
});
|
|
211
|
+
const data = await r.json();
|
|
212
|
+
if (!r.ok) throw new Error(data.error || 'report failed');
|
|
213
|
+
alert('Thank you — your report has been submitted.');
|
|
214
|
+
} catch (err) {
|
|
215
|
+
alert('Report failed: ' + err.message);
|
|
216
|
+
}
|
|
217
|
+
}
|
|
218
|
+
|
|
219
|
+
el('startCam').addEventListener('click', startCamera);
|
|
220
|
+
el('stopCam').addEventListener('click', stopCamera);
|
|
221
|
+
el('manualScan').addEventListener('click', () => {
|
|
222
|
+
const u = el('manualUrl').value.trim();
|
|
223
|
+
if (u) verify(u);
|
|
224
|
+
});
|
|
225
|
+
el('manualUrl').addEventListener('keydown', (e) => {
|
|
226
|
+
if (e.key === 'Enter') { e.preventDefault(); el('manualScan').click(); }
|
|
227
|
+
});
|
|
228
|
+
})();
|
|
229
|
+
</script>
|
|
230
|
+
</body>
|
|
231
|
+
</html>
|
package/public/sitemap.xml
CHANGED
|
@@ -2,16 +2,34 @@
|
|
|
2
2
|
<urlset xmlns="http://www.sitemaps.org/schemas/sitemap/0.9">
|
|
3
3
|
<url>
|
|
4
4
|
<loc>https://webagentbridge.com/</loc>
|
|
5
|
-
<lastmod>2026-
|
|
5
|
+
<lastmod>2026-05-13</lastmod>
|
|
6
6
|
<changefreq>weekly</changefreq>
|
|
7
7
|
<priority>1.0</priority>
|
|
8
8
|
</url>
|
|
9
|
+
<url>
|
|
10
|
+
<loc>https://webagentbridge.com/ring4</loc>
|
|
11
|
+
<lastmod>2026-05-13</lastmod>
|
|
12
|
+
<changefreq>weekly</changefreq>
|
|
13
|
+
<priority>0.95</priority>
|
|
14
|
+
</url>
|
|
15
|
+
<url>
|
|
16
|
+
<loc>https://webagentbridge.com/milestones</loc>
|
|
17
|
+
<lastmod>2026-05-13</lastmod>
|
|
18
|
+
<changefreq>weekly</changefreq>
|
|
19
|
+
<priority>0.9</priority>
|
|
20
|
+
</url>
|
|
9
21
|
<url>
|
|
10
22
|
<loc>https://webagentbridge.com/docs</loc>
|
|
11
23
|
<lastmod>2026-03-28</lastmod>
|
|
12
24
|
<changefreq>weekly</changefreq>
|
|
13
25
|
<priority>0.9</priority>
|
|
14
26
|
</url>
|
|
27
|
+
<url>
|
|
28
|
+
<loc>https://webagentbridge.com/whitepaper</loc>
|
|
29
|
+
<lastmod>2026-05-03</lastmod>
|
|
30
|
+
<changefreq>yearly</changefreq>
|
|
31
|
+
<priority>0.95</priority>
|
|
32
|
+
</url>
|
|
15
33
|
<url>
|
|
16
34
|
<loc>https://webagentbridge.com/premium</loc>
|
|
17
35
|
<lastmod>2026-03-28</lastmod>
|